|
@@ -45,6 +45,7 @@
|
|
|
#include <linux/fs.h>
|
|
|
#include <linux/namei.h>
|
|
|
#include <linux/vfs.h>
|
|
|
+#include <linux/utsname.h>
|
|
|
#include <linux/sunrpc/xdr.h>
|
|
|
#include <linux/sunrpc/svc.h>
|
|
|
#include <linux/sunrpc/clnt.h>
|
|
@@ -188,6 +189,11 @@ static __be32 *read_buf(struct nfsd4_compoundargs *argp, u32 nbytes)
|
|
|
return p;
|
|
|
}
|
|
|
|
|
|
+static int zero_clientid(clientid_t *clid)
|
|
|
+{
|
|
|
+ return (clid->cl_boot == 0) && (clid->cl_id == 0);
|
|
|
+}
|
|
|
+
|
|
|
static int
|
|
|
defer_free(struct nfsd4_compoundargs *argp,
|
|
|
void (*release)(const void *), void *p)
|
|
@@ -230,6 +236,7 @@ nfsd4_decode_bitmap(struct nfsd4_compoundargs *argp, u32 *bmval)
|
|
|
|
|
|
bmval[0] = 0;
|
|
|
bmval[1] = 0;
|
|
|
+ bmval[2] = 0;
|
|
|
|
|
|
READ_BUF(4);
|
|
|
READ32(bmlen);
|
|
@@ -241,13 +248,27 @@ nfsd4_decode_bitmap(struct nfsd4_compoundargs *argp, u32 *bmval)
|
|
|
READ32(bmval[0]);
|
|
|
if (bmlen > 1)
|
|
|
READ32(bmval[1]);
|
|
|
+ if (bmlen > 2)
|
|
|
+ READ32(bmval[2]);
|
|
|
|
|
|
DECODE_TAIL;
|
|
|
}
|
|
|
|
|
|
+static u32 nfsd_attrmask[] = {
|
|
|
+ NFSD_WRITEABLE_ATTRS_WORD0,
|
|
|
+ NFSD_WRITEABLE_ATTRS_WORD1,
|
|
|
+ NFSD_WRITEABLE_ATTRS_WORD2
|
|
|
+};
|
|
|
+
|
|
|
+static u32 nfsd41_ex_attrmask[] = {
|
|
|
+ NFSD_SUPPATTR_EXCLCREAT_WORD0,
|
|
|
+ NFSD_SUPPATTR_EXCLCREAT_WORD1,
|
|
|
+ NFSD_SUPPATTR_EXCLCREAT_WORD2
|
|
|
+};
|
|
|
+
|
|
|
static __be32
|
|
|
-nfsd4_decode_fattr(struct nfsd4_compoundargs *argp, u32 *bmval, struct iattr *iattr,
|
|
|
- struct nfs4_acl **acl)
|
|
|
+nfsd4_decode_fattr(struct nfsd4_compoundargs *argp, u32 *bmval, u32 *writable,
|
|
|
+ struct iattr *iattr, struct nfs4_acl **acl)
|
|
|
{
|
|
|
int expected_len, len = 0;
|
|
|
u32 dummy32;
|
|
@@ -263,9 +284,12 @@ nfsd4_decode_fattr(struct nfsd4_compoundargs *argp, u32 *bmval, struct iattr *ia
|
|
|
* According to spec, unsupported attributes return ERR_ATTRNOTSUPP;
|
|
|
* read-only attributes return ERR_INVAL.
|
|
|
*/
|
|
|
- if ((bmval[0] & ~NFSD_SUPPORTED_ATTRS_WORD0) || (bmval[1] & ~NFSD_SUPPORTED_ATTRS_WORD1))
|
|
|
+ if ((bmval[0] & ~nfsd_suppattrs0(argp->minorversion)) ||
|
|
|
+ (bmval[1] & ~nfsd_suppattrs1(argp->minorversion)) ||
|
|
|
+ (bmval[2] & ~nfsd_suppattrs2(argp->minorversion)))
|
|
|
return nfserr_attrnotsupp;
|
|
|
- if ((bmval[0] & ~NFSD_WRITEABLE_ATTRS_WORD0) || (bmval[1] & ~NFSD_WRITEABLE_ATTRS_WORD1))
|
|
|
+ if ((bmval[0] & ~writable[0]) || (bmval[1] & ~writable[1]) ||
|
|
|
+ (bmval[2] & ~writable[2]))
|
|
|
return nfserr_inval;
|
|
|
|
|
|
READ_BUF(4);
|
|
@@ -400,6 +424,7 @@ nfsd4_decode_fattr(struct nfsd4_compoundargs *argp, u32 *bmval, struct iattr *ia
|
|
|
goto xdr_error;
|
|
|
}
|
|
|
}
|
|
|
+ BUG_ON(bmval[2]); /* no such writeable attr supported yet */
|
|
|
if (len != expected_len)
|
|
|
goto xdr_error;
|
|
|
|
|
@@ -493,7 +518,9 @@ nfsd4_decode_create(struct nfsd4_compoundargs *argp, struct nfsd4_create *create
|
|
|
if ((status = check_filename(create->cr_name, create->cr_namelen, nfserr_inval)))
|
|
|
return status;
|
|
|
|
|
|
- if ((status = nfsd4_decode_fattr(argp, create->cr_bmval, &create->cr_iattr, &create->cr_acl)))
|
|
|
+ status = nfsd4_decode_fattr(argp, create->cr_bmval, nfsd_attrmask,
|
|
|
+ &create->cr_iattr, &create->cr_acl);
|
|
|
+ if (status)
|
|
|
goto out;
|
|
|
|
|
|
DECODE_TAIL;
|
|
@@ -583,6 +610,8 @@ nfsd4_decode_lockt(struct nfsd4_compoundargs *argp, struct nfsd4_lockt *lockt)
|
|
|
READ_BUF(lockt->lt_owner.len);
|
|
|
READMEM(lockt->lt_owner.data, lockt->lt_owner.len);
|
|
|
|
|
|
+ if (argp->minorversion && !zero_clientid(&lockt->lt_clientid))
|
|
|
+ return nfserr_inval;
|
|
|
DECODE_TAIL;
|
|
|
}
|
|
|
|
|
@@ -652,13 +681,26 @@ nfsd4_decode_open(struct nfsd4_compoundargs *argp, struct nfsd4_open *open)
|
|
|
switch (open->op_createmode) {
|
|
|
case NFS4_CREATE_UNCHECKED:
|
|
|
case NFS4_CREATE_GUARDED:
|
|
|
- if ((status = nfsd4_decode_fattr(argp, open->op_bmval, &open->op_iattr, &open->op_acl)))
|
|
|
+ status = nfsd4_decode_fattr(argp, open->op_bmval,
|
|
|
+ nfsd_attrmask, &open->op_iattr, &open->op_acl);
|
|
|
+ if (status)
|
|
|
goto out;
|
|
|
break;
|
|
|
case NFS4_CREATE_EXCLUSIVE:
|
|
|
READ_BUF(8);
|
|
|
COPYMEM(open->op_verf.data, 8);
|
|
|
break;
|
|
|
+ case NFS4_CREATE_EXCLUSIVE4_1:
|
|
|
+ if (argp->minorversion < 1)
|
|
|
+ goto xdr_error;
|
|
|
+ READ_BUF(8);
|
|
|
+ COPYMEM(open->op_verf.data, 8);
|
|
|
+ status = nfsd4_decode_fattr(argp, open->op_bmval,
|
|
|
+ nfsd41_ex_attrmask, &open->op_iattr,
|
|
|
+ &open->op_acl);
|
|
|
+ if (status)
|
|
|
+ goto out;
|
|
|
+ break;
|
|
|
default:
|
|
|
goto xdr_error;
|
|
|
}
|
|
@@ -851,7 +893,7 @@ nfsd4_decode_setattr(struct nfsd4_compoundargs *argp, struct nfsd4_setattr *seta
|
|
|
status = nfsd4_decode_stateid(argp, &setattr->sa_stateid);
|
|
|
if (status)
|
|
|
return status;
|
|
|
- return nfsd4_decode_fattr(argp, setattr->sa_bmval,
|
|
|
+ return nfsd4_decode_fattr(argp, setattr->sa_bmval, nfsd_attrmask,
|
|
|
&setattr->sa_iattr, &setattr->sa_acl);
|
|
|
}
|
|
|
|
|
@@ -993,6 +1035,241 @@ nfsd4_decode_release_lockowner(struct nfsd4_compoundargs *argp, struct nfsd4_rel
|
|
|
READ_BUF(rlockowner->rl_owner.len);
|
|
|
READMEM(rlockowner->rl_owner.data, rlockowner->rl_owner.len);
|
|
|
|
|
|
+ if (argp->minorversion && !zero_clientid(&rlockowner->rl_clientid))
|
|
|
+ return nfserr_inval;
|
|
|
+ DECODE_TAIL;
|
|
|
+}
|
|
|
+
|
|
|
+static __be32
|
|
|
+nfsd4_decode_exchange_id(struct nfsd4_compoundargs *argp,
|
|
|
+ struct nfsd4_exchange_id *exid)
|
|
|
+{
|
|
|
+ int dummy;
|
|
|
+ DECODE_HEAD;
|
|
|
+
|
|
|
+ READ_BUF(NFS4_VERIFIER_SIZE);
|
|
|
+ COPYMEM(exid->verifier.data, NFS4_VERIFIER_SIZE);
|
|
|
+
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(exid->clname.len);
|
|
|
+
|
|
|
+ READ_BUF(exid->clname.len);
|
|
|
+ SAVEMEM(exid->clname.data, exid->clname.len);
|
|
|
+
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(exid->flags);
|
|
|
+
|
|
|
+ /* Ignore state_protect4_a */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(exid->spa_how);
|
|
|
+ switch (exid->spa_how) {
|
|
|
+ case SP4_NONE:
|
|
|
+ break;
|
|
|
+ case SP4_MACH_CRED:
|
|
|
+ /* spo_must_enforce */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy * 4);
|
|
|
+ p += dummy;
|
|
|
+
|
|
|
+ /* spo_must_allow */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy * 4);
|
|
|
+ p += dummy;
|
|
|
+ break;
|
|
|
+ case SP4_SSV:
|
|
|
+ /* ssp_ops */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy * 4);
|
|
|
+ p += dummy;
|
|
|
+
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy * 4);
|
|
|
+ p += dummy;
|
|
|
+
|
|
|
+ /* ssp_hash_algs<> */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ p += XDR_QUADLEN(dummy);
|
|
|
+
|
|
|
+ /* ssp_encr_algs<> */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ p += XDR_QUADLEN(dummy);
|
|
|
+
|
|
|
+ /* ssp_window and ssp_num_gss_handles */
|
|
|
+ READ_BUF(8);
|
|
|
+ READ32(dummy);
|
|
|
+ READ32(dummy);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ goto xdr_error;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Ignore Implementation ID */
|
|
|
+ READ_BUF(4); /* nfs_impl_id4 array length */
|
|
|
+ READ32(dummy);
|
|
|
+
|
|
|
+ if (dummy > 1)
|
|
|
+ goto xdr_error;
|
|
|
+
|
|
|
+ if (dummy == 1) {
|
|
|
+ /* nii_domain */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ p += XDR_QUADLEN(dummy);
|
|
|
+
|
|
|
+ /* nii_name */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ p += XDR_QUADLEN(dummy);
|
|
|
+
|
|
|
+ /* nii_date */
|
|
|
+ READ_BUF(12);
|
|
|
+ p += 3;
|
|
|
+ }
|
|
|
+ DECODE_TAIL;
|
|
|
+}
|
|
|
+
|
|
|
+static __be32
|
|
|
+nfsd4_decode_create_session(struct nfsd4_compoundargs *argp,
|
|
|
+ struct nfsd4_create_session *sess)
|
|
|
+{
|
|
|
+ DECODE_HEAD;
|
|
|
+
|
|
|
+ u32 dummy;
|
|
|
+ char *machine_name;
|
|
|
+ int i;
|
|
|
+ int nr_secflavs;
|
|
|
+
|
|
|
+ READ_BUF(16);
|
|
|
+ COPYMEM(&sess->clientid, 8);
|
|
|
+ READ32(sess->seqid);
|
|
|
+ READ32(sess->flags);
|
|
|
+
|
|
|
+ /* Fore channel attrs */
|
|
|
+ READ_BUF(28);
|
|
|
+ READ32(dummy); /* headerpadsz is always 0 */
|
|
|
+ READ32(sess->fore_channel.maxreq_sz);
|
|
|
+ READ32(sess->fore_channel.maxresp_sz);
|
|
|
+ READ32(sess->fore_channel.maxresp_cached);
|
|
|
+ READ32(sess->fore_channel.maxops);
|
|
|
+ READ32(sess->fore_channel.maxreqs);
|
|
|
+ READ32(sess->fore_channel.nr_rdma_attrs);
|
|
|
+ if (sess->fore_channel.nr_rdma_attrs == 1) {
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(sess->fore_channel.rdma_attrs);
|
|
|
+ } else if (sess->fore_channel.nr_rdma_attrs > 1) {
|
|
|
+ dprintk("Too many fore channel attr bitmaps!\n");
|
|
|
+ goto xdr_error;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Back channel attrs */
|
|
|
+ READ_BUF(28);
|
|
|
+ READ32(dummy); /* headerpadsz is always 0 */
|
|
|
+ READ32(sess->back_channel.maxreq_sz);
|
|
|
+ READ32(sess->back_channel.maxresp_sz);
|
|
|
+ READ32(sess->back_channel.maxresp_cached);
|
|
|
+ READ32(sess->back_channel.maxops);
|
|
|
+ READ32(sess->back_channel.maxreqs);
|
|
|
+ READ32(sess->back_channel.nr_rdma_attrs);
|
|
|
+ if (sess->back_channel.nr_rdma_attrs == 1) {
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(sess->back_channel.rdma_attrs);
|
|
|
+ } else if (sess->back_channel.nr_rdma_attrs > 1) {
|
|
|
+ dprintk("Too many back channel attr bitmaps!\n");
|
|
|
+ goto xdr_error;
|
|
|
+ }
|
|
|
+
|
|
|
+ READ_BUF(8);
|
|
|
+ READ32(sess->callback_prog);
|
|
|
+
|
|
|
+ /* callback_sec_params4 */
|
|
|
+ READ32(nr_secflavs);
|
|
|
+ for (i = 0; i < nr_secflavs; ++i) {
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ switch (dummy) {
|
|
|
+ case RPC_AUTH_NULL:
|
|
|
+ /* Nothing to read */
|
|
|
+ break;
|
|
|
+ case RPC_AUTH_UNIX:
|
|
|
+ READ_BUF(8);
|
|
|
+ /* stamp */
|
|
|
+ READ32(dummy);
|
|
|
+
|
|
|
+ /* machine name */
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ SAVEMEM(machine_name, dummy);
|
|
|
+
|
|
|
+ /* uid, gid */
|
|
|
+ READ_BUF(8);
|
|
|
+ READ32(sess->uid);
|
|
|
+ READ32(sess->gid);
|
|
|
+
|
|
|
+ /* more gids */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy * 4);
|
|
|
+ for (i = 0; i < dummy; ++i)
|
|
|
+ READ32(dummy);
|
|
|
+ break;
|
|
|
+ case RPC_AUTH_GSS:
|
|
|
+ dprintk("RPC_AUTH_GSS callback secflavor "
|
|
|
+ "not supported!\n");
|
|
|
+ READ_BUF(8);
|
|
|
+ /* gcbp_service */
|
|
|
+ READ32(dummy);
|
|
|
+ /* gcbp_handle_from_server */
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ p += XDR_QUADLEN(dummy);
|
|
|
+ /* gcbp_handle_from_client */
|
|
|
+ READ_BUF(4);
|
|
|
+ READ32(dummy);
|
|
|
+ READ_BUF(dummy);
|
|
|
+ p += XDR_QUADLEN(dummy);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ dprintk("Illegal callback secflavor\n");
|
|
|
+ return nfserr_inval;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ DECODE_TAIL;
|
|
|
+}
|
|
|
+
|
|
|
+static __be32
|
|
|
+nfsd4_decode_destroy_session(struct nfsd4_compoundargs *argp,
|
|
|
+ struct nfsd4_destroy_session *destroy_session)
|
|
|
+{
|
|
|
+ DECODE_HEAD;
|
|
|
+ READ_BUF(NFS4_MAX_SESSIONID_LEN);
|
|
|
+ COPYMEM(destroy_session->sessionid.data, NFS4_MAX_SESSIONID_LEN);
|
|
|
+
|
|
|
+ DECODE_TAIL;
|
|
|
+}
|
|
|
+
|
|
|
+static __be32
|
|
|
+nfsd4_decode_sequence(struct nfsd4_compoundargs *argp,
|
|
|
+ struct nfsd4_sequence *seq)
|
|
|
+{
|
|
|
+ DECODE_HEAD;
|
|
|
+
|
|
|
+ READ_BUF(NFS4_MAX_SESSIONID_LEN + 16);
|
|
|
+ COPYMEM(seq->sessionid.data, NFS4_MAX_SESSIONID_LEN);
|
|
|
+ READ32(seq->seqid);
|
|
|
+ READ32(seq->slotid);
|
|
|
+ READ32(seq->maxslots);
|
|
|
+ READ32(seq->cachethis);
|
|
|
+
|
|
|
DECODE_TAIL;
|
|
|
}
|
|
|
|
|
@@ -1005,7 +1282,7 @@ nfsd4_decode_noop(struct nfsd4_compoundargs *argp, void *p)
|
|
|
static __be32
|
|
|
nfsd4_decode_notsupp(struct nfsd4_compoundargs *argp, void *p)
|
|
|
{
|
|
|
- return nfserr_opnotsupp;
|
|
|
+ return nfserr_notsupp;
|
|
|
}
|
|
|
|
|
|
typedef __be32(*nfsd4_dec)(struct nfsd4_compoundargs *argp, void *);
|
|
@@ -1031,7 +1308,7 @@ static nfsd4_dec nfsd4_dec_ops[] = {
|
|
|
[OP_OPEN_CONFIRM] = (nfsd4_dec)nfsd4_decode_open_confirm,
|
|
|
[OP_OPEN_DOWNGRADE] = (nfsd4_dec)nfsd4_decode_open_downgrade,
|
|
|
[OP_PUTFH] = (nfsd4_dec)nfsd4_decode_putfh,
|
|
|
- [OP_PUTPUBFH] = (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_PUTPUBFH] = (nfsd4_dec)nfsd4_decode_noop,
|
|
|
[OP_PUTROOTFH] = (nfsd4_dec)nfsd4_decode_noop,
|
|
|
[OP_READ] = (nfsd4_dec)nfsd4_decode_read,
|
|
|
[OP_READDIR] = (nfsd4_dec)nfsd4_decode_readdir,
|
|
@@ -1050,6 +1327,67 @@ static nfsd4_dec nfsd4_dec_ops[] = {
|
|
|
[OP_RELEASE_LOCKOWNER] = (nfsd4_dec)nfsd4_decode_release_lockowner,
|
|
|
};
|
|
|
|
|
|
+static nfsd4_dec nfsd41_dec_ops[] = {
|
|
|
+ [OP_ACCESS] (nfsd4_dec)nfsd4_decode_access,
|
|
|
+ [OP_CLOSE] (nfsd4_dec)nfsd4_decode_close,
|
|
|
+ [OP_COMMIT] (nfsd4_dec)nfsd4_decode_commit,
|
|
|
+ [OP_CREATE] (nfsd4_dec)nfsd4_decode_create,
|
|
|
+ [OP_DELEGPURGE] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_DELEGRETURN] (nfsd4_dec)nfsd4_decode_delegreturn,
|
|
|
+ [OP_GETATTR] (nfsd4_dec)nfsd4_decode_getattr,
|
|
|
+ [OP_GETFH] (nfsd4_dec)nfsd4_decode_noop,
|
|
|
+ [OP_LINK] (nfsd4_dec)nfsd4_decode_link,
|
|
|
+ [OP_LOCK] (nfsd4_dec)nfsd4_decode_lock,
|
|
|
+ [OP_LOCKT] (nfsd4_dec)nfsd4_decode_lockt,
|
|
|
+ [OP_LOCKU] (nfsd4_dec)nfsd4_decode_locku,
|
|
|
+ [OP_LOOKUP] (nfsd4_dec)nfsd4_decode_lookup,
|
|
|
+ [OP_LOOKUPP] (nfsd4_dec)nfsd4_decode_noop,
|
|
|
+ [OP_NVERIFY] (nfsd4_dec)nfsd4_decode_verify,
|
|
|
+ [OP_OPEN] (nfsd4_dec)nfsd4_decode_open,
|
|
|
+ [OP_OPENATTR] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_OPEN_CONFIRM] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_OPEN_DOWNGRADE] (nfsd4_dec)nfsd4_decode_open_downgrade,
|
|
|
+ [OP_PUTFH] (nfsd4_dec)nfsd4_decode_putfh,
|
|
|
+ [OP_PUTPUBFH] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_PUTROOTFH] (nfsd4_dec)nfsd4_decode_noop,
|
|
|
+ [OP_READ] (nfsd4_dec)nfsd4_decode_read,
|
|
|
+ [OP_READDIR] (nfsd4_dec)nfsd4_decode_readdir,
|
|
|
+ [OP_READLINK] (nfsd4_dec)nfsd4_decode_noop,
|
|
|
+ [OP_REMOVE] (nfsd4_dec)nfsd4_decode_remove,
|
|
|
+ [OP_RENAME] (nfsd4_dec)nfsd4_decode_rename,
|
|
|
+ [OP_RENEW] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_RESTOREFH] (nfsd4_dec)nfsd4_decode_noop,
|
|
|
+ [OP_SAVEFH] (nfsd4_dec)nfsd4_decode_noop,
|
|
|
+ [OP_SECINFO] (nfsd4_dec)nfsd4_decode_secinfo,
|
|
|
+ [OP_SETATTR] (nfsd4_dec)nfsd4_decode_setattr,
|
|
|
+ [OP_SETCLIENTID] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_SETCLIENTID_CONFIRM](nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_VERIFY] (nfsd4_dec)nfsd4_decode_verify,
|
|
|
+ [OP_WRITE] (nfsd4_dec)nfsd4_decode_write,
|
|
|
+ [OP_RELEASE_LOCKOWNER] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+
|
|
|
+ /* new operations for NFSv4.1 */
|
|
|
+ [OP_BACKCHANNEL_CTL] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_BIND_CONN_TO_SESSION](nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_EXCHANGE_ID] (nfsd4_dec)nfsd4_decode_exchange_id,
|
|
|
+ [OP_CREATE_SESSION] (nfsd4_dec)nfsd4_decode_create_session,
|
|
|
+ [OP_DESTROY_SESSION] (nfsd4_dec)nfsd4_decode_destroy_session,
|
|
|
+ [OP_FREE_STATEID] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_GET_DIR_DELEGATION] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_GETDEVICEINFO] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_GETDEVICELIST] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_LAYOUTCOMMIT] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_LAYOUTGET] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_LAYOUTRETURN] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_SECINFO_NO_NAME] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_SEQUENCE] (nfsd4_dec)nfsd4_decode_sequence,
|
|
|
+ [OP_SET_SSV] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_TEST_STATEID] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_WANT_DELEGATION] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_DESTROY_CLIENTID] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+ [OP_RECLAIM_COMPLETE] (nfsd4_dec)nfsd4_decode_notsupp,
|
|
|
+};
|
|
|
+
|
|
|
struct nfsd4_minorversion_ops {
|
|
|
nfsd4_dec *decoders;
|
|
|
int nops;
|
|
@@ -1057,6 +1395,7 @@ struct nfsd4_minorversion_ops {
|
|
|
|
|
|
static struct nfsd4_minorversion_ops nfsd4_minorversion[] = {
|
|
|
[0] = { nfsd4_dec_ops, ARRAY_SIZE(nfsd4_dec_ops) },
|
|
|
+ [1] = { nfsd41_dec_ops, ARRAY_SIZE(nfsd41_dec_ops) },
|
|
|
};
|
|
|
|
|
|
static __be32
|
|
@@ -1412,6 +1751,7 @@ nfsd4_encode_fattr(struct svc_fh *fhp, struct svc_export *exp,
|
|
|
{
|
|
|
u32 bmval0 = bmval[0];
|
|
|
u32 bmval1 = bmval[1];
|
|
|
+ u32 bmval2 = bmval[2];
|
|
|
struct kstat stat;
|
|
|
struct svc_fh tempfh;
|
|
|
struct kstatfs statfs;
|
|
@@ -1425,12 +1765,16 @@ nfsd4_encode_fattr(struct svc_fh *fhp, struct svc_export *exp,
|
|
|
int err;
|
|
|
int aclsupport = 0;
|
|
|
struct nfs4_acl *acl = NULL;
|
|
|
+ struct nfsd4_compoundres *resp = rqstp->rq_resp;
|
|
|
+ u32 minorversion = resp->cstate.minorversion;
|
|
|
|
|
|
BUG_ON(bmval1 & NFSD_WRITEONLY_ATTRS_WORD1);
|
|
|
- BUG_ON(bmval0 & ~NFSD_SUPPORTED_ATTRS_WORD0);
|
|
|
- BUG_ON(bmval1 & ~NFSD_SUPPORTED_ATTRS_WORD1);
|
|
|
+ BUG_ON(bmval0 & ~nfsd_suppattrs0(minorversion));
|
|
|
+ BUG_ON(bmval1 & ~nfsd_suppattrs1(minorversion));
|
|
|
+ BUG_ON(bmval2 & ~nfsd_suppattrs2(minorversion));
|
|
|
|
|
|
if (exp->ex_fslocs.migrated) {
|
|
|
+ BUG_ON(bmval[2]);
|
|
|
status = fattr_handle_absent_fs(&bmval0, &bmval1, &rdattr_err);
|
|
|
if (status)
|
|
|
goto out;
|
|
@@ -1476,22 +1820,42 @@ nfsd4_encode_fattr(struct svc_fh *fhp, struct svc_export *exp,
|
|
|
if ((buflen -= 16) < 0)
|
|
|
goto out_resource;
|
|
|
|
|
|
- WRITE32(2);
|
|
|
- WRITE32(bmval0);
|
|
|
- WRITE32(bmval1);
|
|
|
+ if (unlikely(bmval2)) {
|
|
|
+ WRITE32(3);
|
|
|
+ WRITE32(bmval0);
|
|
|
+ WRITE32(bmval1);
|
|
|
+ WRITE32(bmval2);
|
|
|
+ } else if (likely(bmval1)) {
|
|
|
+ WRITE32(2);
|
|
|
+ WRITE32(bmval0);
|
|
|
+ WRITE32(bmval1);
|
|
|
+ } else {
|
|
|
+ WRITE32(1);
|
|
|
+ WRITE32(bmval0);
|
|
|
+ }
|
|
|
attrlenp = p++; /* to be backfilled later */
|
|
|
|
|
|
if (bmval0 & FATTR4_WORD0_SUPPORTED_ATTRS) {
|
|
|
- u32 word0 = NFSD_SUPPORTED_ATTRS_WORD0;
|
|
|
+ u32 word0 = nfsd_suppattrs0(minorversion);
|
|
|
+ u32 word1 = nfsd_suppattrs1(minorversion);
|
|
|
+ u32 word2 = nfsd_suppattrs2(minorversion);
|
|
|
+
|
|
|
if ((buflen -= 12) < 0)
|
|
|
goto out_resource;
|
|
|
if (!aclsupport)
|
|
|
word0 &= ~FATTR4_WORD0_ACL;
|
|
|
if (!exp->ex_fslocs.locations)
|
|
|
word0 &= ~FATTR4_WORD0_FS_LOCATIONS;
|
|
|
- WRITE32(2);
|
|
|
- WRITE32(word0);
|
|
|
- WRITE32(NFSD_SUPPORTED_ATTRS_WORD1);
|
|
|
+ if (!word2) {
|
|
|
+ WRITE32(2);
|
|
|
+ WRITE32(word0);
|
|
|
+ WRITE32(word1);
|
|
|
+ } else {
|
|
|
+ WRITE32(3);
|
|
|
+ WRITE32(word0);
|
|
|
+ WRITE32(word1);
|
|
|
+ WRITE32(word2);
|
|
|
+ }
|
|
|
}
|
|
|
if (bmval0 & FATTR4_WORD0_TYPE) {
|
|
|
if ((buflen -= 4) < 0)
|
|
@@ -1801,6 +2165,13 @@ out_acl:
|
|
|
}
|
|
|
WRITE64(stat.ino);
|
|
|
}
|
|
|
+ if (bmval2 & FATTR4_WORD2_SUPPATTR_EXCLCREAT) {
|
|
|
+ WRITE32(3);
|
|
|
+ WRITE32(NFSD_SUPPATTR_EXCLCREAT_WORD0);
|
|
|
+ WRITE32(NFSD_SUPPATTR_EXCLCREAT_WORD1);
|
|
|
+ WRITE32(NFSD_SUPPATTR_EXCLCREAT_WORD2);
|
|
|
+ }
|
|
|
+
|
|
|
*attrlenp = htonl((char *)p - (char *)attrlenp - 4);
|
|
|
*countp = p - buffer;
|
|
|
status = nfs_ok;
|
|
@@ -2571,6 +2942,143 @@ nfsd4_encode_write(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd4_w
|
|
|
return nfserr;
|
|
|
}
|
|
|
|
|
|
+static __be32
|
|
|
+nfsd4_encode_exchange_id(struct nfsd4_compoundres *resp, int nfserr,
|
|
|
+ struct nfsd4_exchange_id *exid)
|
|
|
+{
|
|
|
+ ENCODE_HEAD;
|
|
|
+ char *major_id;
|
|
|
+ char *server_scope;
|
|
|
+ int major_id_sz;
|
|
|
+ int server_scope_sz;
|
|
|
+ uint64_t minor_id = 0;
|
|
|
+
|
|
|
+ if (nfserr)
|
|
|
+ return nfserr;
|
|
|
+
|
|
|
+ major_id = utsname()->nodename;
|
|
|
+ major_id_sz = strlen(major_id);
|
|
|
+ server_scope = utsname()->nodename;
|
|
|
+ server_scope_sz = strlen(server_scope);
|
|
|
+
|
|
|
+ RESERVE_SPACE(
|
|
|
+ 8 /* eir_clientid */ +
|
|
|
+ 4 /* eir_sequenceid */ +
|
|
|
+ 4 /* eir_flags */ +
|
|
|
+ 4 /* spr_how (SP4_NONE) */ +
|
|
|
+ 8 /* so_minor_id */ +
|
|
|
+ 4 /* so_major_id.len */ +
|
|
|
+ (XDR_QUADLEN(major_id_sz) * 4) +
|
|
|
+ 4 /* eir_server_scope.len */ +
|
|
|
+ (XDR_QUADLEN(server_scope_sz) * 4) +
|
|
|
+ 4 /* eir_server_impl_id.count (0) */);
|
|
|
+
|
|
|
+ WRITEMEM(&exid->clientid, 8);
|
|
|
+ WRITE32(exid->seqid);
|
|
|
+ WRITE32(exid->flags);
|
|
|
+
|
|
|
+ /* state_protect4_r. Currently only support SP4_NONE */
|
|
|
+ BUG_ON(exid->spa_how != SP4_NONE);
|
|
|
+ WRITE32(exid->spa_how);
|
|
|
+
|
|
|
+ /* The server_owner struct */
|
|
|
+ WRITE64(minor_id); /* Minor id */
|
|
|
+ /* major id */
|
|
|
+ WRITE32(major_id_sz);
|
|
|
+ WRITEMEM(major_id, major_id_sz);
|
|
|
+
|
|
|
+ /* Server scope */
|
|
|
+ WRITE32(server_scope_sz);
|
|
|
+ WRITEMEM(server_scope, server_scope_sz);
|
|
|
+
|
|
|
+ /* Implementation id */
|
|
|
+ WRITE32(0); /* zero length nfs_impl_id4 array */
|
|
|
+ ADJUST_ARGS();
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static __be32
|
|
|
+nfsd4_encode_create_session(struct nfsd4_compoundres *resp, int nfserr,
|
|
|
+ struct nfsd4_create_session *sess)
|
|
|
+{
|
|
|
+ ENCODE_HEAD;
|
|
|
+
|
|
|
+ if (nfserr)
|
|
|
+ return nfserr;
|
|
|
+
|
|
|
+ RESERVE_SPACE(24);
|
|
|
+ WRITEMEM(sess->sessionid.data, NFS4_MAX_SESSIONID_LEN);
|
|
|
+ WRITE32(sess->seqid);
|
|
|
+ WRITE32(sess->flags);
|
|
|
+ ADJUST_ARGS();
|
|
|
+
|
|
|
+ RESERVE_SPACE(28);
|
|
|
+ WRITE32(0); /* headerpadsz */
|
|
|
+ WRITE32(sess->fore_channel.maxreq_sz);
|
|
|
+ WRITE32(sess->fore_channel.maxresp_sz);
|
|
|
+ WRITE32(sess->fore_channel.maxresp_cached);
|
|
|
+ WRITE32(sess->fore_channel.maxops);
|
|
|
+ WRITE32(sess->fore_channel.maxreqs);
|
|
|
+ WRITE32(sess->fore_channel.nr_rdma_attrs);
|
|
|
+ ADJUST_ARGS();
|
|
|
+
|
|
|
+ if (sess->fore_channel.nr_rdma_attrs) {
|
|
|
+ RESERVE_SPACE(4);
|
|
|
+ WRITE32(sess->fore_channel.rdma_attrs);
|
|
|
+ ADJUST_ARGS();
|
|
|
+ }
|
|
|
+
|
|
|
+ RESERVE_SPACE(28);
|
|
|
+ WRITE32(0); /* headerpadsz */
|
|
|
+ WRITE32(sess->back_channel.maxreq_sz);
|
|
|
+ WRITE32(sess->back_channel.maxresp_sz);
|
|
|
+ WRITE32(sess->back_channel.maxresp_cached);
|
|
|
+ WRITE32(sess->back_channel.maxops);
|
|
|
+ WRITE32(sess->back_channel.maxreqs);
|
|
|
+ WRITE32(sess->back_channel.nr_rdma_attrs);
|
|
|
+ ADJUST_ARGS();
|
|
|
+
|
|
|
+ if (sess->back_channel.nr_rdma_attrs) {
|
|
|
+ RESERVE_SPACE(4);
|
|
|
+ WRITE32(sess->back_channel.rdma_attrs);
|
|
|
+ ADJUST_ARGS();
|
|
|
+ }
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static __be32
|
|
|
+nfsd4_encode_destroy_session(struct nfsd4_compoundres *resp, int nfserr,
|
|
|
+ struct nfsd4_destroy_session *destroy_session)
|
|
|
+{
|
|
|
+ return nfserr;
|
|
|
+}
|
|
|
+
|
|
|
+__be32
|
|
|
+nfsd4_encode_sequence(struct nfsd4_compoundres *resp, int nfserr,
|
|
|
+ struct nfsd4_sequence *seq)
|
|
|
+{
|
|
|
+ ENCODE_HEAD;
|
|
|
+
|
|
|
+ if (nfserr)
|
|
|
+ return nfserr;
|
|
|
+
|
|
|
+ RESERVE_SPACE(NFS4_MAX_SESSIONID_LEN + 20);
|
|
|
+ WRITEMEM(seq->sessionid.data, NFS4_MAX_SESSIONID_LEN);
|
|
|
+ WRITE32(seq->seqid);
|
|
|
+ WRITE32(seq->slotid);
|
|
|
+ WRITE32(seq->maxslots);
|
|
|
+ /*
|
|
|
+ * FIXME: for now:
|
|
|
+ * target_maxslots = maxslots
|
|
|
+ * status_flags = 0
|
|
|
+ */
|
|
|
+ WRITE32(seq->maxslots);
|
|
|
+ WRITE32(0);
|
|
|
+
|
|
|
+ ADJUST_ARGS();
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
static __be32
|
|
|
nfsd4_encode_noop(struct nfsd4_compoundres *resp, __be32 nfserr, void *p)
|
|
|
{
|
|
@@ -2579,6 +3087,11 @@ nfsd4_encode_noop(struct nfsd4_compoundres *resp, __be32 nfserr, void *p)
|
|
|
|
|
|
typedef __be32(* nfsd4_enc)(struct nfsd4_compoundres *, __be32, void *);
|
|
|
|
|
|
+/*
|
|
|
+ * Note: nfsd4_enc_ops vector is shared for v4.0 and v4.1
|
|
|
+ * since we don't need to filter out obsolete ops as this is
|
|
|
+ * done in the decoding phase.
|
|
|
+ */
|
|
|
static nfsd4_enc nfsd4_enc_ops[] = {
|
|
|
[OP_ACCESS] = (nfsd4_enc)nfsd4_encode_access,
|
|
|
[OP_CLOSE] = (nfsd4_enc)nfsd4_encode_close,
|
|
@@ -2617,8 +3130,77 @@ static nfsd4_enc nfsd4_enc_ops[] = {
|
|
|
[OP_VERIFY] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
[OP_WRITE] = (nfsd4_enc)nfsd4_encode_write,
|
|
|
[OP_RELEASE_LOCKOWNER] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+
|
|
|
+ /* NFSv4.1 operations */
|
|
|
+ [OP_BACKCHANNEL_CTL] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_BIND_CONN_TO_SESSION] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_EXCHANGE_ID] = (nfsd4_enc)nfsd4_encode_exchange_id,
|
|
|
+ [OP_CREATE_SESSION] = (nfsd4_enc)nfsd4_encode_create_session,
|
|
|
+ [OP_DESTROY_SESSION] = (nfsd4_enc)nfsd4_encode_destroy_session,
|
|
|
+ [OP_FREE_STATEID] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_GET_DIR_DELEGATION] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_GETDEVICEINFO] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_GETDEVICELIST] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_LAYOUTCOMMIT] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_LAYOUTGET] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_LAYOUTRETURN] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_SECINFO_NO_NAME] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_SEQUENCE] = (nfsd4_enc)nfsd4_encode_sequence,
|
|
|
+ [OP_SET_SSV] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_TEST_STATEID] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_WANT_DELEGATION] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_DESTROY_CLIENTID] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
+ [OP_RECLAIM_COMPLETE] = (nfsd4_enc)nfsd4_encode_noop,
|
|
|
};
|
|
|
|
|
|
+/*
|
|
|
+ * Calculate the total amount of memory that the compound response has taken
|
|
|
+ * after encoding the current operation.
|
|
|
+ *
|
|
|
+ * pad: add on 8 bytes for the next operation's op_code and status so that
|
|
|
+ * there is room to cache a failure on the next operation.
|
|
|
+ *
|
|
|
+ * Compare this length to the session se_fmaxresp_cached.
|
|
|
+ *
|
|
|
+ * Our se_fmaxresp_cached will always be a multiple of PAGE_SIZE, and so
|
|
|
+ * will be at least a page and will therefore hold the xdr_buf head.
|
|
|
+ */
|
|
|
+static int nfsd4_check_drc_limit(struct nfsd4_compoundres *resp)
|
|
|
+{
|
|
|
+ int status = 0;
|
|
|
+ struct xdr_buf *xb = &resp->rqstp->rq_res;
|
|
|
+ struct nfsd4_compoundargs *args = resp->rqstp->rq_argp;
|
|
|
+ struct nfsd4_session *session = NULL;
|
|
|
+ struct nfsd4_slot *slot = resp->cstate.slot;
|
|
|
+ u32 length, tlen = 0, pad = 8;
|
|
|
+
|
|
|
+ if (!nfsd4_has_session(&resp->cstate))
|
|
|
+ return status;
|
|
|
+
|
|
|
+ session = resp->cstate.session;
|
|
|
+ if (session == NULL || slot->sl_cache_entry.ce_cachethis == 0)
|
|
|
+ return status;
|
|
|
+
|
|
|
+ if (resp->opcnt >= args->opcnt)
|
|
|
+ pad = 0; /* this is the last operation */
|
|
|
+
|
|
|
+ if (xb->page_len == 0) {
|
|
|
+ length = (char *)resp->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;
|
|
|
+
|
|
|
+ length = xb->head[0].iov_len + xb->page_len + tlen + pad;
|
|
|
+ }
|
|
|
+ dprintk("%s length %u, xb->page_len %u tlen %u pad %u\n", __func__,
|
|
|
+ length, xb->page_len, tlen, pad);
|
|
|
+
|
|
|
+ if (length <= session->se_fmaxresp_cached)
|
|
|
+ return status;
|
|
|
+ else
|
|
|
+ return nfserr_rep_too_big_to_cache;
|
|
|
+}
|
|
|
+
|
|
|
void
|
|
|
nfsd4_encode_operation(struct nfsd4_compoundres *resp, struct nfsd4_op *op)
|
|
|
{
|
|
@@ -2635,6 +3217,9 @@ nfsd4_encode_operation(struct nfsd4_compoundres *resp, struct nfsd4_op *op)
|
|
|
BUG_ON(op->opnum < 0 || op->opnum >= ARRAY_SIZE(nfsd4_enc_ops) ||
|
|
|
!nfsd4_enc_ops[op->opnum]);
|
|
|
op->status = nfsd4_enc_ops[op->opnum](resp, op->status, &op->u);
|
|
|
+ /* nfsd4_check_drc_limit guarantees enough room for error status */
|
|
|
+ if (!op->status && nfsd4_check_drc_limit(resp))
|
|
|
+ op->status = nfserr_rep_too_big_to_cache;
|
|
|
status:
|
|
|
/*
|
|
|
* Note: We write the status directly, instead of using WRITE32(),
|
|
@@ -2735,6 +3320,18 @@ nfs4svc_encode_compoundres(struct svc_rqst *rqstp, __be32 *p, struct nfsd4_compo
|
|
|
iov = &rqstp->rq_res.head[0];
|
|
|
iov->iov_len = ((char*)resp->p) - (char*)iov->iov_base;
|
|
|
BUG_ON(iov->iov_len > PAGE_SIZE);
|
|
|
+ if (nfsd4_has_session(&resp->cstate)) {
|
|
|
+ if (resp->cstate.status == nfserr_replay_cache &&
|
|
|
+ !nfsd4_not_cached(resp)) {
|
|
|
+ iov->iov_len = resp->cstate.iovlen;
|
|
|
+ } else {
|
|
|
+ nfsd4_store_cache_entry(resp);
|
|
|
+ dprintk("%s: SET SLOT STATE TO AVAILABLE\n", __func__);
|
|
|
+ resp->cstate.slot->sl_inuse = 0;
|
|
|
+ }
|
|
|
+ if (resp->cstate.session)
|
|
|
+ nfsd4_put_session(resp->cstate.session);
|
|
|
+ }
|
|
|
return 1;
|
|
|
}
|
|
|
|