ocfs2: Trivial cleanup of jbd compatibility layer removal
[safe/jmp/linux-2.6] / fs / nfs / mount_clnt.c
index 8429885..0adefc4 100644 (file)
 #define encode_dirpath_sz      (1 + XDR_QUADLEN(MNTPATHLEN))
 #define MNT_status_sz          (1)
 #define MNT_fhs_status_sz      (1)
+#define MNT_fhandle_sz         XDR_QUADLEN(NFS2_FHSIZE)
+#define MNT_fhandle3_sz                (1 + XDR_QUADLEN(NFS3_FHSIZE))
+#define MNT_authflav3_sz       (1 + NFS_MAX_SECFLAVORS)
 
 /*
  * XDR argument and result sizes
  */
 #define MNT_enc_dirpath_sz     encode_dirpath_sz
+#define MNT_dec_mountres_sz    (MNT_status_sz + MNT_fhandle_sz)
+#define MNT_dec_mountres3_sz   (MNT_status_sz + MNT_fhandle_sz + \
+                                MNT_authflav3_sz)
 
 /*
  * Defined by RFC 1094, section A.5
@@ -120,6 +126,8 @@ static struct {
 struct mountres {
        int errno;
        struct nfs_fh *fh;
+       unsigned int *auth_count;
+       rpc_authflavor_t *auth_flavors;
 };
 
 struct mnt_fhstatus {
@@ -135,8 +143,10 @@ struct mnt_fhstatus {
  */
 int nfs_mount(struct nfs_mount_request *info)
 {
-       struct mnt_fhstatus     result = {
-               .fh             = info->fh
+       struct mountres result = {
+               .fh             = info->fh,
+               .auth_count     = info->auth_flav_len,
+               .auth_flavors   = info->auth_flavs,
        };
        struct rpc_message msg  = {
                .rpc_argp       = info->dirpath,
@@ -175,7 +185,7 @@ int nfs_mount(struct nfs_mount_request *info)
 
        if (status < 0)
                goto out_call_err;
-       if (result.status != 0)
+       if (result.errno != 0)
                goto out_mnt_err;
 
        dprintk("NFS: MNT request succeeded\n");
@@ -186,19 +196,84 @@ out:
 
 out_clnt_err:
        status = PTR_ERR(mnt_clnt);
-       dprintk("NFS: failed to create RPC client, status=%d\n", status);
+       dprintk("NFS: failed to create MNT RPC client, status=%d\n", status);
        goto out;
 
 out_call_err:
-       dprintk("NFS: failed to start MNT request, status=%d\n", status);
+       dprintk("NFS: MNT request failed, status=%d\n", status);
        goto out;
 
 out_mnt_err:
-       dprintk("NFS: MNT server returned result %d\n", result.status);
-       status = nfs_stat_to_errno(result.status);
+       dprintk("NFS: MNT server returned result %d\n", result.errno);
+       status = result.errno;
        goto out;
 }
 
+/**
+ * nfs_umount - Notify a server that we have unmounted this export
+ * @info: pointer to umount request arguments
+ *
+ * MOUNTPROC_UMNT is advisory, so we set a short timeout, and always
+ * use UDP.
+ */
+void nfs_umount(const struct nfs_mount_request *info)
+{
+       static const struct rpc_timeout nfs_umnt_timeout = {
+               .to_initval = 1 * HZ,
+               .to_maxval = 3 * HZ,
+               .to_retries = 2,
+       };
+       struct rpc_create_args args = {
+               .protocol       = IPPROTO_UDP,
+               .address        = info->sap,
+               .addrsize       = info->salen,
+               .timeout        = &nfs_umnt_timeout,
+               .servername     = info->hostname,
+               .program        = &mnt_program,
+               .version        = info->version,
+               .authflavor     = RPC_AUTH_UNIX,
+               .flags          = RPC_CLNT_CREATE_NOPING,
+       };
+       struct mountres result;
+       struct rpc_message msg  = {
+               .rpc_argp       = info->dirpath,
+               .rpc_resp       = &result,
+       };
+       struct rpc_clnt *clnt;
+       int status;
+
+       if (info->noresvport)
+               args.flags |= RPC_CLNT_CREATE_NONPRIVPORT;
+
+       clnt = rpc_create(&args);
+       if (unlikely(IS_ERR(clnt)))
+               goto out_clnt_err;
+
+       dprintk("NFS: sending UMNT request for %s:%s\n",
+               (info->hostname ? info->hostname : "server"), info->dirpath);
+
+       if (info->version == NFS_MNT3_VERSION)
+               msg.rpc_proc = &clnt->cl_procinfo[MOUNTPROC3_UMNT];
+       else
+               msg.rpc_proc = &clnt->cl_procinfo[MOUNTPROC_UMNT];
+
+       status = rpc_call_sync(clnt, &msg, 0);
+       rpc_shutdown_client(clnt);
+
+       if (unlikely(status < 0))
+               goto out_call_err;
+
+       return;
+
+out_clnt_err:
+       dprintk("NFS: failed to create UMNT RPC client, status=%ld\n",
+                       PTR_ERR(clnt));
+       return;
+
+out_call_err:
+       dprintk("NFS: UMNT request failed, status=%d\n", status);
+}
+
 /*
  * XDR encode/decode functions for MOUNT
  */
@@ -228,18 +303,6 @@ static int mnt_enc_dirpath(struct rpc_rqst *req, __be32 *p,
        return encode_mntdirpath(&xdr, dirpath);
 }
 
-static int xdr_decode_fhstatus(struct rpc_rqst *req, __be32 *p,
-                              struct mnt_fhstatus *res)
-{
-       struct nfs_fh *fh = res->fh;
-
-       if ((res->status = ntohl(*p++)) == 0) {
-               fh->size = NFS2_FHSIZE;
-               memcpy(fh->data, p, NFS2_FHSIZE);
-       }
-       return 0;
-}
-
 /*
  * RFC 1094: "A non-zero status indicates some sort of error.  In this
  * case, the status is a UNIX error number."  This can be problematic
@@ -260,7 +323,7 @@ static int decode_status(struct xdr_stream *xdr, struct mountres *res)
                return -EIO;
        status = ntohl(*p);
 
-       for (i = 0; i <= ARRAY_SIZE(mnt_errtbl); i++) {
+       for (i = 0; i < ARRAY_SIZE(mnt_errtbl); i++) {
                if (mnt_errtbl[i].status == status) {
                        res->errno = mnt_errtbl[i].errno;
                        return 0;
@@ -272,6 +335,34 @@ static int decode_status(struct xdr_stream *xdr, struct mountres *res)
        return 0;
 }
 
+static int decode_fhandle(struct xdr_stream *xdr, struct mountres *res)
+{
+       struct nfs_fh *fh = res->fh;
+       __be32 *p;
+
+       p = xdr_inline_decode(xdr, NFS2_FHSIZE);
+       if (unlikely(p == NULL))
+               return -EIO;
+
+       fh->size = NFS2_FHSIZE;
+       memcpy(fh->data, p, NFS2_FHSIZE);
+       return 0;
+}
+
+static int mnt_dec_mountres(struct rpc_rqst *req, __be32 *p,
+                           struct mountres *res)
+{
+       struct xdr_stream xdr;
+       int status;
+
+       xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
+
+       status = decode_status(&xdr, res);
+       if (unlikely(status != 0 || res->errno != 0))
+               return status;
+       return decode_fhandle(&xdr, res);
+}
+
 static int decode_fhs_status(struct xdr_stream *xdr, struct mountres *res)
 {
        unsigned int i;
@@ -283,7 +374,7 @@ static int decode_fhs_status(struct xdr_stream *xdr, struct mountres *res)
                return -EIO;
        status = ntohl(*p);
 
-       for (i = 0; i <= ARRAY_SIZE(mnt3_errtbl); i++) {
+       for (i = 0; i < ARRAY_SIZE(mnt3_errtbl); i++) {
                if (mnt3_errtbl[i].status == status) {
                        res->errno = mnt3_errtbl[i].errno;
                        return 0;
@@ -295,48 +386,118 @@ static int decode_fhs_status(struct xdr_stream *xdr, struct mountres *res)
        return 0;
 }
 
-static int xdr_decode_fhstatus3(struct rpc_rqst *req, __be32 *p,
-                               struct mnt_fhstatus *res)
+static int decode_fhandle3(struct xdr_stream *xdr, struct mountres *res)
 {
        struct nfs_fh *fh = res->fh;
-       unsigned size;
-
-       if ((res->status = ntohl(*p++)) == 0) {
-               size = ntohl(*p++);
-               if (size <= NFS3_FHSIZE && size != 0) {
-                       fh->size = size;
-                       memcpy(fh->data, p, size);
-               } else
-                       res->status = -EBADHANDLE;
+       u32 size;
+       __be32 *p;
+
+       p = xdr_inline_decode(xdr, sizeof(size));
+       if (unlikely(p == NULL))
+               return -EIO;
+
+       size = ntohl(*p++);
+       if (size > NFS3_FHSIZE || size == 0)
+               return -EIO;
+
+       p = xdr_inline_decode(xdr, size);
+       if (unlikely(p == NULL))
+               return -EIO;
+
+       fh->size = size;
+       memcpy(fh->data, p, size);
+       return 0;
+}
+
+static int decode_auth_flavors(struct xdr_stream *xdr, struct mountres *res)
+{
+       rpc_authflavor_t *flavors = res->auth_flavors;
+       unsigned int *count = res->auth_count;
+       u32 entries, i;
+       __be32 *p;
+
+       if (*count == 0)
+               return 0;
+
+       p = xdr_inline_decode(xdr, sizeof(entries));
+       if (unlikely(p == NULL))
+               return -EIO;
+       entries = ntohl(*p);
+       dprintk("NFS: received %u auth flavors\n", entries);
+       if (entries > NFS_MAX_SECFLAVORS)
+               entries = NFS_MAX_SECFLAVORS;
+
+       p = xdr_inline_decode(xdr, sizeof(u32) * entries);
+       if (unlikely(p == NULL))
+               return -EIO;
+
+       if (entries > *count)
+               entries = *count;
+
+       for (i = 0; i < entries; i++) {
+               flavors[i] = ntohl(*p++);
+               dprintk("NFS:\tflavor %u: %d\n", i, flavors[i]);
        }
+       *count = i;
+
        return 0;
 }
 
-#define MNT_fhstatus_sz                (1 + 8)
-#define MNT_fhstatus3_sz       (1 + 16)
+static int mnt_dec_mountres3(struct rpc_rqst *req, __be32 *p,
+                            struct mountres *res)
+{
+       struct xdr_stream xdr;
+       int status;
+
+       xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
+
+       status = decode_fhs_status(&xdr, res);
+       if (unlikely(status != 0 || res->errno != 0))
+               return status;
+       status = decode_fhandle3(&xdr, res);
+       if (unlikely(status != 0)) {
+               res->errno = -EBADHANDLE;
+               return 0;
+       }
+       return decode_auth_flavors(&xdr, res);
+}
 
 static struct rpc_procinfo mnt_procedures[] = {
        [MOUNTPROC_MNT] = {
                .p_proc         = MOUNTPROC_MNT,
                .p_encode       = (kxdrproc_t)mnt_enc_dirpath,
-               .p_decode       = (kxdrproc_t) xdr_decode_fhstatus,
+               .p_decode       = (kxdrproc_t)mnt_dec_mountres,
                .p_arglen       = MNT_enc_dirpath_sz,
-               .p_replen       = MNT_fhstatus_sz,
+               .p_replen       = MNT_dec_mountres_sz,
                .p_statidx      = MOUNTPROC_MNT,
                .p_name         = "MOUNT",
        },
+       [MOUNTPROC_UMNT] = {
+               .p_proc         = MOUNTPROC_UMNT,
+               .p_encode       = (kxdrproc_t)mnt_enc_dirpath,
+               .p_arglen       = MNT_enc_dirpath_sz,
+               .p_statidx      = MOUNTPROC_UMNT,
+               .p_name         = "UMOUNT",
+       },
 };
 
 static struct rpc_procinfo mnt3_procedures[] = {
        [MOUNTPROC3_MNT] = {
                .p_proc         = MOUNTPROC3_MNT,
                .p_encode       = (kxdrproc_t)mnt_enc_dirpath,
-               .p_decode       = (kxdrproc_t) xdr_decode_fhstatus3,
+               .p_decode       = (kxdrproc_t)mnt_dec_mountres3,
                .p_arglen       = MNT_enc_dirpath_sz,
-               .p_replen       = MNT_fhstatus3_sz,
+               .p_replen       = MNT_dec_mountres3_sz,
                .p_statidx      = MOUNTPROC3_MNT,
                .p_name         = "MOUNT",
        },
+       [MOUNTPROC3_UMNT] = {
+               .p_proc         = MOUNTPROC3_UMNT,
+               .p_encode       = (kxdrproc_t)mnt_enc_dirpath,
+               .p_arglen       = MNT_enc_dirpath_sz,
+               .p_statidx      = MOUNTPROC3_UMNT,
+               .p_name         = "UMOUNT",
+       },
 };