nfsd4: tweak nfsd4_encode_getattr to take xdr_stream
[firefly-linux-kernel-4.4.55.git] / fs / nfsd / nfs4xdr.c
index 2723c1badd01276f9c1802d6cac210aa40f8f0a3..187925065d162933e3abbb5c45a63dcc36118b27 100644 (file)
@@ -1677,11 +1677,6 @@ nfsd4_decode_compound(struct nfsd4_compoundargs *argp)
                        op->opnum = OP_ILLEGAL;
                        op->status = nfserr_op_illegal;
                }
-
-               if (op->status) {
-                       argp->opcnt = i+1;
-                       break;
-               }
                /*
                 * We'll try to cache the result in the DRC if any one
                 * op in the compound wants to be cached:
@@ -1689,6 +1684,11 @@ nfsd4_decode_compound(struct nfsd4_compoundargs *argp)
                cachethis |= nfsd4_cache_this_op(op);
 
                max_reply = max(max_reply, nfsd4_max_reply(op->opnum));
+
+               if (op->status) {
+                       argp->opcnt = i+1;
+                       break;
+               }
        }
        /* Sessions make the DRC unnecessary: */
        if (argp->minorversion)
@@ -1747,10 +1747,10 @@ static void write_cinfo(__be32 **p, struct nfsd4_change_info *c)
 }
 
 #define RESERVE_SPACE(nbytes)  do {                            \
-       p = resp->p;                                            \
-       BUG_ON(p + XDR_QUADLEN(nbytes) > resp->end);            \
+       p = resp->xdr.p;                                                \
+       BUG_ON(p + XDR_QUADLEN(nbytes) > resp->xdr.end);                \
 } while (0)
-#define ADJUST_ARGS()          resp->p = p
+#define ADJUST_ARGS()          resp->xdr.p = p
 
 /* Encode as an array of strings the string given with components
  * separated @sep, escaped with esc_enter and esc_exit.
@@ -2045,12 +2045,11 @@ static int get_parent_attributes(struct svc_export *exp, struct kstat *stat)
 /*
  * Note: @fhp can be NULL; in this case, we might have to compose the filehandle
  * ourselves.
- *
- * countp is the buffer size in _words_
  */
 __be32
-nfsd4_encode_fattr(struct svc_fh *fhp, struct svc_export *exp,
-               struct dentry *dentry, __be32 **buffer, int count, u32 *bmval,
+nfsd4_encode_fattr(struct xdr_stream *xdr, struct svc_fh *fhp,
+               struct svc_export *exp,
+               struct dentry *dentry, u32 *bmval,
                struct svc_rqst *rqstp, int ignore_crossmnt)
 {
        u32 bmval0 = bmval[0];
@@ -2059,12 +2058,12 @@ nfsd4_encode_fattr(struct svc_fh *fhp, struct svc_export *exp,
        struct kstat stat;
        struct svc_fh *tempfh = NULL;
        struct kstatfs statfs;
-       int buflen = count << 2;
+       __be32 *p = xdr->p;
+       int buflen = xdr->buf->buflen;
        __be32 *attrlenp;
        u32 dummy;
        u64 dummy64;
        u32 rdattr_err = 0;
-       __be32 *p = *buffer;
        __be32 status;
        int err;
        int aclsupport = 0;
@@ -2491,7 +2490,7 @@ out_acl:
        }
 
        *attrlenp = htonl((char *)p - (char *)attrlenp - 4);
-       *buffer = p;
+       xdr->p = p;
        status = nfs_ok;
 
 out:
@@ -2513,6 +2512,27 @@ out_resource:
        goto out;
 }
 
+__be32 nfsd4_encode_fattr_to_buf(__be32 **p, int words,
+                       struct svc_fh *fhp, struct svc_export *exp,
+                       struct dentry *dentry, u32 *bmval,
+                       struct svc_rqst *rqstp, int ignore_crossmnt)
+{
+       struct xdr_buf dummy = {
+                       .head[0] = {
+                               .iov_base = *p,
+                       },
+                       .buflen = words << 2,
+               };
+       struct xdr_stream xdr;
+       __be32 ret;
+
+       xdr_init_encode(&xdr, &dummy, NULL);
+       ret = nfsd4_encode_fattr(&xdr, fhp, exp, dentry, bmval, rqstp,
+                                                       ignore_crossmnt);
+       *p = xdr.p;
+       return ret;
+}
+
 static inline int attributes_need_mount(u32 *bmval)
 {
        if (bmval[0] & ~(FATTR4_WORD0_RDATTR_ERROR | FATTR4_WORD0_LEASE_TIME))
@@ -2576,7 +2596,8 @@ nfsd4_encode_dirent_fattr(struct nfsd4_readdir *cd,
 
        }
 out_encode:
-       nfserr = nfsd4_encode_fattr(NULL, exp, dentry, p, buflen, cd->rd_bmval,
+       nfserr = nfsd4_encode_fattr_to_buf(p, buflen, NULL, exp, dentry,
+                                       cd->rd_bmval,
                                        cd->rd_rqstp, ignore_crossmnt);
 out_put:
        dput(dentry);
@@ -2746,14 +2767,16 @@ static __be32
 nfsd4_encode_getattr(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd4_getattr *getattr)
 {
        struct svc_fh *fhp = getattr->ga_fhp;
-       int buflen;
+       struct xdr_stream *xdr = &resp->xdr;
+       struct xdr_buf *buf = resp->xdr.buf;
 
        if (nfserr)
                return nfserr;
 
-       buflen = resp->end - resp->p - (COMPOUND_ERR_SLACK_SPACE >> 2);
-       nfserr = nfsd4_encode_fattr(fhp, fhp->fh_export, fhp->fh_dentry,
-                                   &resp->p, buflen, getattr->ga_bmval,
+       buf->buflen = (void *)resp->xdr.end - (void *)resp->xdr.p
+                       - COMPOUND_ERR_SLACK_SPACE;
+       nfserr = nfsd4_encode_fattr(xdr, fhp, fhp->fh_export, fhp->fh_dentry,
+                                   getattr->ga_bmval,
                                    resp->rqstp, 0);
        return nfserr;
 }
@@ -2953,7 +2976,7 @@ nfsd4_encode_read(struct nfsd4_compoundres *resp, __be32 nfserr,
 
        if (nfserr)
                return nfserr;
-       if (resp->xbuf->page_len)
+       if (resp->xdr.buf->page_len)
                return nfserr_resource;
 
        RESERVE_SPACE(8); /* eof flag and byte count */
@@ -2991,18 +3014,18 @@ nfsd4_encode_read(struct nfsd4_compoundres *resp, __be32 nfserr,
        WRITE32(eof);
        WRITE32(maxcount);
        ADJUST_ARGS();
-       resp->xbuf->head[0].iov_len = (char*)p
-                                       - (char*)resp->xbuf->head[0].iov_base;
-       resp->xbuf->page_len = maxcount;
+       resp->xdr.buf->head[0].iov_len = (char *)p
+                               - (char *)resp->xdr.buf->head[0].iov_base;
+       resp->xdr.buf->page_len = maxcount;
 
        /* Use rest of head for padding and remaining ops: */
-       resp->xbuf->tail[0].iov_base = p;
-       resp->xbuf->tail[0].iov_len = 0;
+       resp->xdr.buf->tail[0].iov_base = p;
+       resp->xdr.buf->tail[0].iov_len = 0;
        if (maxcount&3) {
                RESERVE_SPACE(4);
                WRITE32(0);
-               resp->xbuf->tail[0].iov_base += maxcount&3;
-               resp->xbuf->tail[0].iov_len = 4 - (maxcount&3);
+               resp->xdr.buf->tail[0].iov_base += maxcount&3;
+               resp->xdr.buf->tail[0].iov_len = 4 - (maxcount&3);
                ADJUST_ARGS();
        }
        return 0;
@@ -3017,7 +3040,7 @@ nfsd4_encode_readlink(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd
 
        if (nfserr)
                return nfserr;
-       if (resp->xbuf->page_len)
+       if (resp->xdr.buf->page_len)
                return nfserr_resource;
        if (!*resp->rqstp->rq_next_page)
                return nfserr_resource;
@@ -3041,18 +3064,18 @@ nfsd4_encode_readlink(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd
 
        WRITE32(maxcount);
        ADJUST_ARGS();
-       resp->xbuf->head[0].iov_len = (char*)p
-                               - (char*)resp->xbuf->head[0].iov_base;
-       resp->xbuf->page_len = maxcount;
+       resp->xdr.buf->head[0].iov_len = (char *)p
+                               - (char *)resp->xdr.buf->head[0].iov_base;
+       resp->xdr.buf->page_len = maxcount;
 
        /* Use rest of head for padding and remaining ops: */
-       resp->xbuf->tail[0].iov_base = p;
-       resp->xbuf->tail[0].iov_len = 0;
+       resp->xdr.buf->tail[0].iov_base = p;
+       resp->xdr.buf->tail[0].iov_len = 0;
        if (maxcount&3) {
                RESERVE_SPACE(4);
                WRITE32(0);
-               resp->xbuf->tail[0].iov_base += maxcount&3;
-               resp->xbuf->tail[0].iov_len = 4 - (maxcount&3);
+               resp->xdr.buf->tail[0].iov_base += maxcount&3;
+               resp->xdr.buf->tail[0].iov_len = 4 - (maxcount&3);
                ADJUST_ARGS();
        }
        return 0;
@@ -3068,7 +3091,7 @@ nfsd4_encode_readdir(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd4
 
        if (nfserr)
                return nfserr;
-       if (resp->xbuf->page_len)
+       if (resp->xdr.buf->page_len)
                return nfserr_resource;
        if (!*resp->rqstp->rq_next_page)
                return nfserr_resource;
@@ -3080,7 +3103,8 @@ nfsd4_encode_readdir(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd4
        WRITE32(0);
        WRITE32(0);
        ADJUST_ARGS();
-       resp->xbuf->head[0].iov_len = ((char*)resp->p) - (char*)resp->xbuf->head[0].iov_base;
+       resp->xdr.buf->head[0].iov_len = ((char *)resp->xdr.p)
+                               - (char *)resp->xdr.buf->head[0].iov_base;
        tailbase = p;
 
        maxcount = PAGE_SIZE;
@@ -3121,14 +3145,15 @@ nfsd4_encode_readdir(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd4
        p = readdir->buffer;
        *p++ = 0;       /* no more entries */
        *p++ = htonl(readdir->common.err == nfserr_eof);
-       resp->xbuf->page_len = ((char*)p) -
+       resp->xdr.buf->page_len = ((char *)p) -
                (char*)page_address(*(resp->rqstp->rq_next_page-1));
 
        /* Use rest of head for padding and remaining ops: */
-       resp->xbuf->tail[0].iov_base = tailbase;
-       resp->xbuf->tail[0].iov_len = 0;
-       resp->p = resp->xbuf->tail[0].iov_base;
-       resp->end = resp->p + (PAGE_SIZE - resp->xbuf->head[0].iov_len)/4;
+       resp->xdr.buf->tail[0].iov_base = tailbase;
+       resp->xdr.buf->tail[0].iov_len = 0;
+       resp->xdr.p = resp->xdr.buf->tail[0].iov_base;
+       resp->xdr.end = resp->xdr.p +
+                       (PAGE_SIZE - resp->xdr.buf->head[0].iov_len)/4;
 
        return 0;
 err_no_verf:
@@ -3587,10 +3612,10 @@ __be32 nfsd4_check_resp_size(struct nfsd4_compoundres *resp, u32 pad)
        session = resp->cstate.session;
 
        if (xb->page_len == 0) {
-               length = (char *)resp->p - (char *)xb->head[0].iov_base + pad;
+               length = (char *)resp->xdr.p - (char *)xb->head[0].iov_base + pad;
        } else {
                if (xb->tail[0].iov_base && xb->tail[0].iov_len > 0)
-                       tlen = (char *)resp->p - (char *)xb->tail[0].iov_base;
+                       tlen = (char *)resp->xdr.p - (char *)xb->tail[0].iov_base;
 
                length = xb->head[0].iov_len + xb->page_len + tlen + pad;
        }
@@ -3627,17 +3652,10 @@ nfsd4_encode_operation(struct nfsd4_compoundres *resp, struct nfsd4_op *op)
        /* nfsd4_check_resp_size guarantees enough room for error status */
        if (!op->status)
                op->status = nfsd4_check_resp_size(resp, 0);
-       if (op->status == nfserr_resource && nfsd4_has_session(&resp->cstate)) {
-               struct nfsd4_slot *slot = resp->cstate.slot;
-
-               if (slot->sl_flags & NFSD4_SLOT_CACHETHIS)
-                       op->status = nfserr_rep_too_big_to_cache;
-               else
-                       op->status = nfserr_rep_too_big;
-       }
        if (so) {
                so->so_replay.rp_status = op->status;
-               so->so_replay.rp_buflen = (char *)resp->p - (char *)(statp+1);
+               so->so_replay.rp_buflen = (char *)resp->xdr.p
+                                               - (char *)(statp+1);
                memcpy(so->so_replay.rp_buf, statp+1, so->so_replay.rp_buflen);
        }
 status:
@@ -3739,7 +3757,7 @@ nfs4svc_encode_compoundres(struct svc_rqst *rqstp, __be32 *p, struct nfsd4_compo
                iov = &rqstp->rq_res.tail[0];
        else
                iov = &rqstp->rq_res.head[0];
-       iov->iov_len = ((char*)resp->p) - (char*)iov->iov_base;
+       iov->iov_len = ((char *)resp->xdr.p) - (char *)iov->iov_base;
        BUG_ON(iov->iov_len > PAGE_SIZE);
        if (nfsd4_has_session(cs)) {
                struct nfsd_net *nn = net_generic(SVC_NET(rqstp), nfsd_net_id);