From: David Howells <[email protected]>
Split various bits relating to mount parameterisation out from
fs/nfs/super.c into their own file to form the basis of filesystem context
handling for NFS.
No other changes are made to the code beyond removing 'static' qualifiers.
Signed-off-by: David Howells <[email protected]>
Signed-off-by: Al Viro <[email protected]>
---
fs/nfs/Makefile | 2 +-
fs/nfs/fs_context.c | 1418 +++++++++++++++++++++++++++++++++++++++++++
fs/nfs/internal.h | 29 +
fs/nfs/super.c | 1411 ------------------------------------------
4 files changed, 1448 insertions(+), 1412 deletions(-)
create mode 100644 fs/nfs/fs_context.c
diff --git a/fs/nfs/Makefile b/fs/nfs/Makefile
index 34cdeaecccf6..2433c3e03cfa 100644
--- a/fs/nfs/Makefile
+++ b/fs/nfs/Makefile
@@ -9,7 +9,7 @@ CFLAGS_nfstrace.o += -I$(src)
nfs-y := client.o dir.o file.o getroot.o inode.o super.o \
io.o direct.o pagelist.o read.o symlink.o unlink.o \
write.o namespace.o mount_clnt.o nfstrace.o \
- export.o sysfs.o
+ export.o sysfs.o fs_context.o
nfs-$(CONFIG_ROOT_NFS) += nfsroot.o
nfs-$(CONFIG_SYSCTL) += sysctl.o
nfs-$(CONFIG_NFS_FSCACHE) += fscache.o fscache-index.o
diff --git a/fs/nfs/fs_context.c b/fs/nfs/fs_context.c
new file mode 100644
index 000000000000..82b312a5cdde
--- /dev/null
+++ b/fs/nfs/fs_context.c
@@ -0,0 +1,1418 @@
+/* NFS mount handling.
+ *
+ * Copyright (C) 2018 Red Hat, Inc. All Rights Reserved.
+ * Written by David Howells ([email protected])
+ *
+ * Split from fs/nfs/super.c:
+ *
+ * Copyright (C) 1992 Rick Sladkey
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public Licence
+ * as published by the Free Software Foundation; either version
+ * 2 of the Licence, or (at your option) any later version.
+ */
+
+#include <linux/module.h>
+#include <linux/fs.h>
+#include <linux/parser.h>
+#include <linux/nfs_fs.h>
+#include <linux/nfs_mount.h>
+#include <linux/nfs4_mount.h>
+#include "nfs.h"
+#include "internal.h"
+
+#define NFSDBG_FACILITY NFSDBG_MOUNT
+
+#if IS_ENABLED(CONFIG_NFS_V3)
+#define NFS_DEFAULT_VERSION 3
+#else
+#define NFS_DEFAULT_VERSION 2
+#endif
+
+#define NFS_MAX_CONNECTIONS 16
+
+enum {
+ /* Mount options that take no arguments */
+ Opt_soft, Opt_softerr, Opt_hard,
+ Opt_posix, Opt_noposix,
+ Opt_cto, Opt_nocto,
+ Opt_ac, Opt_noac,
+ Opt_lock, Opt_nolock,
+ Opt_udp, Opt_tcp, Opt_rdma,
+ Opt_acl, Opt_noacl,
+ Opt_rdirplus, Opt_nordirplus,
+ Opt_sharecache, Opt_nosharecache,
+ Opt_resvport, Opt_noresvport,
+ Opt_fscache, Opt_nofscache,
+ Opt_migration, Opt_nomigration,
+
+ /* Mount options that take integer arguments */
+ Opt_port,
+ Opt_rsize, Opt_wsize, Opt_bsize,
+ Opt_timeo, Opt_retrans,
+ Opt_acregmin, Opt_acregmax,
+ Opt_acdirmin, Opt_acdirmax,
+ Opt_actimeo,
+ Opt_namelen,
+ Opt_mountport,
+ Opt_mountvers,
+ Opt_minorversion,
+
+ /* Mount options that take string arguments */
+ Opt_nfsvers,
+ Opt_sec, Opt_proto, Opt_mountproto, Opt_mounthost,
+ Opt_addr, Opt_mountaddr, Opt_clientaddr,
+ Opt_nconnect,
+ Opt_lookupcache,
+ Opt_fscache_uniq,
+ Opt_local_lock,
+
+ /* Special mount options */
+ Opt_userspace, Opt_deprecated, Opt_sloppy,
+
+ Opt_err
+};
+
+static const match_table_t nfs_mount_option_tokens = {
+ { Opt_userspace, "bg" },
+ { Opt_userspace, "fg" },
+ { Opt_userspace, "retry=%s" },
+
+ { Opt_sloppy, "sloppy" },
+
+ { Opt_soft, "soft" },
+ { Opt_softerr, "softerr" },
+ { Opt_hard, "hard" },
+ { Opt_deprecated, "intr" },
+ { Opt_deprecated, "nointr" },
+ { Opt_posix, "posix" },
+ { Opt_noposix, "noposix" },
+ { Opt_cto, "cto" },
+ { Opt_nocto, "nocto" },
+ { Opt_ac, "ac" },
+ { Opt_noac, "noac" },
+ { Opt_lock, "lock" },
+ { Opt_nolock, "nolock" },
+ { Opt_udp, "udp" },
+ { Opt_tcp, "tcp" },
+ { Opt_rdma, "rdma" },
+ { Opt_acl, "acl" },
+ { Opt_noacl, "noacl" },
+ { Opt_rdirplus, "rdirplus" },
+ { Opt_nordirplus, "nordirplus" },
+ { Opt_sharecache, "sharecache" },
+ { Opt_nosharecache, "nosharecache" },
+ { Opt_resvport, "resvport" },
+ { Opt_noresvport, "noresvport" },
+ { Opt_fscache, "fsc" },
+ { Opt_nofscache, "nofsc" },
+ { Opt_migration, "migration" },
+ { Opt_nomigration, "nomigration" },
+
+ { Opt_port, "port=%s" },
+ { Opt_rsize, "rsize=%s" },
+ { Opt_wsize, "wsize=%s" },
+ { Opt_bsize, "bsize=%s" },
+ { Opt_timeo, "timeo=%s" },
+ { Opt_retrans, "retrans=%s" },
+ { Opt_acregmin, "acregmin=%s" },
+ { Opt_acregmax, "acregmax=%s" },
+ { Opt_acdirmin, "acdirmin=%s" },
+ { Opt_acdirmax, "acdirmax=%s" },
+ { Opt_actimeo, "actimeo=%s" },
+ { Opt_namelen, "namlen=%s" },
+ { Opt_mountport, "mountport=%s" },
+ { Opt_mountvers, "mountvers=%s" },
+ { Opt_minorversion, "minorversion=%s" },
+
+ { Opt_nfsvers, "nfsvers=%s" },
+ { Opt_nfsvers, "vers=%s" },
+
+ { Opt_sec, "sec=%s" },
+ { Opt_proto, "proto=%s" },
+ { Opt_mountproto, "mountproto=%s" },
+ { Opt_addr, "addr=%s" },
+ { Opt_clientaddr, "clientaddr=%s" },
+ { Opt_mounthost, "mounthost=%s" },
+ { Opt_mountaddr, "mountaddr=%s" },
+
+ { Opt_nconnect, "nconnect=%s" },
+
+ { Opt_lookupcache, "lookupcache=%s" },
+ { Opt_fscache_uniq, "fsc=%s" },
+ { Opt_local_lock, "local_lock=%s" },
+
+ /* The following needs to be listed after all other options */
+ { Opt_nfsvers, "v%s" },
+
+ { Opt_err, NULL }
+};
+
+enum {
+ Opt_xprt_udp, Opt_xprt_udp6, Opt_xprt_tcp, Opt_xprt_tcp6, Opt_xprt_rdma,
+ Opt_xprt_rdma6,
+
+ Opt_xprt_err
+};
+
+static const match_table_t nfs_xprt_protocol_tokens = {
+ { Opt_xprt_udp, "udp" },
+ { Opt_xprt_udp6, "udp6" },
+ { Opt_xprt_tcp, "tcp" },
+ { Opt_xprt_tcp6, "tcp6" },
+ { Opt_xprt_rdma, "rdma" },
+ { Opt_xprt_rdma6, "rdma6" },
+
+ { Opt_xprt_err, NULL }
+};
+
+enum {
+ Opt_sec_none, Opt_sec_sys,
+ Opt_sec_krb5, Opt_sec_krb5i, Opt_sec_krb5p,
+ Opt_sec_lkey, Opt_sec_lkeyi, Opt_sec_lkeyp,
+ Opt_sec_spkm, Opt_sec_spkmi, Opt_sec_spkmp,
+
+ Opt_sec_err
+};
+
+static const match_table_t nfs_secflavor_tokens = {
+ { Opt_sec_none, "none" },
+ { Opt_sec_none, "null" },
+ { Opt_sec_sys, "sys" },
+
+ { Opt_sec_krb5, "krb5" },
+ { Opt_sec_krb5i, "krb5i" },
+ { Opt_sec_krb5p, "krb5p" },
+
+ { Opt_sec_lkey, "lkey" },
+ { Opt_sec_lkeyi, "lkeyi" },
+ { Opt_sec_lkeyp, "lkeyp" },
+
+ { Opt_sec_spkm, "spkm3" },
+ { Opt_sec_spkmi, "spkm3i" },
+ { Opt_sec_spkmp, "spkm3p" },
+
+ { Opt_sec_err, NULL }
+};
+
+enum {
+ Opt_lookupcache_all, Opt_lookupcache_positive,
+ Opt_lookupcache_none,
+
+ Opt_lookupcache_err
+};
+
+static match_table_t nfs_lookupcache_tokens = {
+ { Opt_lookupcache_all, "all" },
+ { Opt_lookupcache_positive, "pos" },
+ { Opt_lookupcache_positive, "positive" },
+ { Opt_lookupcache_none, "none" },
+
+ { Opt_lookupcache_err, NULL }
+};
+
+enum {
+ Opt_local_lock_all, Opt_local_lock_flock, Opt_local_lock_posix,
+ Opt_local_lock_none,
+
+ Opt_local_lock_err
+};
+
+static match_table_t nfs_local_lock_tokens = {
+ { Opt_local_lock_all, "all" },
+ { Opt_local_lock_flock, "flock" },
+ { Opt_local_lock_posix, "posix" },
+ { Opt_local_lock_none, "none" },
+
+ { Opt_local_lock_err, NULL }
+};
+
+enum {
+ Opt_vers_2, Opt_vers_3, Opt_vers_4, Opt_vers_4_0,
+ Opt_vers_4_1, Opt_vers_4_2,
+
+ Opt_vers_err
+};
+
+static match_table_t nfs_vers_tokens = {
+ { Opt_vers_2, "2" },
+ { Opt_vers_3, "3" },
+ { Opt_vers_4, "4" },
+ { Opt_vers_4_0, "4.0" },
+ { Opt_vers_4_1, "4.1" },
+ { Opt_vers_4_2, "4.2" },
+
+ { Opt_vers_err, NULL }
+};
+
+struct nfs_parsed_mount_data *nfs_alloc_parsed_mount_data(void)
+{
+ struct nfs_parsed_mount_data *data;
+
+ data = kzalloc(sizeof(*data), GFP_KERNEL);
+ if (data) {
+ data->timeo = NFS_UNSPEC_TIMEO;
+ data->retrans = NFS_UNSPEC_RETRANS;
+ data->acregmin = NFS_DEF_ACREGMIN;
+ data->acregmax = NFS_DEF_ACREGMAX;
+ data->acdirmin = NFS_DEF_ACDIRMIN;
+ data->acdirmax = NFS_DEF_ACDIRMAX;
+ data->mount_server.port = NFS_UNSPEC_PORT;
+ data->nfs_server.port = NFS_UNSPEC_PORT;
+ data->nfs_server.protocol = XPRT_TRANSPORT_TCP;
+ data->selected_flavor = RPC_AUTH_MAXFLAVOR;
+ data->minorversion = 0;
+ data->need_mount = true;
+ data->net = current->nsproxy->net_ns;
+ data->lsm_opts = NULL;
+ }
+ return data;
+}
+
+void nfs_free_parsed_mount_data(struct nfs_parsed_mount_data *data)
+{
+ if (data) {
+ kfree(data->client_address);
+ kfree(data->mount_server.hostname);
+ kfree(data->nfs_server.export_path);
+ kfree(data->nfs_server.hostname);
+ kfree(data->fscache_uniq);
+ security_free_mnt_opts(&data->lsm_opts);
+ kfree(data);
+ }
+}
+
+/*
+ * Sanity-check a server address provided by the mount command.
+ *
+ * Address family must be initialized, and address must not be
+ * the ANY address for that family.
+ */
+static int nfs_verify_server_address(struct sockaddr *addr)
+{
+ switch (addr->sa_family) {
+ case AF_INET: {
+ struct sockaddr_in *sa = (struct sockaddr_in *)addr;
+ return sa->sin_addr.s_addr != htonl(INADDR_ANY);
+ }
+ case AF_INET6: {
+ struct in6_addr *sa = &((struct sockaddr_in6 *)addr)->sin6_addr;
+ return !ipv6_addr_any(sa);
+ }
+ }
+
+ dfprintk(MOUNT, "NFS: Invalid IP address specified\n");
+ return 0;
+}
+
+/*
+ * Sanity check the NFS transport protocol.
+ *
+ */
+static void nfs_validate_transport_protocol(struct nfs_parsed_mount_data *mnt)
+{
+ switch (mnt->nfs_server.protocol) {
+ case XPRT_TRANSPORT_UDP:
+ case XPRT_TRANSPORT_TCP:
+ case XPRT_TRANSPORT_RDMA:
+ break;
+ default:
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
+ }
+}
+
+/*
+ * For text based NFSv2/v3 mounts, the mount protocol transport default
+ * settings should depend upon the specified NFS transport.
+ */
+static void nfs_set_mount_transport_protocol(struct nfs_parsed_mount_data *mnt)
+{
+ nfs_validate_transport_protocol(mnt);
+
+ if (mnt->mount_server.protocol == XPRT_TRANSPORT_UDP ||
+ mnt->mount_server.protocol == XPRT_TRANSPORT_TCP)
+ return;
+ switch (mnt->nfs_server.protocol) {
+ case XPRT_TRANSPORT_UDP:
+ mnt->mount_server.protocol = XPRT_TRANSPORT_UDP;
+ break;
+ case XPRT_TRANSPORT_TCP:
+ case XPRT_TRANSPORT_RDMA:
+ mnt->mount_server.protocol = XPRT_TRANSPORT_TCP;
+ }
+}
+
+/*
+ * Add 'flavor' to 'auth_info' if not already present.
+ * Returns true if 'flavor' ends up in the list, false otherwise
+ */
+static bool nfs_auth_info_add(struct nfs_auth_info *auth_info,
+ rpc_authflavor_t flavor)
+{
+ unsigned int i;
+ unsigned int max_flavor_len = ARRAY_SIZE(auth_info->flavors);
+
+ /* make sure this flavor isn't already in the list */
+ for (i = 0; i < auth_info->flavor_len; i++) {
+ if (flavor == auth_info->flavors[i])
+ return true;
+ }
+
+ if (auth_info->flavor_len + 1 >= max_flavor_len) {
+ dfprintk(MOUNT, "NFS: too many sec= flavors\n");
+ return false;
+ }
+
+ auth_info->flavors[auth_info->flavor_len++] = flavor;
+ return true;
+}
+
+/*
+ * Parse the value of the 'sec=' option.
+ */
+static int nfs_parse_security_flavors(char *value,
+ struct nfs_parsed_mount_data *mnt)
+{
+ substring_t args[MAX_OPT_ARGS];
+ rpc_authflavor_t pseudoflavor;
+ char *p;
+
+ dfprintk(MOUNT, "NFS: parsing sec=%s option\n", value);
+
+ while ((p = strsep(&value, ":")) != NULL) {
+ switch (match_token(p, nfs_secflavor_tokens, args)) {
+ case Opt_sec_none:
+ pseudoflavor = RPC_AUTH_NULL;
+ break;
+ case Opt_sec_sys:
+ pseudoflavor = RPC_AUTH_UNIX;
+ break;
+ case Opt_sec_krb5:
+ pseudoflavor = RPC_AUTH_GSS_KRB5;
+ break;
+ case Opt_sec_krb5i:
+ pseudoflavor = RPC_AUTH_GSS_KRB5I;
+ break;
+ case Opt_sec_krb5p:
+ pseudoflavor = RPC_AUTH_GSS_KRB5P;
+ break;
+ case Opt_sec_lkey:
+ pseudoflavor = RPC_AUTH_GSS_LKEY;
+ break;
+ case Opt_sec_lkeyi:
+ pseudoflavor = RPC_AUTH_GSS_LKEYI;
+ break;
+ case Opt_sec_lkeyp:
+ pseudoflavor = RPC_AUTH_GSS_LKEYP;
+ break;
+ case Opt_sec_spkm:
+ pseudoflavor = RPC_AUTH_GSS_SPKM;
+ break;
+ case Opt_sec_spkmi:
+ pseudoflavor = RPC_AUTH_GSS_SPKMI;
+ break;
+ case Opt_sec_spkmp:
+ pseudoflavor = RPC_AUTH_GSS_SPKMP;
+ break;
+ default:
+ dfprintk(MOUNT,
+ "NFS: sec= option '%s' not recognized\n", p);
+ return 0;
+ }
+
+ if (!nfs_auth_info_add(&mnt->auth_info, pseudoflavor))
+ return 0;
+ }
+
+ return 1;
+}
+
+static int nfs_parse_version_string(char *string,
+ struct nfs_parsed_mount_data *mnt,
+ substring_t *args)
+{
+ mnt->flags &= ~NFS_MOUNT_VER3;
+ switch (match_token(string, nfs_vers_tokens, args)) {
+ case Opt_vers_2:
+ mnt->version = 2;
+ break;
+ case Opt_vers_3:
+ mnt->flags |= NFS_MOUNT_VER3;
+ mnt->version = 3;
+ break;
+ case Opt_vers_4:
+ /* Backward compatibility option. In future,
+ * the mount program should always supply
+ * a NFSv4 minor version number.
+ */
+ mnt->version = 4;
+ break;
+ case Opt_vers_4_0:
+ mnt->version = 4;
+ mnt->minorversion = 0;
+ break;
+ case Opt_vers_4_1:
+ mnt->version = 4;
+ mnt->minorversion = 1;
+ break;
+ case Opt_vers_4_2:
+ mnt->version = 4;
+ mnt->minorversion = 2;
+ break;
+ default:
+ return 0;
+ }
+ return 1;
+}
+
+static int nfs_get_option_str(substring_t args[], char **option)
+{
+ kfree(*option);
+ *option = match_strdup(args);
+ return !*option;
+}
+
+static int nfs_get_option_ul(substring_t args[], unsigned long *option)
+{
+ int rc;
+ char *string;
+
+ string = match_strdup(args);
+ if (string == NULL)
+ return -ENOMEM;
+ rc = kstrtoul(string, 10, option);
+ kfree(string);
+
+ return rc;
+}
+
+static int nfs_get_option_ul_bound(substring_t args[], unsigned long *option,
+ unsigned long l_bound, unsigned long u_bound)
+{
+ int ret;
+
+ ret = nfs_get_option_ul(args, option);
+ if (ret != 0)
+ return ret;
+ if (*option < l_bound || *option > u_bound)
+ return -ERANGE;
+ return 0;
+}
+
+/*
+ * Error-check and convert a string of mount options from user space into
+ * a data structure. The whole mount string is processed; bad options are
+ * skipped as they are encountered. If there were no errors, return 1;
+ * otherwise return 0 (zero).
+ */
+int nfs_parse_mount_options(char *raw, struct nfs_parsed_mount_data *mnt)
+{
+ char *p, *string;
+ int rc, sloppy = 0, invalid_option = 0;
+ unsigned short protofamily = AF_UNSPEC;
+ unsigned short mountfamily = AF_UNSPEC;
+
+ if (!raw) {
+ dfprintk(MOUNT, "NFS: mount options string was NULL.\n");
+ return 1;
+ }
+ dfprintk(MOUNT, "NFS: nfs mount opts='%s'\n", raw);
+
+ rc = security_sb_eat_lsm_opts(raw, &mnt->lsm_opts);
+ if (rc)
+ goto out_security_failure;
+
+ while ((p = strsep(&raw, ",")) != NULL) {
+ substring_t args[MAX_OPT_ARGS];
+ unsigned long option;
+ int token;
+
+ if (!*p)
+ continue;
+
+ dfprintk(MOUNT, "NFS: parsing nfs mount option '%s'\n", p);
+
+ token = match_token(p, nfs_mount_option_tokens, args);
+ switch (token) {
+
+ /*
+ * boolean options: foo/nofoo
+ */
+ case Opt_soft:
+ mnt->flags |= NFS_MOUNT_SOFT;
+ mnt->flags &= ~NFS_MOUNT_SOFTERR;
+ break;
+ case Opt_softerr:
+ mnt->flags |= NFS_MOUNT_SOFTERR;
+ mnt->flags &= ~NFS_MOUNT_SOFT;
+ break;
+ case Opt_hard:
+ mnt->flags &= ~(NFS_MOUNT_SOFT|NFS_MOUNT_SOFTERR);
+ break;
+ case Opt_posix:
+ mnt->flags |= NFS_MOUNT_POSIX;
+ break;
+ case Opt_noposix:
+ mnt->flags &= ~NFS_MOUNT_POSIX;
+ break;
+ case Opt_cto:
+ mnt->flags &= ~NFS_MOUNT_NOCTO;
+ break;
+ case Opt_nocto:
+ mnt->flags |= NFS_MOUNT_NOCTO;
+ break;
+ case Opt_ac:
+ mnt->flags &= ~NFS_MOUNT_NOAC;
+ break;
+ case Opt_noac:
+ mnt->flags |= NFS_MOUNT_NOAC;
+ break;
+ case Opt_lock:
+ mnt->flags &= ~NFS_MOUNT_NONLM;
+ mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
+ NFS_MOUNT_LOCAL_FCNTL);
+ break;
+ case Opt_nolock:
+ mnt->flags |= NFS_MOUNT_NONLM;
+ mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
+ NFS_MOUNT_LOCAL_FCNTL);
+ break;
+ case Opt_udp:
+ mnt->flags &= ~NFS_MOUNT_TCP;
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP;
+ break;
+ case Opt_tcp:
+ mnt->flags |= NFS_MOUNT_TCP;
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
+ break;
+ case Opt_rdma:
+ mnt->flags |= NFS_MOUNT_TCP; /* for side protocols */
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_RDMA;
+ xprt_load_transport(p);
+ break;
+ case Opt_acl:
+ mnt->flags &= ~NFS_MOUNT_NOACL;
+ break;
+ case Opt_noacl:
+ mnt->flags |= NFS_MOUNT_NOACL;
+ break;
+ case Opt_rdirplus:
+ mnt->flags &= ~NFS_MOUNT_NORDIRPLUS;
+ break;
+ case Opt_nordirplus:
+ mnt->flags |= NFS_MOUNT_NORDIRPLUS;
+ break;
+ case Opt_sharecache:
+ mnt->flags &= ~NFS_MOUNT_UNSHARED;
+ break;
+ case Opt_nosharecache:
+ mnt->flags |= NFS_MOUNT_UNSHARED;
+ break;
+ case Opt_resvport:
+ mnt->flags &= ~NFS_MOUNT_NORESVPORT;
+ break;
+ case Opt_noresvport:
+ mnt->flags |= NFS_MOUNT_NORESVPORT;
+ break;
+ case Opt_fscache:
+ mnt->options |= NFS_OPTION_FSCACHE;
+ kfree(mnt->fscache_uniq);
+ mnt->fscache_uniq = NULL;
+ break;
+ case Opt_nofscache:
+ mnt->options &= ~NFS_OPTION_FSCACHE;
+ kfree(mnt->fscache_uniq);
+ mnt->fscache_uniq = NULL;
+ break;
+ case Opt_migration:
+ mnt->options |= NFS_OPTION_MIGRATION;
+ break;
+ case Opt_nomigration:
+ mnt->options &= ~NFS_OPTION_MIGRATION;
+ break;
+
+ /*
+ * options that take numeric values
+ */
+ case Opt_port:
+ if (nfs_get_option_ul(args, &option) ||
+ option > USHRT_MAX)
+ goto out_invalid_value;
+ mnt->nfs_server.port = option;
+ break;
+ case Opt_rsize:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->rsize = option;
+ break;
+ case Opt_wsize:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->wsize = option;
+ break;
+ case Opt_bsize:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->bsize = option;
+ break;
+ case Opt_timeo:
+ if (nfs_get_option_ul_bound(args, &option, 1, INT_MAX))
+ goto out_invalid_value;
+ mnt->timeo = option;
+ break;
+ case Opt_retrans:
+ if (nfs_get_option_ul_bound(args, &option, 0, INT_MAX))
+ goto out_invalid_value;
+ mnt->retrans = option;
+ break;
+ case Opt_acregmin:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->acregmin = option;
+ break;
+ case Opt_acregmax:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->acregmax = option;
+ break;
+ case Opt_acdirmin:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->acdirmin = option;
+ break;
+ case Opt_acdirmax:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->acdirmax = option;
+ break;
+ case Opt_actimeo:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->acregmin = mnt->acregmax =
+ mnt->acdirmin = mnt->acdirmax = option;
+ break;
+ case Opt_namelen:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ mnt->namlen = option;
+ break;
+ case Opt_mountport:
+ if (nfs_get_option_ul(args, &option) ||
+ option > USHRT_MAX)
+ goto out_invalid_value;
+ mnt->mount_server.port = option;
+ break;
+ case Opt_mountvers:
+ if (nfs_get_option_ul(args, &option) ||
+ option < NFS_MNT_VERSION ||
+ option > NFS_MNT3_VERSION)
+ goto out_invalid_value;
+ mnt->mount_server.version = option;
+ break;
+ case Opt_minorversion:
+ if (nfs_get_option_ul(args, &option))
+ goto out_invalid_value;
+ if (option > NFS4_MAX_MINOR_VERSION)
+ goto out_invalid_value;
+ mnt->minorversion = option;
+ break;
+
+ /*
+ * options that take text values
+ */
+ case Opt_nfsvers:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ rc = nfs_parse_version_string(string, mnt, args);
+ kfree(string);
+ if (!rc)
+ goto out_invalid_value;
+ break;
+ case Opt_sec:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ rc = nfs_parse_security_flavors(string, mnt);
+ kfree(string);
+ if (!rc) {
+ dfprintk(MOUNT, "NFS: unrecognized "
+ "security flavor\n");
+ return 0;
+ }
+ break;
+ case Opt_proto:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ token = match_token(string,
+ nfs_xprt_protocol_tokens, args);
+
+ protofamily = AF_INET;
+ switch (token) {
+ case Opt_xprt_udp6:
+ protofamily = AF_INET6;
+ /* fall through */
+ case Opt_xprt_udp:
+ mnt->flags &= ~NFS_MOUNT_TCP;
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP;
+ break;
+ case Opt_xprt_tcp6:
+ protofamily = AF_INET6;
+ /* fall through */
+ case Opt_xprt_tcp:
+ mnt->flags |= NFS_MOUNT_TCP;
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
+ break;
+ case Opt_xprt_rdma6:
+ protofamily = AF_INET6;
+ /* fall through */
+ case Opt_xprt_rdma:
+ /* vector side protocols to TCP */
+ mnt->flags |= NFS_MOUNT_TCP;
+ mnt->nfs_server.protocol = XPRT_TRANSPORT_RDMA;
+ xprt_load_transport(string);
+ break;
+ default:
+ dfprintk(MOUNT, "NFS: unrecognized "
+ "transport protocol\n");
+ kfree(string);
+ return 0;
+ }
+ kfree(string);
+ break;
+ case Opt_mountproto:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ token = match_token(string,
+ nfs_xprt_protocol_tokens, args);
+ kfree(string);
+
+ mountfamily = AF_INET;
+ switch (token) {
+ case Opt_xprt_udp6:
+ mountfamily = AF_INET6;
+ /* fall through */
+ case Opt_xprt_udp:
+ mnt->mount_server.protocol = XPRT_TRANSPORT_UDP;
+ break;
+ case Opt_xprt_tcp6:
+ mountfamily = AF_INET6;
+ /* fall through */
+ case Opt_xprt_tcp:
+ mnt->mount_server.protocol = XPRT_TRANSPORT_TCP;
+ break;
+ case Opt_xprt_rdma: /* not used for side protocols */
+ default:
+ dfprintk(MOUNT, "NFS: unrecognized "
+ "transport protocol\n");
+ return 0;
+ }
+ break;
+ case Opt_addr:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ mnt->nfs_server.addrlen =
+ rpc_pton(mnt->net, string, strlen(string),
+ (struct sockaddr *)
+ &mnt->nfs_server.address,
+ sizeof(mnt->nfs_server.address));
+ kfree(string);
+ if (mnt->nfs_server.addrlen == 0)
+ goto out_invalid_address;
+ break;
+ case Opt_clientaddr:
+ if (nfs_get_option_str(args, &mnt->client_address))
+ goto out_nomem;
+ break;
+ case Opt_mounthost:
+ if (nfs_get_option_str(args,
+ &mnt->mount_server.hostname))
+ goto out_nomem;
+ break;
+ case Opt_mountaddr:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ mnt->mount_server.addrlen =
+ rpc_pton(mnt->net, string, strlen(string),
+ (struct sockaddr *)
+ &mnt->mount_server.address,
+ sizeof(mnt->mount_server.address));
+ kfree(string);
+ if (mnt->mount_server.addrlen == 0)
+ goto out_invalid_address;
+ break;
+ case Opt_nconnect:
+ if (nfs_get_option_ul_bound(args, &option, 1, NFS_MAX_CONNECTIONS))
+ goto out_invalid_value;
+ mnt->nfs_server.nconnect = option;
+ break;
+ case Opt_lookupcache:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ token = match_token(string,
+ nfs_lookupcache_tokens, args);
+ kfree(string);
+ switch (token) {
+ case Opt_lookupcache_all:
+ mnt->flags &= ~(NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE);
+ break;
+ case Opt_lookupcache_positive:
+ mnt->flags &= ~NFS_MOUNT_LOOKUP_CACHE_NONE;
+ mnt->flags |= NFS_MOUNT_LOOKUP_CACHE_NONEG;
+ break;
+ case Opt_lookupcache_none:
+ mnt->flags |= NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE;
+ break;
+ default:
+ dfprintk(MOUNT, "NFS: invalid "
+ "lookupcache argument\n");
+ return 0;
+ };
+ break;
+ case Opt_fscache_uniq:
+ if (nfs_get_option_str(args, &mnt->fscache_uniq))
+ goto out_nomem;
+ mnt->options |= NFS_OPTION_FSCACHE;
+ break;
+ case Opt_local_lock:
+ string = match_strdup(args);
+ if (string == NULL)
+ goto out_nomem;
+ token = match_token(string, nfs_local_lock_tokens,
+ args);
+ kfree(string);
+ switch (token) {
+ case Opt_local_lock_all:
+ mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
+ NFS_MOUNT_LOCAL_FCNTL);
+ break;
+ case Opt_local_lock_flock:
+ mnt->flags |= NFS_MOUNT_LOCAL_FLOCK;
+ break;
+ case Opt_local_lock_posix:
+ mnt->flags |= NFS_MOUNT_LOCAL_FCNTL;
+ break;
+ case Opt_local_lock_none:
+ mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
+ NFS_MOUNT_LOCAL_FCNTL);
+ break;
+ default:
+ dfprintk(MOUNT, "NFS: invalid "
+ "local_lock argument\n");
+ return 0;
+ };
+ break;
+
+ /*
+ * Special options
+ */
+ case Opt_sloppy:
+ sloppy = 1;
+ dfprintk(MOUNT, "NFS: relaxing parsing rules\n");
+ break;
+ case Opt_userspace:
+ case Opt_deprecated:
+ dfprintk(MOUNT, "NFS: ignoring mount option "
+ "'%s'\n", p);
+ break;
+
+ default:
+ invalid_option = 1;
+ dfprintk(MOUNT, "NFS: unrecognized mount option "
+ "'%s'\n", p);
+ }
+ }
+
+ if (!sloppy && invalid_option)
+ return 0;
+
+ if (mnt->minorversion && mnt->version != 4)
+ goto out_minorversion_mismatch;
+
+ if (mnt->options & NFS_OPTION_MIGRATION &&
+ (mnt->version != 4 || mnt->minorversion != 0))
+ goto out_migration_misuse;
+
+ /*
+ * verify that any proto=/mountproto= options match the address
+ * families in the addr=/mountaddr= options.
+ */
+ if (protofamily != AF_UNSPEC &&
+ protofamily != mnt->nfs_server.address.ss_family)
+ goto out_proto_mismatch;
+
+ if (mountfamily != AF_UNSPEC) {
+ if (mnt->mount_server.addrlen) {
+ if (mountfamily != mnt->mount_server.address.ss_family)
+ goto out_mountproto_mismatch;
+ } else {
+ if (mountfamily != mnt->nfs_server.address.ss_family)
+ goto out_mountproto_mismatch;
+ }
+ }
+
+ return 1;
+
+out_mountproto_mismatch:
+ printk(KERN_INFO "NFS: mount server address does not match mountproto= "
+ "option\n");
+ return 0;
+out_proto_mismatch:
+ printk(KERN_INFO "NFS: server address does not match proto= option\n");
+ return 0;
+out_invalid_address:
+ printk(KERN_INFO "NFS: bad IP address specified: %s\n", p);
+ return 0;
+out_invalid_value:
+ printk(KERN_INFO "NFS: bad mount option value specified: %s\n", p);
+ return 0;
+out_minorversion_mismatch:
+ printk(KERN_INFO "NFS: mount option vers=%u does not support "
+ "minorversion=%u\n", mnt->version, mnt->minorversion);
+ return 0;
+out_migration_misuse:
+ printk(KERN_INFO
+ "NFS: 'migration' not supported for this NFS version\n");
+ return 0;
+out_nomem:
+ printk(KERN_INFO "NFS: not enough memory to parse option\n");
+ return 0;
+out_security_failure:
+ printk(KERN_INFO "NFS: security options invalid: %d\n", rc);
+ return 0;
+}
+
+/*
+ * Split "dev_name" into "hostname:export_path".
+ *
+ * The leftmost colon demarks the split between the server's hostname
+ * and the export path. If the hostname starts with a left square
+ * bracket, then it may contain colons.
+ *
+ * Note: caller frees hostname and export path, even on error.
+ */
+static int nfs_parse_devname(const char *dev_name,
+ char **hostname, size_t maxnamlen,
+ char **export_path, size_t maxpathlen)
+{
+ size_t len;
+ char *end;
+
+ if (unlikely(!dev_name || !*dev_name)) {
+ dfprintk(MOUNT, "NFS: device name not specified\n");
+ return -EINVAL;
+ }
+
+ /* Is the host name protected with square brakcets? */
+ if (*dev_name == '[') {
+ end = strchr(++dev_name, ']');
+ if (end == NULL || end[1] != ':')
+ goto out_bad_devname;
+
+ len = end - dev_name;
+ end++;
+ } else {
+ char *comma;
+
+ end = strchr(dev_name, ':');
+ if (end == NULL)
+ goto out_bad_devname;
+ len = end - dev_name;
+
+ /* kill possible hostname list: not supported */
+ comma = strchr(dev_name, ',');
+ if (comma != NULL && comma < end)
+ len = comma - dev_name;
+ }
+
+ if (len > maxnamlen)
+ goto out_hostname;
+
+ /* N.B. caller will free nfs_server.hostname in all cases */
+ *hostname = kstrndup(dev_name, len, GFP_KERNEL);
+ if (*hostname == NULL)
+ goto out_nomem;
+ len = strlen(++end);
+ if (len > maxpathlen)
+ goto out_path;
+ *export_path = kstrndup(end, len, GFP_KERNEL);
+ if (!*export_path)
+ goto out_nomem;
+
+ dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", *export_path);
+ return 0;
+
+out_bad_devname:
+ dfprintk(MOUNT, "NFS: device name not in host:path format\n");
+ return -EINVAL;
+
+out_nomem:
+ dfprintk(MOUNT, "NFS: not enough memory to parse device name\n");
+ return -ENOMEM;
+
+out_hostname:
+ dfprintk(MOUNT, "NFS: server hostname too long\n");
+ return -ENAMETOOLONG;
+
+out_path:
+ dfprintk(MOUNT, "NFS: export pathname too long\n");
+ return -ENAMETOOLONG;
+}
+
+/*
+ * Validate the NFS2/NFS3 mount data
+ * - fills in the mount root filehandle
+ *
+ * For option strings, user space handles the following behaviors:
+ *
+ * + DNS: mapping server host name to IP address ("addr=" option)
+ *
+ * + failure mode: how to behave if a mount request can't be handled
+ * immediately ("fg/bg" option)
+ *
+ * + retry: how often to retry a mount request ("retry=" option)
+ *
+ * + breaking back: trying proto=udp after proto=tcp, v2 after v3,
+ * mountproto=tcp after mountproto=udp, and so on
+ */
+static int nfs23_validate_mount_data(void *options,
+ struct nfs_parsed_mount_data *args,
+ struct nfs_fh *mntfh,
+ const char *dev_name)
+{
+ struct nfs_mount_data *data = (struct nfs_mount_data *)options;
+ struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address;
+ int extra_flags = NFS_MOUNT_LEGACY_INTERFACE;
+
+ if (data == NULL)
+ goto out_no_data;
+
+ args->version = NFS_DEFAULT_VERSION;
+ switch (data->version) {
+ case 1:
+ data->namlen = 0; /* fall through */
+ case 2:
+ data->bsize = 0; /* fall through */
+ case 3:
+ if (data->flags & NFS_MOUNT_VER3)
+ goto out_no_v3;
+ data->root.size = NFS2_FHSIZE;
+ memcpy(data->root.data, data->old_root.data, NFS2_FHSIZE);
+ /* Turn off security negotiation */
+ extra_flags |= NFS_MOUNT_SECFLAVOUR;
+ /* fall through */
+ case 4:
+ if (data->flags & NFS_MOUNT_SECFLAVOUR)
+ goto out_no_sec;
+ /* fall through */
+ case 5:
+ memset(data->context, 0, sizeof(data->context));
+ /* fall through */
+ case 6:
+ if (data->flags & NFS_MOUNT_VER3) {
+ if (data->root.size > NFS3_FHSIZE || data->root.size == 0)
+ goto out_invalid_fh;
+ mntfh->size = data->root.size;
+ args->version = 3;
+ } else {
+ mntfh->size = NFS2_FHSIZE;
+ args->version = 2;
+ }
+
+
+ memcpy(mntfh->data, data->root.data, mntfh->size);
+ if (mntfh->size < sizeof(mntfh->data))
+ memset(mntfh->data + mntfh->size, 0,
+ sizeof(mntfh->data) - mntfh->size);
+
+ /*
+ * Translate to nfs_parsed_mount_data, which nfs_fill_super
+ * can deal with.
+ */
+ args->flags = data->flags & NFS_MOUNT_FLAGMASK;
+ args->flags |= extra_flags;
+ args->rsize = data->rsize;
+ args->wsize = data->wsize;
+ args->timeo = data->timeo;
+ args->retrans = data->retrans;
+ args->acregmin = data->acregmin;
+ args->acregmax = data->acregmax;
+ args->acdirmin = data->acdirmin;
+ args->acdirmax = data->acdirmax;
+ args->need_mount = false;
+
+ memcpy(sap, &data->addr, sizeof(data->addr));
+ args->nfs_server.addrlen = sizeof(data->addr);
+ args->nfs_server.port = ntohs(data->addr.sin_port);
+ if (sap->sa_family != AF_INET ||
+ !nfs_verify_server_address(sap))
+ goto out_no_address;
+
+ if (!(data->flags & NFS_MOUNT_TCP))
+ args->nfs_server.protocol = XPRT_TRANSPORT_UDP;
+ /* N.B. caller will free nfs_server.hostname in all cases */
+ args->nfs_server.hostname = kstrdup(data->hostname, GFP_KERNEL);
+ args->namlen = data->namlen;
+ args->bsize = data->bsize;
+
+ if (data->flags & NFS_MOUNT_SECFLAVOUR)
+ args->selected_flavor = data->pseudoflavor;
+ else
+ args->selected_flavor = RPC_AUTH_UNIX;
+ if (!args->nfs_server.hostname)
+ goto out_nomem;
+
+ if (!(data->flags & NFS_MOUNT_NONLM))
+ args->flags &= ~(NFS_MOUNT_LOCAL_FLOCK|
+ NFS_MOUNT_LOCAL_FCNTL);
+ else
+ args->flags |= (NFS_MOUNT_LOCAL_FLOCK|
+ NFS_MOUNT_LOCAL_FCNTL);
+ /*
+ * The legacy version 6 binary mount data from userspace has a
+ * field used only to transport selinux information into the
+ * the kernel. To continue to support that functionality we
+ * have a touch of selinux knowledge here in the NFS code. The
+ * userspace code converted context=blah to just blah so we are
+ * converting back to the full string selinux understands.
+ */
+ if (data->context[0]){
+#ifdef CONFIG_SECURITY_SELINUX
+ int rc;
+ data->context[NFS_MAX_CONTEXT_LEN] = '\0';
+ rc = security_add_mnt_opt("context", data->context,
+ strlen(data->context), &args->lsm_opts);
+ if (rc)
+ return rc;
+#else
+ return -EINVAL;
+#endif
+ }
+
+ break;
+ default:
+ return NFS_TEXT_DATA;
+ }
+
+ return 0;
+
+out_no_data:
+ dfprintk(MOUNT, "NFS: mount program didn't pass any mount data\n");
+ return -EINVAL;
+
+out_no_v3:
+ dfprintk(MOUNT, "NFS: nfs_mount_data version %d does not support v3\n",
+ data->version);
+ return -EINVAL;
+
+out_no_sec:
+ dfprintk(MOUNT, "NFS: nfs_mount_data version supports only AUTH_SYS\n");
+ return -EINVAL;
+
+out_nomem:
+ dfprintk(MOUNT, "NFS: not enough memory to handle mount options\n");
+ return -ENOMEM;
+
+out_no_address:
+ dfprintk(MOUNT, "NFS: mount program didn't pass remote address\n");
+ return -EINVAL;
+
+out_invalid_fh:
+ dfprintk(MOUNT, "NFS: invalid root filehandle\n");
+ return -EINVAL;
+}
+
+#if IS_ENABLED(CONFIG_NFS_V4)
+
+static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data *args)
+{
+ args->flags &= ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3|
+ NFS_MOUNT_LOCAL_FLOCK|NFS_MOUNT_LOCAL_FCNTL);
+}
+
+/*
+ * Validate NFSv4 mount options
+ */
+static int nfs4_validate_mount_data(void *options,
+ struct nfs_parsed_mount_data *args,
+ const char *dev_name)
+{
+ struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address;
+ struct nfs4_mount_data *data = (struct nfs4_mount_data *)options;
+ char *c;
+
+ if (data == NULL)
+ goto out_no_data;
+
+ args->version = 4;
+
+ switch (data->version) {
+ case 1:
+ if (data->host_addrlen > sizeof(args->nfs_server.address))
+ goto out_no_address;
+ if (data->host_addrlen == 0)
+ goto out_no_address;
+ args->nfs_server.addrlen = data->host_addrlen;
+ if (copy_from_user(sap, data->host_addr, data->host_addrlen))
+ return -EFAULT;
+ if (!nfs_verify_server_address(sap))
+ goto out_no_address;
+ args->nfs_server.port = ntohs(((struct sockaddr_in *)sap)->sin_port);
+
+ if (data->auth_flavourlen) {
+ rpc_authflavor_t pseudoflavor;
+ if (data->auth_flavourlen > 1)
+ goto out_inval_auth;
+ if (copy_from_user(&pseudoflavor,
+ data->auth_flavours,
+ sizeof(pseudoflavor)))
+ return -EFAULT;
+ args->selected_flavor = pseudoflavor;
+ } else
+ args->selected_flavor = RPC_AUTH_UNIX;
+
+ c = strndup_user(data->hostname.data, NFS4_MAXNAMLEN);
+ if (IS_ERR(c))
+ return PTR_ERR(c);
+ args->nfs_server.hostname = c;
+
+ c = strndup_user(data->mnt_path.data, NFS4_MAXPATHLEN);
+ if (IS_ERR(c))
+ return PTR_ERR(c);
+ args->nfs_server.export_path = c;
+ dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", c);
+
+ c = strndup_user(data->client_addr.data, 16);
+ if (IS_ERR(c))
+ return PTR_ERR(c);
+ args->client_address = c;
+
+ /*
+ * Translate to nfs_parsed_mount_data, which nfs4_fill_super
+ * can deal with.
+ */
+
+ args->flags = data->flags & NFS4_MOUNT_FLAGMASK;
+ args->rsize = data->rsize;
+ args->wsize = data->wsize;
+ args->timeo = data->timeo;
+ args->retrans = data->retrans;
+ args->acregmin = data->acregmin;
+ args->acregmax = data->acregmax;
+ args->acdirmin = data->acdirmin;
+ args->acdirmax = data->acdirmax;
+ args->nfs_server.protocol = data->proto;
+ nfs_validate_transport_protocol(args);
+ if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
+ goto out_invalid_transport_udp;
+
+ break;
+ default:
+ return NFS_TEXT_DATA;
+ }
+
+ return 0;
+
+out_no_data:
+ dfprintk(MOUNT, "NFS4: mount program didn't pass any mount data\n");
+ return -EINVAL;
+
+out_inval_auth:
+ dfprintk(MOUNT, "NFS4: Invalid number of RPC auth flavours %d\n",
+ data->auth_flavourlen);
+ return -EINVAL;
+
+out_no_address:
+ dfprintk(MOUNT, "NFS4: mount program didn't pass remote address\n");
+ return -EINVAL;
+
+out_invalid_transport_udp:
+ dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
+ return -EINVAL;
+}
+
+int nfs_validate_mount_data(struct file_system_type *fs_type,
+ void *options,
+ struct nfs_parsed_mount_data *args,
+ struct nfs_fh *mntfh,
+ const char *dev_name)
+{
+ if (fs_type == &nfs_fs_type)
+ return nfs23_validate_mount_data(options, args, mntfh, dev_name);
+ return nfs4_validate_mount_data(options, args, dev_name);
+}
+#else
+int nfs_validate_mount_data(struct file_system_type *fs_type,
+ void *options,
+ struct nfs_parsed_mount_data *args,
+ struct nfs_fh *mntfh,
+ const char *dev_name)
+{
+ return nfs23_validate_mount_data(options, args, mntfh, dev_name);
+}
+#endif
+
+int nfs_validate_text_mount_data(void *options,
+ struct nfs_parsed_mount_data *args,
+ const char *dev_name)
+{
+ int port = 0;
+ int max_namelen = PAGE_SIZE;
+ int max_pathlen = NFS_MAXPATHLEN;
+ struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address;
+
+ if (nfs_parse_mount_options((char *)options, args) == 0)
+ return -EINVAL;
+
+ if (!nfs_verify_server_address(sap))
+ goto out_no_address;
+
+ if (args->version == 4) {
+#if IS_ENABLED(CONFIG_NFS_V4)
+ if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
+ port = NFS_RDMA_PORT;
+ else
+ port = NFS_PORT;
+ max_namelen = NFS4_MAXNAMLEN;
+ max_pathlen = NFS4_MAXPATHLEN;
+ nfs_validate_transport_protocol(args);
+ if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
+ goto out_invalid_transport_udp;
+ nfs4_validate_mount_flags(args);
+#else
+ goto out_v4_not_compiled;
+#endif /* CONFIG_NFS_V4 */
+ } else {
+ nfs_set_mount_transport_protocol(args);
+ if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
+ port = NFS_RDMA_PORT;
+ }
+
+ nfs_set_port(sap, &args->nfs_server.port, port);
+
+ return nfs_parse_devname(dev_name,
+ &args->nfs_server.hostname,
+ max_namelen,
+ &args->nfs_server.export_path,
+ max_pathlen);
+
+#if !IS_ENABLED(CONFIG_NFS_V4)
+out_v4_not_compiled:
+ dfprintk(MOUNT, "NFS: NFSv4 is not compiled into kernel\n");
+ return -EPROTONOSUPPORT;
+#else
+out_invalid_transport_udp:
+ dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
+ return -EINVAL;
+#endif /* !CONFIG_NFS_V4 */
+
+out_no_address:
+ dfprintk(MOUNT, "NFS: mount program didn't pass remote address\n");
+ return -EINVAL;
+}
diff --git a/fs/nfs/internal.h b/fs/nfs/internal.h
index d512ec394559..b66fd35993b3 100644
--- a/fs/nfs/internal.h
+++ b/fs/nfs/internal.h
@@ -7,6 +7,7 @@
#include <linux/mount.h>
#include <linux/security.h>
#include <linux/crc32.h>
+#include <linux/sunrpc/addr.h>
#include <linux/nfs_page.h>
#include <linux/wait_bit.h>
@@ -232,6 +233,22 @@ extern const struct svc_version nfs4_callback_version1;
extern const struct svc_version nfs4_callback_version4;
struct nfs_pageio_descriptor;
+
+/* mount.c */
+#define NFS_TEXT_DATA 1
+
+extern struct nfs_parsed_mount_data *nfs_alloc_parsed_mount_data(void);
+extern void nfs_free_parsed_mount_data(struct nfs_parsed_mount_data *data);
+extern int nfs_parse_mount_options(char *raw, struct nfs_parsed_mount_data *mnt);
+extern int nfs_validate_mount_data(struct file_system_type *fs_type,
+ void *options,
+ struct nfs_parsed_mount_data *args,
+ struct nfs_fh *mntfh,
+ const char *dev_name);
+extern int nfs_validate_text_mount_data(void *options,
+ struct nfs_parsed_mount_data *args,
+ const char *dev_name);
+
/* pagelist.c */
extern int __init nfs_init_nfspagecache(void);
extern void nfs_destroy_nfspagecache(void);
@@ -763,3 +780,15 @@ static inline bool nfs_error_is_fatal(int err)
}
}
+/*
+ * Select between a default port value and a user-specified port value.
+ * If a zero value is set, then autobind will be used.
+ */
+static inline void nfs_set_port(struct sockaddr *sap, int *port,
+ const unsigned short default_port)
+{
+ if (*port == NFS_UNSPEC_PORT)
+ *port = default_port;
+
+ rpc_set_port(sap, *port);
+}
diff --git a/fs/nfs/super.c b/fs/nfs/super.c
index d8702e57f7fc..886220d2da4e 100644
--- a/fs/nfs/super.c
+++ b/fs/nfs/super.c
@@ -69,229 +69,6 @@
#include "nfs.h"
#define NFSDBG_FACILITY NFSDBG_VFS
-#define NFS_TEXT_DATA 1
-
-#if IS_ENABLED(CONFIG_NFS_V3)
-#define NFS_DEFAULT_VERSION 3
-#else
-#define NFS_DEFAULT_VERSION 2
-#endif
-
-#define NFS_MAX_CONNECTIONS 16
-
-enum {
- /* Mount options that take no arguments */
- Opt_soft, Opt_softerr, Opt_hard,
- Opt_posix, Opt_noposix,
- Opt_cto, Opt_nocto,
- Opt_ac, Opt_noac,
- Opt_lock, Opt_nolock,
- Opt_udp, Opt_tcp, Opt_rdma,
- Opt_acl, Opt_noacl,
- Opt_rdirplus, Opt_nordirplus,
- Opt_sharecache, Opt_nosharecache,
- Opt_resvport, Opt_noresvport,
- Opt_fscache, Opt_nofscache,
- Opt_migration, Opt_nomigration,
-
- /* Mount options that take integer arguments */
- Opt_port,
- Opt_rsize, Opt_wsize, Opt_bsize,
- Opt_timeo, Opt_retrans,
- Opt_acregmin, Opt_acregmax,
- Opt_acdirmin, Opt_acdirmax,
- Opt_actimeo,
- Opt_namelen,
- Opt_mountport,
- Opt_mountvers,
- Opt_minorversion,
-
- /* Mount options that take string arguments */
- Opt_nfsvers,
- Opt_sec, Opt_proto, Opt_mountproto, Opt_mounthost,
- Opt_addr, Opt_mountaddr, Opt_clientaddr,
- Opt_nconnect,
- Opt_lookupcache,
- Opt_fscache_uniq,
- Opt_local_lock,
-
- /* Special mount options */
- Opt_userspace, Opt_deprecated, Opt_sloppy,
-
- Opt_err
-};
-
-static const match_table_t nfs_mount_option_tokens = {
- { Opt_userspace, "bg" },
- { Opt_userspace, "fg" },
- { Opt_userspace, "retry=%s" },
-
- { Opt_sloppy, "sloppy" },
-
- { Opt_soft, "soft" },
- { Opt_softerr, "softerr" },
- { Opt_hard, "hard" },
- { Opt_deprecated, "intr" },
- { Opt_deprecated, "nointr" },
- { Opt_posix, "posix" },
- { Opt_noposix, "noposix" },
- { Opt_cto, "cto" },
- { Opt_nocto, "nocto" },
- { Opt_ac, "ac" },
- { Opt_noac, "noac" },
- { Opt_lock, "lock" },
- { Opt_nolock, "nolock" },
- { Opt_udp, "udp" },
- { Opt_tcp, "tcp" },
- { Opt_rdma, "rdma" },
- { Opt_acl, "acl" },
- { Opt_noacl, "noacl" },
- { Opt_rdirplus, "rdirplus" },
- { Opt_nordirplus, "nordirplus" },
- { Opt_sharecache, "sharecache" },
- { Opt_nosharecache, "nosharecache" },
- { Opt_resvport, "resvport" },
- { Opt_noresvport, "noresvport" },
- { Opt_fscache, "fsc" },
- { Opt_nofscache, "nofsc" },
- { Opt_migration, "migration" },
- { Opt_nomigration, "nomigration" },
-
- { Opt_port, "port=%s" },
- { Opt_rsize, "rsize=%s" },
- { Opt_wsize, "wsize=%s" },
- { Opt_bsize, "bsize=%s" },
- { Opt_timeo, "timeo=%s" },
- { Opt_retrans, "retrans=%s" },
- { Opt_acregmin, "acregmin=%s" },
- { Opt_acregmax, "acregmax=%s" },
- { Opt_acdirmin, "acdirmin=%s" },
- { Opt_acdirmax, "acdirmax=%s" },
- { Opt_actimeo, "actimeo=%s" },
- { Opt_namelen, "namlen=%s" },
- { Opt_mountport, "mountport=%s" },
- { Opt_mountvers, "mountvers=%s" },
- { Opt_minorversion, "minorversion=%s" },
-
- { Opt_nfsvers, "nfsvers=%s" },
- { Opt_nfsvers, "vers=%s" },
-
- { Opt_sec, "sec=%s" },
- { Opt_proto, "proto=%s" },
- { Opt_mountproto, "mountproto=%s" },
- { Opt_addr, "addr=%s" },
- { Opt_clientaddr, "clientaddr=%s" },
- { Opt_mounthost, "mounthost=%s" },
- { Opt_mountaddr, "mountaddr=%s" },
-
- { Opt_nconnect, "nconnect=%s" },
-
- { Opt_lookupcache, "lookupcache=%s" },
- { Opt_fscache_uniq, "fsc=%s" },
- { Opt_local_lock, "local_lock=%s" },
-
- /* The following needs to be listed after all other options */
- { Opt_nfsvers, "v%s" },
-
- { Opt_err, NULL }
-};
-
-enum {
- Opt_xprt_udp, Opt_xprt_udp6, Opt_xprt_tcp, Opt_xprt_tcp6, Opt_xprt_rdma,
- Opt_xprt_rdma6,
-
- Opt_xprt_err
-};
-
-static const match_table_t nfs_xprt_protocol_tokens = {
- { Opt_xprt_udp, "udp" },
- { Opt_xprt_udp6, "udp6" },
- { Opt_xprt_tcp, "tcp" },
- { Opt_xprt_tcp6, "tcp6" },
- { Opt_xprt_rdma, "rdma" },
- { Opt_xprt_rdma6, "rdma6" },
-
- { Opt_xprt_err, NULL }
-};
-
-enum {
- Opt_sec_none, Opt_sec_sys,
- Opt_sec_krb5, Opt_sec_krb5i, Opt_sec_krb5p,
- Opt_sec_lkey, Opt_sec_lkeyi, Opt_sec_lkeyp,
- Opt_sec_spkm, Opt_sec_spkmi, Opt_sec_spkmp,
-
- Opt_sec_err
-};
-
-static const match_table_t nfs_secflavor_tokens = {
- { Opt_sec_none, "none" },
- { Opt_sec_none, "null" },
- { Opt_sec_sys, "sys" },
-
- { Opt_sec_krb5, "krb5" },
- { Opt_sec_krb5i, "krb5i" },
- { Opt_sec_krb5p, "krb5p" },
-
- { Opt_sec_lkey, "lkey" },
- { Opt_sec_lkeyi, "lkeyi" },
- { Opt_sec_lkeyp, "lkeyp" },
-
- { Opt_sec_spkm, "spkm3" },
- { Opt_sec_spkmi, "spkm3i" },
- { Opt_sec_spkmp, "spkm3p" },
-
- { Opt_sec_err, NULL }
-};
-
-enum {
- Opt_lookupcache_all, Opt_lookupcache_positive,
- Opt_lookupcache_none,
-
- Opt_lookupcache_err
-};
-
-static match_table_t nfs_lookupcache_tokens = {
- { Opt_lookupcache_all, "all" },
- { Opt_lookupcache_positive, "pos" },
- { Opt_lookupcache_positive, "positive" },
- { Opt_lookupcache_none, "none" },
-
- { Opt_lookupcache_err, NULL }
-};
-
-enum {
- Opt_local_lock_all, Opt_local_lock_flock, Opt_local_lock_posix,
- Opt_local_lock_none,
-
- Opt_local_lock_err
-};
-
-static match_table_t nfs_local_lock_tokens = {
- { Opt_local_lock_all, "all" },
- { Opt_local_lock_flock, "flock" },
- { Opt_local_lock_posix, "posix" },
- { Opt_local_lock_none, "none" },
-
- { Opt_local_lock_err, NULL }
-};
-
-enum {
- Opt_vers_2, Opt_vers_3, Opt_vers_4, Opt_vers_4_0,
- Opt_vers_4_1, Opt_vers_4_2,
-
- Opt_vers_err
-};
-
-static match_table_t nfs_vers_tokens = {
- { Opt_vers_2, "2" },
- { Opt_vers_3, "3" },
- { Opt_vers_4, "4" },
- { Opt_vers_4_0, "4.0" },
- { Opt_vers_4_1, "4.1" },
- { Opt_vers_4_2, "4.2" },
-
- { Opt_vers_err, NULL }
-};
static struct dentry *nfs_prepared_mount(struct file_system_type *fs_type,
int flags, const char *dev_name, void *raw_data);
@@ -332,10 +109,6 @@ const struct super_operations nfs_sops = {
EXPORT_SYMBOL_GPL(nfs_sops);
#if IS_ENABLED(CONFIG_NFS_V4)
-static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data *);
-static int nfs4_validate_mount_data(void *options,
- struct nfs_parsed_mount_data *args, const char *dev_name);
-
struct file_system_type nfs4_fs_type = {
.owner = THIS_MODULE,
.name = "nfs4",
@@ -932,141 +705,6 @@ void nfs_umount_begin(struct super_block *sb)
}
EXPORT_SYMBOL_GPL(nfs_umount_begin);
-static struct nfs_parsed_mount_data *nfs_alloc_parsed_mount_data(void)
-{
- struct nfs_parsed_mount_data *data;
-
- data = kzalloc(sizeof(*data), GFP_KERNEL);
- if (data) {
- data->timeo = NFS_UNSPEC_TIMEO;
- data->retrans = NFS_UNSPEC_RETRANS;
- data->acregmin = NFS_DEF_ACREGMIN;
- data->acregmax = NFS_DEF_ACREGMAX;
- data->acdirmin = NFS_DEF_ACDIRMIN;
- data->acdirmax = NFS_DEF_ACDIRMAX;
- data->mount_server.port = NFS_UNSPEC_PORT;
- data->nfs_server.port = NFS_UNSPEC_PORT;
- data->nfs_server.protocol = XPRT_TRANSPORT_TCP;
- data->selected_flavor = RPC_AUTH_MAXFLAVOR;
- data->minorversion = 0;
- data->need_mount = true;
- data->net = current->nsproxy->net_ns;
- data->lsm_opts = NULL;
- }
- return data;
-}
-
-static void nfs_free_parsed_mount_data(struct nfs_parsed_mount_data *data)
-{
- if (data) {
- kfree(data->client_address);
- kfree(data->mount_server.hostname);
- kfree(data->nfs_server.export_path);
- kfree(data->nfs_server.hostname);
- kfree(data->fscache_uniq);
- security_free_mnt_opts(&data->lsm_opts);
- kfree(data);
- }
-}
-
-/*
- * Sanity-check a server address provided by the mount command.
- *
- * Address family must be initialized, and address must not be
- * the ANY address for that family.
- */
-static int nfs_verify_server_address(struct sockaddr *addr)
-{
- switch (addr->sa_family) {
- case AF_INET: {
- struct sockaddr_in *sa = (struct sockaddr_in *)addr;
- return sa->sin_addr.s_addr != htonl(INADDR_ANY);
- }
- case AF_INET6: {
- struct in6_addr *sa = &((struct sockaddr_in6 *)addr)->sin6_addr;
- return !ipv6_addr_any(sa);
- }
- }
-
- dfprintk(MOUNT, "NFS: Invalid IP address specified\n");
- return 0;
-}
-
-/*
- * Select between a default port value and a user-specified port value.
- * If a zero value is set, then autobind will be used.
- */
-static void nfs_set_port(struct sockaddr *sap, int *port,
- const unsigned short default_port)
-{
- if (*port == NFS_UNSPEC_PORT)
- *port = default_port;
-
- rpc_set_port(sap, *port);
-}
-
-/*
- * Sanity check the NFS transport protocol.
- *
- */
-static void nfs_validate_transport_protocol(struct nfs_parsed_mount_data *mnt)
-{
- switch (mnt->nfs_server.protocol) {
- case XPRT_TRANSPORT_UDP:
- case XPRT_TRANSPORT_TCP:
- case XPRT_TRANSPORT_RDMA:
- break;
- default:
- mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
- }
-}
-
-/*
- * For text based NFSv2/v3 mounts, the mount protocol transport default
- * settings should depend upon the specified NFS transport.
- */
-static void nfs_set_mount_transport_protocol(struct nfs_parsed_mount_data *mnt)
-{
- nfs_validate_transport_protocol(mnt);
-
- if (mnt->mount_server.protocol == XPRT_TRANSPORT_UDP ||
- mnt->mount_server.protocol == XPRT_TRANSPORT_TCP)
- return;
- switch (mnt->nfs_server.protocol) {
- case XPRT_TRANSPORT_UDP:
- mnt->mount_server.protocol = XPRT_TRANSPORT_UDP;
- break;
- case XPRT_TRANSPORT_TCP:
- case XPRT_TRANSPORT_RDMA:
- mnt->mount_server.protocol = XPRT_TRANSPORT_TCP;
- }
-}
-
-/*
- * Add 'flavor' to 'auth_info' if not already present.
- * Returns true if 'flavor' ends up in the list, false otherwise
- */
-static bool nfs_auth_info_add(struct nfs_auth_info *auth_info,
- rpc_authflavor_t flavor)
-{
- unsigned int i;
- unsigned int max_flavor_len = ARRAY_SIZE(auth_info->flavors);
-
- /* make sure this flavor isn't already in the list */
- for (i = 0; i < auth_info->flavor_len; i++) {
- if (flavor == auth_info->flavors[i])
- return true;
- }
-
- if (auth_info->flavor_len + 1 >= max_flavor_len) {
- dfprintk(MOUNT, "NFS: too many sec= flavors\n");
- return false;
- }
-
- auth_info->flavors[auth_info->flavor_len++] = flavor;
- return true;
-}
-
/*
* Return true if 'match' is in auth_info or auth_info is empty.
* Return false otherwise.
@@ -1087,627 +725,6 @@ bool nfs_auth_info_match(const struct nfs_auth_info *auth_info,
}
EXPORT_SYMBOL_GPL(nfs_auth_info_match);
-/*
- * Parse the value of the 'sec=' option.
- */
-static int nfs_parse_security_flavors(char *value,
- struct nfs_parsed_mount_data *mnt)
-{
- substring_t args[MAX_OPT_ARGS];
- rpc_authflavor_t pseudoflavor;
- char *p;
-
- dfprintk(MOUNT, "NFS: parsing sec=%s option\n", value);
-
- while ((p = strsep(&value, ":")) != NULL) {
- switch (match_token(p, nfs_secflavor_tokens, args)) {
- case Opt_sec_none:
- pseudoflavor = RPC_AUTH_NULL;
- break;
- case Opt_sec_sys:
- pseudoflavor = RPC_AUTH_UNIX;
- break;
- case Opt_sec_krb5:
- pseudoflavor = RPC_AUTH_GSS_KRB5;
- break;
- case Opt_sec_krb5i:
- pseudoflavor = RPC_AUTH_GSS_KRB5I;
- break;
- case Opt_sec_krb5p:
- pseudoflavor = RPC_AUTH_GSS_KRB5P;
- break;
- case Opt_sec_lkey:
- pseudoflavor = RPC_AUTH_GSS_LKEY;
- break;
- case Opt_sec_lkeyi:
- pseudoflavor = RPC_AUTH_GSS_LKEYI;
- break;
- case Opt_sec_lkeyp:
- pseudoflavor = RPC_AUTH_GSS_LKEYP;
- break;
- case Opt_sec_spkm:
- pseudoflavor = RPC_AUTH_GSS_SPKM;
- break;
- case Opt_sec_spkmi:
- pseudoflavor = RPC_AUTH_GSS_SPKMI;
- break;
- case Opt_sec_spkmp:
- pseudoflavor = RPC_AUTH_GSS_SPKMP;
- break;
- default:
- dfprintk(MOUNT,
- "NFS: sec= option '%s' not recognized\n", p);
- return 0;
- }
-
- if (!nfs_auth_info_add(&mnt->auth_info, pseudoflavor))
- return 0;
- }
-
- return 1;
-}
-
-static int nfs_parse_version_string(char *string,
- struct nfs_parsed_mount_data *mnt,
- substring_t *args)
-{
- mnt->flags &= ~NFS_MOUNT_VER3;
- switch (match_token(string, nfs_vers_tokens, args)) {
- case Opt_vers_2:
- mnt->version = 2;
- break;
- case Opt_vers_3:
- mnt->flags |= NFS_MOUNT_VER3;
- mnt->version = 3;
- break;
- case Opt_vers_4:
- /* Backward compatibility option. In future,
- * the mount program should always supply
- * a NFSv4 minor version number.
- */
- mnt->version = 4;
- break;
- case Opt_vers_4_0:
- mnt->version = 4;
- mnt->minorversion = 0;
- break;
- case Opt_vers_4_1:
- mnt->version = 4;
- mnt->minorversion = 1;
- break;
- case Opt_vers_4_2:
- mnt->version = 4;
- mnt->minorversion = 2;
- break;
- default:
- return 0;
- }
- return 1;
-}
-
-static int nfs_get_option_str(substring_t args[], char **option)
-{
- kfree(*option);
- *option = match_strdup(args);
- return !*option;
-}
-
-static int nfs_get_option_ul(substring_t args[], unsigned long *option)
-{
- int rc;
- char *string;
-
- string = match_strdup(args);
- if (string == NULL)
- return -ENOMEM;
- rc = kstrtoul(string, 10, option);
- kfree(string);
-
- return rc;
-}
-
-static int nfs_get_option_ul_bound(substring_t args[], unsigned long *option,
- unsigned long l_bound, unsigned long u_bound)
-{
- int ret;
-
- ret = nfs_get_option_ul(args, option);
- if (ret != 0)
- return ret;
- if (*option < l_bound || *option > u_bound)
- return -ERANGE;
- return 0;
-}
-
-/*
- * Error-check and convert a string of mount options from user space into
- * a data structure. The whole mount string is processed; bad options are
- * skipped as they are encountered. If there were no errors, return 1;
- * otherwise return 0 (zero).
- */
-static int nfs_parse_mount_options(char *raw,
- struct nfs_parsed_mount_data *mnt)
-{
- char *p, *string;
- int rc, sloppy = 0, invalid_option = 0;
- unsigned short protofamily = AF_UNSPEC;
- unsigned short mountfamily = AF_UNSPEC;
-
- if (!raw) {
- dfprintk(MOUNT, "NFS: mount options string was NULL.\n");
- return 1;
- }
- dfprintk(MOUNT, "NFS: nfs mount opts='%s'\n", raw);
-
- rc = security_sb_eat_lsm_opts(raw, &mnt->lsm_opts);
- if (rc)
- goto out_security_failure;
-
- while ((p = strsep(&raw, ",")) != NULL) {
- substring_t args[MAX_OPT_ARGS];
- unsigned long option;
- int token;
-
- if (!*p)
- continue;
-
- dfprintk(MOUNT, "NFS: parsing nfs mount option '%s'\n", p);
-
- token = match_token(p, nfs_mount_option_tokens, args);
- switch (token) {
-
- /*
- * boolean options: foo/nofoo
- */
- case Opt_soft:
- mnt->flags |= NFS_MOUNT_SOFT;
- mnt->flags &= ~NFS_MOUNT_SOFTERR;
- break;
- case Opt_softerr:
- mnt->flags |= NFS_MOUNT_SOFTERR;
- mnt->flags &= ~NFS_MOUNT_SOFT;
- break;
- case Opt_hard:
- mnt->flags &= ~(NFS_MOUNT_SOFT|NFS_MOUNT_SOFTERR);
- break;
- case Opt_posix:
- mnt->flags |= NFS_MOUNT_POSIX;
- break;
- case Opt_noposix:
- mnt->flags &= ~NFS_MOUNT_POSIX;
- break;
- case Opt_cto:
- mnt->flags &= ~NFS_MOUNT_NOCTO;
- break;
- case Opt_nocto:
- mnt->flags |= NFS_MOUNT_NOCTO;
- break;
- case Opt_ac:
- mnt->flags &= ~NFS_MOUNT_NOAC;
- break;
- case Opt_noac:
- mnt->flags |= NFS_MOUNT_NOAC;
- break;
- case Opt_lock:
- mnt->flags &= ~NFS_MOUNT_NONLM;
- mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
- NFS_MOUNT_LOCAL_FCNTL);
- break;
- case Opt_nolock:
- mnt->flags |= NFS_MOUNT_NONLM;
- mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
- NFS_MOUNT_LOCAL_FCNTL);
- break;
- case Opt_udp:
- mnt->flags &= ~NFS_MOUNT_TCP;
- mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP;
- break;
- case Opt_tcp:
- mnt->flags |= NFS_MOUNT_TCP;
- mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
- break;
- case Opt_rdma:
- mnt->flags |= NFS_MOUNT_TCP; /* for side protocols */
- mnt->nfs_server.protocol = XPRT_TRANSPORT_RDMA;
- xprt_load_transport(p);
- break;
- case Opt_acl:
- mnt->flags &= ~NFS_MOUNT_NOACL;
- break;
- case Opt_noacl:
- mnt->flags |= NFS_MOUNT_NOACL;
- break;
- case Opt_rdirplus:
- mnt->flags &= ~NFS_MOUNT_NORDIRPLUS;
- break;
- case Opt_nordirplus:
- mnt->flags |= NFS_MOUNT_NORDIRPLUS;
- break;
- case Opt_sharecache:
- mnt->flags &= ~NFS_MOUNT_UNSHARED;
- break;
- case Opt_nosharecache:
- mnt->flags |= NFS_MOUNT_UNSHARED;
- break;
- case Opt_resvport:
- mnt->flags &= ~NFS_MOUNT_NORESVPORT;
- break;
- case Opt_noresvport:
- mnt->flags |= NFS_MOUNT_NORESVPORT;
- break;
- case Opt_fscache:
- mnt->options |= NFS_OPTION_FSCACHE;
- kfree(mnt->fscache_uniq);
- mnt->fscache_uniq = NULL;
- break;
- case Opt_nofscache:
- mnt->options &= ~NFS_OPTION_FSCACHE;
- kfree(mnt->fscache_uniq);
- mnt->fscache_uniq = NULL;
- break;
- case Opt_migration:
- mnt->options |= NFS_OPTION_MIGRATION;
- break;
- case Opt_nomigration:
- mnt->options &= ~NFS_OPTION_MIGRATION;
- break;
-
- /*
- * options that take numeric values
- */
- case Opt_port:
- if (nfs_get_option_ul(args, &option) ||
- option > USHRT_MAX)
- goto out_invalid_value;
- mnt->nfs_server.port = option;
- break;
- case Opt_rsize:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->rsize = option;
- break;
- case Opt_wsize:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->wsize = option;
- break;
- case Opt_bsize:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->bsize = option;
- break;
- case Opt_timeo:
- if (nfs_get_option_ul_bound(args, &option, 1, INT_MAX))
- goto out_invalid_value;
- mnt->timeo = option;
- break;
- case Opt_retrans:
- if (nfs_get_option_ul_bound(args, &option, 0, INT_MAX))
- goto out_invalid_value;
- mnt->retrans = option;
- break;
- case Opt_acregmin:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->acregmin = option;
- break;
- case Opt_acregmax:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->acregmax = option;
- break;
- case Opt_acdirmin:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->acdirmin = option;
- break;
- case Opt_acdirmax:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->acdirmax = option;
- break;
- case Opt_actimeo:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->acregmin = mnt->acregmax =
- mnt->acdirmin = mnt->acdirmax = option;
- break;
- case Opt_namelen:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- mnt->namlen = option;
- break;
- case Opt_mountport:
- if (nfs_get_option_ul(args, &option) ||
- option > USHRT_MAX)
- goto out_invalid_value;
- mnt->mount_server.port = option;
- break;
- case Opt_mountvers:
- if (nfs_get_option_ul(args, &option) ||
- option < NFS_MNT_VERSION ||
- option > NFS_MNT3_VERSION)
- goto out_invalid_value;
- mnt->mount_server.version = option;
- break;
- case Opt_minorversion:
- if (nfs_get_option_ul(args, &option))
- goto out_invalid_value;
- if (option > NFS4_MAX_MINOR_VERSION)
- goto out_invalid_value;
- mnt->minorversion = option;
- break;
-
- /*
- * options that take text values
- */
- case Opt_nfsvers:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- rc = nfs_parse_version_string(string, mnt, args);
- kfree(string);
- if (!rc)
- goto out_invalid_value;
- break;
- case Opt_sec:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- rc = nfs_parse_security_flavors(string, mnt);
- kfree(string);
- if (!rc) {
- dfprintk(MOUNT, "NFS: unrecognized "
- "security flavor\n");
- return 0;
- }
- break;
- case Opt_proto:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- token = match_token(string,
- nfs_xprt_protocol_tokens, args);
-
- protofamily = AF_INET;
- switch (token) {
- case Opt_xprt_udp6:
- protofamily = AF_INET6;
- /* fall through */
- case Opt_xprt_udp:
- mnt->flags &= ~NFS_MOUNT_TCP;
- mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP;
- break;
- case Opt_xprt_tcp6:
- protofamily = AF_INET6;
- /* fall through */
- case Opt_xprt_tcp:
- mnt->flags |= NFS_MOUNT_TCP;
- mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
- break;
- case Opt_xprt_rdma6:
- protofamily = AF_INET6;
- /* fall through */
- case Opt_xprt_rdma:
- /* vector side protocols to TCP */
- mnt->flags |= NFS_MOUNT_TCP;
- mnt->nfs_server.protocol = XPRT_TRANSPORT_RDMA;
- xprt_load_transport(string);
- break;
- default:
- dfprintk(MOUNT, "NFS: unrecognized "
- "transport protocol\n");
- kfree(string);
- return 0;
- }
- kfree(string);
- break;
- case Opt_mountproto:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- token = match_token(string,
- nfs_xprt_protocol_tokens, args);
- kfree(string);
-
- mountfamily = AF_INET;
- switch (token) {
- case Opt_xprt_udp6:
- mountfamily = AF_INET6;
- /* fall through */
- case Opt_xprt_udp:
- mnt->mount_server.protocol = XPRT_TRANSPORT_UDP;
- break;
- case Opt_xprt_tcp6:
- mountfamily = AF_INET6;
- /* fall through */
- case Opt_xprt_tcp:
- mnt->mount_server.protocol = XPRT_TRANSPORT_TCP;
- break;
- case Opt_xprt_rdma: /* not used for side protocols */
- default:
- dfprintk(MOUNT, "NFS: unrecognized "
- "transport protocol\n");
- return 0;
- }
- break;
- case Opt_addr:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- mnt->nfs_server.addrlen =
- rpc_pton(mnt->net, string, strlen(string),
- (struct sockaddr *)
- &mnt->nfs_server.address,
- sizeof(mnt->nfs_server.address));
- kfree(string);
- if (mnt->nfs_server.addrlen == 0)
- goto out_invalid_address;
- break;
- case Opt_clientaddr:
- if (nfs_get_option_str(args, &mnt->client_address))
- goto out_nomem;
- break;
- case Opt_mounthost:
- if (nfs_get_option_str(args,
- &mnt->mount_server.hostname))
- goto out_nomem;
- break;
- case Opt_mountaddr:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- mnt->mount_server.addrlen =
- rpc_pton(mnt->net, string, strlen(string),
- (struct sockaddr *)
- &mnt->mount_server.address,
- sizeof(mnt->mount_server.address));
- kfree(string);
- if (mnt->mount_server.addrlen == 0)
- goto out_invalid_address;
- break;
- case Opt_nconnect:
- if (nfs_get_option_ul_bound(args, &option, 1, NFS_MAX_CONNECTIONS))
- goto out_invalid_value;
- mnt->nfs_server.nconnect = option;
- break;
- case Opt_lookupcache:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- token = match_token(string,
- nfs_lookupcache_tokens, args);
- kfree(string);
- switch (token) {
- case Opt_lookupcache_all:
- mnt->flags &= ~(NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE);
- break;
- case Opt_lookupcache_positive:
- mnt->flags &= ~NFS_MOUNT_LOOKUP_CACHE_NONE;
- mnt->flags |= NFS_MOUNT_LOOKUP_CACHE_NONEG;
- break;
- case Opt_lookupcache_none:
- mnt->flags |= NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE;
- break;
- default:
- dfprintk(MOUNT, "NFS: invalid "
- "lookupcache argument\n");
- return 0;
- };
- break;
- case Opt_fscache_uniq:
- if (nfs_get_option_str(args, &mnt->fscache_uniq))
- goto out_nomem;
- mnt->options |= NFS_OPTION_FSCACHE;
- break;
- case Opt_local_lock:
- string = match_strdup(args);
- if (string == NULL)
- goto out_nomem;
- token = match_token(string, nfs_local_lock_tokens,
- args);
- kfree(string);
- switch (token) {
- case Opt_local_lock_all:
- mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
- NFS_MOUNT_LOCAL_FCNTL);
- break;
- case Opt_local_lock_flock:
- mnt->flags |= NFS_MOUNT_LOCAL_FLOCK;
- break;
- case Opt_local_lock_posix:
- mnt->flags |= NFS_MOUNT_LOCAL_FCNTL;
- break;
- case Opt_local_lock_none:
- mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
- NFS_MOUNT_LOCAL_FCNTL);
- break;
- default:
- dfprintk(MOUNT, "NFS: invalid "
- "local_lock argument\n");
- return 0;
- };
- break;
-
- /*
- * Special options
- */
- case Opt_sloppy:
- sloppy = 1;
- dfprintk(MOUNT, "NFS: relaxing parsing rules\n");
- break;
- case Opt_userspace:
- case Opt_deprecated:
- dfprintk(MOUNT, "NFS: ignoring mount option "
- "'%s'\n", p);
- break;
-
- default:
- invalid_option = 1;
- dfprintk(MOUNT, "NFS: unrecognized mount option "
- "'%s'\n", p);
- }
- }
-
- if (!sloppy && invalid_option)
- return 0;
-
- if (mnt->minorversion && mnt->version != 4)
- goto out_minorversion_mismatch;
-
- if (mnt->options & NFS_OPTION_MIGRATION &&
- (mnt->version != 4 || mnt->minorversion != 0))
- goto out_migration_misuse;
-
- /*
- * verify that any proto=/mountproto= options match the address
- * families in the addr=/mountaddr= options.
- */
- if (protofamily != AF_UNSPEC &&
- protofamily != mnt->nfs_server.address.ss_family)
- goto out_proto_mismatch;
-
- if (mountfamily != AF_UNSPEC) {
- if (mnt->mount_server.addrlen) {
- if (mountfamily != mnt->mount_server.address.ss_family)
- goto out_mountproto_mismatch;
- } else {
- if (mountfamily != mnt->nfs_server.address.ss_family)
- goto out_mountproto_mismatch;
- }
- }
-
- return 1;
-
-out_mountproto_mismatch:
- printk(KERN_INFO "NFS: mount server address does not match mountproto= "
- "option\n");
- return 0;
-out_proto_mismatch:
- printk(KERN_INFO "NFS: server address does not match proto= option\n");
- return 0;
-out_invalid_address:
- printk(KERN_INFO "NFS: bad IP address specified: %s\n", p);
- return 0;
-out_invalid_value:
- printk(KERN_INFO "NFS: bad mount option value specified: %s\n", p);
- return 0;
-out_minorversion_mismatch:
- printk(KERN_INFO "NFS: mount option vers=%u does not support "
- "minorversion=%u\n", mnt->version, mnt->minorversion);
- return 0;
-out_migration_misuse:
- printk(KERN_INFO
- "NFS: 'migration' not supported for this NFS version\n");
- return 0;
-out_nomem:
- printk(KERN_INFO "NFS: not enough memory to parse option\n");
- return 0;
-out_security_failure:
- printk(KERN_INFO "NFS: security options invalid: %d\n", rc);
- return 0;
-}
-
/*
* Ensure that a specified authtype in args->auth_info is supported by
* the server. Returns 0 and sets args->selected_flavor if it's ok, and
@@ -1908,327 +925,6 @@ struct dentry *nfs_try_mount(int flags, const char *dev_name,
}
EXPORT_SYMBOL_GPL(nfs_try_mount);
-/*
- * Split "dev_name" into "hostname:export_path".
- *
- * The leftmost colon demarks the split between the server's hostname
- * and the export path. If the hostname starts with a left square
- * bracket, then it may contain colons.
- *
- * Note: caller frees hostname and export path, even on error.
- */
-static int nfs_parse_devname(const char *dev_name,
- char **hostname, size_t maxnamlen,
- char **export_path, size_t maxpathlen)
-{
- size_t len;
- char *end;
-
- if (unlikely(!dev_name || !*dev_name)) {
- dfprintk(MOUNT, "NFS: device name not specified\n");
- return -EINVAL;
- }
-
- /* Is the host name protected with square brakcets? */
- if (*dev_name == '[') {
- end = strchr(++dev_name, ']');
- if (end == NULL || end[1] != ':')
- goto out_bad_devname;
-
- len = end - dev_name;
- end++;
- } else {
- char *comma;
-
- end = strchr(dev_name, ':');
- if (end == NULL)
- goto out_bad_devname;
- len = end - dev_name;
-
- /* kill possible hostname list: not supported */
- comma = strchr(dev_name, ',');
- if (comma != NULL && comma < end)
- len = comma - dev_name;
- }
-
- if (len > maxnamlen)
- goto out_hostname;
-
- /* N.B. caller will free nfs_server.hostname in all cases */
- *hostname = kstrndup(dev_name, len, GFP_KERNEL);
- if (*hostname == NULL)
- goto out_nomem;
- len = strlen(++end);
- if (len > maxpathlen)
- goto out_path;
- *export_path = kstrndup(end, len, GFP_KERNEL);
- if (!*export_path)
- goto out_nomem;
-
- dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", *export_path);
- return 0;
-
-out_bad_devname:
- dfprintk(MOUNT, "NFS: device name not in host:path format\n");
- return -EINVAL;
-
-out_nomem:
- dfprintk(MOUNT, "NFS: not enough memory to parse device name\n");
- return -ENOMEM;
-
-out_hostname:
- dfprintk(MOUNT, "NFS: server hostname too long\n");
- return -ENAMETOOLONG;
-
-out_path:
- dfprintk(MOUNT, "NFS: export pathname too long\n");
- return -ENAMETOOLONG;
-}
-
-/*
- * Validate the NFS2/NFS3 mount data
- * - fills in the mount root filehandle
- *
- * For option strings, user space handles the following behaviors:
- *
- * + DNS: mapping server host name to IP address ("addr=" option)
- *
- * + failure mode: how to behave if a mount request can't be handled
- * immediately ("fg/bg" option)
- *
- * + retry: how often to retry a mount request ("retry=" option)
- *
- * + breaking back: trying proto=udp after proto=tcp, v2 after v3,
- * mountproto=tcp after mountproto=udp, and so on
- */
-static int nfs23_validate_mount_data(void *options,
- struct nfs_parsed_mount_data *args,
- struct nfs_fh *mntfh,
- const char *dev_name)
-{
- struct nfs_mount_data *data = (struct nfs_mount_data *)options;
- struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address;
- int extra_flags = NFS_MOUNT_LEGACY_INTERFACE;
-
- if (data == NULL)
- goto out_no_data;
-
- args->version = NFS_DEFAULT_VERSION;
- switch (data->version) {
- case 1:
- data->namlen = 0; /* fall through */
- case 2:
- data->bsize = 0; /* fall through */
- case 3:
- if (data->flags & NFS_MOUNT_VER3)
- goto out_no_v3;
- data->root.size = NFS2_FHSIZE;
- memcpy(data->root.data, data->old_root.data, NFS2_FHSIZE);
- /* Turn off security negotiation */
- extra_flags |= NFS_MOUNT_SECFLAVOUR;
- /* fall through */
- case 4:
- if (data->flags & NFS_MOUNT_SECFLAVOUR)
- goto out_no_sec;
- /* fall through */
- case 5:
- memset(data->context, 0, sizeof(data->context));
- /* fall through */
- case 6:
- if (data->flags & NFS_MOUNT_VER3) {
- if (data->root.size > NFS3_FHSIZE || data->root.size == 0)
- goto out_invalid_fh;
- mntfh->size = data->root.size;
- args->version = 3;
- } else {
- mntfh->size = NFS2_FHSIZE;
- args->version = 2;
- }
-
-
- memcpy(mntfh->data, data->root.data, mntfh->size);
- if (mntfh->size < sizeof(mntfh->data))
- memset(mntfh->data + mntfh->size, 0,
- sizeof(mntfh->data) - mntfh->size);
-
- /*
- * Translate to nfs_parsed_mount_data, which nfs_fill_super
- * can deal with.
- */
- args->flags = data->flags & NFS_MOUNT_FLAGMASK;
- args->flags |= extra_flags;
- args->rsize = data->rsize;
- args->wsize = data->wsize;
- args->timeo = data->timeo;
- args->retrans = data->retrans;
- args->acregmin = data->acregmin;
- args->acregmax = data->acregmax;
- args->acdirmin = data->acdirmin;
- args->acdirmax = data->acdirmax;
- args->need_mount = false;
-
- memcpy(sap, &data->addr, sizeof(data->addr));
- args->nfs_server.addrlen = sizeof(data->addr);
- args->nfs_server.port = ntohs(data->addr.sin_port);
- if (sap->sa_family != AF_INET ||
- !nfs_verify_server_address(sap))
- goto out_no_address;
-
- if (!(data->flags & NFS_MOUNT_TCP))
- args->nfs_server.protocol = XPRT_TRANSPORT_UDP;
- /* N.B. caller will free nfs_server.hostname in all cases */
- args->nfs_server.hostname = kstrdup(data->hostname, GFP_KERNEL);
- args->namlen = data->namlen;
- args->bsize = data->bsize;
-
- if (data->flags & NFS_MOUNT_SECFLAVOUR)
- args->selected_flavor = data->pseudoflavor;
- else
- args->selected_flavor = RPC_AUTH_UNIX;
- if (!args->nfs_server.hostname)
- goto out_nomem;
-
- if (!(data->flags & NFS_MOUNT_NONLM))
- args->flags &= ~(NFS_MOUNT_LOCAL_FLOCK|
- NFS_MOUNT_LOCAL_FCNTL);
- else
- args->flags |= (NFS_MOUNT_LOCAL_FLOCK|
- NFS_MOUNT_LOCAL_FCNTL);
- /*
- * The legacy version 6 binary mount data from userspace has a
- * field used only to transport selinux information into the
- * the kernel. To continue to support that functionality we
- * have a touch of selinux knowledge here in the NFS code. The
- * userspace code converted context=blah to just blah so we are
- * converting back to the full string selinux understands.
- */
- if (data->context[0]){
-#ifdef CONFIG_SECURITY_SELINUX
- int rc;
- data->context[NFS_MAX_CONTEXT_LEN] = '\0';
- rc = security_add_mnt_opt("context", data->context,
- strlen(data->context), &args->lsm_opts);
- if (rc)
- return rc;
-#else
- return -EINVAL;
-#endif
- }
-
- break;
- default:
- return NFS_TEXT_DATA;
- }
-
- return 0;
-
-out_no_data:
- dfprintk(MOUNT, "NFS: mount program didn't pass any mount data\n");
- return -EINVAL;
-
-out_no_v3:
- dfprintk(MOUNT, "NFS: nfs_mount_data version %d does not support v3\n",
- data->version);
- return -EINVAL;
-
-out_no_sec:
- dfprintk(MOUNT, "NFS: nfs_mount_data version supports only AUTH_SYS\n");
- return -EINVAL;
-
-out_nomem:
- dfprintk(MOUNT, "NFS: not enough memory to handle mount options\n");
- return -ENOMEM;
-
-out_no_address:
- dfprintk(MOUNT, "NFS: mount program didn't pass remote address\n");
- return -EINVAL;
-
-out_invalid_fh:
- dfprintk(MOUNT, "NFS: invalid root filehandle\n");
- return -EINVAL;
-}
-
-#if IS_ENABLED(CONFIG_NFS_V4)
-static int nfs_validate_mount_data(struct file_system_type *fs_type,
- void *options,
- struct nfs_parsed_mount_data *args,
- struct nfs_fh *mntfh,
- const char *dev_name)
-{
- if (fs_type == &nfs_fs_type)
- return nfs23_validate_mount_data(options, args, mntfh, dev_name);
- return nfs4_validate_mount_data(options, args, dev_name);
-}
-#else
-static int nfs_validate_mount_data(struct file_system_type *fs_type,
- void *options,
- struct nfs_parsed_mount_data *args,
- struct nfs_fh *mntfh,
- const char *dev_name)
-{
- return nfs23_validate_mount_data(options, args, mntfh, dev_name);
-}
-#endif
-
-static int nfs_validate_text_mount_data(void *options,
- struct nfs_parsed_mount_data *args,
- const char *dev_name)
-{
- int port = 0;
- int max_namelen = PAGE_SIZE;
- int max_pathlen = NFS_MAXPATHLEN;
- struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address;
-
- if (nfs_parse_mount_options((char *)options, args) == 0)
- return -EINVAL;
-
- if (!nfs_verify_server_address(sap))
- goto out_no_address;
-
- if (args->version == 4) {
-#if IS_ENABLED(CONFIG_NFS_V4)
- if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
- port = NFS_RDMA_PORT;
- else
- port = NFS_PORT;
- max_namelen = NFS4_MAXNAMLEN;
- max_pathlen = NFS4_MAXPATHLEN;
- nfs_validate_transport_protocol(args);
- if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
- goto out_invalid_transport_udp;
- nfs4_validate_mount_flags(args);
-#else
- goto out_v4_not_compiled;
-#endif /* CONFIG_NFS_V4 */
- } else {
- nfs_set_mount_transport_protocol(args);
- if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
- port = NFS_RDMA_PORT;
- }
-
- nfs_set_port(sap, &args->nfs_server.port, port);
-
- return nfs_parse_devname(dev_name,
- &args->nfs_server.hostname,
- max_namelen,
- &args->nfs_server.export_path,
- max_pathlen);
-
-#if !IS_ENABLED(CONFIG_NFS_V4)
-out_v4_not_compiled:
- dfprintk(MOUNT, "NFS: NFSv4 is not compiled into kernel\n");
- return -EPROTONOSUPPORT;
-#else
-out_invalid_transport_udp:
- dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
- return -EINVAL;
-#endif /* !CONFIG_NFS_V4 */
-
-out_no_address:
- dfprintk(MOUNT, "NFS: mount program didn't pass remote address\n");
- return -EINVAL;
-}
-
#define NFS_REMOUNT_CMP_FLAGMASK ~(NFS_MOUNT_INTR \
| NFS_MOUNT_SECURE \
| NFS_MOUNT_TCP \
@@ -2719,113 +1415,6 @@ nfs_prepared_mount(struct file_system_type *fs_type, int flags,
#if IS_ENABLED(CONFIG_NFS_V4)
-static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data *args)
-{
- args->flags &= ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3|
- NFS_MOUNT_LOCAL_FLOCK|NFS_MOUNT_LOCAL_FCNTL);
-}
-
-/*
- * Validate NFSv4 mount options
- */
-static int nfs4_validate_mount_data(void *options,
- struct nfs_parsed_mount_data *args,
- const char *dev_name)
-{
- struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address;
- struct nfs4_mount_data *data = (struct nfs4_mount_data *)options;
- char *c;
-
- if (data == NULL)
- goto out_no_data;
-
- args->version = 4;
-
- switch (data->version) {
- case 1:
- if (data->host_addrlen > sizeof(args->nfs_server.address))
- goto out_no_address;
- if (data->host_addrlen == 0)
- goto out_no_address;
- args->nfs_server.addrlen = data->host_addrlen;
- if (copy_from_user(sap, data->host_addr, data->host_addrlen))
- return -EFAULT;
- if (!nfs_verify_server_address(sap))
- goto out_no_address;
- args->nfs_server.port = ntohs(((struct sockaddr_in *)sap)->sin_port);
-
- if (data->auth_flavourlen) {
- rpc_authflavor_t pseudoflavor;
- if (data->auth_flavourlen > 1)
- goto out_inval_auth;
- if (copy_from_user(&pseudoflavor,
- data->auth_flavours,
- sizeof(pseudoflavor)))
- return -EFAULT;
- args->selected_flavor = pseudoflavor;
- } else
- args->selected_flavor = RPC_AUTH_UNIX;
-
- c = strndup_user(data->hostname.data, NFS4_MAXNAMLEN);
- if (IS_ERR(c))
- return PTR_ERR(c);
- args->nfs_server.hostname = c;
-
- c = strndup_user(data->mnt_path.data, NFS4_MAXPATHLEN);
- if (IS_ERR(c))
- return PTR_ERR(c);
- args->nfs_server.export_path = c;
- dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", c);
-
- c = strndup_user(data->client_addr.data, 16);
- if (IS_ERR(c))
- return PTR_ERR(c);
- args->client_address = c;
-
- /*
- * Translate to nfs_parsed_mount_data, which nfs4_fill_super
- * can deal with.
- */
-
- args->flags = data->flags & NFS4_MOUNT_FLAGMASK;
- args->rsize = data->rsize;
- args->wsize = data->wsize;
- args->timeo = data->timeo;
- args->retrans = data->retrans;
- args->acregmin = data->acregmin;
- args->acregmax = data->acregmax;
- args->acdirmin = data->acdirmin;
- args->acdirmax = data->acdirmax;
- args->nfs_server.protocol = data->proto;
- nfs_validate_transport_protocol(args);
- if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
- goto out_invalid_transport_udp;
-
- break;
- default:
- return NFS_TEXT_DATA;
- }
-
- return 0;
-
-out_no_data:
- dfprintk(MOUNT, "NFS4: mount program didn't pass any mount data\n");
- return -EINVAL;
-
-out_inval_auth:
- dfprintk(MOUNT, "NFS4: Invalid number of RPC auth flavours %d\n",
- data->auth_flavourlen);
- return -EINVAL;
-
-out_no_address:
- dfprintk(MOUNT, "NFS4: mount program didn't pass remote address\n");
- return -EINVAL;
-
-out_invalid_transport_udp:
- dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
- return -EINVAL;
-}
-
/*
* NFS v4 module parameters need to stay in the
* NFS client for backwards compatibility
--
2.17.2
On Wed, 2019-09-11 at 14:24 -0400, Chuck Lever wrote:
> > On Sep 11, 2019, at 12:16 PM, Scott Mayhew <[email protected]>
> > wrote:
> >
> > From: David Howells <[email protected]>
> >
> > Split various bits relating to mount parameterisation out from
> > fs/nfs/super.c into their own file to form the basis of filesystem
> > context
> > handling for NFS.
> >
> > No other changes are made to the code beyond removing 'static'
> > qualifiers.
> >
> > Signed-off-by: David Howells <[email protected]>
> > Signed-off-by: Al Viro <[email protected]>
> > ---
> > fs/nfs/Makefile | 2 +-
> > fs/nfs/fs_context.c | 1418
> > +++++++++++++++++++++++++++++++++++++++++++
> > fs/nfs/internal.h | 29 +
> > fs/nfs/super.c | 1411 ----------------------------------------
> > --
> > 4 files changed, 1448 insertions(+), 1412 deletions(-)
> > create mode 100644 fs/nfs/fs_context.c
> >
> > diff --git a/fs/nfs/Makefile b/fs/nfs/Makefile
> > index 34cdeaecccf6..2433c3e03cfa 100644
> > --- a/fs/nfs/Makefile
> > +++ b/fs/nfs/Makefile
> > @@ -9,7 +9,7 @@ CFLAGS_nfstrace.o += -I$(src)
> > nfs-y := client.o dir.o file.o getroot.o
> > inode.o super.o \
> > io.o direct.o pagelist.o read.o symlink.o
> > unlink.o \
> > write.o namespace.o mount_clnt.o nfstrace.o
> > \
> > - export.o sysfs.o
> > + export.o sysfs.o fs_context.o
> > nfs-$(CONFIG_ROOT_NFS) += nfsroot.o
> > nfs-$(CONFIG_SYSCTL) += sysctl.o
> > nfs-$(CONFIG_NFS_FSCACHE) += fscache.o fscache-index.o
> > diff --git a/fs/nfs/fs_context.c b/fs/nfs/fs_context.c
> > new file mode 100644
> > index 000000000000..82b312a5cdde
> > --- /dev/null
> > +++ b/fs/nfs/fs_context.c
> > @@ -0,0 +1,1418 @@
> > +/* NFS mount handling.
> > + *
> > + * Copyright (C) 2018 Red Hat, Inc. All Rights Reserved.
> > + * Written by David Howells ([email protected])
> > + *
> > + * Split from fs/nfs/super.c:
> > + *
> > + * Copyright (C) 1992 Rick Sladkey
> > + *
> > + * This program is free software; you can redistribute it and/or
> > + * modify it under the terms of the GNU General Public Licence
> > + * as published by the Free Software Foundation; either version
> > + * 2 of the Licence, or (at your option) any later version.
> > + */
>
> New source files should have an SPDX tag instead of boilerplate.
> I suggest:
>
> // SPDX-License-Identifier: GPL-2.0-only
>
Agreed. It is also quite a long stretch to claim authorship of the
entire file as implied above. Given that this is mostly a copy-paste
effort, then most of the actual copyrights belong to the people who've
contributed to super.c (and to inode.c before it). David is one of
those authors, but he is one of many.
> > +#include <linux/parser.h>
> > +#include <linux/nfs_fs.h>
> > +#include <linux/nfs_mount.h>
> > +#include <linux/nfs4_mount.h>
> > +#include "nfs.h"
> > +#include "internal.h"
> > +
> > +#define NFSDBG_FACILITY NFSDBG_MOUNT
> > +
> > +#if IS_ENABLED(CONFIG_NFS_V3)
> > +#define NFS_DEFAULT_VERSION 3
> > +#else
> > +#define NFS_DEFAULT_VERSION 2
> > +#endif
> > +
> > +#define NFS_MAX_CONNECTIONS 16
> > +
> > +enum {
> > + /* Mount options that take no arguments */
> > + Opt_soft, Opt_softerr, Opt_hard,
> > + Opt_posix, Opt_noposix,
> > + Opt_cto, Opt_nocto,
> > + Opt_ac, Opt_noac,
> > + Opt_lock, Opt_nolock,
> > + Opt_udp, Opt_tcp, Opt_rdma,
> > + Opt_acl, Opt_noacl,
> > + Opt_rdirplus, Opt_nordirplus,
> > + Opt_sharecache, Opt_nosharecache,
> > + Opt_resvport, Opt_noresvport,
> > + Opt_fscache, Opt_nofscache,
> > + Opt_migration, Opt_nomigration,
> > +
> > + /* Mount options that take integer arguments */
> > + Opt_port,
> > + Opt_rsize, Opt_wsize, Opt_bsize,
> > + Opt_timeo, Opt_retrans,
> > + Opt_acregmin, Opt_acregmax,
> > + Opt_acdirmin, Opt_acdirmax,
> > + Opt_actimeo,
> > + Opt_namelen,
> > + Opt_mountport,
> > + Opt_mountvers,
> > + Opt_minorversion,
> > +
> > + /* Mount options that take string arguments */
> > + Opt_nfsvers,
> > + Opt_sec, Opt_proto, Opt_mountproto, Opt_mounthost,
> > + Opt_addr, Opt_mountaddr, Opt_clientaddr,
> > + Opt_nconnect,
> > + Opt_lookupcache,
> > + Opt_fscache_uniq,
> > + Opt_local_lock,
> > +
> > + /* Special mount options */
> > + Opt_userspace, Opt_deprecated, Opt_sloppy,
> > +
> > + Opt_err
> > +};
> > +
> > +static const match_table_t nfs_mount_option_tokens = {
> > + { Opt_userspace, "bg" },
> > + { Opt_userspace, "fg" },
> > + { Opt_userspace, "retry=%s" },
> > +
> > + { Opt_sloppy, "sloppy" },
> > +
> > + { Opt_soft, "soft" },
> > + { Opt_softerr, "softerr" },
> > + { Opt_hard, "hard" },
> > + { Opt_deprecated, "intr" },
> > + { Opt_deprecated, "nointr" },
> > + { Opt_posix, "posix" },
> > + { Opt_noposix, "noposix" },
> > + { Opt_cto, "cto" },
> > + { Opt_nocto, "nocto" },
> > + { Opt_ac, "ac" },
> > + { Opt_noac, "noac" },
> > + { Opt_lock, "lock" },
> > + { Opt_nolock, "nolock" },
> > + { Opt_udp, "udp" },
> > + { Opt_tcp, "tcp" },
> > + { Opt_rdma, "rdma" },
> > + { Opt_acl, "acl" },
> > + { Opt_noacl, "noacl" },
> > + { Opt_rdirplus, "rdirplus" },
> > + { Opt_nordirplus, "nordirplus" },
> > + { Opt_sharecache, "sharecache" },
> > + { Opt_nosharecache, "nosharecache" },
> > + { Opt_resvport, "resvport" },
> > + { Opt_noresvport, "noresvport" },
> > + { Opt_fscache, "fsc" },
> > + { Opt_nofscache, "nofsc" },
> > + { Opt_migration, "migration" },
> > + { Opt_nomigration, "nomigration" },
> > +
> > + { Opt_port, "port=%s" },
> > + { Opt_rsize, "rsize=%s" },
> > + { Opt_wsize, "wsize=%s" },
> > + { Opt_bsize, "bsize=%s" },
> > + { Opt_timeo, "timeo=%s" },
> > + { Opt_retrans, "retrans=%s" },
> > + { Opt_acregmin, "acregmin=%s" },
> > + { Opt_acregmax, "acregmax=%s" },
> > + { Opt_acdirmin, "acdirmin=%s" },
> > + { Opt_acdirmax, "acdirmax=%s" },
> > + { Opt_actimeo, "actimeo=%s" },
> > + { Opt_namelen, "namlen=%s" },
> > + { Opt_mountport, "mountport=%s" },
> > + { Opt_mountvers, "mountvers=%s" },
> > + { Opt_minorversion, "minorversion=%s" },
> > +
> > + { Opt_nfsvers, "nfsvers=%s" },
> > + { Opt_nfsvers, "vers=%s" },
> > +
> > + { Opt_sec, "sec=%s" },
> > + { Opt_proto, "proto=%s" },
> > + { Opt_mountproto, "mountproto=%s" },
> > + { Opt_addr, "addr=%s" },
> > + { Opt_clientaddr, "clientaddr=%s" },
> > + { Opt_mounthost, "mounthost=%s" },
> > + { Opt_mountaddr, "mountaddr=%s" },
> > +
> > + { Opt_nconnect, "nconnect=%s" },
> > +
> > + { Opt_lookupcache, "lookupcache=%s" },
> > + { Opt_fscache_uniq, "fsc=%s" },
> > + { Opt_local_lock, "local_lock=%s" },
> > +
> > + /* The following needs to be listed after all other options */
> > + { Opt_nfsvers, "v%s" },
> > +
> > + { Opt_err, NULL }
> > +};
> > +
> > +enum {
> > + Opt_xprt_udp, Opt_xprt_udp6, Opt_xprt_tcp, Opt_xprt_tcp6,
> > Opt_xprt_rdma,
> > + Opt_xprt_rdma6,
> > +
> > + Opt_xprt_err
> > +};
> > +
> > +static const match_table_t nfs_xprt_protocol_tokens = {
> > + { Opt_xprt_udp, "udp" },
> > + { Opt_xprt_udp6, "udp6" },
> > + { Opt_xprt_tcp, "tcp" },
> > + { Opt_xprt_tcp6, "tcp6" },
> > + { Opt_xprt_rdma, "rdma" },
> > + { Opt_xprt_rdma6, "rdma6" },
> > +
> > + { Opt_xprt_err, NULL }
> > +};
> > +
> > +enum {
> > + Opt_sec_none, Opt_sec_sys,
> > + Opt_sec_krb5, Opt_sec_krb5i, Opt_sec_krb5p,
> > + Opt_sec_lkey, Opt_sec_lkeyi, Opt_sec_lkeyp,
> > + Opt_sec_spkm, Opt_sec_spkmi, Opt_sec_spkmp,
> > +
> > + Opt_sec_err
> > +};
> > +
> > +static const match_table_t nfs_secflavor_tokens = {
> > + { Opt_sec_none, "none" },
> > + { Opt_sec_none, "null" },
> > + { Opt_sec_sys, "sys" },
> > +
> > + { Opt_sec_krb5, "krb5" },
> > + { Opt_sec_krb5i, "krb5i" },
> > + { Opt_sec_krb5p, "krb5p" },
> > +
> > + { Opt_sec_lkey, "lkey" },
> > + { Opt_sec_lkeyi, "lkeyi" },
> > + { Opt_sec_lkeyp, "lkeyp" },
> > +
> > + { Opt_sec_spkm, "spkm3" },
> > + { Opt_sec_spkmi, "spkm3i" },
> > + { Opt_sec_spkmp, "spkm3p" },
> > +
> > + { Opt_sec_err, NULL }
> > +};
> > +
> > +enum {
> > + Opt_lookupcache_all, Opt_lookupcache_positive,
> > + Opt_lookupcache_none,
> > +
> > + Opt_lookupcache_err
> > +};
> > +
> > +static match_table_t nfs_lookupcache_tokens = {
> > + { Opt_lookupcache_all, "all" },
> > + { Opt_lookupcache_positive, "pos" },
> > + { Opt_lookupcache_positive, "positive" },
> > + { Opt_lookupcache_none, "none" },
> > +
> > + { Opt_lookupcache_err, NULL }
> > +};
> > +
> > +enum {
> > + Opt_local_lock_all, Opt_local_lock_flock, Opt_local_lock_posix,
> > + Opt_local_lock_none,
> > +
> > + Opt_local_lock_err
> > +};
> > +
> > +static match_table_t nfs_local_lock_tokens = {
> > + { Opt_local_lock_all, "all" },
> > + { Opt_local_lock_flock, "flock" },
> > + { Opt_local_lock_posix, "posix" },
> > + { Opt_local_lock_none, "none" },
> > +
> > + { Opt_local_lock_err, NULL }
> > +};
> > +
> > +enum {
> > + Opt_vers_2, Opt_vers_3, Opt_vers_4, Opt_vers_4_0,
> > + Opt_vers_4_1, Opt_vers_4_2,
> > +
> > + Opt_vers_err
> > +};
> > +
> > +static match_table_t nfs_vers_tokens = {
> > + { Opt_vers_2, "2" },
> > + { Opt_vers_3, "3" },
> > + { Opt_vers_4, "4" },
> > + { Opt_vers_4_0, "4.0" },
> > + { Opt_vers_4_1, "4.1" },
> > + { Opt_vers_4_2, "4.2" },
> > +
> > + { Opt_vers_err, NULL }
> > +};
> > +
> > +struct nfs_parsed_mount_data *nfs_alloc_parsed_mount_data(void)
> > +{
> > + struct nfs_parsed_mount_data *data;
> > +
> > + data = kzalloc(sizeof(*data), GFP_KERNEL);
> > + if (data) {
> > + data->timeo = NFS_UNSPEC_TIMEO;
> > + data->retrans = NFS_UNSPEC_RETRANS;
> > + data->acregmin = NFS_DEF_ACREGMIN;
> > + data->acregmax = NFS_DEF_ACREGMAX;
> > + data->acdirmin = NFS_DEF_ACDIRMIN;
> > + data->acdirmax = NFS_DEF_ACDIRMAX;
> > + data->mount_server.port = NFS_UNSPEC_PORT;
> > + data->nfs_server.port = NFS_UNSPEC_PORT;
> > + data->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > + data->selected_flavor = RPC_AUTH_MAXFLAVOR;
> > + data->minorversion = 0;
> > + data->need_mount = true;
> > + data->net = current->nsproxy->net_ns;
> > + data->lsm_opts = NULL;
> > + }
> > + return data;
> > +}
> > +
> > +void nfs_free_parsed_mount_data(struct nfs_parsed_mount_data
> > *data)
> > +{
> > + if (data) {
> > + kfree(data->client_address);
> > + kfree(data->mount_server.hostname);
> > + kfree(data->nfs_server.export_path);
> > + kfree(data->nfs_server.hostname);
> > + kfree(data->fscache_uniq);
> > + security_free_mnt_opts(&data->lsm_opts);
> > + kfree(data);
> > + }
> > +}
> > +
> > +/*
> > + * Sanity-check a server address provided by the mount command.
> > + *
> > + * Address family must be initialized, and address must not be
> > + * the ANY address for that family.
> > + */
> > +static int nfs_verify_server_address(struct sockaddr *addr)
> > +{
> > + switch (addr->sa_family) {
> > + case AF_INET: {
> > + struct sockaddr_in *sa = (struct sockaddr_in *)addr;
> > + return sa->sin_addr.s_addr != htonl(INADDR_ANY);
> > + }
> > + case AF_INET6: {
> > + struct in6_addr *sa = &((struct sockaddr_in6 *)addr)-
> > >sin6_addr;
> > + return !ipv6_addr_any(sa);
> > + }
> > + }
> > +
> > + dfprintk(MOUNT, "NFS: Invalid IP address specified\n");
> > + return 0;
> > +}
> > +
> > +/*
> > + * Sanity check the NFS transport protocol.
> > + *
> > + */
> > +static void nfs_validate_transport_protocol(struct
> > nfs_parsed_mount_data *mnt)
> > +{
> > + switch (mnt->nfs_server.protocol) {
> > + case XPRT_TRANSPORT_UDP:
> > + case XPRT_TRANSPORT_TCP:
> > + case XPRT_TRANSPORT_RDMA:
> > + break;
> > + default:
> > + mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > + }
> > +}
> > +
> > +/*
> > + * For text based NFSv2/v3 mounts, the mount protocol transport
> > default
> > + * settings should depend upon the specified NFS transport.
> > + */
> > +static void nfs_set_mount_transport_protocol(struct
> > nfs_parsed_mount_data *mnt)
> > +{
> > + nfs_validate_transport_protocol(mnt);
> > +
> > + if (mnt->mount_server.protocol == XPRT_TRANSPORT_UDP ||
> > + mnt->mount_server.protocol == XPRT_TRANSPORT_TCP)
> > + return;
> > + switch (mnt->nfs_server.protocol) {
> > + case XPRT_TRANSPORT_UDP:
> > + mnt->mount_server.protocol = XPRT_TRANSPORT_UDP;
> > + break;
> > + case XPRT_TRANSPORT_TCP:
> > + case XPRT_TRANSPORT_RDMA:
> > + mnt->mount_server.protocol = XPRT_TRANSPORT_TCP;
> > + }
> > +}
> > +
> > +/*
> > + * Add 'flavor' to 'auth_info' if not already present.
> > + * Returns true if 'flavor' ends up in the list, false otherwise
> > + */
> > +static bool nfs_auth_info_add(struct nfs_auth_info *auth_info,
> > + rpc_authflavor_t flavor)
> > +{
> > + unsigned int i;
> > + unsigned int max_flavor_len = ARRAY_SIZE(auth_info->flavors);
> > +
> > + /* make sure this flavor isn't already in the list */
> > + for (i = 0; i < auth_info->flavor_len; i++) {
> > + if (flavor == auth_info->flavors[i])
> > + return true;
> > + }
> > +
> > + if (auth_info->flavor_len + 1 >= max_flavor_len) {
> > + dfprintk(MOUNT, "NFS: too many sec= flavors\n");
> > + return false;
> > + }
> > +
> > + auth_info->flavors[auth_info->flavor_len++] = flavor;
> > + return true;
> > +}
> > +
> > +/*
> > + * Parse the value of the 'sec=' option.
> > + */
> > +static int nfs_parse_security_flavors(char *value,
> > + struct nfs_parsed_mount_data
> > *mnt)
> > +{
> > + substring_t args[MAX_OPT_ARGS];
> > + rpc_authflavor_t pseudoflavor;
> > + char *p;
> > +
> > + dfprintk(MOUNT, "NFS: parsing sec=%s option\n", value);
> > +
> > + while ((p = strsep(&value, ":")) != NULL) {
> > + switch (match_token(p, nfs_secflavor_tokens, args)) {
> > + case Opt_sec_none:
> > + pseudoflavor = RPC_AUTH_NULL;
> > + break;
> > + case Opt_sec_sys:
> > + pseudoflavor = RPC_AUTH_UNIX;
> > + break;
> > + case Opt_sec_krb5:
> > + pseudoflavor = RPC_AUTH_GSS_KRB5;
> > + break;
> > + case Opt_sec_krb5i:
> > + pseudoflavor = RPC_AUTH_GSS_KRB5I;
> > + break;
> > + case Opt_sec_krb5p:
> > + pseudoflavor = RPC_AUTH_GSS_KRB5P;
> > + break;
> > + case Opt_sec_lkey:
> > + pseudoflavor = RPC_AUTH_GSS_LKEY;
> > + break;
> > + case Opt_sec_lkeyi:
> > + pseudoflavor = RPC_AUTH_GSS_LKEYI;
> > + break;
> > + case Opt_sec_lkeyp:
> > + pseudoflavor = RPC_AUTH_GSS_LKEYP;
> > + break;
> > + case Opt_sec_spkm:
> > + pseudoflavor = RPC_AUTH_GSS_SPKM;
> > + break;
> > + case Opt_sec_spkmi:
> > + pseudoflavor = RPC_AUTH_GSS_SPKMI;
> > + break;
> > + case Opt_sec_spkmp:
> > + pseudoflavor = RPC_AUTH_GSS_SPKMP;
> > + break;
> > + default:
> > + dfprintk(MOUNT,
> > + "NFS: sec= option '%s' not
> > recognized\n", p);
> > + return 0;
> > + }
> > +
> > + if (!nfs_auth_info_add(&mnt->auth_info, pseudoflavor))
> > + return 0;
> > + }
> > +
> > + return 1;
> > +}
> > +
> > +static int nfs_parse_version_string(char *string,
> > + struct nfs_parsed_mount_data *mnt,
> > + substring_t *args)
> > +{
> > + mnt->flags &= ~NFS_MOUNT_VER3;
> > + switch (match_token(string, nfs_vers_tokens, args)) {
> > + case Opt_vers_2:
> > + mnt->version = 2;
> > + break;
> > + case Opt_vers_3:
> > + mnt->flags |= NFS_MOUNT_VER3;
> > + mnt->version = 3;
> > + break;
> > + case Opt_vers_4:
> > + /* Backward compatibility option. In future,
> > + * the mount program should always supply
> > + * a NFSv4 minor version number.
> > + */
> > + mnt->version = 4;
> > + break;
> > + case Opt_vers_4_0:
> > + mnt->version = 4;
> > + mnt->minorversion = 0;
> > + break;
> > + case Opt_vers_4_1:
> > + mnt->version = 4;
> > + mnt->minorversion = 1;
> > + break;
> > + case Opt_vers_4_2:
> > + mnt->version = 4;
> > + mnt->minorversion = 2;
> > + break;
> > + default:
> > + return 0;
> > + }
> > + return 1;
> > +}
> > +
> > +static int nfs_get_option_str(substring_t args[], char **option)
> > +{
> > + kfree(*option);
> > + *option = match_strdup(args);
> > + return !*option;
> > +}
> > +
> > +static int nfs_get_option_ul(substring_t args[], unsigned long
> > *option)
> > +{
> > + int rc;
> > + char *string;
> > +
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + return -ENOMEM;
> > + rc = kstrtoul(string, 10, option);
> > + kfree(string);
> > +
> > + return rc;
> > +}
> > +
> > +static int nfs_get_option_ul_bound(substring_t args[], unsigned
> > long *option,
> > + unsigned long l_bound, unsigned long u_bound)
> > +{
> > + int ret;
> > +
> > + ret = nfs_get_option_ul(args, option);
> > + if (ret != 0)
> > + return ret;
> > + if (*option < l_bound || *option > u_bound)
> > + return -ERANGE;
> > + return 0;
> > +}
> > +
> > +/*
> > + * Error-check and convert a string of mount options from user
> > space into
> > + * a data structure. The whole mount string is processed; bad
> > options are
> > + * skipped as they are encountered. If there were no errors,
> > return 1;
> > + * otherwise return 0 (zero).
> > + */
> > +int nfs_parse_mount_options(char *raw, struct
> > nfs_parsed_mount_data *mnt)
> > +{
> > + char *p, *string;
> > + int rc, sloppy = 0, invalid_option = 0;
> > + unsigned short protofamily = AF_UNSPEC;
> > + unsigned short mountfamily = AF_UNSPEC;
> > +
> > + if (!raw) {
> > + dfprintk(MOUNT, "NFS: mount options string was
> > NULL.\n");
> > + return 1;
> > + }
> > + dfprintk(MOUNT, "NFS: nfs mount opts='%s'\n", raw);
> > +
> > + rc = security_sb_eat_lsm_opts(raw, &mnt->lsm_opts);
> > + if (rc)
> > + goto out_security_failure;
> > +
> > + while ((p = strsep(&raw, ",")) != NULL) {
> > + substring_t args[MAX_OPT_ARGS];
> > + unsigned long option;
> > + int token;
> > +
> > + if (!*p)
> > + continue;
> > +
> > + dfprintk(MOUNT, "NFS: parsing nfs mount option
> > '%s'\n", p);
> > +
> > + token = match_token(p, nfs_mount_option_tokens, args);
> > + switch (token) {
> > +
> > + /*
> > + * boolean options: foo/nofoo
> > + */
> > + case Opt_soft:
> > + mnt->flags |= NFS_MOUNT_SOFT;
> > + mnt->flags &= ~NFS_MOUNT_SOFTERR;
> > + break;
> > + case Opt_softerr:
> > + mnt->flags |= NFS_MOUNT_SOFTERR;
> > + mnt->flags &= ~NFS_MOUNT_SOFT;
> > + break;
> > + case Opt_hard:
> > + mnt->flags &=
> > ~(NFS_MOUNT_SOFT|NFS_MOUNT_SOFTERR);
> > + break;
> > + case Opt_posix:
> > + mnt->flags |= NFS_MOUNT_POSIX;
> > + break;
> > + case Opt_noposix:
> > + mnt->flags &= ~NFS_MOUNT_POSIX;
> > + break;
> > + case Opt_cto:
> > + mnt->flags &= ~NFS_MOUNT_NOCTO;
> > + break;
> > + case Opt_nocto:
> > + mnt->flags |= NFS_MOUNT_NOCTO;
> > + break;
> > + case Opt_ac:
> > + mnt->flags &= ~NFS_MOUNT_NOAC;
> > + break;
> > + case Opt_noac:
> > + mnt->flags |= NFS_MOUNT_NOAC;
> > + break;
> > + case Opt_lock:
> > + mnt->flags &= ~NFS_MOUNT_NONLM;
> > + mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
> > + NFS_MOUNT_LOCAL_FCNTL);
> > + break;
> > + case Opt_nolock:
> > + mnt->flags |= NFS_MOUNT_NONLM;
> > + mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
> > + NFS_MOUNT_LOCAL_FCNTL);
> > + break;
> > + case Opt_udp:
> > + mnt->flags &= ~NFS_MOUNT_TCP;
> > + mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP;
> > + break;
> > + case Opt_tcp:
> > + mnt->flags |= NFS_MOUNT_TCP;
> > + mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > + break;
> > + case Opt_rdma:
> > + mnt->flags |= NFS_MOUNT_TCP; /* for side
> > protocols */
> > + mnt->nfs_server.protocol = XPRT_TRANSPORT_RDMA;
> > + xprt_load_transport(p);
> > + break;
> > + case Opt_acl:
> > + mnt->flags &= ~NFS_MOUNT_NOACL;
> > + break;
> > + case Opt_noacl:
> > + mnt->flags |= NFS_MOUNT_NOACL;
> > + break;
> > + case Opt_rdirplus:
> > + mnt->flags &= ~NFS_MOUNT_NORDIRPLUS;
> > + break;
> > + case Opt_nordirplus:
> > + mnt->flags |= NFS_MOUNT_NORDIRPLUS;
> > + break;
> > + case Opt_sharecache:
> > + mnt->flags &= ~NFS_MOUNT_UNSHARED;
> > + break;
> > + case Opt_nosharecache:
> > + mnt->flags |= NFS_MOUNT_UNSHARED;
> > + break;
> > + case Opt_resvport:
> > + mnt->flags &= ~NFS_MOUNT_NORESVPORT;
> > + break;
> > + case Opt_noresvport:
> > + mnt->flags |= NFS_MOUNT_NORESVPORT;
> > + break;
> > + case Opt_fscache:
> > + mnt->options |= NFS_OPTION_FSCACHE;
> > + kfree(mnt->fscache_uniq);
> > + mnt->fscache_uniq = NULL;
> > + break;
> > + case Opt_nofscache:
> > + mnt->options &= ~NFS_OPTION_FSCACHE;
> > + kfree(mnt->fscache_uniq);
> > + mnt->fscache_uniq = NULL;
> > + break;
> > + case Opt_migration:
> > + mnt->options |= NFS_OPTION_MIGRATION;
> > + break;
> > + case Opt_nomigration:
> > + mnt->options &= ~NFS_OPTION_MIGRATION;
> > + break;
> > +
> > + /*
> > + * options that take numeric values
> > + */
> > + case Opt_port:
> > + if (nfs_get_option_ul(args, &option) ||
> > + option > USHRT_MAX)
> > + goto out_invalid_value;
> > + mnt->nfs_server.port = option;
> > + break;
> > + case Opt_rsize:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->rsize = option;
> > + break;
> > + case Opt_wsize:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->wsize = option;
> > + break;
> > + case Opt_bsize:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->bsize = option;
> > + break;
> > + case Opt_timeo:
> > + if (nfs_get_option_ul_bound(args, &option, 1,
> > INT_MAX))
> > + goto out_invalid_value;
> > + mnt->timeo = option;
> > + break;
> > + case Opt_retrans:
> > + if (nfs_get_option_ul_bound(args, &option, 0,
> > INT_MAX))
> > + goto out_invalid_value;
> > + mnt->retrans = option;
> > + break;
> > + case Opt_acregmin:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->acregmin = option;
> > + break;
> > + case Opt_acregmax:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->acregmax = option;
> > + break;
> > + case Opt_acdirmin:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->acdirmin = option;
> > + break;
> > + case Opt_acdirmax:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->acdirmax = option;
> > + break;
> > + case Opt_actimeo:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->acregmin = mnt->acregmax =
> > + mnt->acdirmin = mnt->acdirmax = option;
> > + break;
> > + case Opt_namelen:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + mnt->namlen = option;
> > + break;
> > + case Opt_mountport:
> > + if (nfs_get_option_ul(args, &option) ||
> > + option > USHRT_MAX)
> > + goto out_invalid_value;
> > + mnt->mount_server.port = option;
> > + break;
> > + case Opt_mountvers:
> > + if (nfs_get_option_ul(args, &option) ||
> > + option < NFS_MNT_VERSION ||
> > + option > NFS_MNT3_VERSION)
> > + goto out_invalid_value;
> > + mnt->mount_server.version = option;
> > + break;
> > + case Opt_minorversion:
> > + if (nfs_get_option_ul(args, &option))
> > + goto out_invalid_value;
> > + if (option > NFS4_MAX_MINOR_VERSION)
> > + goto out_invalid_value;
> > + mnt->minorversion = option;
> > + break;
> > +
> > + /*
> > + * options that take text values
> > + */
> > + case Opt_nfsvers:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + rc = nfs_parse_version_string(string, mnt,
> > args);
> > + kfree(string);
> > + if (!rc)
> > + goto out_invalid_value;
> > + break;
> > + case Opt_sec:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + rc = nfs_parse_security_flavors(string, mnt);
> > + kfree(string);
> > + if (!rc) {
> > + dfprintk(MOUNT, "NFS: unrecognized "
> > + "security flavor\n");
> > + return 0;
> > + }
> > + break;
> > + case Opt_proto:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + token = match_token(string,
> > + nfs_xprt_protocol_tokens,
> > args);
> > +
> > + protofamily = AF_INET;
> > + switch (token) {
> > + case Opt_xprt_udp6:
> > + protofamily = AF_INET6;
> > + /* fall through */
> > + case Opt_xprt_udp:
> > + mnt->flags &= ~NFS_MOUNT_TCP;
> > + mnt->nfs_server.protocol =
> > XPRT_TRANSPORT_UDP;
> > + break;
> > + case Opt_xprt_tcp6:
> > + protofamily = AF_INET6;
> > + /* fall through */
> > + case Opt_xprt_tcp:
> > + mnt->flags |= NFS_MOUNT_TCP;
> > + mnt->nfs_server.protocol =
> > XPRT_TRANSPORT_TCP;
> > + break;
> > + case Opt_xprt_rdma6:
> > + protofamily = AF_INET6;
> > + /* fall through */
> > + case Opt_xprt_rdma:
> > + /* vector side protocols to TCP */
> > + mnt->flags |= NFS_MOUNT_TCP;
> > + mnt->nfs_server.protocol =
> > XPRT_TRANSPORT_RDMA;
> > + xprt_load_transport(string);
> > + break;
> > + default:
> > + dfprintk(MOUNT, "NFS: unrecognized "
> > + "transport
> > protocol\n");
> > + kfree(string);
> > + return 0;
> > + }
> > + kfree(string);
> > + break;
> > + case Opt_mountproto:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + token = match_token(string,
> > + nfs_xprt_protocol_tokens,
> > args);
> > + kfree(string);
> > +
> > + mountfamily = AF_INET;
> > + switch (token) {
> > + case Opt_xprt_udp6:
> > + mountfamily = AF_INET6;
> > + /* fall through */
> > + case Opt_xprt_udp:
> > + mnt->mount_server.protocol =
> > XPRT_TRANSPORT_UDP;
> > + break;
> > + case Opt_xprt_tcp6:
> > + mountfamily = AF_INET6;
> > + /* fall through */
> > + case Opt_xprt_tcp:
> > + mnt->mount_server.protocol =
> > XPRT_TRANSPORT_TCP;
> > + break;
> > + case Opt_xprt_rdma: /* not used for side
> > protocols */
> > + default:
> > + dfprintk(MOUNT, "NFS: unrecognized "
> > + "transport
> > protocol\n");
> > + return 0;
> > + }
> > + break;
> > + case Opt_addr:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + mnt->nfs_server.addrlen =
> > + rpc_pton(mnt->net, string,
> > strlen(string),
> > + (struct sockaddr *)
> > + &mnt->nfs_server.address,
> > + sizeof(mnt-
> > >nfs_server.address));
> > + kfree(string);
> > + if (mnt->nfs_server.addrlen == 0)
> > + goto out_invalid_address;
> > + break;
> > + case Opt_clientaddr:
> > + if (nfs_get_option_str(args, &mnt-
> > >client_address))
> > + goto out_nomem;
> > + break;
> > + case Opt_mounthost:
> > + if (nfs_get_option_str(args,
> > + &mnt-
> > >mount_server.hostname))
> > + goto out_nomem;
> > + break;
> > + case Opt_mountaddr:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + mnt->mount_server.addrlen =
> > + rpc_pton(mnt->net, string,
> > strlen(string),
> > + (struct sockaddr *)
> > + &mnt->mount_server.address,
> > + sizeof(mnt-
> > >mount_server.address));
> > + kfree(string);
> > + if (mnt->mount_server.addrlen == 0)
> > + goto out_invalid_address;
> > + break;
> > + case Opt_nconnect:
> > + if (nfs_get_option_ul_bound(args, &option, 1,
> > NFS_MAX_CONNECTIONS))
> > + goto out_invalid_value;
> > + mnt->nfs_server.nconnect = option;
> > + break;
> > + case Opt_lookupcache:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + token = match_token(string,
> > + nfs_lookupcache_tokens, args);
> > + kfree(string);
> > + switch (token) {
> > + case Opt_lookupcache_all:
> > + mnt->flags &=
> > ~(NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE);
> > + break;
> > + case Opt_lookupcache_positive:
> > + mnt->flags &=
> > ~NFS_MOUNT_LOOKUP_CACHE_NONE;
> > + mnt->flags |=
> > NFS_MOUNT_LOOKUP_CACHE_NONEG;
> > + break;
> > + case Opt_lookupcache_none:
> > + mnt->flags |=
> > NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE;
> > + break;
> > + default:
> > + dfprintk(MOUNT, "NFS: invalid
> > "
> > + "lookupcache
> > argument\n");
> > + return 0;
> > + };
> > + break;
> > + case Opt_fscache_uniq:
> > + if (nfs_get_option_str(args, &mnt-
> > >fscache_uniq))
> > + goto out_nomem;
> > + mnt->options |= NFS_OPTION_FSCACHE;
> > + break;
> > + case Opt_local_lock:
> > + string = match_strdup(args);
> > + if (string == NULL)
> > + goto out_nomem;
> > + token = match_token(string,
> > nfs_local_lock_tokens,
> > + args);
> > + kfree(string);
> > + switch (token) {
> > + case Opt_local_lock_all:
> > + mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
> > + NFS_MOUNT_LOCAL_FCNTL);
> > + break;
> > + case Opt_local_lock_flock:
> > + mnt->flags |= NFS_MOUNT_LOCAL_FLOCK;
> > + break;
> > + case Opt_local_lock_posix:
> > + mnt->flags |= NFS_MOUNT_LOCAL_FCNTL;
> > + break;
> > + case Opt_local_lock_none:
> > + mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
> > + NFS_MOUNT_LOCAL_FCNTL);
> > + break;
> > + default:
> > + dfprintk(MOUNT, "NFS: invalid "
> > + "local_lock
> > argument\n");
> > + return 0;
> > + };
> > + break;
> > +
> > + /*
> > + * Special options
> > + */
> > + case Opt_sloppy:
> > + sloppy = 1;
> > + dfprintk(MOUNT, "NFS: relaxing parsing
> > rules\n");
> > + break;
> > + case Opt_userspace:
> > + case Opt_deprecated:
> > + dfprintk(MOUNT, "NFS: ignoring mount option "
> > + "'%s'\n", p);
> > + break;
> > +
> > + default:
> > + invalid_option = 1;
> > + dfprintk(MOUNT, "NFS: unrecognized mount
> > option "
> > + "'%s'\n", p);
> > + }
> > + }
> > +
> > + if (!sloppy && invalid_option)
> > + return 0;
> > +
> > + if (mnt->minorversion && mnt->version != 4)
> > + goto out_minorversion_mismatch;
> > +
> > + if (mnt->options & NFS_OPTION_MIGRATION &&
> > + (mnt->version != 4 || mnt->minorversion != 0))
> > + goto out_migration_misuse;
> > +
> > + /*
> > + * verify that any proto=/mountproto= options match the address
> > + * families in the addr=/mountaddr= options.
> > + */
> > + if (protofamily != AF_UNSPEC &&
> > + protofamily != mnt->nfs_server.address.ss_family)
> > + goto out_proto_mismatch;
> > +
> > + if (mountfamily != AF_UNSPEC) {
> > + if (mnt->mount_server.addrlen) {
> > + if (mountfamily != mnt-
> > >mount_server.address.ss_family)
> > + goto out_mountproto_mismatch;
> > + } else {
> > + if (mountfamily != mnt-
> > >nfs_server.address.ss_family)
> > + goto out_mountproto_mismatch;
> > + }
> > + }
> > +
> > + return 1;
> > +
> > +out_mountproto_mismatch:
> > + printk(KERN_INFO "NFS: mount server address does not match
> > mountproto= "
> > + "option\n");
> > + return 0;
> > +out_proto_mismatch:
> > + printk(KERN_INFO "NFS: server address does not match proto=
> > option\n");
> > + return 0;
> > +out_invalid_address:
> > + printk(KERN_INFO "NFS: bad IP address specified: %s\n", p);
> > + return 0;
> > +out_invalid_value:
> > + printk(KERN_INFO "NFS: bad mount option value specified: %s\n",
> > p);
> > + return 0;
> > +out_minorversion_mismatch:
> > + printk(KERN_INFO "NFS: mount option vers=%u does not support "
> > + "minorversion=%u\n", mnt->version, mnt-
> > >minorversion);
> > + return 0;
> > +out_migration_misuse:
> > + printk(KERN_INFO
> > + "NFS: 'migration' not supported for this NFS
> > version\n");
> > + return 0;
> > +out_nomem:
> > + printk(KERN_INFO "NFS: not enough memory to parse option\n");
> > + return 0;
> > +out_security_failure:
> > + printk(KERN_INFO "NFS: security options invalid: %d\n", rc);
> > + return 0;
> > +}
> > +
> > +/*
> > + * Split "dev_name" into "hostname:export_path".
> > + *
> > + * The leftmost colon demarks the split between the server's
> > hostname
> > + * and the export path. If the hostname starts with a left square
> > + * bracket, then it may contain colons.
> > + *
> > + * Note: caller frees hostname and export path, even on error.
> > + */
> > +static int nfs_parse_devname(const char *dev_name,
> > + char **hostname, size_t maxnamlen,
> > + char **export_path, size_t maxpathlen)
> > +{
> > + size_t len;
> > + char *end;
> > +
> > + if (unlikely(!dev_name || !*dev_name)) {
> > + dfprintk(MOUNT, "NFS: device name not specified\n");
> > + return -EINVAL;
> > + }
> > +
> > + /* Is the host name protected with square brakcets? */
> > + if (*dev_name == '[') {
> > + end = strchr(++dev_name, ']');
> > + if (end == NULL || end[1] != ':')
> > + goto out_bad_devname;
> > +
> > + len = end - dev_name;
> > + end++;
> > + } else {
> > + char *comma;
> > +
> > + end = strchr(dev_name, ':');
> > + if (end == NULL)
> > + goto out_bad_devname;
> > + len = end - dev_name;
> > +
> > + /* kill possible hostname list: not supported */
> > + comma = strchr(dev_name, ',');
> > + if (comma != NULL && comma < end)
> > + len = comma - dev_name;
> > + }
> > +
> > + if (len > maxnamlen)
> > + goto out_hostname;
> > +
> > + /* N.B. caller will free nfs_server.hostname in all cases */
> > + *hostname = kstrndup(dev_name, len, GFP_KERNEL);
> > + if (*hostname == NULL)
> > + goto out_nomem;
> > + len = strlen(++end);
> > + if (len > maxpathlen)
> > + goto out_path;
> > + *export_path = kstrndup(end, len, GFP_KERNEL);
> > + if (!*export_path)
> > + goto out_nomem;
> > +
> > + dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", *export_path);
> > + return 0;
> > +
> > +out_bad_devname:
> > + dfprintk(MOUNT, "NFS: device name not in host:path format\n");
> > + return -EINVAL;
> > +
> > +out_nomem:
> > + dfprintk(MOUNT, "NFS: not enough memory to parse device
> > name\n");
> > + return -ENOMEM;
> > +
> > +out_hostname:
> > + dfprintk(MOUNT, "NFS: server hostname too long\n");
> > + return -ENAMETOOLONG;
> > +
> > +out_path:
> > + dfprintk(MOUNT, "NFS: export pathname too long\n");
> > + return -ENAMETOOLONG;
> > +}
> > +
> > +/*
> > + * Validate the NFS2/NFS3 mount data
> > + * - fills in the mount root filehandle
> > + *
> > + * For option strings, user space handles the following behaviors:
> > + *
> > + * + DNS: mapping server host name to IP address ("addr=" option)
> > + *
> > + * + failure mode: how to behave if a mount request can't be
> > handled
> > + * immediately ("fg/bg" option)
> > + *
> > + * + retry: how often to retry a mount request ("retry=" option)
> > + *
> > + * + breaking back: trying proto=udp after proto=tcp, v2 after v3,
> > + * mountproto=tcp after mountproto=udp, and so on
> > + */
> > +static int nfs23_validate_mount_data(void *options,
> > + struct nfs_parsed_mount_data
> > *args,
> > + struct nfs_fh *mntfh,
> > + const char *dev_name)
> > +{
> > + struct nfs_mount_data *data = (struct nfs_mount_data *)options;
> > + struct sockaddr *sap = (struct sockaddr *)&args-
> > >nfs_server.address;
> > + int extra_flags = NFS_MOUNT_LEGACY_INTERFACE;
> > +
> > + if (data == NULL)
> > + goto out_no_data;
> > +
> > + args->version = NFS_DEFAULT_VERSION;
> > + switch (data->version) {
> > + case 1:
> > + data->namlen = 0; /* fall through */
> > + case 2:
> > + data->bsize = 0; /* fall through */
> > + case 3:
> > + if (data->flags & NFS_MOUNT_VER3)
> > + goto out_no_v3;
> > + data->root.size = NFS2_FHSIZE;
> > + memcpy(data->root.data, data->old_root.data,
> > NFS2_FHSIZE);
> > + /* Turn off security negotiation */
> > + extra_flags |= NFS_MOUNT_SECFLAVOUR;
> > + /* fall through */
> > + case 4:
> > + if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > + goto out_no_sec;
> > + /* fall through */
> > + case 5:
> > + memset(data->context, 0, sizeof(data->context));
> > + /* fall through */
> > + case 6:
> > + if (data->flags & NFS_MOUNT_VER3) {
> > + if (data->root.size > NFS3_FHSIZE || data-
> > >root.size == 0)
> > + goto out_invalid_fh;
> > + mntfh->size = data->root.size;
> > + args->version = 3;
> > + } else {
> > + mntfh->size = NFS2_FHSIZE;
> > + args->version = 2;
> > + }
> > +
> > +
> > + memcpy(mntfh->data, data->root.data, mntfh->size);
> > + if (mntfh->size < sizeof(mntfh->data))
> > + memset(mntfh->data + mntfh->size, 0,
> > + sizeof(mntfh->data) - mntfh->size);
> > +
> > + /*
> > + * Translate to nfs_parsed_mount_data, which
> > nfs_fill_super
> > + * can deal with.
> > + */
> > + args->flags = data->flags & NFS_MOUNT_FLAGMASK;
> > + args->flags |= extra_flags;
> > + args->rsize = data->rsize;
> > + args->wsize = data->wsize;
> > + args->timeo = data->timeo;
> > + args->retrans = data->retrans;
> > + args->acregmin = data->acregmin;
> > + args->acregmax = data->acregmax;
> > + args->acdirmin = data->acdirmin;
> > + args->acdirmax = data->acdirmax;
> > + args->need_mount = false;
> > +
> > + memcpy(sap, &data->addr, sizeof(data->addr));
> > + args->nfs_server.addrlen = sizeof(data->addr);
> > + args->nfs_server.port = ntohs(data->addr.sin_port);
> > + if (sap->sa_family != AF_INET ||
> > + !nfs_verify_server_address(sap))
> > + goto out_no_address;
> > +
> > + if (!(data->flags & NFS_MOUNT_TCP))
> > + args->nfs_server.protocol = XPRT_TRANSPORT_UDP;
> > + /* N.B. caller will free nfs_server.hostname in all
> > cases */
> > + args->nfs_server.hostname = kstrdup(data->hostname,
> > GFP_KERNEL);
> > + args->namlen = data->namlen;
> > + args->bsize = data->bsize;
> > +
> > + if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > + args->selected_flavor = data->pseudoflavor;
> > + else
> > + args->selected_flavor = RPC_AUTH_UNIX;
> > + if (!args->nfs_server.hostname)
> > + goto out_nomem;
> > +
> > + if (!(data->flags & NFS_MOUNT_NONLM))
> > + args->flags &= ~(NFS_MOUNT_LOCAL_FLOCK|
> > + NFS_MOUNT_LOCAL_FCNTL);
> > + else
> > + args->flags |= (NFS_MOUNT_LOCAL_FLOCK|
> > + NFS_MOUNT_LOCAL_FCNTL);
> > + /*
> > + * The legacy version 6 binary mount data from
> > userspace has a
> > + * field used only to transport selinux information
> > into the
> > + * the kernel. To continue to support that
> > functionality we
> > + * have a touch of selinux knowledge here in the NFS
> > code. The
> > + * userspace code converted context=blah to just blah
> > so we are
> > + * converting back to the full string selinux
> > understands.
> > + */
> > + if (data->context[0]){
> > +#ifdef CONFIG_SECURITY_SELINUX
> > + int rc;
> > + data->context[NFS_MAX_CONTEXT_LEN] = '\0';
> > + rc = security_add_mnt_opt("context", data-
> > >context,
> > + strlen(data->context), &args-
> > >lsm_opts);
> > + if (rc)
> > + return rc;
> > +#else
> > + return -EINVAL;
> > +#endif
> > + }
> > +
> > + break;
> > + default:
> > + return NFS_TEXT_DATA;
> > + }
> > +
> > + return 0;
> > +
> > +out_no_data:
> > + dfprintk(MOUNT, "NFS: mount program didn't pass any mount
> > data\n");
> > + return -EINVAL;
> > +
> > +out_no_v3:
> > + dfprintk(MOUNT, "NFS: nfs_mount_data version %d does not
> > support v3\n",
> > + data->version);
> > + return -EINVAL;
> > +
> > +out_no_sec:
> > + dfprintk(MOUNT, "NFS: nfs_mount_data version supports only
> > AUTH_SYS\n");
> > + return -EINVAL;
> > +
> > +out_nomem:
> > + dfprintk(MOUNT, "NFS: not enough memory to handle mount
> > options\n");
> > + return -ENOMEM;
> > +
> > +out_no_address:
> > + dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > address\n");
> > + return -EINVAL;
> > +
> > +out_invalid_fh:
> > + dfprintk(MOUNT, "NFS: invalid root filehandle\n");
> > + return -EINVAL;
> > +}
> > +
> > +#if IS_ENABLED(CONFIG_NFS_V4)
> > +
> > +static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data
> > *args)
> > +{
> > + args->flags &=
> > ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3|
> > + NFS_MOUNT_LOCAL_FLOCK|NFS_MOUNT_LOCAL_FCNTL);
> > +}
> > +
> > +/*
> > + * Validate NFSv4 mount options
> > + */
> > +static int nfs4_validate_mount_data(void *options,
> > + struct nfs_parsed_mount_data *args,
> > + const char *dev_name)
> > +{
> > + struct sockaddr *sap = (struct sockaddr *)&args-
> > >nfs_server.address;
> > + struct nfs4_mount_data *data = (struct nfs4_mount_data
> > *)options;
> > + char *c;
> > +
> > + if (data == NULL)
> > + goto out_no_data;
> > +
> > + args->version = 4;
> > +
> > + switch (data->version) {
> > + case 1:
> > + if (data->host_addrlen > sizeof(args-
> > >nfs_server.address))
> > + goto out_no_address;
> > + if (data->host_addrlen == 0)
> > + goto out_no_address;
> > + args->nfs_server.addrlen = data->host_addrlen;
> > + if (copy_from_user(sap, data->host_addr, data-
> > >host_addrlen))
> > + return -EFAULT;
> > + if (!nfs_verify_server_address(sap))
> > + goto out_no_address;
> > + args->nfs_server.port = ntohs(((struct sockaddr_in
> > *)sap)->sin_port);
> > +
> > + if (data->auth_flavourlen) {
> > + rpc_authflavor_t pseudoflavor;
> > + if (data->auth_flavourlen > 1)
> > + goto out_inval_auth;
> > + if (copy_from_user(&pseudoflavor,
> > + data->auth_flavours,
> > + sizeof(pseudoflavor)))
> > + return -EFAULT;
> > + args->selected_flavor = pseudoflavor;
> > + } else
> > + args->selected_flavor = RPC_AUTH_UNIX;
> > +
> > + c = strndup_user(data->hostname.data, NFS4_MAXNAMLEN);
> > + if (IS_ERR(c))
> > + return PTR_ERR(c);
> > + args->nfs_server.hostname = c;
> > +
> > + c = strndup_user(data->mnt_path.data, NFS4_MAXPATHLEN);
> > + if (IS_ERR(c))
> > + return PTR_ERR(c);
> > + args->nfs_server.export_path = c;
> > + dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", c);
> > +
> > + c = strndup_user(data->client_addr.data, 16);
> > + if (IS_ERR(c))
> > + return PTR_ERR(c);
> > + args->client_address = c;
> > +
> > + /*
> > + * Translate to nfs_parsed_mount_data, which
> > nfs4_fill_super
> > + * can deal with.
> > + */
> > +
> > + args->flags = data->flags & NFS4_MOUNT_FLAGMASK;
> > + args->rsize = data->rsize;
> > + args->wsize = data->wsize;
> > + args->timeo = data->timeo;
> > + args->retrans = data->retrans;
> > + args->acregmin = data->acregmin;
> > + args->acregmax = data->acregmax;
> > + args->acdirmin = data->acdirmin;
> > + args->acdirmax = data->acdirmax;
> > + args->nfs_server.protocol = data->proto;
> > + nfs_validate_transport_protocol(args);
> > + if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
> > + goto out_invalid_transport_udp;
> > +
> > + break;
> > + default:
> > + return NFS_TEXT_DATA;
> > + }
> > +
> > + return 0;
> > +
> > +out_no_data:
> > + dfprintk(MOUNT, "NFS4: mount program didn't pass any mount
> > data\n");
> > + return -EINVAL;
> > +
> > +out_inval_auth:
> > + dfprintk(MOUNT, "NFS4: Invalid number of RPC auth flavours
> > %d\n",
> > + data->auth_flavourlen);
> > + return -EINVAL;
> > +
> > +out_no_address:
> > + dfprintk(MOUNT, "NFS4: mount program didn't pass remote
> > address\n");
> > + return -EINVAL;
> > +
> > +out_invalid_transport_udp:
> > + dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
> > + return -EINVAL;
> > +}
> > +
> > +int nfs_validate_mount_data(struct file_system_type *fs_type,
> > + void *options,
> > + struct nfs_parsed_mount_data *args,
> > + struct nfs_fh *mntfh,
> > + const char *dev_name)
> > +{
> > + if (fs_type == &nfs_fs_type)
> > + return nfs23_validate_mount_data(options, args, mntfh,
> > dev_name);
> > + return nfs4_validate_mount_data(options, args, dev_name);
> > +}
> > +#else
> > +int nfs_validate_mount_data(struct file_system_type *fs_type,
> > + void *options,
> > + struct nfs_parsed_mount_data *args,
> > + struct nfs_fh *mntfh,
> > + const char *dev_name)
> > +{
> > + return nfs23_validate_mount_data(options, args, mntfh,
> > dev_name);
> > +}
> > +#endif
> > +
> > +int nfs_validate_text_mount_data(void *options,
> > + struct nfs_parsed_mount_data *args,
> > + const char *dev_name)
> > +{
> > + int port = 0;
> > + int max_namelen = PAGE_SIZE;
> > + int max_pathlen = NFS_MAXPATHLEN;
> > + struct sockaddr *sap = (struct sockaddr *)&args-
> > >nfs_server.address;
> > +
> > + if (nfs_parse_mount_options((char *)options, args) == 0)
> > + return -EINVAL;
> > +
> > + if (!nfs_verify_server_address(sap))
> > + goto out_no_address;
> > +
> > + if (args->version == 4) {
> > +#if IS_ENABLED(CONFIG_NFS_V4)
> > + if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
> > + port = NFS_RDMA_PORT;
> > + else
> > + port = NFS_PORT;
> > + max_namelen = NFS4_MAXNAMLEN;
> > + max_pathlen = NFS4_MAXPATHLEN;
> > + nfs_validate_transport_protocol(args);
> > + if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
> > + goto out_invalid_transport_udp;
> > + nfs4_validate_mount_flags(args);
> > +#else
> > + goto out_v4_not_compiled;
> > +#endif /* CONFIG_NFS_V4 */
> > + } else {
> > + nfs_set_mount_transport_protocol(args);
> > + if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
> > + port = NFS_RDMA_PORT;
> > + }
> > +
> > + nfs_set_port(sap, &args->nfs_server.port, port);
> > +
> > + return nfs_parse_devname(dev_name,
> > + &args->nfs_server.hostname,
> > + max_namelen,
> > + &args->nfs_server.export_path,
> > + max_pathlen);
> > +
> > +#if !IS_ENABLED(CONFIG_NFS_V4)
> > +out_v4_not_compiled:
> > + dfprintk(MOUNT, "NFS: NFSv4 is not compiled into kernel\n");
> > + return -EPROTONOSUPPORT;
> > +#else
> > +out_invalid_transport_udp:
> > + dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
> > + return -EINVAL;
> > +#endif /* !CONFIG_NFS_V4 */
> > +
> > +out_no_address:
> > + dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > address\n");
> > + return -EINVAL;
> > +}
> > diff --git a/fs/nfs/internal.h b/fs/nfs/internal.h
> > index d512ec394559..b66fd35993b3 100644
> > --- a/fs/nfs/internal.h
> > +++ b/fs/nfs/internal.h
> > @@ -7,6 +7,7 @@
> > #include <linux/mount.h>
> > #include <linux/security.h>
> > #include <linux/crc32.h>
> > +#include <linux/sunrpc/addr.h>
> > #include <linux/nfs_page.h>
> > #include <linux/wait_bit.h>
> >
> > @@ -232,6 +233,22 @@ extern const struct svc_version
> > nfs4_callback_version1;
> > extern const struct svc_version nfs4_callback_version4;
> >
> > struct nfs_pageio_descriptor;
> > +
> > +/* mount.c */
> > +#define NFS_TEXT_DATA 1
> > +
> > +extern struct nfs_parsed_mount_data
> > *nfs_alloc_parsed_mount_data(void);
> > +extern void nfs_free_parsed_mount_data(struct
> > nfs_parsed_mount_data *data);
> > +extern int nfs_parse_mount_options(char *raw, struct
> > nfs_parsed_mount_data *mnt);
> > +extern int nfs_validate_mount_data(struct file_system_type
> > *fs_type,
> > + void *options,
> > + struct nfs_parsed_mount_data *args,
> > + struct nfs_fh *mntfh,
> > + const char *dev_name);
> > +extern int nfs_validate_text_mount_data(void *options,
> > + struct nfs_parsed_mount_data
> > *args,
> > + const char *dev_name);
> > +
> > /* pagelist.c */
> > extern int __init nfs_init_nfspagecache(void);
> > extern void nfs_destroy_nfspagecache(void);
> > @@ -763,3 +780,15 @@ static inline bool nfs_error_is_fatal(int err)
> > }
> > }
> >
> > +/*
> > + * Select between a default port value and a user-specified port
> > value.
> > + * If a zero value is set, then autobind will be used.
> > + */
> > +static inline void nfs_set_port(struct sockaddr *sap, int *port,
> > + const unsigned short default_port)
> > +{
> > + if (*port == NFS_UNSPEC_PORT)
> > + *port = default_port;
> > +
> > + rpc_set_port(sap, *port);
> > +}
> > diff --git a/fs/nfs/super.c b/fs/nfs/super.c
> > index d8702e57f7fc..886220d2da4e 100644
> > --- a/fs/nfs/super.c
> > +++ b/fs/nfs/super.c
> > @@ -69,229 +69,6 @@
> > #include "nfs.h"
> >
> > #define NFSDBG_FACILITY NFSDBG_VFS
> > -#define NFS_TEXT_DATA 1
> > -
> > -#if IS_ENABLED(CONFIG_NFS_V3)
> > -#define NFS_DEFAULT_VERSION 3
> > -#else
> > -#define NFS_DEFAULT_VERSION 2
> > -#endif
> > -
> > -#define NFS_MAX_CONNECTIONS 16
> > -
> > -enum {
> > - /* Mount options that take no arguments */
> > - Opt_soft, Opt_softerr, Opt_hard,
> > - Opt_posix, Opt_noposix,
> > - Opt_cto, Opt_nocto,
> > - Opt_ac, Opt_noac,
> > - Opt_lock, Opt_nolock,
> > - Opt_udp, Opt_tcp, Opt_rdma,
> > - Opt_acl, Opt_noacl,
> > - Opt_rdirplus, Opt_nordirplus,
> > - Opt_sharecache, Opt_nosharecache,
> > - Opt_resvport, Opt_noresvport,
> > - Opt_fscache, Opt_nofscache,
> > - Opt_migration, Opt_nomigration,
> > -
> > - /* Mount options that take integer arguments */
> > - Opt_port,
> > - Opt_rsize, Opt_wsize, Opt_bsize,
> > - Opt_timeo, Opt_retrans,
> > - Opt_acregmin, Opt_acregmax,
> > - Opt_acdirmin, Opt_acdirmax,
> > - Opt_actimeo,
> > - Opt_namelen,
> > - Opt_mountport,
> > - Opt_mountvers,
> > - Opt_minorversion,
> > -
> > - /* Mount options that take string arguments */
> > - Opt_nfsvers,
> > - Opt_sec, Opt_proto, Opt_mountproto, Opt_mounthost,
> > - Opt_addr, Opt_mountaddr, Opt_clientaddr,
> > - Opt_nconnect,
> > - Opt_lookupcache,
> > - Opt_fscache_uniq,
> > - Opt_local_lock,
> > -
> > - /* Special mount options */
> > - Opt_userspace, Opt_deprecated, Opt_sloppy,
> > -
> > - Opt_err
> > -};
> > -
> > -static const match_table_t nfs_mount_option_tokens = {
> > - { Opt_userspace, "bg" },
> > - { Opt_userspace, "fg" },
> > - { Opt_userspace, "retry=%s" },
> > -
> > - { Opt_sloppy, "sloppy" },
> > -
> > - { Opt_soft, "soft" },
> > - { Opt_softerr, "softerr" },
> > - { Opt_hard, "hard" },
> > - { Opt_deprecated, "intr" },
> > - { Opt_deprecated, "nointr" },
> > - { Opt_posix, "posix" },
> > - { Opt_noposix, "noposix" },
> > - { Opt_cto, "cto" },
> > - { Opt_nocto, "nocto" },
> > - { Opt_ac, "ac" },
> > - { Opt_noac, "noac" },
> > - { Opt_lock, "lock" },
> > - { Opt_nolock, "nolock" },
> > - { Opt_udp, "udp" },
> > - { Opt_tcp, "tcp" },
> > - { Opt_rdma, "rdma" },
> > - { Opt_acl, "acl" },
> > - { Opt_noacl, "noacl" },
> > - { Opt_rdirplus, "rdirplus" },
> > - { Opt_nordirplus, "nordirplus" },
> > - { Opt_sharecache, "sharecache" },
> > - { Opt_nosharecache, "nosharecache" },
> > - { Opt_resvport, "resvport" },
> > - { Opt_noresvport, "noresvport" },
> > - { Opt_fscache, "fsc" },
> > - { Opt_nofscache, "nofsc" },
> > - { Opt_migration, "migration" },
> > - { Opt_nomigration, "nomigration" },
> > -
> > - { Opt_port, "port=%s" },
> > - { Opt_rsize, "rsize=%s" },
> > - { Opt_wsize, "wsize=%s" },
> > - { Opt_bsize, "bsize=%s" },
> > - { Opt_timeo, "timeo=%s" },
> > - { Opt_retrans, "retrans=%s" },
> > - { Opt_acregmin, "acregmin=%s" },
> > - { Opt_acregmax, "acregmax=%s" },
> > - { Opt_acdirmin, "acdirmin=%s" },
> > - { Opt_acdirmax, "acdirmax=%s" },
> > - { Opt_actimeo, "actimeo=%s" },
> > - { Opt_namelen, "namlen=%s" },
> > - { Opt_mountport, "mountport=%s" },
> > - { Opt_mountvers, "mountvers=%s" },
> > - { Opt_minorversion, "minorversion=%s" },
> > -
> > - { Opt_nfsvers, "nfsvers=%s" },
> > - { Opt_nfsvers, "vers=%s" },
> > -
> > - { Opt_sec, "sec=%s" },
> > - { Opt_proto, "proto=%s" },
> > - { Opt_mountproto, "mountproto=%s" },
> > - { Opt_addr, "addr=%s" },
> > - { Opt_clientaddr, "clientaddr=%s" },
> > - { Opt_mounthost, "mounthost=%s" },
> > - { Opt_mountaddr, "mountaddr=%s" },
> > -
> > - { Opt_nconnect, "nconnect=%s" },
> > -
> > - { Opt_lookupcache, "lookupcache=%s" },
> > - { Opt_fscache_uniq, "fsc=%s" },
> > - { Opt_local_lock, "local_lock=%s" },
> > -
> > - /* The following needs to be listed after all other options */
> > - { Opt_nfsvers, "v%s" },
> > -
> > - { Opt_err, NULL }
> > -};
> > -
> > -enum {
> > - Opt_xprt_udp, Opt_xprt_udp6, Opt_xprt_tcp, Opt_xprt_tcp6,
> > Opt_xprt_rdma,
> > - Opt_xprt_rdma6,
> > -
> > - Opt_xprt_err
> > -};
> > -
> > -static const match_table_t nfs_xprt_protocol_tokens = {
> > - { Opt_xprt_udp, "udp" },
> > - { Opt_xprt_udp6, "udp6" },
> > - { Opt_xprt_tcp, "tcp" },
> > - { Opt_xprt_tcp6, "tcp6" },
> > - { Opt_xprt_rdma, "rdma" },
> > - { Opt_xprt_rdma6, "rdma6" },
> > -
> > - { Opt_xprt_err, NULL }
> > -};
> > -
> > -enum {
> > - Opt_sec_none, Opt_sec_sys,
> > - Opt_sec_krb5, Opt_sec_krb5i, Opt_sec_krb5p,
> > - Opt_sec_lkey, Opt_sec_lkeyi, Opt_sec_lkeyp,
> > - Opt_sec_spkm, Opt_sec_spkmi, Opt_sec_spkmp,
> > -
> > - Opt_sec_err
> > -};
> > -
> > -static const match_table_t nfs_secflavor_tokens = {
> > - { Opt_sec_none, "none" },
> > - { Opt_sec_none, "null" },
> > - { Opt_sec_sys, "sys" },
> > -
> > - { Opt_sec_krb5, "krb5" },
> > - { Opt_sec_krb5i, "krb5i" },
> > - { Opt_sec_krb5p, "krb5p" },
> > -
> > - { Opt_sec_lkey, "lkey" },
> > - { Opt_sec_lkeyi, "lkeyi" },
> > - { Opt_sec_lkeyp, "lkeyp" },
> > -
> > - { Opt_sec_spkm, "spkm3" },
> > - { Opt_sec_spkmi, "spkm3i" },
> > - { Opt_sec_spkmp, "spkm3p" },
> > -
> > - { Opt_sec_err, NULL }
> > -};
> > -
> > -enum {
> > - Opt_lookupcache_all, Opt_lookupcache_positive,
> > - Opt_lookupcache_none,
> > -
> > - Opt_lookupcache_err
> > -};
> > -
> > -static match_table_t nfs_lookupcache_tokens = {
> > - { Opt_lookupcache_all, "all" },
> > - { Opt_lookupcache_positive, "pos" },
> > - { Opt_lookupcache_positive, "positive" },
> > - { Opt_lookupcache_none, "none" },
> > -
> > - { Opt_lookupcache_err, NULL }
> > -};
> > -
> > -enum {
> > - Opt_local_lock_all, Opt_local_lock_flock, Opt_local_lock_posix,
> > - Opt_local_lock_none,
> > -
> > - Opt_local_lock_err
> > -};
> > -
> > -static match_table_t nfs_local_lock_tokens = {
> > - { Opt_local_lock_all, "all" },
> > - { Opt_local_lock_flock, "flock" },
> > - { Opt_local_lock_posix, "posix" },
> > - { Opt_local_lock_none, "none" },
> > -
> > - { Opt_local_lock_err, NULL }
> > -};
> > -
> > -enum {
> > - Opt_vers_2, Opt_vers_3, Opt_vers_4, Opt_vers_4_0,
> > - Opt_vers_4_1, Opt_vers_4_2,
> > -
> > - Opt_vers_err
> > -};
> > -
> > -static match_table_t nfs_vers_tokens = {
> > - { Opt_vers_2, "2" },
> > - { Opt_vers_3, "3" },
> > - { Opt_vers_4, "4" },
> > - { Opt_vers_4_0, "4.0" },
> > - { Opt_vers_4_1, "4.1" },
> > - { Opt_vers_4_2, "4.2" },
> > -
> > - { Opt_vers_err, NULL }
> > -};
> >
> > static struct dentry *nfs_prepared_mount(struct file_system_type
> > *fs_type,
> > int flags, const char *dev_name, void *raw_data);
> > @@ -332,10 +109,6 @@ const struct super_operations nfs_sops = {
> > EXPORT_SYMBOL_GPL(nfs_sops);
> >
> > #if IS_ENABLED(CONFIG_NFS_V4)
> > -static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data
> > *);
> > -static int nfs4_validate_mount_data(void *options,
> > - struct nfs_parsed_mount_data *args, const char *dev_name);
> > -
> > struct file_system_type nfs4_fs_type = {
> > .owner = THIS_MODULE,
> > .name = "nfs4",
> > @@ -932,141 +705,6 @@ void nfs_umount_begin(struct super_block *sb)
> > }
> > EXPORT_SYMBOL_GPL(nfs_umount_begin);
> >
> > -static struct nfs_parsed_mount_data
> > *nfs_alloc_parsed_mount_data(void)
> > -{
> > - struct nfs_parsed_mount_data *data;
> > -
> > - data = kzalloc(sizeof(*data), GFP_KERNEL);
> > - if (data) {
> > - data->timeo = NFS_UNSPEC_TIMEO;
> > - data->retrans = NFS_UNSPEC_RETRANS;
> > - data->acregmin = NFS_DEF_ACREGMIN;
> > - data->acregmax = NFS_DEF_ACREGMAX;
> > - data->acdirmin = NFS_DEF_ACDIRMIN;
> > - data->acdirmax = NFS_DEF_ACDIRMAX;
> > - data->mount_server.port = NFS_UNSPEC_PORT;
> > - data->nfs_server.port = NFS_UNSPEC_PORT;
> > - data->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > - data->selected_flavor = RPC_AUTH_MAXFLAVOR;
> > - data->minorversion = 0;
> > - data->need_mount = true;
> > - data->net = current->nsproxy->net_ns;
> > - data->lsm_opts = NULL;
> > - }
> > - return data;
> > -}
> > -
> > -static void nfs_free_parsed_mount_data(struct
> > nfs_parsed_mount_data *data)
> > -{
> > - if (data) {
> > - kfree(data->client_address);
> > - kfree(data->mount_server.hostname);
> > - kfree(data->nfs_server.export_path);
> > - kfree(data->nfs_server.hostname);
> > - kfree(data->fscache_uniq);
> > - security_free_mnt_opts(&data->lsm_opts);
> > - kfree(data);
> > - }
> > -}
> > -
> > -/*
> > - * Sanity-check a server address provided by the mount command.
> > - *
> > - * Address family must be initialized, and address must not be
> > - * the ANY address for that family.
> > - */
> > -static int nfs_verify_server_address(struct sockaddr *addr)
> > -{
> > - switch (addr->sa_family) {
> > - case AF_INET: {
> > - struct sockaddr_in *sa = (struct sockaddr_in *)addr;
> > - return sa->sin_addr.s_addr != htonl(INADDR_ANY);
> > - }
> > - case AF_INET6: {
> > - struct in6_addr *sa = &((struct sockaddr_in6 *)addr)-
> > >sin6_addr;
> > - return !ipv6_addr_any(sa);
> > - }
> > - }
> > -
> > - dfprintk(MOUNT, "NFS: Invalid IP address specified\n");
> > - return 0;
> > -}
> > -
> > -/*
> > - * Select between a default port value and a user-specified port
> > value.
> > - * If a zero value is set, then autobind will be used.
> > - */
> > -static void nfs_set_port(struct sockaddr *sap, int *port,
> > - const unsigned short default_port)
> > -{
> > - if (*port == NFS_UNSPEC_PORT)
> > - *port = default_port;
> > -
> > - rpc_set_port(sap, *port);
> > -}
> > -
> > -/*
> > - * Sanity check the NFS transport protocol.
> > - *
> > - */
> > -static void nfs_validate_transport_protocol(struct
> > nfs_parsed_mount_data *mnt)
> > -{
> > - switch (mnt->nfs_server.protocol) {
> > - case XPRT_TRANSPORT_UDP:
> > - case XPRT_TRANSPORT_TCP:
> > - case XPRT_TRANSPORT_RDMA:
> > - break;
> > - default:
> > - mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > - }
> > -}
> > -
> > -/*
> > - * For text based NFSv2/v3 mounts, the mount protocol transport
> > default
> > - * settings should depend upon the specified NFS transport.
> > - */
> > -static void nfs_set_mount_transport_protocol(struct
> > nfs_parsed_mount_data *mnt)
> > -{
> > - nfs_validate_transport_protocol(mnt);
> > -
> > - if (mnt->mount_server.protocol == XPRT_TRANSPORT_UDP ||
> > - mnt->mount_server.protocol == XPRT_TRANSPORT_TCP)
> > - return;
> > - switch (mnt->nfs_server.protocol) {
> > - case XPRT_TRANSPORT_UDP:
> > - mnt->mount_server.protocol = XPRT_TRANSPORT_UDP;
> > - break;
> > - case XPRT_TRANSPORT_TCP:
> > - case XPRT_TRANSPORT_RDMA:
> > - mnt->mount_server.protocol = XPRT_TRANSPORT_TCP;
> > - }
> > -}
> > -
> > -/*
> > - * Add 'flavor' to 'auth_info' if not already present.
> > - * Returns true if 'flavor' ends up in the list, false otherwise
> > - */
> > -static bool nfs_auth_info_add(struct nfs_auth_info *auth_info,
> > - rpc_authflavor_t flavor)
> > -{
> > - unsigned int i;
> > - unsigned int max_flavor_len = ARRAY_SIZE(auth_info->flavors);
> > -
> > - /* make sure this flavor isn't already in the list */
> > - for (i = 0; i < auth_info->flavor_len; i++) {
> > - if (flavor == auth_info->flavors[i])
> > - return true;
> > - }
> > -
> > - if (auth_info->flavor_len + 1 >= max_flavor_len) {
> > - dfprintk(MOUNT, "NFS: too many sec= flavors\n");
> > - return false;
> > - }
> > -
> > - auth_info->flavors[auth_info->flavor_len++] = flavor;
> > - return true;
> > -}
> > -
> > /*
> > * Return true if 'match' is in auth_info or auth_info is empty.
> > * Return false otherwise.
> > @@ -1087,627 +725,6 @@ bool nfs_auth_info_match(const struct
> > nfs_auth_info *auth_info,
> > }
> > EXPORT_SYMBOL_GPL(nfs_auth_info_match);
> >
> > -/*
> > - * Parse the value of the 'sec=' option.
> > - */
> > -static int nfs_parse_security_flavors(char *value,
> > - struct nfs_parsed_mount_data
> > *mnt)
> > -{
> > - substring_t args[MAX_OPT_ARGS];
> > - rpc_authflavor_t pseudoflavor;
> > - char *p;
> > -
> > - dfprintk(MOUNT, "NFS: parsing sec=%s option\n", value);
> > -
> > - while ((p = strsep(&value, ":")) != NULL) {
> > - switch (match_token(p, nfs_secflavor_tokens, args)) {
> > - case Opt_sec_none:
> > - pseudoflavor = RPC_AUTH_NULL;
> > - break;
> > - case Opt_sec_sys:
> > - pseudoflavor = RPC_AUTH_UNIX;
> > - break;
> > - case Opt_sec_krb5:
> > - pseudoflavor = RPC_AUTH_GSS_KRB5;
> > - break;
> > - case Opt_sec_krb5i:
> > - pseudoflavor = RPC_AUTH_GSS_KRB5I;
> > - break;
> > - case Opt_sec_krb5p:
> > - pseudoflavor = RPC_AUTH_GSS_KRB5P;
> > - break;
> > - case Opt_sec_lkey:
> > - pseudoflavor = RPC_AUTH_GSS_LKEY;
> > - break;
> > - case Opt_sec_lkeyi:
> > - pseudoflavor = RPC_AUTH_GSS_LKEYI;
> > - break;
> > - case Opt_sec_lkeyp:
> > - pseudoflavor = RPC_AUTH_GSS_LKEYP;
> > - break;
> > - case Opt_sec_spkm:
> > - pseudoflavor = RPC_AUTH_GSS_SPKM;
> > - break;
> > - case Opt_sec_spkmi:
> > - pseudoflavor = RPC_AUTH_GSS_SPKMI;
> > - break;
> > - case Opt_sec_spkmp:
> > - pseudoflavor = RPC_AUTH_GSS_SPKMP;
> > - break;
> > - default:
> > - dfprintk(MOUNT,
> > - "NFS: sec= option '%s' not
> > recognized\n", p);
> > - return 0;
> > - }
> > -
> > - if (!nfs_auth_info_add(&mnt->auth_info, pseudoflavor))
> > - return 0;
> > - }
> > -
> > - return 1;
> > -}
> > -
> > -static int nfs_parse_version_string(char *string,
> > - struct nfs_parsed_mount_data *mnt,
> > - substring_t *args)
> > -{
> > - mnt->flags &= ~NFS_MOUNT_VER3;
> > - switch (match_token(string, nfs_vers_tokens, args)) {
> > - case Opt_vers_2:
> > - mnt->version = 2;
> > - break;
> > - case Opt_vers_3:
> > - mnt->flags |= NFS_MOUNT_VER3;
> > - mnt->version = 3;
> > - break;
> > - case Opt_vers_4:
> > - /* Backward compatibility option. In future,
> > - * the mount program should always supply
> > - * a NFSv4 minor version number.
> > - */
> > - mnt->version = 4;
> > - break;
> > - case Opt_vers_4_0:
> > - mnt->version = 4;
> > - mnt->minorversion = 0;
> > - break;
> > - case Opt_vers_4_1:
> > - mnt->version = 4;
> > - mnt->minorversion = 1;
> > - break;
> > - case Opt_vers_4_2:
> > - mnt->version = 4;
> > - mnt->minorversion = 2;
> > - break;
> > - default:
> > - return 0;
> > - }
> > - return 1;
> > -}
> > -
> > -static int nfs_get_option_str(substring_t args[], char **option)
> > -{
> > - kfree(*option);
> > - *option = match_strdup(args);
> > - return !*option;
> > -}
> > -
> > -static int nfs_get_option_ul(substring_t args[], unsigned long
> > *option)
> > -{
> > - int rc;
> > - char *string;
> > -
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - return -ENOMEM;
> > - rc = kstrtoul(string, 10, option);
> > - kfree(string);
> > -
> > - return rc;
> > -}
> > -
> > -static int nfs_get_option_ul_bound(substring_t args[], unsigned
> > long *option,
> > - unsigned long l_bound, unsigned long u_bound)
> > -{
> > - int ret;
> > -
> > - ret = nfs_get_option_ul(args, option);
> > - if (ret != 0)
> > - return ret;
> > - if (*option < l_bound || *option > u_bound)
> > - return -ERANGE;
> > - return 0;
> > -}
> > -
> > -/*
> > - * Error-check and convert a string of mount options from user
> > space into
> > - * a data structure. The whole mount string is processed; bad
> > options are
> > - * skipped as they are encountered. If there were no errors,
> > return 1;
> > - * otherwise return 0 (zero).
> > - */
> > -static int nfs_parse_mount_options(char *raw,
> > - struct nfs_parsed_mount_data *mnt)
> > -{
> > - char *p, *string;
> > - int rc, sloppy = 0, invalid_option = 0;
> > - unsigned short protofamily = AF_UNSPEC;
> > - unsigned short mountfamily = AF_UNSPEC;
> > -
> > - if (!raw) {
> > - dfprintk(MOUNT, "NFS: mount options string was
> > NULL.\n");
> > - return 1;
> > - }
> > - dfprintk(MOUNT, "NFS: nfs mount opts='%s'\n", raw);
> > -
> > - rc = security_sb_eat_lsm_opts(raw, &mnt->lsm_opts);
> > - if (rc)
> > - goto out_security_failure;
> > -
> > - while ((p = strsep(&raw, ",")) != NULL) {
> > - substring_t args[MAX_OPT_ARGS];
> > - unsigned long option;
> > - int token;
> > -
> > - if (!*p)
> > - continue;
> > -
> > - dfprintk(MOUNT, "NFS: parsing nfs mount option
> > '%s'\n", p);
> > -
> > - token = match_token(p, nfs_mount_option_tokens, args);
> > - switch (token) {
> > -
> > - /*
> > - * boolean options: foo/nofoo
> > - */
> > - case Opt_soft:
> > - mnt->flags |= NFS_MOUNT_SOFT;
> > - mnt->flags &= ~NFS_MOUNT_SOFTERR;
> > - break;
> > - case Opt_softerr:
> > - mnt->flags |= NFS_MOUNT_SOFTERR;
> > - mnt->flags &= ~NFS_MOUNT_SOFT;
> > - break;
> > - case Opt_hard:
> > - mnt->flags &=
> > ~(NFS_MOUNT_SOFT|NFS_MOUNT_SOFTERR);
> > - break;
> > - case Opt_posix:
> > - mnt->flags |= NFS_MOUNT_POSIX;
> > - break;
> > - case Opt_noposix:
> > - mnt->flags &= ~NFS_MOUNT_POSIX;
> > - break;
> > - case Opt_cto:
> > - mnt->flags &= ~NFS_MOUNT_NOCTO;
> > - break;
> > - case Opt_nocto:
> > - mnt->flags |= NFS_MOUNT_NOCTO;
> > - break;
> > - case Opt_ac:
> > - mnt->flags &= ~NFS_MOUNT_NOAC;
> > - break;
> > - case Opt_noac:
> > - mnt->flags |= NFS_MOUNT_NOAC;
> > - break;
> > - case Opt_lock:
> > - mnt->flags &= ~NFS_MOUNT_NONLM;
> > - mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
> > - NFS_MOUNT_LOCAL_FCNTL);
> > - break;
> > - case Opt_nolock:
> > - mnt->flags |= NFS_MOUNT_NONLM;
> > - mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
> > - NFS_MOUNT_LOCAL_FCNTL);
> > - break;
> > - case Opt_udp:
> > - mnt->flags &= ~NFS_MOUNT_TCP;
> > - mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP;
> > - break;
> > - case Opt_tcp:
> > - mnt->flags |= NFS_MOUNT_TCP;
> > - mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > - break;
> > - case Opt_rdma:
> > - mnt->flags |= NFS_MOUNT_TCP; /* for side
> > protocols */
> > - mnt->nfs_server.protocol = XPRT_TRANSPORT_RDMA;
> > - xprt_load_transport(p);
> > - break;
> > - case Opt_acl:
> > - mnt->flags &= ~NFS_MOUNT_NOACL;
> > - break;
> > - case Opt_noacl:
> > - mnt->flags |= NFS_MOUNT_NOACL;
> > - break;
> > - case Opt_rdirplus:
> > - mnt->flags &= ~NFS_MOUNT_NORDIRPLUS;
> > - break;
> > - case Opt_nordirplus:
> > - mnt->flags |= NFS_MOUNT_NORDIRPLUS;
> > - break;
> > - case Opt_sharecache:
> > - mnt->flags &= ~NFS_MOUNT_UNSHARED;
> > - break;
> > - case Opt_nosharecache:
> > - mnt->flags |= NFS_MOUNT_UNSHARED;
> > - break;
> > - case Opt_resvport:
> > - mnt->flags &= ~NFS_MOUNT_NORESVPORT;
> > - break;
> > - case Opt_noresvport:
> > - mnt->flags |= NFS_MOUNT_NORESVPORT;
> > - break;
> > - case Opt_fscache:
> > - mnt->options |= NFS_OPTION_FSCACHE;
> > - kfree(mnt->fscache_uniq);
> > - mnt->fscache_uniq = NULL;
> > - break;
> > - case Opt_nofscache:
> > - mnt->options &= ~NFS_OPTION_FSCACHE;
> > - kfree(mnt->fscache_uniq);
> > - mnt->fscache_uniq = NULL;
> > - break;
> > - case Opt_migration:
> > - mnt->options |= NFS_OPTION_MIGRATION;
> > - break;
> > - case Opt_nomigration:
> > - mnt->options &= ~NFS_OPTION_MIGRATION;
> > - break;
> > -
> > - /*
> > - * options that take numeric values
> > - */
> > - case Opt_port:
> > - if (nfs_get_option_ul(args, &option) ||
> > - option > USHRT_MAX)
> > - goto out_invalid_value;
> > - mnt->nfs_server.port = option;
> > - break;
> > - case Opt_rsize:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->rsize = option;
> > - break;
> > - case Opt_wsize:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->wsize = option;
> > - break;
> > - case Opt_bsize:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->bsize = option;
> > - break;
> > - case Opt_timeo:
> > - if (nfs_get_option_ul_bound(args, &option, 1,
> > INT_MAX))
> > - goto out_invalid_value;
> > - mnt->timeo = option;
> > - break;
> > - case Opt_retrans:
> > - if (nfs_get_option_ul_bound(args, &option, 0,
> > INT_MAX))
> > - goto out_invalid_value;
> > - mnt->retrans = option;
> > - break;
> > - case Opt_acregmin:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->acregmin = option;
> > - break;
> > - case Opt_acregmax:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->acregmax = option;
> > - break;
> > - case Opt_acdirmin:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->acdirmin = option;
> > - break;
> > - case Opt_acdirmax:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->acdirmax = option;
> > - break;
> > - case Opt_actimeo:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->acregmin = mnt->acregmax =
> > - mnt->acdirmin = mnt->acdirmax = option;
> > - break;
> > - case Opt_namelen:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - mnt->namlen = option;
> > - break;
> > - case Opt_mountport:
> > - if (nfs_get_option_ul(args, &option) ||
> > - option > USHRT_MAX)
> > - goto out_invalid_value;
> > - mnt->mount_server.port = option;
> > - break;
> > - case Opt_mountvers:
> > - if (nfs_get_option_ul(args, &option) ||
> > - option < NFS_MNT_VERSION ||
> > - option > NFS_MNT3_VERSION)
> > - goto out_invalid_value;
> > - mnt->mount_server.version = option;
> > - break;
> > - case Opt_minorversion:
> > - if (nfs_get_option_ul(args, &option))
> > - goto out_invalid_value;
> > - if (option > NFS4_MAX_MINOR_VERSION)
> > - goto out_invalid_value;
> > - mnt->minorversion = option;
> > - break;
> > -
> > - /*
> > - * options that take text values
> > - */
> > - case Opt_nfsvers:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - rc = nfs_parse_version_string(string, mnt,
> > args);
> > - kfree(string);
> > - if (!rc)
> > - goto out_invalid_value;
> > - break;
> > - case Opt_sec:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - rc = nfs_parse_security_flavors(string, mnt);
> > - kfree(string);
> > - if (!rc) {
> > - dfprintk(MOUNT, "NFS: unrecognized "
> > - "security flavor\n");
> > - return 0;
> > - }
> > - break;
> > - case Opt_proto:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - token = match_token(string,
> > - nfs_xprt_protocol_tokens,
> > args);
> > -
> > - protofamily = AF_INET;
> > - switch (token) {
> > - case Opt_xprt_udp6:
> > - protofamily = AF_INET6;
> > - /* fall through */
> > - case Opt_xprt_udp:
> > - mnt->flags &= ~NFS_MOUNT_TCP;
> > - mnt->nfs_server.protocol =
> > XPRT_TRANSPORT_UDP;
> > - break;
> > - case Opt_xprt_tcp6:
> > - protofamily = AF_INET6;
> > - /* fall through */
> > - case Opt_xprt_tcp:
> > - mnt->flags |= NFS_MOUNT_TCP;
> > - mnt->nfs_server.protocol =
> > XPRT_TRANSPORT_TCP;
> > - break;
> > - case Opt_xprt_rdma6:
> > - protofamily = AF_INET6;
> > - /* fall through */
> > - case Opt_xprt_rdma:
> > - /* vector side protocols to TCP */
> > - mnt->flags |= NFS_MOUNT_TCP;
> > - mnt->nfs_server.protocol =
> > XPRT_TRANSPORT_RDMA;
> > - xprt_load_transport(string);
> > - break;
> > - default:
> > - dfprintk(MOUNT, "NFS: unrecognized "
> > - "transport
> > protocol\n");
> > - kfree(string);
> > - return 0;
> > - }
> > - kfree(string);
> > - break;
> > - case Opt_mountproto:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - token = match_token(string,
> > - nfs_xprt_protocol_tokens,
> > args);
> > - kfree(string);
> > -
> > - mountfamily = AF_INET;
> > - switch (token) {
> > - case Opt_xprt_udp6:
> > - mountfamily = AF_INET6;
> > - /* fall through */
> > - case Opt_xprt_udp:
> > - mnt->mount_server.protocol =
> > XPRT_TRANSPORT_UDP;
> > - break;
> > - case Opt_xprt_tcp6:
> > - mountfamily = AF_INET6;
> > - /* fall through */
> > - case Opt_xprt_tcp:
> > - mnt->mount_server.protocol =
> > XPRT_TRANSPORT_TCP;
> > - break;
> > - case Opt_xprt_rdma: /* not used for side
> > protocols */
> > - default:
> > - dfprintk(MOUNT, "NFS: unrecognized "
> > - "transport
> > protocol\n");
> > - return 0;
> > - }
> > - break;
> > - case Opt_addr:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - mnt->nfs_server.addrlen =
> > - rpc_pton(mnt->net, string,
> > strlen(string),
> > - (struct sockaddr *)
> > - &mnt->nfs_server.address,
> > - sizeof(mnt-
> > >nfs_server.address));
> > - kfree(string);
> > - if (mnt->nfs_server.addrlen == 0)
> > - goto out_invalid_address;
> > - break;
> > - case Opt_clientaddr:
> > - if (nfs_get_option_str(args, &mnt-
> > >client_address))
> > - goto out_nomem;
> > - break;
> > - case Opt_mounthost:
> > - if (nfs_get_option_str(args,
> > - &mnt-
> > >mount_server.hostname))
> > - goto out_nomem;
> > - break;
> > - case Opt_mountaddr:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - mnt->mount_server.addrlen =
> > - rpc_pton(mnt->net, string,
> > strlen(string),
> > - (struct sockaddr *)
> > - &mnt->mount_server.address,
> > - sizeof(mnt-
> > >mount_server.address));
> > - kfree(string);
> > - if (mnt->mount_server.addrlen == 0)
> > - goto out_invalid_address;
> > - break;
> > - case Opt_nconnect:
> > - if (nfs_get_option_ul_bound(args, &option, 1,
> > NFS_MAX_CONNECTIONS))
> > - goto out_invalid_value;
> > - mnt->nfs_server.nconnect = option;
> > - break;
> > - case Opt_lookupcache:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - token = match_token(string,
> > - nfs_lookupcache_tokens, args);
> > - kfree(string);
> > - switch (token) {
> > - case Opt_lookupcache_all:
> > - mnt->flags &=
> > ~(NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE);
> > - break;
> > - case Opt_lookupcache_positive:
> > - mnt->flags &=
> > ~NFS_MOUNT_LOOKUP_CACHE_NONE;
> > - mnt->flags |=
> > NFS_MOUNT_LOOKUP_CACHE_NONEG;
> > - break;
> > - case Opt_lookupcache_none:
> > - mnt->flags |=
> > NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE;
> > - break;
> > - default:
> > - dfprintk(MOUNT, "NFS: invalid
> > "
> > - "lookupcache
> > argument\n");
> > - return 0;
> > - };
> > - break;
> > - case Opt_fscache_uniq:
> > - if (nfs_get_option_str(args, &mnt-
> > >fscache_uniq))
> > - goto out_nomem;
> > - mnt->options |= NFS_OPTION_FSCACHE;
> > - break;
> > - case Opt_local_lock:
> > - string = match_strdup(args);
> > - if (string == NULL)
> > - goto out_nomem;
> > - token = match_token(string,
> > nfs_local_lock_tokens,
> > - args);
> > - kfree(string);
> > - switch (token) {
> > - case Opt_local_lock_all:
> > - mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
> > - NFS_MOUNT_LOCAL_FCNTL);
> > - break;
> > - case Opt_local_lock_flock:
> > - mnt->flags |= NFS_MOUNT_LOCAL_FLOCK;
> > - break;
> > - case Opt_local_lock_posix:
> > - mnt->flags |= NFS_MOUNT_LOCAL_FCNTL;
> > - break;
> > - case Opt_local_lock_none:
> > - mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
> > - NFS_MOUNT_LOCAL_FCNTL);
> > - break;
> > - default:
> > - dfprintk(MOUNT, "NFS: invalid "
> > - "local_lock
> > argument\n");
> > - return 0;
> > - };
> > - break;
> > -
> > - /*
> > - * Special options
> > - */
> > - case Opt_sloppy:
> > - sloppy = 1;
> > - dfprintk(MOUNT, "NFS: relaxing parsing
> > rules\n");
> > - break;
> > - case Opt_userspace:
> > - case Opt_deprecated:
> > - dfprintk(MOUNT, "NFS: ignoring mount option "
> > - "'%s'\n", p);
> > - break;
> > -
> > - default:
> > - invalid_option = 1;
> > - dfprintk(MOUNT, "NFS: unrecognized mount
> > option "
> > - "'%s'\n", p);
> > - }
> > - }
> > -
> > - if (!sloppy && invalid_option)
> > - return 0;
> > -
> > - if (mnt->minorversion && mnt->version != 4)
> > - goto out_minorversion_mismatch;
> > -
> > - if (mnt->options & NFS_OPTION_MIGRATION &&
> > - (mnt->version != 4 || mnt->minorversion != 0))
> > - goto out_migration_misuse;
> > -
> > - /*
> > - * verify that any proto=/mountproto= options match the address
> > - * families in the addr=/mountaddr= options.
> > - */
> > - if (protofamily != AF_UNSPEC &&
> > - protofamily != mnt->nfs_server.address.ss_family)
> > - goto out_proto_mismatch;
> > -
> > - if (mountfamily != AF_UNSPEC) {
> > - if (mnt->mount_server.addrlen) {
> > - if (mountfamily != mnt-
> > >mount_server.address.ss_family)
> > - goto out_mountproto_mismatch;
> > - } else {
> > - if (mountfamily != mnt-
> > >nfs_server.address.ss_family)
> > - goto out_mountproto_mismatch;
> > - }
> > - }
> > -
> > - return 1;
> > -
> > -out_mountproto_mismatch:
> > - printk(KERN_INFO "NFS: mount server address does not match
> > mountproto= "
> > - "option\n");
> > - return 0;
> > -out_proto_mismatch:
> > - printk(KERN_INFO "NFS: server address does not match proto=
> > option\n");
> > - return 0;
> > -out_invalid_address:
> > - printk(KERN_INFO "NFS: bad IP address specified: %s\n", p);
> > - return 0;
> > -out_invalid_value:
> > - printk(KERN_INFO "NFS: bad mount option value specified: %s\n",
> > p);
> > - return 0;
> > -out_minorversion_mismatch:
> > - printk(KERN_INFO "NFS: mount option vers=%u does not support "
> > - "minorversion=%u\n", mnt->version, mnt-
> > >minorversion);
> > - return 0;
> > -out_migration_misuse:
> > - printk(KERN_INFO
> > - "NFS: 'migration' not supported for this NFS
> > version\n");
> > - return 0;
> > -out_nomem:
> > - printk(KERN_INFO "NFS: not enough memory to parse option\n");
> > - return 0;
> > -out_security_failure:
> > - printk(KERN_INFO "NFS: security options invalid: %d\n", rc);
> > - return 0;
> > -}
> > -
> > /*
> > * Ensure that a specified authtype in args->auth_info is supported
> > by
> > * the server. Returns 0 and sets args->selected_flavor if it's ok,
> > and
> > @@ -1908,327 +925,6 @@ struct dentry *nfs_try_mount(int flags,
> > const char *dev_name,
> > }
> > EXPORT_SYMBOL_GPL(nfs_try_mount);
> >
> > -/*
> > - * Split "dev_name" into "hostname:export_path".
> > - *
> > - * The leftmost colon demarks the split between the server's
> > hostname
> > - * and the export path. If the hostname starts with a left square
> > - * bracket, then it may contain colons.
> > - *
> > - * Note: caller frees hostname and export path, even on error.
> > - */
> > -static int nfs_parse_devname(const char *dev_name,
> > - char **hostname, size_t maxnamlen,
> > - char **export_path, size_t maxpathlen)
> > -{
> > - size_t len;
> > - char *end;
> > -
> > - if (unlikely(!dev_name || !*dev_name)) {
> > - dfprintk(MOUNT, "NFS: device name not specified\n");
> > - return -EINVAL;
> > - }
> > -
> > - /* Is the host name protected with square brakcets? */
> > - if (*dev_name == '[') {
> > - end = strchr(++dev_name, ']');
> > - if (end == NULL || end[1] != ':')
> > - goto out_bad_devname;
> > -
> > - len = end - dev_name;
> > - end++;
> > - } else {
> > - char *comma;
> > -
> > - end = strchr(dev_name, ':');
> > - if (end == NULL)
> > - goto out_bad_devname;
> > - len = end - dev_name;
> > -
> > - /* kill possible hostname list: not supported */
> > - comma = strchr(dev_name, ',');
> > - if (comma != NULL && comma < end)
> > - len = comma - dev_name;
> > - }
> > -
> > - if (len > maxnamlen)
> > - goto out_hostname;
> > -
> > - /* N.B. caller will free nfs_server.hostname in all cases */
> > - *hostname = kstrndup(dev_name, len, GFP_KERNEL);
> > - if (*hostname == NULL)
> > - goto out_nomem;
> > - len = strlen(++end);
> > - if (len > maxpathlen)
> > - goto out_path;
> > - *export_path = kstrndup(end, len, GFP_KERNEL);
> > - if (!*export_path)
> > - goto out_nomem;
> > -
> > - dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", *export_path);
> > - return 0;
> > -
> > -out_bad_devname:
> > - dfprintk(MOUNT, "NFS: device name not in host:path format\n");
> > - return -EINVAL;
> > -
> > -out_nomem:
> > - dfprintk(MOUNT, "NFS: not enough memory to parse device
> > name\n");
> > - return -ENOMEM;
> > -
> > -out_hostname:
> > - dfprintk(MOUNT, "NFS: server hostname too long\n");
> > - return -ENAMETOOLONG;
> > -
> > -out_path:
> > - dfprintk(MOUNT, "NFS: export pathname too long\n");
> > - return -ENAMETOOLONG;
> > -}
> > -
> > -/*
> > - * Validate the NFS2/NFS3 mount data
> > - * - fills in the mount root filehandle
> > - *
> > - * For option strings, user space handles the following behaviors:
> > - *
> > - * + DNS: mapping server host name to IP address ("addr=" option)
> > - *
> > - * + failure mode: how to behave if a mount request can't be
> > handled
> > - * immediately ("fg/bg" option)
> > - *
> > - * + retry: how often to retry a mount request ("retry=" option)
> > - *
> > - * + breaking back: trying proto=udp after proto=tcp, v2 after v3,
> > - * mountproto=tcp after mountproto=udp, and so on
> > - */
> > -static int nfs23_validate_mount_data(void *options,
> > - struct nfs_parsed_mount_data
> > *args,
> > - struct nfs_fh *mntfh,
> > - const char *dev_name)
> > -{
> > - struct nfs_mount_data *data = (struct nfs_mount_data *)options;
> > - struct sockaddr *sap = (struct sockaddr *)&args-
> > >nfs_server.address;
> > - int extra_flags = NFS_MOUNT_LEGACY_INTERFACE;
> > -
> > - if (data == NULL)
> > - goto out_no_data;
> > -
> > - args->version = NFS_DEFAULT_VERSION;
> > - switch (data->version) {
> > - case 1:
> > - data->namlen = 0; /* fall through */
> > - case 2:
> > - data->bsize = 0; /* fall through */
> > - case 3:
> > - if (data->flags & NFS_MOUNT_VER3)
> > - goto out_no_v3;
> > - data->root.size = NFS2_FHSIZE;
> > - memcpy(data->root.data, data->old_root.data,
> > NFS2_FHSIZE);
> > - /* Turn off security negotiation */
> > - extra_flags |= NFS_MOUNT_SECFLAVOUR;
> > - /* fall through */
> > - case 4:
> > - if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > - goto out_no_sec;
> > - /* fall through */
> > - case 5:
> > - memset(data->context, 0, sizeof(data->context));
> > - /* fall through */
> > - case 6:
> > - if (data->flags & NFS_MOUNT_VER3) {
> > - if (data->root.size > NFS3_FHSIZE || data-
> > >root.size == 0)
> > - goto out_invalid_fh;
> > - mntfh->size = data->root.size;
> > - args->version = 3;
> > - } else {
> > - mntfh->size = NFS2_FHSIZE;
> > - args->version = 2;
> > - }
> > -
> > -
> > - memcpy(mntfh->data, data->root.data, mntfh->size);
> > - if (mntfh->size < sizeof(mntfh->data))
> > - memset(mntfh->data + mntfh->size, 0,
> > - sizeof(mntfh->data) - mntfh->size);
> > -
> > - /*
> > - * Translate to nfs_parsed_mount_data, which
> > nfs_fill_super
> > - * can deal with.
> > - */
> > - args->flags = data->flags & NFS_MOUNT_FLAGMASK;
> > - args->flags |= extra_flags;
> > - args->rsize = data->rsize;
> > - args->wsize = data->wsize;
> > - args->timeo = data->timeo;
> > - args->retrans = data->retrans;
> > - args->acregmin = data->acregmin;
> > - args->acregmax = data->acregmax;
> > - args->acdirmin = data->acdirmin;
> > - args->acdirmax = data->acdirmax;
> > - args->need_mount = false;
> > -
> > - memcpy(sap, &data->addr, sizeof(data->addr));
> > - args->nfs_server.addrlen = sizeof(data->addr);
> > - args->nfs_server.port = ntohs(data->addr.sin_port);
> > - if (sap->sa_family != AF_INET ||
> > - !nfs_verify_server_address(sap))
> > - goto out_no_address;
> > -
> > - if (!(data->flags & NFS_MOUNT_TCP))
> > - args->nfs_server.protocol = XPRT_TRANSPORT_UDP;
> > - /* N.B. caller will free nfs_server.hostname in all
> > cases */
> > - args->nfs_server.hostname = kstrdup(data->hostname,
> > GFP_KERNEL);
> > - args->namlen = data->namlen;
> > - args->bsize = data->bsize;
> > -
> > - if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > - args->selected_flavor = data->pseudoflavor;
> > - else
> > - args->selected_flavor = RPC_AUTH_UNIX;
> > - if (!args->nfs_server.hostname)
> > - goto out_nomem;
> > -
> > - if (!(data->flags & NFS_MOUNT_NONLM))
> > - args->flags &= ~(NFS_MOUNT_LOCAL_FLOCK|
> > - NFS_MOUNT_LOCAL_FCNTL);
> > - else
> > - args->flags |= (NFS_MOUNT_LOCAL_FLOCK|
> > - NFS_MOUNT_LOCAL_FCNTL);
> > - /*
> > - * The legacy version 6 binary mount data from
> > userspace has a
> > - * field used only to transport selinux information
> > into the
> > - * the kernel. To continue to support that
> > functionality we
> > - * have a touch of selinux knowledge here in the NFS
> > code. The
> > - * userspace code converted context=blah to just blah
> > so we are
> > - * converting back to the full string selinux
> > understands.
> > - */
> > - if (data->context[0]){
> > -#ifdef CONFIG_SECURITY_SELINUX
> > - int rc;
> > - data->context[NFS_MAX_CONTEXT_LEN] = '\0';
> > - rc = security_add_mnt_opt("context", data-
> > >context,
> > - strlen(data->context), &args-
> > >lsm_opts);
> > - if (rc)
> > - return rc;
> > -#else
> > - return -EINVAL;
> > -#endif
> > - }
> > -
> > - break;
> > - default:
> > - return NFS_TEXT_DATA;
> > - }
> > -
> > - return 0;
> > -
> > -out_no_data:
> > - dfprintk(MOUNT, "NFS: mount program didn't pass any mount
> > data\n");
> > - return -EINVAL;
> > -
> > -out_no_v3:
> > - dfprintk(MOUNT, "NFS: nfs_mount_data version %d does not
> > support v3\n",
> > - data->version);
> > - return -EINVAL;
> > -
> > -out_no_sec:
> > - dfprintk(MOUNT, "NFS: nfs_mount_data version supports only
> > AUTH_SYS\n");
> > - return -EINVAL;
> > -
> > -out_nomem:
> > - dfprintk(MOUNT, "NFS: not enough memory to handle mount
> > options\n");
> > - return -ENOMEM;
> > -
> > -out_no_address:
> > - dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > address\n");
> > - return -EINVAL;
> > -
> > -out_invalid_fh:
> > - dfprintk(MOUNT, "NFS: invalid root filehandle\n");
> > - return -EINVAL;
> > -}
> > -
> > -#if IS_ENABLED(CONFIG_NFS_V4)
> > -static int nfs_validate_mount_data(struct file_system_type
> > *fs_type,
> > - void *options,
> > - struct nfs_parsed_mount_data *args,
> > - struct nfs_fh *mntfh,
> > - const char *dev_name)
> > -{
> > - if (fs_type == &nfs_fs_type)
> > - return nfs23_validate_mount_data(options, args, mntfh,
> > dev_name);
> > - return nfs4_validate_mount_data(options, args, dev_name);
> > -}
> > -#else
> > -static int nfs_validate_mount_data(struct file_system_type
> > *fs_type,
> > - void *options,
> > - struct nfs_parsed_mount_data *args,
> > - struct nfs_fh *mntfh,
> > - const char *dev_name)
> > -{
> > - return nfs23_validate_mount_data(options, args, mntfh,
> > dev_name);
> > -}
> > -#endif
> > -
> > -static int nfs_validate_text_mount_data(void *options,
> > - struct nfs_parsed_mount_data
> > *args,
> > - const char *dev_name)
> > -{
> > - int port = 0;
> > - int max_namelen = PAGE_SIZE;
> > - int max_pathlen = NFS_MAXPATHLEN;
> > - struct sockaddr *sap = (struct sockaddr *)&args-
> > >nfs_server.address;
> > -
> > - if (nfs_parse_mount_options((char *)options, args) == 0)
> > - return -EINVAL;
> > -
> > - if (!nfs_verify_server_address(sap))
> > - goto out_no_address;
> > -
> > - if (args->version == 4) {
> > -#if IS_ENABLED(CONFIG_NFS_V4)
> > - if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
> > - port = NFS_RDMA_PORT;
> > - else
> > - port = NFS_PORT;
> > - max_namelen = NFS4_MAXNAMLEN;
> > - max_pathlen = NFS4_MAXPATHLEN;
> > - nfs_validate_transport_protocol(args);
> > - if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
> > - goto out_invalid_transport_udp;
> > - nfs4_validate_mount_flags(args);
> > -#else
> > - goto out_v4_not_compiled;
> > -#endif /* CONFIG_NFS_V4 */
> > - } else {
> > - nfs_set_mount_transport_protocol(args);
> > - if (args->nfs_server.protocol == XPRT_TRANSPORT_RDMA)
> > - port = NFS_RDMA_PORT;
> > - }
> > -
> > - nfs_set_port(sap, &args->nfs_server.port, port);
> > -
> > - return nfs_parse_devname(dev_name,
> > - &args->nfs_server.hostname,
> > - max_namelen,
> > - &args->nfs_server.export_path,
> > - max_pathlen);
> > -
> > -#if !IS_ENABLED(CONFIG_NFS_V4)
> > -out_v4_not_compiled:
> > - dfprintk(MOUNT, "NFS: NFSv4 is not compiled into kernel\n");
> > - return -EPROTONOSUPPORT;
> > -#else
> > -out_invalid_transport_udp:
> > - dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
> > - return -EINVAL;
> > -#endif /* !CONFIG_NFS_V4 */
> > -
> > -out_no_address:
> > - dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > address\n");
> > - return -EINVAL;
> > -}
> > -
> > #define NFS_REMOUNT_CMP_FLAGMASK ~(NFS_MOUNT_INTR \
> > | NFS_MOUNT_SECURE \
> > | NFS_MOUNT_TCP \
> > @@ -2719,113 +1415,6 @@ nfs_prepared_mount(struct file_system_type
> > *fs_type, int flags,
> >
> > #if IS_ENABLED(CONFIG_NFS_V4)
> >
> > -static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data
> > *args)
> > -{
> > - args->flags &=
> > ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3|
> > - NFS_MOUNT_LOCAL_FLOCK|NFS_MOUNT_LOCAL_FCNTL);
> > -}
> > -
> > -/*
> > - * Validate NFSv4 mount options
> > - */
> > -static int nfs4_validate_mount_data(void *options,
> > - struct nfs_parsed_mount_data *args,
> > - const char *dev_name)
> > -{
> > - struct sockaddr *sap = (struct sockaddr *)&args-
> > >nfs_server.address;
> > - struct nfs4_mount_data *data = (struct nfs4_mount_data
> > *)options;
> > - char *c;
> > -
> > - if (data == NULL)
> > - goto out_no_data;
> > -
> > - args->version = 4;
> > -
> > - switch (data->version) {
> > - case 1:
> > - if (data->host_addrlen > sizeof(args-
> > >nfs_server.address))
> > - goto out_no_address;
> > - if (data->host_addrlen == 0)
> > - goto out_no_address;
> > - args->nfs_server.addrlen = data->host_addrlen;
> > - if (copy_from_user(sap, data->host_addr, data-
> > >host_addrlen))
> > - return -EFAULT;
> > - if (!nfs_verify_server_address(sap))
> > - goto out_no_address;
> > - args->nfs_server.port = ntohs(((struct sockaddr_in
> > *)sap)->sin_port);
> > -
> > - if (data->auth_flavourlen) {
> > - rpc_authflavor_t pseudoflavor;
> > - if (data->auth_flavourlen > 1)
> > - goto out_inval_auth;
> > - if (copy_from_user(&pseudoflavor,
> > - data->auth_flavours,
> > - sizeof(pseudoflavor)))
> > - return -EFAULT;
> > - args->selected_flavor = pseudoflavor;
> > - } else
> > - args->selected_flavor = RPC_AUTH_UNIX;
> > -
> > - c = strndup_user(data->hostname.data, NFS4_MAXNAMLEN);
> > - if (IS_ERR(c))
> > - return PTR_ERR(c);
> > - args->nfs_server.hostname = c;
> > -
> > - c = strndup_user(data->mnt_path.data, NFS4_MAXPATHLEN);
> > - if (IS_ERR(c))
> > - return PTR_ERR(c);
> > - args->nfs_server.export_path = c;
> > - dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", c);
> > -
> > - c = strndup_user(data->client_addr.data, 16);
> > - if (IS_ERR(c))
> > - return PTR_ERR(c);
> > - args->client_address = c;
> > -
> > - /*
> > - * Translate to nfs_parsed_mount_data, which
> > nfs4_fill_super
> > - * can deal with.
> > - */
> > -
> > - args->flags = data->flags & NFS4_MOUNT_FLAGMASK;
> > - args->rsize = data->rsize;
> > - args->wsize = data->wsize;
> > - args->timeo = data->timeo;
> > - args->retrans = data->retrans;
> > - args->acregmin = data->acregmin;
> > - args->acregmax = data->acregmax;
> > - args->acdirmin = data->acdirmin;
> > - args->acdirmax = data->acdirmax;
> > - args->nfs_server.protocol = data->proto;
> > - nfs_validate_transport_protocol(args);
> > - if (args->nfs_server.protocol == XPRT_TRANSPORT_UDP)
> > - goto out_invalid_transport_udp;
> > -
> > - break;
> > - default:
> > - return NFS_TEXT_DATA;
> > - }
> > -
> > - return 0;
> > -
> > -out_no_data:
> > - dfprintk(MOUNT, "NFS4: mount program didn't pass any mount
> > data\n");
> > - return -EINVAL;
> > -
> > -out_inval_auth:
> > - dfprintk(MOUNT, "NFS4: Invalid number of RPC auth flavours
> > %d\n",
> > - data->auth_flavourlen);
> > - return -EINVAL;
> > -
> > -out_no_address:
> > - dfprintk(MOUNT, "NFS4: mount program didn't pass remote
> > address\n");
> > - return -EINVAL;
> > -
> > -out_invalid_transport_udp:
> > - dfprintk(MOUNT, "NFSv4: Unsupported transport protocol udp\n");
> > - return -EINVAL;
> > -}
> > -
> > /*
> > * NFS v4 module parameters need to stay in the
> > * NFS client for backwards compatibility
> > --
> > 2.17.2
> >
>
> --
> Chuck Lever
> [email protected]
>
>
>
--
Trond Myklebust
Linux NFS client maintainer, Hammerspace
[email protected]
On Thu, 2019-09-12 at 13:36 -0400, Scott Mayhew wrote:
> On Wed, 11 Sep 2019, Trond Myklebust wrote:
>
> > On Wed, 2019-09-11 at 14:24 -0400, Chuck Lever wrote:
> > > > On Sep 11, 2019, at 12:16 PM, Scott Mayhew <[email protected]>
> > > > wrote:
> > > >
> > > > From: David Howells <[email protected]>
> > > >
> > > > Split various bits relating to mount parameterisation out from
> > > > fs/nfs/super.c into their own file to form the basis of
> > > > filesystem
> > > > context
> > > > handling for NFS.
> > > >
> > > > No other changes are made to the code beyond removing 'static'
> > > > qualifiers.
> > > >
> > > > Signed-off-by: David Howells <[email protected]>
> > > > Signed-off-by: Al Viro <[email protected]>
> > > > ---
> > > > fs/nfs/Makefile | 2 +-
> > > > fs/nfs/fs_context.c | 1418
> > > > +++++++++++++++++++++++++++++++++++++++++++
> > > > fs/nfs/internal.h | 29 +
> > > > fs/nfs/super.c | 1411 ------------------------------------
> > > > ----
> > > > --
> > > > 4 files changed, 1448 insertions(+), 1412 deletions(-)
> > > > create mode 100644 fs/nfs/fs_context.c
> > > >
> > > > diff --git a/fs/nfs/Makefile b/fs/nfs/Makefile
> > > > index 34cdeaecccf6..2433c3e03cfa 100644
> > > > --- a/fs/nfs/Makefile
> > > > +++ b/fs/nfs/Makefile
> > > > @@ -9,7 +9,7 @@ CFLAGS_nfstrace.o += -I$(src)
> > > > nfs-y := client.o dir.o file.o getroot.o
> > > > inode.o super.o \
> > > > io.o direct.o pagelist.o read.o
> > > > symlink.o
> > > > unlink.o \
> > > > write.o namespace.o mount_clnt.o
> > > > nfstrace.o
> > > > \
> > > > - export.o sysfs.o
> > > > + export.o sysfs.o fs_context.o
> > > > nfs-$(CONFIG_ROOT_NFS) += nfsroot.o
> > > > nfs-$(CONFIG_SYSCTL) += sysctl.o
> > > > nfs-$(CONFIG_NFS_FSCACHE) += fscache.o fscache-index.o
> > > > diff --git a/fs/nfs/fs_context.c b/fs/nfs/fs_context.c
> > > > new file mode 100644
> > > > index 000000000000..82b312a5cdde
> > > > --- /dev/null
> > > > +++ b/fs/nfs/fs_context.c
> > > > @@ -0,0 +1,1418 @@
> > > > +/* NFS mount handling.
> > > > + *
> > > > + * Copyright (C) 2018 Red Hat, Inc. All Rights Reserved.
> > > > + * Written by David Howells ([email protected])
> > > > + *
> > > > + * Split from fs/nfs/super.c:
> > > > + *
> > > > + * Copyright (C) 1992 Rick Sladkey
> > > > + *
> > > > + * This program is free software; you can redistribute it
> > > > and/or
> > > > + * modify it under the terms of the GNU General Public Licence
> > > > + * as published by the Free Software Foundation; either
> > > > version
> > > > + * 2 of the Licence, or (at your option) any later version.
> > > > + */
> > >
> > > New source files should have an SPDX tag instead of boilerplate.
> > > I suggest:
> > >
> > > // SPDX-License-Identifier: GPL-2.0-only
> > >
> >
> > Agreed. It is also quite a long stretch to claim authorship of the
> > entire file as implied above. Given that this is mostly a copy-
> > paste
> > effort, then most of the actual copyrights belong to the people
> > who've
> > contributed to super.c (and to inode.c before it). David is one of
> > those authors, but he is one of many.
>
> Okay, how about:
>
> // SPDX-License-Identifier: GPL-2.0-only
> /*
> * linux/fs/nfs/fs_context.c
> *
> * Copyright (C) 1992 Rick Sladkey
> *
> * NFS mount handling.
> *
> * Split from fs/nfs/super.c by David Howells <[email protected]>
> */
>
> and have patch 25/26 add a line
>
> * Conversion to new mount api Copyright (C) David Howells
That looks more reasonable to me.
>
> -Scott
> >
> > > > +#include <linux/parser.h>
> > > > +#include <linux/nfs_fs.h>
> > > > +#include <linux/nfs_mount.h>
> > > > +#include <linux/nfs4_mount.h>
> > > > +#include "nfs.h"
> > > > +#include "internal.h"
> > > > +
> > > > +#define NFSDBG_FACILITY NFSDBG_MOUNT
> > > > +
> > > > +#if IS_ENABLED(CONFIG_NFS_V3)
> > > > +#define NFS_DEFAULT_VERSION 3
> > > > +#else
> > > > +#define NFS_DEFAULT_VERSION 2
> > > > +#endif
> > > > +
> > > > +#define NFS_MAX_CONNECTIONS 16
> > > > +
> > > > +enum {
> > > > + /* Mount options that take no arguments */
> > > > + Opt_soft, Opt_softerr, Opt_hard,
> > > > + Opt_posix, Opt_noposix,
> > > > + Opt_cto, Opt_nocto,
> > > > + Opt_ac, Opt_noac,
> > > > + Opt_lock, Opt_nolock,
> > > > + Opt_udp, Opt_tcp, Opt_rdma,
> > > > + Opt_acl, Opt_noacl,
> > > > + Opt_rdirplus, Opt_nordirplus,
> > > > + Opt_sharecache, Opt_nosharecache,
> > > > + Opt_resvport, Opt_noresvport,
> > > > + Opt_fscache, Opt_nofscache,
> > > > + Opt_migration, Opt_nomigration,
> > > > +
> > > > + /* Mount options that take integer arguments */
> > > > + Opt_port,
> > > > + Opt_rsize, Opt_wsize, Opt_bsize,
> > > > + Opt_timeo, Opt_retrans,
> > > > + Opt_acregmin, Opt_acregmax,
> > > > + Opt_acdirmin, Opt_acdirmax,
> > > > + Opt_actimeo,
> > > > + Opt_namelen,
> > > > + Opt_mountport,
> > > > + Opt_mountvers,
> > > > + Opt_minorversion,
> > > > +
> > > > + /* Mount options that take string arguments */
> > > > + Opt_nfsvers,
> > > > + Opt_sec, Opt_proto, Opt_mountproto, Opt_mounthost,
> > > > + Opt_addr, Opt_mountaddr, Opt_clientaddr,
> > > > + Opt_nconnect,
> > > > + Opt_lookupcache,
> > > > + Opt_fscache_uniq,
> > > > + Opt_local_lock,
> > > > +
> > > > + /* Special mount options */
> > > > + Opt_userspace, Opt_deprecated, Opt_sloppy,
> > > > +
> > > > + Opt_err
> > > > +};
> > > > +
> > > > +static const match_table_t nfs_mount_option_tokens = {
> > > > + { Opt_userspace, "bg" },
> > > > + { Opt_userspace, "fg" },
> > > > + { Opt_userspace, "retry=%s" },
> > > > +
> > > > + { Opt_sloppy, "sloppy" },
> > > > +
> > > > + { Opt_soft, "soft" },
> > > > + { Opt_softerr, "softerr" },
> > > > + { Opt_hard, "hard" },
> > > > + { Opt_deprecated, "intr" },
> > > > + { Opt_deprecated, "nointr" },
> > > > + { Opt_posix, "posix" },
> > > > + { Opt_noposix, "noposix" },
> > > > + { Opt_cto, "cto" },
> > > > + { Opt_nocto, "nocto" },
> > > > + { Opt_ac, "ac" },
> > > > + { Opt_noac, "noac" },
> > > > + { Opt_lock, "lock" },
> > > > + { Opt_nolock, "nolock" },
> > > > + { Opt_udp, "udp" },
> > > > + { Opt_tcp, "tcp" },
> > > > + { Opt_rdma, "rdma" },
> > > > + { Opt_acl, "acl" },
> > > > + { Opt_noacl, "noacl" },
> > > > + { Opt_rdirplus, "rdirplus" },
> > > > + { Opt_nordirplus, "nordirplus" },
> > > > + { Opt_sharecache, "sharecache" },
> > > > + { Opt_nosharecache, "nosharecache" },
> > > > + { Opt_resvport, "resvport" },
> > > > + { Opt_noresvport, "noresvport" },
> > > > + { Opt_fscache, "fsc" },
> > > > + { Opt_nofscache, "nofsc" },
> > > > + { Opt_migration, "migration" },
> > > > + { Opt_nomigration, "nomigration" },
> > > > +
> > > > + { Opt_port, "port=%s" },
> > > > + { Opt_rsize, "rsize=%s" },
> > > > + { Opt_wsize, "wsize=%s" },
> > > > + { Opt_bsize, "bsize=%s" },
> > > > + { Opt_timeo, "timeo=%s" },
> > > > + { Opt_retrans, "retrans=%s" },
> > > > + { Opt_acregmin, "acregmin=%s" },
> > > > + { Opt_acregmax, "acregmax=%s" },
> > > > + { Opt_acdirmin, "acdirmin=%s" },
> > > > + { Opt_acdirmax, "acdirmax=%s" },
> > > > + { Opt_actimeo, "actimeo=%s" },
> > > > + { Opt_namelen, "namlen=%s" },
> > > > + { Opt_mountport, "mountport=%s" },
> > > > + { Opt_mountvers, "mountvers=%s" },
> > > > + { Opt_minorversion, "minorversion=%s" },
> > > > +
> > > > + { Opt_nfsvers, "nfsvers=%s" },
> > > > + { Opt_nfsvers, "vers=%s" },
> > > > +
> > > > + { Opt_sec, "sec=%s" },
> > > > + { Opt_proto, "proto=%s" },
> > > > + { Opt_mountproto, "mountproto=%s" },
> > > > + { Opt_addr, "addr=%s" },
> > > > + { Opt_clientaddr, "clientaddr=%s" },
> > > > + { Opt_mounthost, "mounthost=%s" },
> > > > + { Opt_mountaddr, "mountaddr=%s" },
> > > > +
> > > > + { Opt_nconnect, "nconnect=%s" },
> > > > +
> > > > + { Opt_lookupcache, "lookupcache=%s" },
> > > > + { Opt_fscache_uniq, "fsc=%s" },
> > > > + { Opt_local_lock, "local_lock=%s" },
> > > > +
> > > > + /* The following needs to be listed after all other
> > > > options */
> > > > + { Opt_nfsvers, "v%s" },
> > > > +
> > > > + { Opt_err, NULL }
> > > > +};
> > > > +
> > > > +enum {
> > > > + Opt_xprt_udp, Opt_xprt_udp6, Opt_xprt_tcp,
> > > > Opt_xprt_tcp6,
> > > > Opt_xprt_rdma,
> > > > + Opt_xprt_rdma6,
> > > > +
> > > > + Opt_xprt_err
> > > > +};
> > > > +
> > > > +static const match_table_t nfs_xprt_protocol_tokens = {
> > > > + { Opt_xprt_udp, "udp" },
> > > > + { Opt_xprt_udp6, "udp6" },
> > > > + { Opt_xprt_tcp, "tcp" },
> > > > + { Opt_xprt_tcp6, "tcp6" },
> > > > + { Opt_xprt_rdma, "rdma" },
> > > > + { Opt_xprt_rdma6, "rdma6" },
> > > > +
> > > > + { Opt_xprt_err, NULL }
> > > > +};
> > > > +
> > > > +enum {
> > > > + Opt_sec_none, Opt_sec_sys,
> > > > + Opt_sec_krb5, Opt_sec_krb5i, Opt_sec_krb5p,
> > > > + Opt_sec_lkey, Opt_sec_lkeyi, Opt_sec_lkeyp,
> > > > + Opt_sec_spkm, Opt_sec_spkmi, Opt_sec_spkmp,
> > > > +
> > > > + Opt_sec_err
> > > > +};
> > > > +
> > > > +static const match_table_t nfs_secflavor_tokens = {
> > > > + { Opt_sec_none, "none" },
> > > > + { Opt_sec_none, "null" },
> > > > + { Opt_sec_sys, "sys" },
> > > > +
> > > > + { Opt_sec_krb5, "krb5" },
> > > > + { Opt_sec_krb5i, "krb5i" },
> > > > + { Opt_sec_krb5p, "krb5p" },
> > > > +
> > > > + { Opt_sec_lkey, "lkey" },
> > > > + { Opt_sec_lkeyi, "lkeyi" },
> > > > + { Opt_sec_lkeyp, "lkeyp" },
> > > > +
> > > > + { Opt_sec_spkm, "spkm3" },
> > > > + { Opt_sec_spkmi, "spkm3i" },
> > > > + { Opt_sec_spkmp, "spkm3p" },
> > > > +
> > > > + { Opt_sec_err, NULL }
> > > > +};
> > > > +
> > > > +enum {
> > > > + Opt_lookupcache_all, Opt_lookupcache_positive,
> > > > + Opt_lookupcache_none,
> > > > +
> > > > + Opt_lookupcache_err
> > > > +};
> > > > +
> > > > +static match_table_t nfs_lookupcache_tokens = {
> > > > + { Opt_lookupcache_all, "all" },
> > > > + { Opt_lookupcache_positive, "pos" },
> > > > + { Opt_lookupcache_positive, "positive" },
> > > > + { Opt_lookupcache_none, "none" },
> > > > +
> > > > + { Opt_lookupcache_err, NULL }
> > > > +};
> > > > +
> > > > +enum {
> > > > + Opt_local_lock_all, Opt_local_lock_flock,
> > > > Opt_local_lock_posix,
> > > > + Opt_local_lock_none,
> > > > +
> > > > + Opt_local_lock_err
> > > > +};
> > > > +
> > > > +static match_table_t nfs_local_lock_tokens = {
> > > > + { Opt_local_lock_all, "all" },
> > > > + { Opt_local_lock_flock, "flock" },
> > > > + { Opt_local_lock_posix, "posix" },
> > > > + { Opt_local_lock_none, "none" },
> > > > +
> > > > + { Opt_local_lock_err, NULL }
> > > > +};
> > > > +
> > > > +enum {
> > > > + Opt_vers_2, Opt_vers_3, Opt_vers_4, Opt_vers_4_0,
> > > > + Opt_vers_4_1, Opt_vers_4_2,
> > > > +
> > > > + Opt_vers_err
> > > > +};
> > > > +
> > > > +static match_table_t nfs_vers_tokens = {
> > > > + { Opt_vers_2, "2" },
> > > > + { Opt_vers_3, "3" },
> > > > + { Opt_vers_4, "4" },
> > > > + { Opt_vers_4_0, "4.0" },
> > > > + { Opt_vers_4_1, "4.1" },
> > > > + { Opt_vers_4_2, "4.2" },
> > > > +
> > > > + { Opt_vers_err, NULL }
> > > > +};
> > > > +
> > > > +struct nfs_parsed_mount_data
> > > > *nfs_alloc_parsed_mount_data(void)
> > > > +{
> > > > + struct nfs_parsed_mount_data *data;
> > > > +
> > > > + data = kzalloc(sizeof(*data), GFP_KERNEL);
> > > > + if (data) {
> > > > + data->timeo = NFS_UNSPEC_TIMEO;
> > > > + data->retrans = NFS_UNSPEC_RETRANS;
> > > > + data->acregmin = NFS_DEF_ACREGMIN;
> > > > + data->acregmax = NFS_DEF_ACREGMAX;
> > > > + data->acdirmin = NFS_DEF_ACDIRMIN;
> > > > + data->acdirmax = NFS_DEF_ACDIRMAX;
> > > > + data->mount_server.port = NFS_UNSPEC_PORT;
> > > > + data->nfs_server.port = NFS_UNSPEC_PORT;
> > > > + data->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > > > + data->selected_flavor = RPC_AUTH_MAXFLAVOR;
> > > > + data->minorversion = 0;
> > > > + data->need_mount = true;
> > > > + data->net = current->nsproxy-
> > > > >net_ns;
> > > > + data->lsm_opts = NULL;
> > > > + }
> > > > + return data;
> > > > +}
> > > > +
> > > > +void nfs_free_parsed_mount_data(struct nfs_parsed_mount_data
> > > > *data)
> > > > +{
> > > > + if (data) {
> > > > + kfree(data->client_address);
> > > > + kfree(data->mount_server.hostname);
> > > > + kfree(data->nfs_server.export_path);
> > > > + kfree(data->nfs_server.hostname);
> > > > + kfree(data->fscache_uniq);
> > > > + security_free_mnt_opts(&data->lsm_opts);
> > > > + kfree(data);
> > > > + }
> > > > +}
> > > > +
> > > > +/*
> > > > + * Sanity-check a server address provided by the mount
> > > > command.
> > > > + *
> > > > + * Address family must be initialized, and address must not be
> > > > + * the ANY address for that family.
> > > > + */
> > > > +static int nfs_verify_server_address(struct sockaddr *addr)
> > > > +{
> > > > + switch (addr->sa_family) {
> > > > + case AF_INET: {
> > > > + struct sockaddr_in *sa = (struct sockaddr_in
> > > > *)addr;
> > > > + return sa->sin_addr.s_addr !=
> > > > htonl(INADDR_ANY);
> > > > + }
> > > > + case AF_INET6: {
> > > > + struct in6_addr *sa = &((struct sockaddr_in6
> > > > *)addr)-
> > > > > sin6_addr;
> > > > + return !ipv6_addr_any(sa);
> > > > + }
> > > > + }
> > > > +
> > > > + dfprintk(MOUNT, "NFS: Invalid IP address specified\n");
> > > > + return 0;
> > > > +}
> > > > +
> > > > +/*
> > > > + * Sanity check the NFS transport protocol.
> > > > + *
> > > > + */
> > > > +static void nfs_validate_transport_protocol(struct
> > > > nfs_parsed_mount_data *mnt)
> > > > +{
> > > > + switch (mnt->nfs_server.protocol) {
> > > > + case XPRT_TRANSPORT_UDP:
> > > > + case XPRT_TRANSPORT_TCP:
> > > > + case XPRT_TRANSPORT_RDMA:
> > > > + break;
> > > > + default:
> > > > + mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > > > + }
> > > > +}
> > > > +
> > > > +/*
> > > > + * For text based NFSv2/v3 mounts, the mount protocol
> > > > transport
> > > > default
> > > > + * settings should depend upon the specified NFS transport.
> > > > + */
> > > > +static void nfs_set_mount_transport_protocol(struct
> > > > nfs_parsed_mount_data *mnt)
> > > > +{
> > > > + nfs_validate_transport_protocol(mnt);
> > > > +
> > > > + if (mnt->mount_server.protocol == XPRT_TRANSPORT_UDP ||
> > > > + mnt->mount_server.protocol == XPRT_TRANSPORT_TCP)
> > > > + return;
> > > > + switch (mnt->nfs_server.protocol) {
> > > > + case XPRT_TRANSPORT_UDP:
> > > > + mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > + break;
> > > > + case XPRT_TRANSPORT_TCP:
> > > > + case XPRT_TRANSPORT_RDMA:
> > > > + mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > + }
> > > > +}
> > > > +
> > > > +/*
> > > > + * Add 'flavor' to 'auth_info' if not already present.
> > > > + * Returns true if 'flavor' ends up in the list, false
> > > > otherwise
> > > > + */
> > > > +static bool nfs_auth_info_add(struct nfs_auth_info *auth_info,
> > > > + rpc_authflavor_t flavor)
> > > > +{
> > > > + unsigned int i;
> > > > + unsigned int max_flavor_len = ARRAY_SIZE(auth_info-
> > > > >flavors);
> > > > +
> > > > + /* make sure this flavor isn't already in the list */
> > > > + for (i = 0; i < auth_info->flavor_len; i++) {
> > > > + if (flavor == auth_info->flavors[i])
> > > > + return true;
> > > > + }
> > > > +
> > > > + if (auth_info->flavor_len + 1 >= max_flavor_len) {
> > > > + dfprintk(MOUNT, "NFS: too many sec=
> > > > flavors\n");
> > > > + return false;
> > > > + }
> > > > +
> > > > + auth_info->flavors[auth_info->flavor_len++] = flavor;
> > > > + return true;
> > > > +}
> > > > +
> > > > +/*
> > > > + * Parse the value of the 'sec=' option.
> > > > + */
> > > > +static int nfs_parse_security_flavors(char *value,
> > > > + struct
> > > > nfs_parsed_mount_data
> > > > *mnt)
> > > > +{
> > > > + substring_t args[MAX_OPT_ARGS];
> > > > + rpc_authflavor_t pseudoflavor;
> > > > + char *p;
> > > > +
> > > > + dfprintk(MOUNT, "NFS: parsing sec=%s option\n", value);
> > > > +
> > > > + while ((p = strsep(&value, ":")) != NULL) {
> > > > + switch (match_token(p, nfs_secflavor_tokens,
> > > > args)) {
> > > > + case Opt_sec_none:
> > > > + pseudoflavor = RPC_AUTH_NULL;
> > > > + break;
> > > > + case Opt_sec_sys:
> > > > + pseudoflavor = RPC_AUTH_UNIX;
> > > > + break;
> > > > + case Opt_sec_krb5:
> > > > + pseudoflavor = RPC_AUTH_GSS_KRB5;
> > > > + break;
> > > > + case Opt_sec_krb5i:
> > > > + pseudoflavor = RPC_AUTH_GSS_KRB5I;
> > > > + break;
> > > > + case Opt_sec_krb5p:
> > > > + pseudoflavor = RPC_AUTH_GSS_KRB5P;
> > > > + break;
> > > > + case Opt_sec_lkey:
> > > > + pseudoflavor = RPC_AUTH_GSS_LKEY;
> > > > + break;
> > > > + case Opt_sec_lkeyi:
> > > > + pseudoflavor = RPC_AUTH_GSS_LKEYI;
> > > > + break;
> > > > + case Opt_sec_lkeyp:
> > > > + pseudoflavor = RPC_AUTH_GSS_LKEYP;
> > > > + break;
> > > > + case Opt_sec_spkm:
> > > > + pseudoflavor = RPC_AUTH_GSS_SPKM;
> > > > + break;
> > > > + case Opt_sec_spkmi:
> > > > + pseudoflavor = RPC_AUTH_GSS_SPKMI;
> > > > + break;
> > > > + case Opt_sec_spkmp:
> > > > + pseudoflavor = RPC_AUTH_GSS_SPKMP;
> > > > + break;
> > > > + default:
> > > > + dfprintk(MOUNT,
> > > > + "NFS: sec= option '%s' not
> > > > recognized\n", p);
> > > > + return 0;
> > > > + }
> > > > +
> > > > + if (!nfs_auth_info_add(&mnt->auth_info,
> > > > pseudoflavor))
> > > > + return 0;
> > > > + }
> > > > +
> > > > + return 1;
> > > > +}
> > > > +
> > > > +static int nfs_parse_version_string(char *string,
> > > > + struct nfs_parsed_mount_data *mnt,
> > > > + substring_t *args)
> > > > +{
> > > > + mnt->flags &= ~NFS_MOUNT_VER3;
> > > > + switch (match_token(string, nfs_vers_tokens, args)) {
> > > > + case Opt_vers_2:
> > > > + mnt->version = 2;
> > > > + break;
> > > > + case Opt_vers_3:
> > > > + mnt->flags |= NFS_MOUNT_VER3;
> > > > + mnt->version = 3;
> > > > + break;
> > > > + case Opt_vers_4:
> > > > + /* Backward compatibility option. In future,
> > > > + * the mount program should always supply
> > > > + * a NFSv4 minor version number.
> > > > + */
> > > > + mnt->version = 4;
> > > > + break;
> > > > + case Opt_vers_4_0:
> > > > + mnt->version = 4;
> > > > + mnt->minorversion = 0;
> > > > + break;
> > > > + case Opt_vers_4_1:
> > > > + mnt->version = 4;
> > > > + mnt->minorversion = 1;
> > > > + break;
> > > > + case Opt_vers_4_2:
> > > > + mnt->version = 4;
> > > > + mnt->minorversion = 2;
> > > > + break;
> > > > + default:
> > > > + return 0;
> > > > + }
> > > > + return 1;
> > > > +}
> > > > +
> > > > +static int nfs_get_option_str(substring_t args[], char
> > > > **option)
> > > > +{
> > > > + kfree(*option);
> > > > + *option = match_strdup(args);
> > > > + return !*option;
> > > > +}
> > > > +
> > > > +static int nfs_get_option_ul(substring_t args[], unsigned long
> > > > *option)
> > > > +{
> > > > + int rc;
> > > > + char *string;
> > > > +
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + return -ENOMEM;
> > > > + rc = kstrtoul(string, 10, option);
> > > > + kfree(string);
> > > > +
> > > > + return rc;
> > > > +}
> > > > +
> > > > +static int nfs_get_option_ul_bound(substring_t args[],
> > > > unsigned
> > > > long *option,
> > > > + unsigned long l_bound, unsigned long u_bound)
> > > > +{
> > > > + int ret;
> > > > +
> > > > + ret = nfs_get_option_ul(args, option);
> > > > + if (ret != 0)
> > > > + return ret;
> > > > + if (*option < l_bound || *option > u_bound)
> > > > + return -ERANGE;
> > > > + return 0;
> > > > +}
> > > > +
> > > > +/*
> > > > + * Error-check and convert a string of mount options from user
> > > > space into
> > > > + * a data structure. The whole mount string is processed; bad
> > > > options are
> > > > + * skipped as they are encountered. If there were no errors,
> > > > return 1;
> > > > + * otherwise return 0 (zero).
> > > > + */
> > > > +int nfs_parse_mount_options(char *raw, struct
> > > > nfs_parsed_mount_data *mnt)
> > > > +{
> > > > + char *p, *string;
> > > > + int rc, sloppy = 0, invalid_option = 0;
> > > > + unsigned short protofamily = AF_UNSPEC;
> > > > + unsigned short mountfamily = AF_UNSPEC;
> > > > +
> > > > + if (!raw) {
> > > > + dfprintk(MOUNT, "NFS: mount options string was
> > > > NULL.\n");
> > > > + return 1;
> > > > + }
> > > > + dfprintk(MOUNT, "NFS: nfs mount opts='%s'\n", raw);
> > > > +
> > > > + rc = security_sb_eat_lsm_opts(raw, &mnt->lsm_opts);
> > > > + if (rc)
> > > > + goto out_security_failure;
> > > > +
> > > > + while ((p = strsep(&raw, ",")) != NULL) {
> > > > + substring_t args[MAX_OPT_ARGS];
> > > > + unsigned long option;
> > > > + int token;
> > > > +
> > > > + if (!*p)
> > > > + continue;
> > > > +
> > > > + dfprintk(MOUNT, "NFS: parsing nfs mount
> > > > option
> > > > '%s'\n", p);
> > > > +
> > > > + token = match_token(p, nfs_mount_option_tokens,
> > > > args);
> > > > + switch (token) {
> > > > +
> > > > + /*
> > > > + * boolean options: foo/nofoo
> > > > + */
> > > > + case Opt_soft:
> > > > + mnt->flags |= NFS_MOUNT_SOFT;
> > > > + mnt->flags &= ~NFS_MOUNT_SOFTERR;
> > > > + break;
> > > > + case Opt_softerr:
> > > > + mnt->flags |= NFS_MOUNT_SOFTERR;
> > > > + mnt->flags &= ~NFS_MOUNT_SOFT;
> > > > + break;
> > > > + case Opt_hard:
> > > > + mnt->flags &=
> > > > ~(NFS_MOUNT_SOFT|NFS_MOUNT_SOFTERR);
> > > > + break;
> > > > + case Opt_posix:
> > > > + mnt->flags |= NFS_MOUNT_POSIX;
> > > > + break;
> > > > + case Opt_noposix:
> > > > + mnt->flags &= ~NFS_MOUNT_POSIX;
> > > > + break;
> > > > + case Opt_cto:
> > > > + mnt->flags &= ~NFS_MOUNT_NOCTO;
> > > > + break;
> > > > + case Opt_nocto:
> > > > + mnt->flags |= NFS_MOUNT_NOCTO;
> > > > + break;
> > > > + case Opt_ac:
> > > > + mnt->flags &= ~NFS_MOUNT_NOAC;
> > > > + break;
> > > > + case Opt_noac:
> > > > + mnt->flags |= NFS_MOUNT_NOAC;
> > > > + break;
> > > > + case Opt_lock:
> > > > + mnt->flags &= ~NFS_MOUNT_NONLM;
> > > > + mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
> > > > + NFS_MOUNT_LOCAL_FCNTL);
> > > > + break;
> > > > + case Opt_nolock:
> > > > + mnt->flags |= NFS_MOUNT_NONLM;
> > > > + mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
> > > > + NFS_MOUNT_LOCAL_FCNTL);
> > > > + break;
> > > > + case Opt_udp:
> > > > + mnt->flags &= ~NFS_MOUNT_TCP;
> > > > + mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > + break;
> > > > + case Opt_tcp:
> > > > + mnt->flags |= NFS_MOUNT_TCP;
> > > > + mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > + break;
> > > > + case Opt_rdma:
> > > > + mnt->flags |= NFS_MOUNT_TCP; /* for
> > > > side
> > > > protocols */
> > > > + mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_RDMA;
> > > > + xprt_load_transport(p);
> > > > + break;
> > > > + case Opt_acl:
> > > > + mnt->flags &= ~NFS_MOUNT_NOACL;
> > > > + break;
> > > > + case Opt_noacl:
> > > > + mnt->flags |= NFS_MOUNT_NOACL;
> > > > + break;
> > > > + case Opt_rdirplus:
> > > > + mnt->flags &= ~NFS_MOUNT_NORDIRPLUS;
> > > > + break;
> > > > + case Opt_nordirplus:
> > > > + mnt->flags |= NFS_MOUNT_NORDIRPLUS;
> > > > + break;
> > > > + case Opt_sharecache:
> > > > + mnt->flags &= ~NFS_MOUNT_UNSHARED;
> > > > + break;
> > > > + case Opt_nosharecache:
> > > > + mnt->flags |= NFS_MOUNT_UNSHARED;
> > > > + break;
> > > > + case Opt_resvport:
> > > > + mnt->flags &= ~NFS_MOUNT_NORESVPORT;
> > > > + break;
> > > > + case Opt_noresvport:
> > > > + mnt->flags |= NFS_MOUNT_NORESVPORT;
> > > > + break;
> > > > + case Opt_fscache:
> > > > + mnt->options |= NFS_OPTION_FSCACHE;
> > > > + kfree(mnt->fscache_uniq);
> > > > + mnt->fscache_uniq = NULL;
> > > > + break;
> > > > + case Opt_nofscache:
> > > > + mnt->options &= ~NFS_OPTION_FSCACHE;
> > > > + kfree(mnt->fscache_uniq);
> > > > + mnt->fscache_uniq = NULL;
> > > > + break;
> > > > + case Opt_migration:
> > > > + mnt->options |= NFS_OPTION_MIGRATION;
> > > > + break;
> > > > + case Opt_nomigration:
> > > > + mnt->options &= ~NFS_OPTION_MIGRATION;
> > > > + break;
> > > > +
> > > > + /*
> > > > + * options that take numeric values
> > > > + */
> > > > + case Opt_port:
> > > > + if (nfs_get_option_ul(args, &option) ||
> > > > + option > USHRT_MAX)
> > > > + goto out_invalid_value;
> > > > + mnt->nfs_server.port = option;
> > > > + break;
> > > > + case Opt_rsize:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->rsize = option;
> > > > + break;
> > > > + case Opt_wsize:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->wsize = option;
> > > > + break;
> > > > + case Opt_bsize:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->bsize = option;
> > > > + break;
> > > > + case Opt_timeo:
> > > > + if (nfs_get_option_ul_bound(args,
> > > > &option, 1,
> > > > INT_MAX))
> > > > + goto out_invalid_value;
> > > > + mnt->timeo = option;
> > > > + break;
> > > > + case Opt_retrans:
> > > > + if (nfs_get_option_ul_bound(args,
> > > > &option, 0,
> > > > INT_MAX))
> > > > + goto out_invalid_value;
> > > > + mnt->retrans = option;
> > > > + break;
> > > > + case Opt_acregmin:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->acregmin = option;
> > > > + break;
> > > > + case Opt_acregmax:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->acregmax = option;
> > > > + break;
> > > > + case Opt_acdirmin:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->acdirmin = option;
> > > > + break;
> > > > + case Opt_acdirmax:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->acdirmax = option;
> > > > + break;
> > > > + case Opt_actimeo:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->acregmin = mnt->acregmax =
> > > > + mnt->acdirmin = mnt->acdirmax = option;
> > > > + break;
> > > > + case Opt_namelen:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + mnt->namlen = option;
> > > > + break;
> > > > + case Opt_mountport:
> > > > + if (nfs_get_option_ul(args, &option) ||
> > > > + option > USHRT_MAX)
> > > > + goto out_invalid_value;
> > > > + mnt->mount_server.port = option;
> > > > + break;
> > > > + case Opt_mountvers:
> > > > + if (nfs_get_option_ul(args, &option) ||
> > > > + option < NFS_MNT_VERSION ||
> > > > + option > NFS_MNT3_VERSION)
> > > > + goto out_invalid_value;
> > > > + mnt->mount_server.version = option;
> > > > + break;
> > > > + case Opt_minorversion:
> > > > + if (nfs_get_option_ul(args, &option))
> > > > + goto out_invalid_value;
> > > > + if (option > NFS4_MAX_MINOR_VERSION)
> > > > + goto out_invalid_value;
> > > > + mnt->minorversion = option;
> > > > + break;
> > > > +
> > > > + /*
> > > > + * options that take text values
> > > > + */
> > > > + case Opt_nfsvers:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + rc = nfs_parse_version_string(string,
> > > > mnt,
> > > > args);
> > > > + kfree(string);
> > > > + if (!rc)
> > > > + goto out_invalid_value;
> > > > + break;
> > > > + case Opt_sec:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + rc = nfs_parse_security_flavors(string,
> > > > mnt);
> > > > + kfree(string);
> > > > + if (!rc) {
> > > > + dfprintk(MOUNT,
> > > > "NFS: unrecognized "
> > > > + "security
> > > > flavor\n");
> > > > + return 0;
> > > > + }
> > > > + break;
> > > > + case Opt_proto:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + token = match_token(string,
> > > > + nfs_xprt_protocol_t
> > > > okens,
> > > > args);
> > > > +
> > > > + protofamily = AF_INET;
> > > > + switch (token) {
> > > > + case Opt_xprt_udp6:
> > > > + protofamily = AF_INET6;
> > > > + /* fall through */
> > > > + case Opt_xprt_udp:
> > > > + mnt->flags &= ~NFS_MOUNT_TCP;
> > > > + mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > + break;
> > > > + case Opt_xprt_tcp6:
> > > > + protofamily = AF_INET6;
> > > > + /* fall through */
> > > > + case Opt_xprt_tcp:
> > > > + mnt->flags |= NFS_MOUNT_TCP;
> > > > + mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > + break;
> > > > + case Opt_xprt_rdma6:
> > > > + protofamily = AF_INET6;
> > > > + /* fall through */
> > > > + case Opt_xprt_rdma:
> > > > + /* vector side protocols to TCP
> > > > */
> > > > + mnt->flags |= NFS_MOUNT_TCP;
> > > > + mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_RDMA;
> > > > + xprt_load_transport(string);
> > > > + break;
> > > > + default:
> > > > + dfprintk(MOUNT,
> > > > "NFS: unrecognized "
> > > > + "transport
> > > > protocol\n");
> > > > + kfree(string);
> > > > + return 0;
> > > > + }
> > > > + kfree(string);
> > > > + break;
> > > > + case Opt_mountproto:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + token = match_token(string,
> > > > + nfs_xprt_protocol_t
> > > > okens,
> > > > args);
> > > > + kfree(string);
> > > > +
> > > > + mountfamily = AF_INET;
> > > > + switch (token) {
> > > > + case Opt_xprt_udp6:
> > > > + mountfamily = AF_INET6;
> > > > + /* fall through */
> > > > + case Opt_xprt_udp:
> > > > + mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > + break;
> > > > + case Opt_xprt_tcp6:
> > > > + mountfamily = AF_INET6;
> > > > + /* fall through */
> > > > + case Opt_xprt_tcp:
> > > > + mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > + break;
> > > > + case Opt_xprt_rdma: /* not used for
> > > > side
> > > > protocols */
> > > > + default:
> > > > + dfprintk(MOUNT,
> > > > "NFS: unrecognized "
> > > > + "transport
> > > > protocol\n");
> > > > + return 0;
> > > > + }
> > > > + break;
> > > > + case Opt_addr:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + mnt->nfs_server.addrlen =
> > > > + rpc_pton(mnt->net, string,
> > > > strlen(string),
> > > > + (struct sockaddr *)
> > > > + &mnt-
> > > > >nfs_server.address,
> > > > + sizeof(mnt-
> > > > > nfs_server.address));
> > > > + kfree(string);
> > > > + if (mnt->nfs_server.addrlen == 0)
> > > > + goto out_invalid_address;
> > > > + break;
> > > > + case Opt_clientaddr:
> > > > + if (nfs_get_option_str(args, &mnt-
> > > > > client_address))
> > > > + goto out_nomem;
> > > > + break;
> > > > + case Opt_mounthost:
> > > > + if (nfs_get_option_str(args,
> > > > + &mnt-
> > > > > mount_server.hostname))
> > > > + goto out_nomem;
> > > > + break;
> > > > + case Opt_mountaddr:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + mnt->mount_server.addrlen =
> > > > + rpc_pton(mnt->net, string,
> > > > strlen(string),
> > > > + (struct sockaddr *)
> > > > + &mnt-
> > > > >mount_server.address,
> > > > + sizeof(mnt-
> > > > > mount_server.address));
> > > > + kfree(string);
> > > > + if (mnt->mount_server.addrlen == 0)
> > > > + goto out_invalid_address;
> > > > + break;
> > > > + case Opt_nconnect:
> > > > + if (nfs_get_option_ul_bound(args,
> > > > &option, 1,
> > > > NFS_MAX_CONNECTIONS))
> > > > + goto out_invalid_value;
> > > > + mnt->nfs_server.nconnect = option;
> > > > + break;
> > > > + case Opt_lookupcache:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + token = match_token(string,
> > > > + nfs_lookupcache_tokens,
> > > > args);
> > > > + kfree(string);
> > > > + switch (token) {
> > > > + case Opt_lookupcache_all:
> > > > + mnt->flags &=
> > > > ~(NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE);
> > > > + break;
> > > > + case Opt_lookupcache_positive:
> > > > + mnt->flags &=
> > > > ~NFS_MOUNT_LOOKUP_CACHE_NONE;
> > > > + mnt->flags |=
> > > > NFS_MOUNT_LOOKUP_CACHE_NONEG;
> > > > + break;
> > > > + case Opt_lookupcache_none:
> > > > + mnt->flags |=
> > > > NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE;
> > > > + break;
> > > > + default:
> > > > + dfprintk(MOUNT,
> > > > "NFS: invalid
> > > > "
> > > > + "lookup
> > > > cache
> > > > argument\n");
> > > > + return 0;
> > > > + };
> > > > + break;
> > > > + case Opt_fscache_uniq:
> > > > + if (nfs_get_option_str(args, &mnt-
> > > > > fscache_uniq))
> > > > + goto out_nomem;
> > > > + mnt->options |= NFS_OPTION_FSCACHE;
> > > > + break;
> > > > + case Opt_local_lock:
> > > > + string = match_strdup(args);
> > > > + if (string == NULL)
> > > > + goto out_nomem;
> > > > + token = match_token(string,
> > > > nfs_local_lock_tokens,
> > > > + args);
> > > > + kfree(string);
> > > > + switch (token) {
> > > > + case Opt_local_lock_all:
> > > > + mnt->flags |=
> > > > (NFS_MOUNT_LOCAL_FLOCK |
> > > > + NFS_MOUNT_LOCAL_
> > > > FCNTL);
> > > > + break;
> > > > + case Opt_local_lock_flock:
> > > > + mnt->flags |=
> > > > NFS_MOUNT_LOCAL_FLOCK;
> > > > + break;
> > > > + case Opt_local_lock_posix:
> > > > + mnt->flags |=
> > > > NFS_MOUNT_LOCAL_FCNTL;
> > > > + break;
> > > > + case Opt_local_lock_none:
> > > > + mnt->flags &=
> > > > ~(NFS_MOUNT_LOCAL_FLOCK |
> > > > + NFS_MOUNT_LOCAL
> > > > _FCNTL);
> > > > + break;
> > > > + default:
> > > > + dfprintk(MOUNT, "NFS: invalid
> > > > "
> > > > + "local_lock
> > > > argument\n");
> > > > + return 0;
> > > > + };
> > > > + break;
> > > > +
> > > > + /*
> > > > + * Special options
> > > > + */
> > > > + case Opt_sloppy:
> > > > + sloppy = 1;
> > > > + dfprintk(MOUNT, "NFS: relaxing
> > > > parsing
> > > > rules\n");
> > > > + break;
> > > > + case Opt_userspace:
> > > > + case Opt_deprecated:
> > > > + dfprintk(MOUNT, "NFS: ignoring mount
> > > > option "
> > > > + "'%s'\n", p);
> > > > + break;
> > > > +
> > > > + default:
> > > > + invalid_option = 1;
> > > > + dfprintk(MOUNT, "NFS: unrecognized
> > > > mount
> > > > option "
> > > > + "'%s'\n", p);
> > > > + }
> > > > + }
> > > > +
> > > > + if (!sloppy && invalid_option)
> > > > + return 0;
> > > > +
> > > > + if (mnt->minorversion && mnt->version != 4)
> > > > + goto out_minorversion_mismatch;
> > > > +
> > > > + if (mnt->options & NFS_OPTION_MIGRATION &&
> > > > + (mnt->version != 4 || mnt->minorversion != 0))
> > > > + goto out_migration_misuse;
> > > > +
> > > > + /*
> > > > + * verify that any proto=/mountproto= options match the
> > > > address
> > > > + * families in the addr=/mountaddr= options.
> > > > + */
> > > > + if (protofamily != AF_UNSPEC &&
> > > > + protofamily != mnt->nfs_server.address.ss_family)
> > > > + goto out_proto_mismatch;
> > > > +
> > > > + if (mountfamily != AF_UNSPEC) {
> > > > + if (mnt->mount_server.addrlen) {
> > > > + if (mountfamily != mnt-
> > > > > mount_server.address.ss_family)
> > > > + goto out_mountproto_mismatch;
> > > > + } else {
> > > > + if (mountfamily != mnt-
> > > > > nfs_server.address.ss_family)
> > > > + goto out_mountproto_mismatch;
> > > > + }
> > > > + }
> > > > +
> > > > + return 1;
> > > > +
> > > > +out_mountproto_mismatch:
> > > > + printk(KERN_INFO "NFS: mount server address does not
> > > > match
> > > > mountproto= "
> > > > + "option\n");
> > > > + return 0;
> > > > +out_proto_mismatch:
> > > > + printk(KERN_INFO "NFS: server address does not match
> > > > proto=
> > > > option\n");
> > > > + return 0;
> > > > +out_invalid_address:
> > > > + printk(KERN_INFO "NFS: bad IP address specified: %s\n",
> > > > p);
> > > > + return 0;
> > > > +out_invalid_value:
> > > > + printk(KERN_INFO "NFS: bad mount option value
> > > > specified: %s\n",
> > > > p);
> > > > + return 0;
> > > > +out_minorversion_mismatch:
> > > > + printk(KERN_INFO "NFS: mount option vers=%u does not
> > > > support "
> > > > + "minorversion=%u\n", mnt->version,
> > > > mnt-
> > > > > minorversion);
> > > > + return 0;
> > > > +out_migration_misuse:
> > > > + printk(KERN_INFO
> > > > + "NFS: 'migration' not supported for this NFS
> > > > version\n");
> > > > + return 0;
> > > > +out_nomem:
> > > > + printk(KERN_INFO "NFS: not enough memory to parse
> > > > option\n");
> > > > + return 0;
> > > > +out_security_failure:
> > > > + printk(KERN_INFO "NFS: security options invalid: %d\n",
> > > > rc);
> > > > + return 0;
> > > > +}
> > > > +
> > > > +/*
> > > > + * Split "dev_name" into "hostname:export_path".
> > > > + *
> > > > + * The leftmost colon demarks the split between the server's
> > > > hostname
> > > > + * and the export path. If the hostname starts with a left
> > > > square
> > > > + * bracket, then it may contain colons.
> > > > + *
> > > > + * Note: caller frees hostname and export path, even on error.
> > > > + */
> > > > +static int nfs_parse_devname(const char *dev_name,
> > > > + char **hostname, size_t maxnamlen,
> > > > + char **export_path, size_t
> > > > maxpathlen)
> > > > +{
> > > > + size_t len;
> > > > + char *end;
> > > > +
> > > > + if (unlikely(!dev_name || !*dev_name)) {
> > > > + dfprintk(MOUNT, "NFS: device name not
> > > > specified\n");
> > > > + return -EINVAL;
> > > > + }
> > > > +
> > > > + /* Is the host name protected with square brakcets? */
> > > > + if (*dev_name == '[') {
> > > > + end = strchr(++dev_name, ']');
> > > > + if (end == NULL || end[1] != ':')
> > > > + goto out_bad_devname;
> > > > +
> > > > + len = end - dev_name;
> > > > + end++;
> > > > + } else {
> > > > + char *comma;
> > > > +
> > > > + end = strchr(dev_name, ':');
> > > > + if (end == NULL)
> > > > + goto out_bad_devname;
> > > > + len = end - dev_name;
> > > > +
> > > > + /* kill possible hostname list: not supported
> > > > */
> > > > + comma = strchr(dev_name, ',');
> > > > + if (comma != NULL && comma < end)
> > > > + len = comma - dev_name;
> > > > + }
> > > > +
> > > > + if (len > maxnamlen)
> > > > + goto out_hostname;
> > > > +
> > > > + /* N.B. caller will free nfs_server.hostname in all
> > > > cases */
> > > > + *hostname = kstrndup(dev_name, len, GFP_KERNEL);
> > > > + if (*hostname == NULL)
> > > > + goto out_nomem;
> > > > + len = strlen(++end);
> > > > + if (len > maxpathlen)
> > > > + goto out_path;
> > > > + *export_path = kstrndup(end, len, GFP_KERNEL);
> > > > + if (!*export_path)
> > > > + goto out_nomem;
> > > > +
> > > > + dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", *export_path);
> > > > + return 0;
> > > > +
> > > > +out_bad_devname:
> > > > + dfprintk(MOUNT, "NFS: device name not in host:path
> > > > format\n");
> > > > + return -EINVAL;
> > > > +
> > > > +out_nomem:
> > > > + dfprintk(MOUNT, "NFS: not enough memory to parse device
> > > > name\n");
> > > > + return -ENOMEM;
> > > > +
> > > > +out_hostname:
> > > > + dfprintk(MOUNT, "NFS: server hostname too long\n");
> > > > + return -ENAMETOOLONG;
> > > > +
> > > > +out_path:
> > > > + dfprintk(MOUNT, "NFS: export pathname too long\n");
> > > > + return -ENAMETOOLONG;
> > > > +}
> > > > +
> > > > +/*
> > > > + * Validate the NFS2/NFS3 mount data
> > > > + * - fills in the mount root filehandle
> > > > + *
> > > > + * For option strings, user space handles the following
> > > > behaviors:
> > > > + *
> > > > + * + DNS: mapping server host name to IP address ("addr="
> > > > option)
> > > > + *
> > > > + * + failure mode: how to behave if a mount request can't be
> > > > handled
> > > > + * immediately ("fg/bg" option)
> > > > + *
> > > > + * + retry: how often to retry a mount request ("retry="
> > > > option)
> > > > + *
> > > > + * + breaking back: trying proto=udp after proto=tcp, v2 after
> > > > v3,
> > > > + * mountproto=tcp after mountproto=udp, and so on
> > > > + */
> > > > +static int nfs23_validate_mount_data(void *options,
> > > > + struct
> > > > nfs_parsed_mount_data
> > > > *args,
> > > > + struct nfs_fh *mntfh,
> > > > + const char *dev_name)
> > > > +{
> > > > + struct nfs_mount_data *data = (struct nfs_mount_data
> > > > *)options;
> > > > + struct sockaddr *sap = (struct sockaddr *)&args-
> > > > > nfs_server.address;
> > > > + int extra_flags = NFS_MOUNT_LEGACY_INTERFACE;
> > > > +
> > > > + if (data == NULL)
> > > > + goto out_no_data;
> > > > +
> > > > + args->version = NFS_DEFAULT_VERSION;
> > > > + switch (data->version) {
> > > > + case 1:
> > > > + data->namlen = 0; /* fall through */
> > > > + case 2:
> > > > + data->bsize = 0; /* fall through */
> > > > + case 3:
> > > > + if (data->flags & NFS_MOUNT_VER3)
> > > > + goto out_no_v3;
> > > > + data->root.size = NFS2_FHSIZE;
> > > > + memcpy(data->root.data, data->old_root.data,
> > > > NFS2_FHSIZE);
> > > > + /* Turn off security negotiation */
> > > > + extra_flags |= NFS_MOUNT_SECFLAVOUR;
> > > > + /* fall through */
> > > > + case 4:
> > > > + if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > > > + goto out_no_sec;
> > > > + /* fall through */
> > > > + case 5:
> > > > + memset(data->context, 0, sizeof(data-
> > > > >context));
> > > > + /* fall through */
> > > > + case 6:
> > > > + if (data->flags & NFS_MOUNT_VER3) {
> > > > + if (data->root.size > NFS3_FHSIZE ||
> > > > data-
> > > > > root.size == 0)
> > > > + goto out_invalid_fh;
> > > > + mntfh->size = data->root.size;
> > > > + args->version = 3;
> > > > + } else {
> > > > + mntfh->size = NFS2_FHSIZE;
> > > > + args->version = 2;
> > > > + }
> > > > +
> > > > +
> > > > + memcpy(mntfh->data, data->root.data, mntfh-
> > > > >size);
> > > > + if (mntfh->size < sizeof(mntfh->data))
> > > > + memset(mntfh->data + mntfh->size, 0,
> > > > + sizeof(mntfh->data) - mntfh-
> > > > >size);
> > > > +
> > > > + /*
> > > > + * Translate to nfs_parsed_mount_data, which
> > > > nfs_fill_super
> > > > + * can deal with.
> > > > + */
> > > > + args->flags = data->flags &
> > > > NFS_MOUNT_FLAGMASK;
> > > > + args->flags |= extra_flags;
> > > > + args->rsize = data->rsize;
> > > > + args->wsize = data->wsize;
> > > > + args->timeo = data->timeo;
> > > > + args->retrans = data->retrans;
> > > > + args->acregmin = data->acregmin;
> > > > + args->acregmax = data->acregmax;
> > > > + args->acdirmin = data->acdirmin;
> > > > + args->acdirmax = data->acdirmax;
> > > > + args->need_mount = false;
> > > > +
> > > > + memcpy(sap, &data->addr, sizeof(data->addr));
> > > > + args->nfs_server.addrlen = sizeof(data->addr);
> > > > + args->nfs_server.port = ntohs(data-
> > > > >addr.sin_port);
> > > > + if (sap->sa_family != AF_INET ||
> > > > + !nfs_verify_server_address(sap))
> > > > + goto out_no_address;
> > > > +
> > > > + if (!(data->flags & NFS_MOUNT_TCP))
> > > > + args->nfs_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > + /* N.B. caller will free nfs_server.hostname in
> > > > all
> > > > cases */
> > > > + args->nfs_server.hostname = kstrdup(data-
> > > > >hostname,
> > > > GFP_KERNEL);
> > > > + args->namlen = data->namlen;
> > > > + args->bsize = data->bsize;
> > > > +
> > > > + if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > > > + args->selected_flavor = data-
> > > > >pseudoflavor;
> > > > + else
> > > > + args->selected_flavor = RPC_AUTH_UNIX;
> > > > + if (!args->nfs_server.hostname)
> > > > + goto out_nomem;
> > > > +
> > > > + if (!(data->flags & NFS_MOUNT_NONLM))
> > > > + args->flags &= ~(NFS_MOUNT_LOCAL_FLOCK|
> > > > + NFS_MOUNT_LOCAL_FCNTL)
> > > > ;
> > > > + else
> > > > + args->flags |= (NFS_MOUNT_LOCAL_FLOCK|
> > > > + NFS_MOUNT_LOCAL_FCNTL);
> > > > + /*
> > > > + * The legacy version 6 binary mount data from
> > > > userspace has a
> > > > + * field used only to transport selinux
> > > > information
> > > > into the
> > > > + * the kernel. To continue to support that
> > > > functionality we
> > > > + * have a touch of selinux knowledge here in
> > > > the NFS
> > > > code. The
> > > > + * userspace code converted context=blah to
> > > > just blah
> > > > so we are
> > > > + * converting back to the full string selinux
> > > > understands.
> > > > + */
> > > > + if (data->context[0]){
> > > > +#ifdef CONFIG_SECURITY_SELINUX
> > > > + int rc;
> > > > + data->context[NFS_MAX_CONTEXT_LEN] =
> > > > '\0';
> > > > + rc = security_add_mnt_opt("context",
> > > > data-
> > > > > context,
> > > > + strlen(data->context),
> > > > &args-
> > > > > lsm_opts);
> > > > + if (rc)
> > > > + return rc;
> > > > +#else
> > > > + return -EINVAL;
> > > > +#endif
> > > > + }
> > > > +
> > > > + break;
> > > > + default:
> > > > + return NFS_TEXT_DATA;
> > > > + }
> > > > +
> > > > + return 0;
> > > > +
> > > > +out_no_data:
> > > > + dfprintk(MOUNT, "NFS: mount program didn't pass any
> > > > mount
> > > > data\n");
> > > > + return -EINVAL;
> > > > +
> > > > +out_no_v3:
> > > > + dfprintk(MOUNT, "NFS: nfs_mount_data version %d does
> > > > not
> > > > support v3\n",
> > > > + data->version);
> > > > + return -EINVAL;
> > > > +
> > > > +out_no_sec:
> > > > + dfprintk(MOUNT, "NFS: nfs_mount_data version supports
> > > > only
> > > > AUTH_SYS\n");
> > > > + return -EINVAL;
> > > > +
> > > > +out_nomem:
> > > > + dfprintk(MOUNT, "NFS: not enough memory to handle mount
> > > > options\n");
> > > > + return -ENOMEM;
> > > > +
> > > > +out_no_address:
> > > > + dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > > > address\n");
> > > > + return -EINVAL;
> > > > +
> > > > +out_invalid_fh:
> > > > + dfprintk(MOUNT, "NFS: invalid root filehandle\n");
> > > > + return -EINVAL;
> > > > +}
> > > > +
> > > > +#if IS_ENABLED(CONFIG_NFS_V4)
> > > > +
> > > > +static void nfs4_validate_mount_flags(struct
> > > > nfs_parsed_mount_data
> > > > *args)
> > > > +{
> > > > + args->flags &=
> > > > ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3|
> > > > + NFS_MOUNT_LOCAL_FLOCK|NFS_MOUNT_LOCAL_
> > > > FCNTL);
> > > > +}
> > > > +
> > > > +/*
> > > > + * Validate NFSv4 mount options
> > > > + */
> > > > +static int nfs4_validate_mount_data(void *options,
> > > > + struct
> > > > nfs_parsed_mount_data *args,
> > > > + const char *dev_name)
> > > > +{
> > > > + struct sockaddr *sap = (struct sockaddr *)&args-
> > > > > nfs_server.address;
> > > > + struct nfs4_mount_data *data = (struct nfs4_mount_data
> > > > *)options;
> > > > + char *c;
> > > > +
> > > > + if (data == NULL)
> > > > + goto out_no_data;
> > > > +
> > > > + args->version = 4;
> > > > +
> > > > + switch (data->version) {
> > > > + case 1:
> > > > + if (data->host_addrlen > sizeof(args-
> > > > > nfs_server.address))
> > > > + goto out_no_address;
> > > > + if (data->host_addrlen == 0)
> > > > + goto out_no_address;
> > > > + args->nfs_server.addrlen = data->host_addrlen;
> > > > + if (copy_from_user(sap, data->host_addr, data-
> > > > > host_addrlen))
> > > > + return -EFAULT;
> > > > + if (!nfs_verify_server_address(sap))
> > > > + goto out_no_address;
> > > > + args->nfs_server.port = ntohs(((struct
> > > > sockaddr_in
> > > > *)sap)->sin_port);
> > > > +
> > > > + if (data->auth_flavourlen) {
> > > > + rpc_authflavor_t pseudoflavor;
> > > > + if (data->auth_flavourlen > 1)
> > > > + goto out_inval_auth;
> > > > + if (copy_from_user(&pseudoflavor,
> > > > + data->auth_flavours,
> > > > + sizeof(pseudoflavor)
> > > > ))
> > > > + return -EFAULT;
> > > > + args->selected_flavor = pseudoflavor;
> > > > + } else
> > > > + args->selected_flavor = RPC_AUTH_UNIX;
> > > > +
> > > > + c = strndup_user(data->hostname.data,
> > > > NFS4_MAXNAMLEN);
> > > > + if (IS_ERR(c))
> > > > + return PTR_ERR(c);
> > > > + args->nfs_server.hostname = c;
> > > > +
> > > > + c = strndup_user(data->mnt_path.data,
> > > > NFS4_MAXPATHLEN);
> > > > + if (IS_ERR(c))
> > > > + return PTR_ERR(c);
> > > > + args->nfs_server.export_path = c;
> > > > + dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", c);
> > > > +
> > > > + c = strndup_user(data->client_addr.data, 16);
> > > > + if (IS_ERR(c))
> > > > + return PTR_ERR(c);
> > > > + args->client_address = c;
> > > > +
> > > > + /*
> > > > + * Translate to nfs_parsed_mount_data, which
> > > > nfs4_fill_super
> > > > + * can deal with.
> > > > + */
> > > > +
> > > > + args->flags = data->flags &
> > > > NFS4_MOUNT_FLAGMASK;
> > > > + args->rsize = data->rsize;
> > > > + args->wsize = data->wsize;
> > > > + args->timeo = data->timeo;
> > > > + args->retrans = data->retrans;
> > > > + args->acregmin = data->acregmin;
> > > > + args->acregmax = data->acregmax;
> > > > + args->acdirmin = data->acdirmin;
> > > > + args->acdirmax = data->acdirmax;
> > > > + args->nfs_server.protocol = data->proto;
> > > > + nfs_validate_transport_protocol(args);
> > > > + if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_UDP)
> > > > + goto out_invalid_transport_udp;
> > > > +
> > > > + break;
> > > > + default:
> > > > + return NFS_TEXT_DATA;
> > > > + }
> > > > +
> > > > + return 0;
> > > > +
> > > > +out_no_data:
> > > > + dfprintk(MOUNT, "NFS4: mount program didn't pass any
> > > > mount
> > > > data\n");
> > > > + return -EINVAL;
> > > > +
> > > > +out_inval_auth:
> > > > + dfprintk(MOUNT, "NFS4: Invalid number of RPC auth
> > > > flavours
> > > > %d\n",
> > > > + data->auth_flavourlen);
> > > > + return -EINVAL;
> > > > +
> > > > +out_no_address:
> > > > + dfprintk(MOUNT, "NFS4: mount program didn't pass remote
> > > > address\n");
> > > > + return -EINVAL;
> > > > +
> > > > +out_invalid_transport_udp:
> > > > + dfprintk(MOUNT, "NFSv4: Unsupported transport protocol
> > > > udp\n");
> > > > + return -EINVAL;
> > > > +}
> > > > +
> > > > +int nfs_validate_mount_data(struct file_system_type *fs_type,
> > > > + void *options,
> > > > + struct nfs_parsed_mount_data *args,
> > > > + struct nfs_fh *mntfh,
> > > > + const char *dev_name)
> > > > +{
> > > > + if (fs_type == &nfs_fs_type)
> > > > + return nfs23_validate_mount_data(options, args,
> > > > mntfh,
> > > > dev_name);
> > > > + return nfs4_validate_mount_data(options, args,
> > > > dev_name);
> > > > +}
> > > > +#else
> > > > +int nfs_validate_mount_data(struct file_system_type *fs_type,
> > > > + void *options,
> > > > + struct nfs_parsed_mount_data *args,
> > > > + struct nfs_fh *mntfh,
> > > > + const char *dev_name)
> > > > +{
> > > > + return nfs23_validate_mount_data(options, args, mntfh,
> > > > dev_name);
> > > > +}
> > > > +#endif
> > > > +
> > > > +int nfs_validate_text_mount_data(void *options,
> > > > + struct nfs_parsed_mount_data
> > > > *args,
> > > > + const char *dev_name)
> > > > +{
> > > > + int port = 0;
> > > > + int max_namelen = PAGE_SIZE;
> > > > + int max_pathlen = NFS_MAXPATHLEN;
> > > > + struct sockaddr *sap = (struct sockaddr *)&args-
> > > > > nfs_server.address;
> > > > +
> > > > + if (nfs_parse_mount_options((char *)options, args) ==
> > > > 0)
> > > > + return -EINVAL;
> > > > +
> > > > + if (!nfs_verify_server_address(sap))
> > > > + goto out_no_address;
> > > > +
> > > > + if (args->version == 4) {
> > > > +#if IS_ENABLED(CONFIG_NFS_V4)
> > > > + if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_RDMA)
> > > > + port = NFS_RDMA_PORT;
> > > > + else
> > > > + port = NFS_PORT;
> > > > + max_namelen = NFS4_MAXNAMLEN;
> > > > + max_pathlen = NFS4_MAXPATHLEN;
> > > > + nfs_validate_transport_protocol(args);
> > > > + if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_UDP)
> > > > + goto out_invalid_transport_udp;
> > > > + nfs4_validate_mount_flags(args);
> > > > +#else
> > > > + goto out_v4_not_compiled;
> > > > +#endif /* CONFIG_NFS_V4 */
> > > > + } else {
> > > > + nfs_set_mount_transport_protocol(args);
> > > > + if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_RDMA)
> > > > + port = NFS_RDMA_PORT;
> > > > + }
> > > > +
> > > > + nfs_set_port(sap, &args->nfs_server.port, port);
> > > > +
> > > > + return nfs_parse_devname(dev_name,
> > > > + &args->nfs_server.hostname,
> > > > + max_namelen,
> > > > + &args-
> > > > >nfs_server.export_path,
> > > > + max_pathlen);
> > > > +
> > > > +#if !IS_ENABLED(CONFIG_NFS_V4)
> > > > +out_v4_not_compiled:
> > > > + dfprintk(MOUNT, "NFS: NFSv4 is not compiled into
> > > > kernel\n");
> > > > + return -EPROTONOSUPPORT;
> > > > +#else
> > > > +out_invalid_transport_udp:
> > > > + dfprintk(MOUNT, "NFSv4: Unsupported transport protocol
> > > > udp\n");
> > > > + return -EINVAL;
> > > > +#endif /* !CONFIG_NFS_V4 */
> > > > +
> > > > +out_no_address:
> > > > + dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > > > address\n");
> > > > + return -EINVAL;
> > > > +}
> > > > diff --git a/fs/nfs/internal.h b/fs/nfs/internal.h
> > > > index d512ec394559..b66fd35993b3 100644
> > > > --- a/fs/nfs/internal.h
> > > > +++ b/fs/nfs/internal.h
> > > > @@ -7,6 +7,7 @@
> > > > #include <linux/mount.h>
> > > > #include <linux/security.h>
> > > > #include <linux/crc32.h>
> > > > +#include <linux/sunrpc/addr.h>
> > > > #include <linux/nfs_page.h>
> > > > #include <linux/wait_bit.h>
> > > >
> > > > @@ -232,6 +233,22 @@ extern const struct svc_version
> > > > nfs4_callback_version1;
> > > > extern const struct svc_version nfs4_callback_version4;
> > > >
> > > > struct nfs_pageio_descriptor;
> > > > +
> > > > +/* mount.c */
> > > > +#define NFS_TEXT_DATA 1
> > > > +
> > > > +extern struct nfs_parsed_mount_data
> > > > *nfs_alloc_parsed_mount_data(void);
> > > > +extern void nfs_free_parsed_mount_data(struct
> > > > nfs_parsed_mount_data *data);
> > > > +extern int nfs_parse_mount_options(char *raw, struct
> > > > nfs_parsed_mount_data *mnt);
> > > > +extern int nfs_validate_mount_data(struct file_system_type
> > > > *fs_type,
> > > > + void *options,
> > > > + struct nfs_parsed_mount_data
> > > > *args,
> > > > + struct nfs_fh *mntfh,
> > > > + const char *dev_name);
> > > > +extern int nfs_validate_text_mount_data(void *options,
> > > > + struct
> > > > nfs_parsed_mount_data
> > > > *args,
> > > > + const char *dev_name);
> > > > +
> > > > /* pagelist.c */
> > > > extern int __init nfs_init_nfspagecache(void);
> > > > extern void nfs_destroy_nfspagecache(void);
> > > > @@ -763,3 +780,15 @@ static inline bool nfs_error_is_fatal(int
> > > > err)
> > > > }
> > > > }
> > > >
> > > > +/*
> > > > + * Select between a default port value and a user-specified
> > > > port
> > > > value.
> > > > + * If a zero value is set, then autobind will be used.
> > > > + */
> > > > +static inline void nfs_set_port(struct sockaddr *sap, int
> > > > *port,
> > > > + const unsigned short
> > > > default_port)
> > > > +{
> > > > + if (*port == NFS_UNSPEC_PORT)
> > > > + *port = default_port;
> > > > +
> > > > + rpc_set_port(sap, *port);
> > > > +}
> > > > diff --git a/fs/nfs/super.c b/fs/nfs/super.c
> > > > index d8702e57f7fc..886220d2da4e 100644
> > > > --- a/fs/nfs/super.c
> > > > +++ b/fs/nfs/super.c
> > > > @@ -69,229 +69,6 @@
> > > > #include "nfs.h"
> > > >
> > > > #define NFSDBG_FACILITY NFSDBG_VFS
> > > > -#define NFS_TEXT_DATA 1
> > > > -
> > > > -#if IS_ENABLED(CONFIG_NFS_V3)
> > > > -#define NFS_DEFAULT_VERSION 3
> > > > -#else
> > > > -#define NFS_DEFAULT_VERSION 2
> > > > -#endif
> > > > -
> > > > -#define NFS_MAX_CONNECTIONS 16
> > > > -
> > > > -enum {
> > > > - /* Mount options that take no arguments */
> > > > - Opt_soft, Opt_softerr, Opt_hard,
> > > > - Opt_posix, Opt_noposix,
> > > > - Opt_cto, Opt_nocto,
> > > > - Opt_ac, Opt_noac,
> > > > - Opt_lock, Opt_nolock,
> > > > - Opt_udp, Opt_tcp, Opt_rdma,
> > > > - Opt_acl, Opt_noacl,
> > > > - Opt_rdirplus, Opt_nordirplus,
> > > > - Opt_sharecache, Opt_nosharecache,
> > > > - Opt_resvport, Opt_noresvport,
> > > > - Opt_fscache, Opt_nofscache,
> > > > - Opt_migration, Opt_nomigration,
> > > > -
> > > > - /* Mount options that take integer arguments */
> > > > - Opt_port,
> > > > - Opt_rsize, Opt_wsize, Opt_bsize,
> > > > - Opt_timeo, Opt_retrans,
> > > > - Opt_acregmin, Opt_acregmax,
> > > > - Opt_acdirmin, Opt_acdirmax,
> > > > - Opt_actimeo,
> > > > - Opt_namelen,
> > > > - Opt_mountport,
> > > > - Opt_mountvers,
> > > > - Opt_minorversion,
> > > > -
> > > > - /* Mount options that take string arguments */
> > > > - Opt_nfsvers,
> > > > - Opt_sec, Opt_proto, Opt_mountproto, Opt_mounthost,
> > > > - Opt_addr, Opt_mountaddr, Opt_clientaddr,
> > > > - Opt_nconnect,
> > > > - Opt_lookupcache,
> > > > - Opt_fscache_uniq,
> > > > - Opt_local_lock,
> > > > -
> > > > - /* Special mount options */
> > > > - Opt_userspace, Opt_deprecated, Opt_sloppy,
> > > > -
> > > > - Opt_err
> > > > -};
> > > > -
> > > > -static const match_table_t nfs_mount_option_tokens = {
> > > > - { Opt_userspace, "bg" },
> > > > - { Opt_userspace, "fg" },
> > > > - { Opt_userspace, "retry=%s" },
> > > > -
> > > > - { Opt_sloppy, "sloppy" },
> > > > -
> > > > - { Opt_soft, "soft" },
> > > > - { Opt_softerr, "softerr" },
> > > > - { Opt_hard, "hard" },
> > > > - { Opt_deprecated, "intr" },
> > > > - { Opt_deprecated, "nointr" },
> > > > - { Opt_posix, "posix" },
> > > > - { Opt_noposix, "noposix" },
> > > > - { Opt_cto, "cto" },
> > > > - { Opt_nocto, "nocto" },
> > > > - { Opt_ac, "ac" },
> > > > - { Opt_noac, "noac" },
> > > > - { Opt_lock, "lock" },
> > > > - { Opt_nolock, "nolock" },
> > > > - { Opt_udp, "udp" },
> > > > - { Opt_tcp, "tcp" },
> > > > - { Opt_rdma, "rdma" },
> > > > - { Opt_acl, "acl" },
> > > > - { Opt_noacl, "noacl" },
> > > > - { Opt_rdirplus, "rdirplus" },
> > > > - { Opt_nordirplus, "nordirplus" },
> > > > - { Opt_sharecache, "sharecache" },
> > > > - { Opt_nosharecache, "nosharecache" },
> > > > - { Opt_resvport, "resvport" },
> > > > - { Opt_noresvport, "noresvport" },
> > > > - { Opt_fscache, "fsc" },
> > > > - { Opt_nofscache, "nofsc" },
> > > > - { Opt_migration, "migration" },
> > > > - { Opt_nomigration, "nomigration" },
> > > > -
> > > > - { Opt_port, "port=%s" },
> > > > - { Opt_rsize, "rsize=%s" },
> > > > - { Opt_wsize, "wsize=%s" },
> > > > - { Opt_bsize, "bsize=%s" },
> > > > - { Opt_timeo, "timeo=%s" },
> > > > - { Opt_retrans, "retrans=%s" },
> > > > - { Opt_acregmin, "acregmin=%s" },
> > > > - { Opt_acregmax, "acregmax=%s" },
> > > > - { Opt_acdirmin, "acdirmin=%s" },
> > > > - { Opt_acdirmax, "acdirmax=%s" },
> > > > - { Opt_actimeo, "actimeo=%s" },
> > > > - { Opt_namelen, "namlen=%s" },
> > > > - { Opt_mountport, "mountport=%s" },
> > > > - { Opt_mountvers, "mountvers=%s" },
> > > > - { Opt_minorversion, "minorversion=%s" },
> > > > -
> > > > - { Opt_nfsvers, "nfsvers=%s" },
> > > > - { Opt_nfsvers, "vers=%s" },
> > > > -
> > > > - { Opt_sec, "sec=%s" },
> > > > - { Opt_proto, "proto=%s" },
> > > > - { Opt_mountproto, "mountproto=%s" },
> > > > - { Opt_addr, "addr=%s" },
> > > > - { Opt_clientaddr, "clientaddr=%s" },
> > > > - { Opt_mounthost, "mounthost=%s" },
> > > > - { Opt_mountaddr, "mountaddr=%s" },
> > > > -
> > > > - { Opt_nconnect, "nconnect=%s" },
> > > > -
> > > > - { Opt_lookupcache, "lookupcache=%s" },
> > > > - { Opt_fscache_uniq, "fsc=%s" },
> > > > - { Opt_local_lock, "local_lock=%s" },
> > > > -
> > > > - /* The following needs to be listed after all other
> > > > options */
> > > > - { Opt_nfsvers, "v%s" },
> > > > -
> > > > - { Opt_err, NULL }
> > > > -};
> > > > -
> > > > -enum {
> > > > - Opt_xprt_udp, Opt_xprt_udp6, Opt_xprt_tcp,
> > > > Opt_xprt_tcp6,
> > > > Opt_xprt_rdma,
> > > > - Opt_xprt_rdma6,
> > > > -
> > > > - Opt_xprt_err
> > > > -};
> > > > -
> > > > -static const match_table_t nfs_xprt_protocol_tokens = {
> > > > - { Opt_xprt_udp, "udp" },
> > > > - { Opt_xprt_udp6, "udp6" },
> > > > - { Opt_xprt_tcp, "tcp" },
> > > > - { Opt_xprt_tcp6, "tcp6" },
> > > > - { Opt_xprt_rdma, "rdma" },
> > > > - { Opt_xprt_rdma6, "rdma6" },
> > > > -
> > > > - { Opt_xprt_err, NULL }
> > > > -};
> > > > -
> > > > -enum {
> > > > - Opt_sec_none, Opt_sec_sys,
> > > > - Opt_sec_krb5, Opt_sec_krb5i, Opt_sec_krb5p,
> > > > - Opt_sec_lkey, Opt_sec_lkeyi, Opt_sec_lkeyp,
> > > > - Opt_sec_spkm, Opt_sec_spkmi, Opt_sec_spkmp,
> > > > -
> > > > - Opt_sec_err
> > > > -};
> > > > -
> > > > -static const match_table_t nfs_secflavor_tokens = {
> > > > - { Opt_sec_none, "none" },
> > > > - { Opt_sec_none, "null" },
> > > > - { Opt_sec_sys, "sys" },
> > > > -
> > > > - { Opt_sec_krb5, "krb5" },
> > > > - { Opt_sec_krb5i, "krb5i" },
> > > > - { Opt_sec_krb5p, "krb5p" },
> > > > -
> > > > - { Opt_sec_lkey, "lkey" },
> > > > - { Opt_sec_lkeyi, "lkeyi" },
> > > > - { Opt_sec_lkeyp, "lkeyp" },
> > > > -
> > > > - { Opt_sec_spkm, "spkm3" },
> > > > - { Opt_sec_spkmi, "spkm3i" },
> > > > - { Opt_sec_spkmp, "spkm3p" },
> > > > -
> > > > - { Opt_sec_err, NULL }
> > > > -};
> > > > -
> > > > -enum {
> > > > - Opt_lookupcache_all, Opt_lookupcache_positive,
> > > > - Opt_lookupcache_none,
> > > > -
> > > > - Opt_lookupcache_err
> > > > -};
> > > > -
> > > > -static match_table_t nfs_lookupcache_tokens = {
> > > > - { Opt_lookupcache_all, "all" },
> > > > - { Opt_lookupcache_positive, "pos" },
> > > > - { Opt_lookupcache_positive, "positive" },
> > > > - { Opt_lookupcache_none, "none" },
> > > > -
> > > > - { Opt_lookupcache_err, NULL }
> > > > -};
> > > > -
> > > > -enum {
> > > > - Opt_local_lock_all, Opt_local_lock_flock,
> > > > Opt_local_lock_posix,
> > > > - Opt_local_lock_none,
> > > > -
> > > > - Opt_local_lock_err
> > > > -};
> > > > -
> > > > -static match_table_t nfs_local_lock_tokens = {
> > > > - { Opt_local_lock_all, "all" },
> > > > - { Opt_local_lock_flock, "flock" },
> > > > - { Opt_local_lock_posix, "posix" },
> > > > - { Opt_local_lock_none, "none" },
> > > > -
> > > > - { Opt_local_lock_err, NULL }
> > > > -};
> > > > -
> > > > -enum {
> > > > - Opt_vers_2, Opt_vers_3, Opt_vers_4, Opt_vers_4_0,
> > > > - Opt_vers_4_1, Opt_vers_4_2,
> > > > -
> > > > - Opt_vers_err
> > > > -};
> > > > -
> > > > -static match_table_t nfs_vers_tokens = {
> > > > - { Opt_vers_2, "2" },
> > > > - { Opt_vers_3, "3" },
> > > > - { Opt_vers_4, "4" },
> > > > - { Opt_vers_4_0, "4.0" },
> > > > - { Opt_vers_4_1, "4.1" },
> > > > - { Opt_vers_4_2, "4.2" },
> > > > -
> > > > - { Opt_vers_err, NULL }
> > > > -};
> > > >
> > > > static struct dentry *nfs_prepared_mount(struct
> > > > file_system_type
> > > > *fs_type,
> > > > int flags, const char *dev_name, void
> > > > *raw_data);
> > > > @@ -332,10 +109,6 @@ const struct super_operations nfs_sops = {
> > > > EXPORT_SYMBOL_GPL(nfs_sops);
> > > >
> > > > #if IS_ENABLED(CONFIG_NFS_V4)
> > > > -static void nfs4_validate_mount_flags(struct
> > > > nfs_parsed_mount_data
> > > > *);
> > > > -static int nfs4_validate_mount_data(void *options,
> > > > - struct nfs_parsed_mount_data *args, const char
> > > > *dev_name);
> > > > -
> > > > struct file_system_type nfs4_fs_type = {
> > > > .owner = THIS_MODULE,
> > > > .name = "nfs4",
> > > > @@ -932,141 +705,6 @@ void nfs_umount_begin(struct super_block
> > > > *sb)
> > > > }
> > > > EXPORT_SYMBOL_GPL(nfs_umount_begin);
> > > >
> > > > -static struct nfs_parsed_mount_data
> > > > *nfs_alloc_parsed_mount_data(void)
> > > > -{
> > > > - struct nfs_parsed_mount_data *data;
> > > > -
> > > > - data = kzalloc(sizeof(*data), GFP_KERNEL);
> > > > - if (data) {
> > > > - data->timeo = NFS_UNSPEC_TIMEO;
> > > > - data->retrans = NFS_UNSPEC_RETRANS;
> > > > - data->acregmin = NFS_DEF_ACREGMIN;
> > > > - data->acregmax = NFS_DEF_ACREGMAX;
> > > > - data->acdirmin = NFS_DEF_ACDIRMIN;
> > > > - data->acdirmax = NFS_DEF_ACDIRMAX;
> > > > - data->mount_server.port = NFS_UNSPEC_PORT;
> > > > - data->nfs_server.port = NFS_UNSPEC_PORT;
> > > > - data->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > > > - data->selected_flavor = RPC_AUTH_MAXFLAVOR;
> > > > - data->minorversion = 0;
> > > > - data->need_mount = true;
> > > > - data->net = current->nsproxy-
> > > > >net_ns;
> > > > - data->lsm_opts = NULL;
> > > > - }
> > > > - return data;
> > > > -}
> > > > -
> > > > -static void nfs_free_parsed_mount_data(struct
> > > > nfs_parsed_mount_data *data)
> > > > -{
> > > > - if (data) {
> > > > - kfree(data->client_address);
> > > > - kfree(data->mount_server.hostname);
> > > > - kfree(data->nfs_server.export_path);
> > > > - kfree(data->nfs_server.hostname);
> > > > - kfree(data->fscache_uniq);
> > > > - security_free_mnt_opts(&data->lsm_opts);
> > > > - kfree(data);
> > > > - }
> > > > -}
> > > > -
> > > > -/*
> > > > - * Sanity-check a server address provided by the mount
> > > > command.
> > > > - *
> > > > - * Address family must be initialized, and address must not be
> > > > - * the ANY address for that family.
> > > > - */
> > > > -static int nfs_verify_server_address(struct sockaddr *addr)
> > > > -{
> > > > - switch (addr->sa_family) {
> > > > - case AF_INET: {
> > > > - struct sockaddr_in *sa = (struct sockaddr_in
> > > > *)addr;
> > > > - return sa->sin_addr.s_addr !=
> > > > htonl(INADDR_ANY);
> > > > - }
> > > > - case AF_INET6: {
> > > > - struct in6_addr *sa = &((struct sockaddr_in6
> > > > *)addr)-
> > > > > sin6_addr;
> > > > - return !ipv6_addr_any(sa);
> > > > - }
> > > > - }
> > > > -
> > > > - dfprintk(MOUNT, "NFS: Invalid IP address specified\n");
> > > > - return 0;
> > > > -}
> > > > -
> > > > -/*
> > > > - * Select between a default port value and a user-specified
> > > > port
> > > > value.
> > > > - * If a zero value is set, then autobind will be used.
> > > > - */
> > > > -static void nfs_set_port(struct sockaddr *sap, int *port,
> > > > - const unsigned short
> > > > default_port)
> > > > -{
> > > > - if (*port == NFS_UNSPEC_PORT)
> > > > - *port = default_port;
> > > > -
> > > > - rpc_set_port(sap, *port);
> > > > -}
> > > > -
> > > > -/*
> > > > - * Sanity check the NFS transport protocol.
> > > > - *
> > > > - */
> > > > -static void nfs_validate_transport_protocol(struct
> > > > nfs_parsed_mount_data *mnt)
> > > > -{
> > > > - switch (mnt->nfs_server.protocol) {
> > > > - case XPRT_TRANSPORT_UDP:
> > > > - case XPRT_TRANSPORT_TCP:
> > > > - case XPRT_TRANSPORT_RDMA:
> > > > - break;
> > > > - default:
> > > > - mnt->nfs_server.protocol = XPRT_TRANSPORT_TCP;
> > > > - }
> > > > -}
> > > > -
> > > > -/*
> > > > - * For text based NFSv2/v3 mounts, the mount protocol
> > > > transport
> > > > default
> > > > - * settings should depend upon the specified NFS transport.
> > > > - */
> > > > -static void nfs_set_mount_transport_protocol(struct
> > > > nfs_parsed_mount_data *mnt)
> > > > -{
> > > > - nfs_validate_transport_protocol(mnt);
> > > > -
> > > > - if (mnt->mount_server.protocol == XPRT_TRANSPORT_UDP ||
> > > > - mnt->mount_server.protocol == XPRT_TRANSPORT_TCP)
> > > > - return;
> > > > - switch (mnt->nfs_server.protocol) {
> > > > - case XPRT_TRANSPORT_UDP:
> > > > - mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > - break;
> > > > - case XPRT_TRANSPORT_TCP:
> > > > - case XPRT_TRANSPORT_RDMA:
> > > > - mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > - }
> > > > -}
> > > > -
> > > > -/*
> > > > - * Add 'flavor' to 'auth_info' if not already present.
> > > > - * Returns true if 'flavor' ends up in the list, false
> > > > otherwise
> > > > - */
> > > > -static bool nfs_auth_info_add(struct nfs_auth_info *auth_info,
> > > > - rpc_authflavor_t flavor)
> > > > -{
> > > > - unsigned int i;
> > > > - unsigned int max_flavor_len = ARRAY_SIZE(auth_info-
> > > > >flavors);
> > > > -
> > > > - /* make sure this flavor isn't already in the list */
> > > > - for (i = 0; i < auth_info->flavor_len; i++) {
> > > > - if (flavor == auth_info->flavors[i])
> > > > - return true;
> > > > - }
> > > > -
> > > > - if (auth_info->flavor_len + 1 >= max_flavor_len) {
> > > > - dfprintk(MOUNT, "NFS: too many sec=
> > > > flavors\n");
> > > > - return false;
> > > > - }
> > > > -
> > > > - auth_info->flavors[auth_info->flavor_len++] = flavor;
> > > > - return true;
> > > > -}
> > > > -
> > > > /*
> > > > * Return true if 'match' is in auth_info or auth_info is
> > > > empty.
> > > > * Return false otherwise.
> > > > @@ -1087,627 +725,6 @@ bool nfs_auth_info_match(const struct
> > > > nfs_auth_info *auth_info,
> > > > }
> > > > EXPORT_SYMBOL_GPL(nfs_auth_info_match);
> > > >
> > > > -/*
> > > > - * Parse the value of the 'sec=' option.
> > > > - */
> > > > -static int nfs_parse_security_flavors(char *value,
> > > > - struct
> > > > nfs_parsed_mount_data
> > > > *mnt)
> > > > -{
> > > > - substring_t args[MAX_OPT_ARGS];
> > > > - rpc_authflavor_t pseudoflavor;
> > > > - char *p;
> > > > -
> > > > - dfprintk(MOUNT, "NFS: parsing sec=%s option\n", value);
> > > > -
> > > > - while ((p = strsep(&value, ":")) != NULL) {
> > > > - switch (match_token(p, nfs_secflavor_tokens,
> > > > args)) {
> > > > - case Opt_sec_none:
> > > > - pseudoflavor = RPC_AUTH_NULL;
> > > > - break;
> > > > - case Opt_sec_sys:
> > > > - pseudoflavor = RPC_AUTH_UNIX;
> > > > - break;
> > > > - case Opt_sec_krb5:
> > > > - pseudoflavor = RPC_AUTH_GSS_KRB5;
> > > > - break;
> > > > - case Opt_sec_krb5i:
> > > > - pseudoflavor = RPC_AUTH_GSS_KRB5I;
> > > > - break;
> > > > - case Opt_sec_krb5p:
> > > > - pseudoflavor = RPC_AUTH_GSS_KRB5P;
> > > > - break;
> > > > - case Opt_sec_lkey:
> > > > - pseudoflavor = RPC_AUTH_GSS_LKEY;
> > > > - break;
> > > > - case Opt_sec_lkeyi:
> > > > - pseudoflavor = RPC_AUTH_GSS_LKEYI;
> > > > - break;
> > > > - case Opt_sec_lkeyp:
> > > > - pseudoflavor = RPC_AUTH_GSS_LKEYP;
> > > > - break;
> > > > - case Opt_sec_spkm:
> > > > - pseudoflavor = RPC_AUTH_GSS_SPKM;
> > > > - break;
> > > > - case Opt_sec_spkmi:
> > > > - pseudoflavor = RPC_AUTH_GSS_SPKMI;
> > > > - break;
> > > > - case Opt_sec_spkmp:
> > > > - pseudoflavor = RPC_AUTH_GSS_SPKMP;
> > > > - break;
> > > > - default:
> > > > - dfprintk(MOUNT,
> > > > - "NFS: sec= option '%s' not
> > > > recognized\n", p);
> > > > - return 0;
> > > > - }
> > > > -
> > > > - if (!nfs_auth_info_add(&mnt->auth_info,
> > > > pseudoflavor))
> > > > - return 0;
> > > > - }
> > > > -
> > > > - return 1;
> > > > -}
> > > > -
> > > > -static int nfs_parse_version_string(char *string,
> > > > - struct nfs_parsed_mount_data *mnt,
> > > > - substring_t *args)
> > > > -{
> > > > - mnt->flags &= ~NFS_MOUNT_VER3;
> > > > - switch (match_token(string, nfs_vers_tokens, args)) {
> > > > - case Opt_vers_2:
> > > > - mnt->version = 2;
> > > > - break;
> > > > - case Opt_vers_3:
> > > > - mnt->flags |= NFS_MOUNT_VER3;
> > > > - mnt->version = 3;
> > > > - break;
> > > > - case Opt_vers_4:
> > > > - /* Backward compatibility option. In future,
> > > > - * the mount program should always supply
> > > > - * a NFSv4 minor version number.
> > > > - */
> > > > - mnt->version = 4;
> > > > - break;
> > > > - case Opt_vers_4_0:
> > > > - mnt->version = 4;
> > > > - mnt->minorversion = 0;
> > > > - break;
> > > > - case Opt_vers_4_1:
> > > > - mnt->version = 4;
> > > > - mnt->minorversion = 1;
> > > > - break;
> > > > - case Opt_vers_4_2:
> > > > - mnt->version = 4;
> > > > - mnt->minorversion = 2;
> > > > - break;
> > > > - default:
> > > > - return 0;
> > > > - }
> > > > - return 1;
> > > > -}
> > > > -
> > > > -static int nfs_get_option_str(substring_t args[], char
> > > > **option)
> > > > -{
> > > > - kfree(*option);
> > > > - *option = match_strdup(args);
> > > > - return !*option;
> > > > -}
> > > > -
> > > > -static int nfs_get_option_ul(substring_t args[], unsigned long
> > > > *option)
> > > > -{
> > > > - int rc;
> > > > - char *string;
> > > > -
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - return -ENOMEM;
> > > > - rc = kstrtoul(string, 10, option);
> > > > - kfree(string);
> > > > -
> > > > - return rc;
> > > > -}
> > > > -
> > > > -static int nfs_get_option_ul_bound(substring_t args[],
> > > > unsigned
> > > > long *option,
> > > > - unsigned long l_bound, unsigned long u_bound)
> > > > -{
> > > > - int ret;
> > > > -
> > > > - ret = nfs_get_option_ul(args, option);
> > > > - if (ret != 0)
> > > > - return ret;
> > > > - if (*option < l_bound || *option > u_bound)
> > > > - return -ERANGE;
> > > > - return 0;
> > > > -}
> > > > -
> > > > -/*
> > > > - * Error-check and convert a string of mount options from user
> > > > space into
> > > > - * a data structure. The whole mount string is processed; bad
> > > > options are
> > > > - * skipped as they are encountered. If there were no errors,
> > > > return 1;
> > > > - * otherwise return 0 (zero).
> > > > - */
> > > > -static int nfs_parse_mount_options(char *raw,
> > > > - struct nfs_parsed_mount_data
> > > > *mnt)
> > > > -{
> > > > - char *p, *string;
> > > > - int rc, sloppy = 0, invalid_option = 0;
> > > > - unsigned short protofamily = AF_UNSPEC;
> > > > - unsigned short mountfamily = AF_UNSPEC;
> > > > -
> > > > - if (!raw) {
> > > > - dfprintk(MOUNT, "NFS: mount options string was
> > > > NULL.\n");
> > > > - return 1;
> > > > - }
> > > > - dfprintk(MOUNT, "NFS: nfs mount opts='%s'\n", raw);
> > > > -
> > > > - rc = security_sb_eat_lsm_opts(raw, &mnt->lsm_opts);
> > > > - if (rc)
> > > > - goto out_security_failure;
> > > > -
> > > > - while ((p = strsep(&raw, ",")) != NULL) {
> > > > - substring_t args[MAX_OPT_ARGS];
> > > > - unsigned long option;
> > > > - int token;
> > > > -
> > > > - if (!*p)
> > > > - continue;
> > > > -
> > > > - dfprintk(MOUNT, "NFS: parsing nfs mount
> > > > option
> > > > '%s'\n", p);
> > > > -
> > > > - token = match_token(p, nfs_mount_option_tokens,
> > > > args);
> > > > - switch (token) {
> > > > -
> > > > - /*
> > > > - * boolean options: foo/nofoo
> > > > - */
> > > > - case Opt_soft:
> > > > - mnt->flags |= NFS_MOUNT_SOFT;
> > > > - mnt->flags &= ~NFS_MOUNT_SOFTERR;
> > > > - break;
> > > > - case Opt_softerr:
> > > > - mnt->flags |= NFS_MOUNT_SOFTERR;
> > > > - mnt->flags &= ~NFS_MOUNT_SOFT;
> > > > - break;
> > > > - case Opt_hard:
> > > > - mnt->flags &=
> > > > ~(NFS_MOUNT_SOFT|NFS_MOUNT_SOFTERR);
> > > > - break;
> > > > - case Opt_posix:
> > > > - mnt->flags |= NFS_MOUNT_POSIX;
> > > > - break;
> > > > - case Opt_noposix:
> > > > - mnt->flags &= ~NFS_MOUNT_POSIX;
> > > > - break;
> > > > - case Opt_cto:
> > > > - mnt->flags &= ~NFS_MOUNT_NOCTO;
> > > > - break;
> > > > - case Opt_nocto:
> > > > - mnt->flags |= NFS_MOUNT_NOCTO;
> > > > - break;
> > > > - case Opt_ac:
> > > > - mnt->flags &= ~NFS_MOUNT_NOAC;
> > > > - break;
> > > > - case Opt_noac:
> > > > - mnt->flags |= NFS_MOUNT_NOAC;
> > > > - break;
> > > > - case Opt_lock:
> > > > - mnt->flags &= ~NFS_MOUNT_NONLM;
> > > > - mnt->flags &= ~(NFS_MOUNT_LOCAL_FLOCK |
> > > > - NFS_MOUNT_LOCAL_FCNTL);
> > > > - break;
> > > > - case Opt_nolock:
> > > > - mnt->flags |= NFS_MOUNT_NONLM;
> > > > - mnt->flags |= (NFS_MOUNT_LOCAL_FLOCK |
> > > > - NFS_MOUNT_LOCAL_FCNTL);
> > > > - break;
> > > > - case Opt_udp:
> > > > - mnt->flags &= ~NFS_MOUNT_TCP;
> > > > - mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > - break;
> > > > - case Opt_tcp:
> > > > - mnt->flags |= NFS_MOUNT_TCP;
> > > > - mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > - break;
> > > > - case Opt_rdma:
> > > > - mnt->flags |= NFS_MOUNT_TCP; /* for
> > > > side
> > > > protocols */
> > > > - mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_RDMA;
> > > > - xprt_load_transport(p);
> > > > - break;
> > > > - case Opt_acl:
> > > > - mnt->flags &= ~NFS_MOUNT_NOACL;
> > > > - break;
> > > > - case Opt_noacl:
> > > > - mnt->flags |= NFS_MOUNT_NOACL;
> > > > - break;
> > > > - case Opt_rdirplus:
> > > > - mnt->flags &= ~NFS_MOUNT_NORDIRPLUS;
> > > > - break;
> > > > - case Opt_nordirplus:
> > > > - mnt->flags |= NFS_MOUNT_NORDIRPLUS;
> > > > - break;
> > > > - case Opt_sharecache:
> > > > - mnt->flags &= ~NFS_MOUNT_UNSHARED;
> > > > - break;
> > > > - case Opt_nosharecache:
> > > > - mnt->flags |= NFS_MOUNT_UNSHARED;
> > > > - break;
> > > > - case Opt_resvport:
> > > > - mnt->flags &= ~NFS_MOUNT_NORESVPORT;
> > > > - break;
> > > > - case Opt_noresvport:
> > > > - mnt->flags |= NFS_MOUNT_NORESVPORT;
> > > > - break;
> > > > - case Opt_fscache:
> > > > - mnt->options |= NFS_OPTION_FSCACHE;
> > > > - kfree(mnt->fscache_uniq);
> > > > - mnt->fscache_uniq = NULL;
> > > > - break;
> > > > - case Opt_nofscache:
> > > > - mnt->options &= ~NFS_OPTION_FSCACHE;
> > > > - kfree(mnt->fscache_uniq);
> > > > - mnt->fscache_uniq = NULL;
> > > > - break;
> > > > - case Opt_migration:
> > > > - mnt->options |= NFS_OPTION_MIGRATION;
> > > > - break;
> > > > - case Opt_nomigration:
> > > > - mnt->options &= ~NFS_OPTION_MIGRATION;
> > > > - break;
> > > > -
> > > > - /*
> > > > - * options that take numeric values
> > > > - */
> > > > - case Opt_port:
> > > > - if (nfs_get_option_ul(args, &option) ||
> > > > - option > USHRT_MAX)
> > > > - goto out_invalid_value;
> > > > - mnt->nfs_server.port = option;
> > > > - break;
> > > > - case Opt_rsize:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->rsize = option;
> > > > - break;
> > > > - case Opt_wsize:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->wsize = option;
> > > > - break;
> > > > - case Opt_bsize:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->bsize = option;
> > > > - break;
> > > > - case Opt_timeo:
> > > > - if (nfs_get_option_ul_bound(args,
> > > > &option, 1,
> > > > INT_MAX))
> > > > - goto out_invalid_value;
> > > > - mnt->timeo = option;
> > > > - break;
> > > > - case Opt_retrans:
> > > > - if (nfs_get_option_ul_bound(args,
> > > > &option, 0,
> > > > INT_MAX))
> > > > - goto out_invalid_value;
> > > > - mnt->retrans = option;
> > > > - break;
> > > > - case Opt_acregmin:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->acregmin = option;
> > > > - break;
> > > > - case Opt_acregmax:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->acregmax = option;
> > > > - break;
> > > > - case Opt_acdirmin:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->acdirmin = option;
> > > > - break;
> > > > - case Opt_acdirmax:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->acdirmax = option;
> > > > - break;
> > > > - case Opt_actimeo:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->acregmin = mnt->acregmax =
> > > > - mnt->acdirmin = mnt->acdirmax = option;
> > > > - break;
> > > > - case Opt_namelen:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - mnt->namlen = option;
> > > > - break;
> > > > - case Opt_mountport:
> > > > - if (nfs_get_option_ul(args, &option) ||
> > > > - option > USHRT_MAX)
> > > > - goto out_invalid_value;
> > > > - mnt->mount_server.port = option;
> > > > - break;
> > > > - case Opt_mountvers:
> > > > - if (nfs_get_option_ul(args, &option) ||
> > > > - option < NFS_MNT_VERSION ||
> > > > - option > NFS_MNT3_VERSION)
> > > > - goto out_invalid_value;
> > > > - mnt->mount_server.version = option;
> > > > - break;
> > > > - case Opt_minorversion:
> > > > - if (nfs_get_option_ul(args, &option))
> > > > - goto out_invalid_value;
> > > > - if (option > NFS4_MAX_MINOR_VERSION)
> > > > - goto out_invalid_value;
> > > > - mnt->minorversion = option;
> > > > - break;
> > > > -
> > > > - /*
> > > > - * options that take text values
> > > > - */
> > > > - case Opt_nfsvers:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - rc = nfs_parse_version_string(string,
> > > > mnt,
> > > > args);
> > > > - kfree(string);
> > > > - if (!rc)
> > > > - goto out_invalid_value;
> > > > - break;
> > > > - case Opt_sec:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - rc = nfs_parse_security_flavors(string,
> > > > mnt);
> > > > - kfree(string);
> > > > - if (!rc) {
> > > > - dfprintk(MOUNT,
> > > > "NFS: unrecognized "
> > > > - "security
> > > > flavor\n");
> > > > - return 0;
> > > > - }
> > > > - break;
> > > > - case Opt_proto:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - token = match_token(string,
> > > > - nfs_xprt_protocol_t
> > > > okens,
> > > > args);
> > > > -
> > > > - protofamily = AF_INET;
> > > > - switch (token) {
> > > > - case Opt_xprt_udp6:
> > > > - protofamily = AF_INET6;
> > > > - /* fall through */
> > > > - case Opt_xprt_udp:
> > > > - mnt->flags &= ~NFS_MOUNT_TCP;
> > > > - mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > - break;
> > > > - case Opt_xprt_tcp6:
> > > > - protofamily = AF_INET6;
> > > > - /* fall through */
> > > > - case Opt_xprt_tcp:
> > > > - mnt->flags |= NFS_MOUNT_TCP;
> > > > - mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > - break;
> > > > - case Opt_xprt_rdma6:
> > > > - protofamily = AF_INET6;
> > > > - /* fall through */
> > > > - case Opt_xprt_rdma:
> > > > - /* vector side protocols to TCP
> > > > */
> > > > - mnt->flags |= NFS_MOUNT_TCP;
> > > > - mnt->nfs_server.protocol =
> > > > XPRT_TRANSPORT_RDMA;
> > > > - xprt_load_transport(string);
> > > > - break;
> > > > - default:
> > > > - dfprintk(MOUNT,
> > > > "NFS: unrecognized "
> > > > - "transport
> > > > protocol\n");
> > > > - kfree(string);
> > > > - return 0;
> > > > - }
> > > > - kfree(string);
> > > > - break;
> > > > - case Opt_mountproto:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - token = match_token(string,
> > > > - nfs_xprt_protocol_t
> > > > okens,
> > > > args);
> > > > - kfree(string);
> > > > -
> > > > - mountfamily = AF_INET;
> > > > - switch (token) {
> > > > - case Opt_xprt_udp6:
> > > > - mountfamily = AF_INET6;
> > > > - /* fall through */
> > > > - case Opt_xprt_udp:
> > > > - mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > - break;
> > > > - case Opt_xprt_tcp6:
> > > > - mountfamily = AF_INET6;
> > > > - /* fall through */
> > > > - case Opt_xprt_tcp:
> > > > - mnt->mount_server.protocol =
> > > > XPRT_TRANSPORT_TCP;
> > > > - break;
> > > > - case Opt_xprt_rdma: /* not used for
> > > > side
> > > > protocols */
> > > > - default:
> > > > - dfprintk(MOUNT,
> > > > "NFS: unrecognized "
> > > > - "transport
> > > > protocol\n");
> > > > - return 0;
> > > > - }
> > > > - break;
> > > > - case Opt_addr:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - mnt->nfs_server.addrlen =
> > > > - rpc_pton(mnt->net, string,
> > > > strlen(string),
> > > > - (struct sockaddr *)
> > > > - &mnt-
> > > > >nfs_server.address,
> > > > - sizeof(mnt-
> > > > > nfs_server.address));
> > > > - kfree(string);
> > > > - if (mnt->nfs_server.addrlen == 0)
> > > > - goto out_invalid_address;
> > > > - break;
> > > > - case Opt_clientaddr:
> > > > - if (nfs_get_option_str(args, &mnt-
> > > > > client_address))
> > > > - goto out_nomem;
> > > > - break;
> > > > - case Opt_mounthost:
> > > > - if (nfs_get_option_str(args,
> > > > - &mnt-
> > > > > mount_server.hostname))
> > > > - goto out_nomem;
> > > > - break;
> > > > - case Opt_mountaddr:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - mnt->mount_server.addrlen =
> > > > - rpc_pton(mnt->net, string,
> > > > strlen(string),
> > > > - (struct sockaddr *)
> > > > - &mnt-
> > > > >mount_server.address,
> > > > - sizeof(mnt-
> > > > > mount_server.address));
> > > > - kfree(string);
> > > > - if (mnt->mount_server.addrlen == 0)
> > > > - goto out_invalid_address;
> > > > - break;
> > > > - case Opt_nconnect:
> > > > - if (nfs_get_option_ul_bound(args,
> > > > &option, 1,
> > > > NFS_MAX_CONNECTIONS))
> > > > - goto out_invalid_value;
> > > > - mnt->nfs_server.nconnect = option;
> > > > - break;
> > > > - case Opt_lookupcache:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - token = match_token(string,
> > > > - nfs_lookupcache_tokens,
> > > > args);
> > > > - kfree(string);
> > > > - switch (token) {
> > > > - case Opt_lookupcache_all:
> > > > - mnt->flags &=
> > > > ~(NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE);
> > > > - break;
> > > > - case Opt_lookupcache_positive:
> > > > - mnt->flags &=
> > > > ~NFS_MOUNT_LOOKUP_CACHE_NONE;
> > > > - mnt->flags |=
> > > > NFS_MOUNT_LOOKUP_CACHE_NONEG;
> > > > - break;
> > > > - case Opt_lookupcache_none:
> > > > - mnt->flags |=
> > > > NFS_MOUNT_LOOKUP_CACHE_NONEG|NFS_MOUNT_LOOKUP_CACHE_NONE;
> > > > - break;
> > > > - default:
> > > > - dfprintk(MOUNT,
> > > > "NFS: invalid
> > > > "
> > > > - "lookup
> > > > cache
> > > > argument\n");
> > > > - return 0;
> > > > - };
> > > > - break;
> > > > - case Opt_fscache_uniq:
> > > > - if (nfs_get_option_str(args, &mnt-
> > > > > fscache_uniq))
> > > > - goto out_nomem;
> > > > - mnt->options |= NFS_OPTION_FSCACHE;
> > > > - break;
> > > > - case Opt_local_lock:
> > > > - string = match_strdup(args);
> > > > - if (string == NULL)
> > > > - goto out_nomem;
> > > > - token = match_token(string,
> > > > nfs_local_lock_tokens,
> > > > - args);
> > > > - kfree(string);
> > > > - switch (token) {
> > > > - case Opt_local_lock_all:
> > > > - mnt->flags |=
> > > > (NFS_MOUNT_LOCAL_FLOCK |
> > > > - NFS_MOUNT_LOCAL_
> > > > FCNTL);
> > > > - break;
> > > > - case Opt_local_lock_flock:
> > > > - mnt->flags |=
> > > > NFS_MOUNT_LOCAL_FLOCK;
> > > > - break;
> > > > - case Opt_local_lock_posix:
> > > > - mnt->flags |=
> > > > NFS_MOUNT_LOCAL_FCNTL;
> > > > - break;
> > > > - case Opt_local_lock_none:
> > > > - mnt->flags &=
> > > > ~(NFS_MOUNT_LOCAL_FLOCK |
> > > > - NFS_MOUNT_LOCAL
> > > > _FCNTL);
> > > > - break;
> > > > - default:
> > > > - dfprintk(MOUNT, "NFS: invalid
> > > > "
> > > > - "local_lock
> > > > argument\n");
> > > > - return 0;
> > > > - };
> > > > - break;
> > > > -
> > > > - /*
> > > > - * Special options
> > > > - */
> > > > - case Opt_sloppy:
> > > > - sloppy = 1;
> > > > - dfprintk(MOUNT, "NFS: relaxing
> > > > parsing
> > > > rules\n");
> > > > - break;
> > > > - case Opt_userspace:
> > > > - case Opt_deprecated:
> > > > - dfprintk(MOUNT, "NFS: ignoring mount
> > > > option "
> > > > - "'%s'\n", p);
> > > > - break;
> > > > -
> > > > - default:
> > > > - invalid_option = 1;
> > > > - dfprintk(MOUNT, "NFS: unrecognized
> > > > mount
> > > > option "
> > > > - "'%s'\n", p);
> > > > - }
> > > > - }
> > > > -
> > > > - if (!sloppy && invalid_option)
> > > > - return 0;
> > > > -
> > > > - if (mnt->minorversion && mnt->version != 4)
> > > > - goto out_minorversion_mismatch;
> > > > -
> > > > - if (mnt->options & NFS_OPTION_MIGRATION &&
> > > > - (mnt->version != 4 || mnt->minorversion != 0))
> > > > - goto out_migration_misuse;
> > > > -
> > > > - /*
> > > > - * verify that any proto=/mountproto= options match the
> > > > address
> > > > - * families in the addr=/mountaddr= options.
> > > > - */
> > > > - if (protofamily != AF_UNSPEC &&
> > > > - protofamily != mnt->nfs_server.address.ss_family)
> > > > - goto out_proto_mismatch;
> > > > -
> > > > - if (mountfamily != AF_UNSPEC) {
> > > > - if (mnt->mount_server.addrlen) {
> > > > - if (mountfamily != mnt-
> > > > > mount_server.address.ss_family)
> > > > - goto out_mountproto_mismatch;
> > > > - } else {
> > > > - if (mountfamily != mnt-
> > > > > nfs_server.address.ss_family)
> > > > - goto out_mountproto_mismatch;
> > > > - }
> > > > - }
> > > > -
> > > > - return 1;
> > > > -
> > > > -out_mountproto_mismatch:
> > > > - printk(KERN_INFO "NFS: mount server address does not
> > > > match
> > > > mountproto= "
> > > > - "option\n");
> > > > - return 0;
> > > > -out_proto_mismatch:
> > > > - printk(KERN_INFO "NFS: server address does not match
> > > > proto=
> > > > option\n");
> > > > - return 0;
> > > > -out_invalid_address:
> > > > - printk(KERN_INFO "NFS: bad IP address specified: %s\n",
> > > > p);
> > > > - return 0;
> > > > -out_invalid_value:
> > > > - printk(KERN_INFO "NFS: bad mount option value
> > > > specified: %s\n",
> > > > p);
> > > > - return 0;
> > > > -out_minorversion_mismatch:
> > > > - printk(KERN_INFO "NFS: mount option vers=%u does not
> > > > support "
> > > > - "minorversion=%u\n", mnt->version,
> > > > mnt-
> > > > > minorversion);
> > > > - return 0;
> > > > -out_migration_misuse:
> > > > - printk(KERN_INFO
> > > > - "NFS: 'migration' not supported for this NFS
> > > > version\n");
> > > > - return 0;
> > > > -out_nomem:
> > > > - printk(KERN_INFO "NFS: not enough memory to parse
> > > > option\n");
> > > > - return 0;
> > > > -out_security_failure:
> > > > - printk(KERN_INFO "NFS: security options invalid: %d\n",
> > > > rc);
> > > > - return 0;
> > > > -}
> > > > -
> > > > /*
> > > > * Ensure that a specified authtype in args->auth_info is
> > > > supported
> > > > by
> > > > * the server. Returns 0 and sets args->selected_flavor if it's
> > > > ok,
> > > > and
> > > > @@ -1908,327 +925,6 @@ struct dentry *nfs_try_mount(int flags,
> > > > const char *dev_name,
> > > > }
> > > > EXPORT_SYMBOL_GPL(nfs_try_mount);
> > > >
> > > > -/*
> > > > - * Split "dev_name" into "hostname:export_path".
> > > > - *
> > > > - * The leftmost colon demarks the split between the server's
> > > > hostname
> > > > - * and the export path. If the hostname starts with a left
> > > > square
> > > > - * bracket, then it may contain colons.
> > > > - *
> > > > - * Note: caller frees hostname and export path, even on error.
> > > > - */
> > > > -static int nfs_parse_devname(const char *dev_name,
> > > > - char **hostname, size_t maxnamlen,
> > > > - char **export_path, size_t
> > > > maxpathlen)
> > > > -{
> > > > - size_t len;
> > > > - char *end;
> > > > -
> > > > - if (unlikely(!dev_name || !*dev_name)) {
> > > > - dfprintk(MOUNT, "NFS: device name not
> > > > specified\n");
> > > > - return -EINVAL;
> > > > - }
> > > > -
> > > > - /* Is the host name protected with square brakcets? */
> > > > - if (*dev_name == '[') {
> > > > - end = strchr(++dev_name, ']');
> > > > - if (end == NULL || end[1] != ':')
> > > > - goto out_bad_devname;
> > > > -
> > > > - len = end - dev_name;
> > > > - end++;
> > > > - } else {
> > > > - char *comma;
> > > > -
> > > > - end = strchr(dev_name, ':');
> > > > - if (end == NULL)
> > > > - goto out_bad_devname;
> > > > - len = end - dev_name;
> > > > -
> > > > - /* kill possible hostname list: not supported
> > > > */
> > > > - comma = strchr(dev_name, ',');
> > > > - if (comma != NULL && comma < end)
> > > > - len = comma - dev_name;
> > > > - }
> > > > -
> > > > - if (len > maxnamlen)
> > > > - goto out_hostname;
> > > > -
> > > > - /* N.B. caller will free nfs_server.hostname in all
> > > > cases */
> > > > - *hostname = kstrndup(dev_name, len, GFP_KERNEL);
> > > > - if (*hostname == NULL)
> > > > - goto out_nomem;
> > > > - len = strlen(++end);
> > > > - if (len > maxpathlen)
> > > > - goto out_path;
> > > > - *export_path = kstrndup(end, len, GFP_KERNEL);
> > > > - if (!*export_path)
> > > > - goto out_nomem;
> > > > -
> > > > - dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", *export_path);
> > > > - return 0;
> > > > -
> > > > -out_bad_devname:
> > > > - dfprintk(MOUNT, "NFS: device name not in host:path
> > > > format\n");
> > > > - return -EINVAL;
> > > > -
> > > > -out_nomem:
> > > > - dfprintk(MOUNT, "NFS: not enough memory to parse device
> > > > name\n");
> > > > - return -ENOMEM;
> > > > -
> > > > -out_hostname:
> > > > - dfprintk(MOUNT, "NFS: server hostname too long\n");
> > > > - return -ENAMETOOLONG;
> > > > -
> > > > -out_path:
> > > > - dfprintk(MOUNT, "NFS: export pathname too long\n");
> > > > - return -ENAMETOOLONG;
> > > > -}
> > > > -
> > > > -/*
> > > > - * Validate the NFS2/NFS3 mount data
> > > > - * - fills in the mount root filehandle
> > > > - *
> > > > - * For option strings, user space handles the following
> > > > behaviors:
> > > > - *
> > > > - * + DNS: mapping server host name to IP address ("addr="
> > > > option)
> > > > - *
> > > > - * + failure mode: how to behave if a mount request can't be
> > > > handled
> > > > - * immediately ("fg/bg" option)
> > > > - *
> > > > - * + retry: how often to retry a mount request ("retry="
> > > > option)
> > > > - *
> > > > - * + breaking back: trying proto=udp after proto=tcp, v2 after
> > > > v3,
> > > > - * mountproto=tcp after mountproto=udp, and so on
> > > > - */
> > > > -static int nfs23_validate_mount_data(void *options,
> > > > - struct
> > > > nfs_parsed_mount_data
> > > > *args,
> > > > - struct nfs_fh *mntfh,
> > > > - const char *dev_name)
> > > > -{
> > > > - struct nfs_mount_data *data = (struct nfs_mount_data
> > > > *)options;
> > > > - struct sockaddr *sap = (struct sockaddr *)&args-
> > > > > nfs_server.address;
> > > > - int extra_flags = NFS_MOUNT_LEGACY_INTERFACE;
> > > > -
> > > > - if (data == NULL)
> > > > - goto out_no_data;
> > > > -
> > > > - args->version = NFS_DEFAULT_VERSION;
> > > > - switch (data->version) {
> > > > - case 1:
> > > > - data->namlen = 0; /* fall through */
> > > > - case 2:
> > > > - data->bsize = 0; /* fall through */
> > > > - case 3:
> > > > - if (data->flags & NFS_MOUNT_VER3)
> > > > - goto out_no_v3;
> > > > - data->root.size = NFS2_FHSIZE;
> > > > - memcpy(data->root.data, data->old_root.data,
> > > > NFS2_FHSIZE);
> > > > - /* Turn off security negotiation */
> > > > - extra_flags |= NFS_MOUNT_SECFLAVOUR;
> > > > - /* fall through */
> > > > - case 4:
> > > > - if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > > > - goto out_no_sec;
> > > > - /* fall through */
> > > > - case 5:
> > > > - memset(data->context, 0, sizeof(data-
> > > > >context));
> > > > - /* fall through */
> > > > - case 6:
> > > > - if (data->flags & NFS_MOUNT_VER3) {
> > > > - if (data->root.size > NFS3_FHSIZE ||
> > > > data-
> > > > > root.size == 0)
> > > > - goto out_invalid_fh;
> > > > - mntfh->size = data->root.size;
> > > > - args->version = 3;
> > > > - } else {
> > > > - mntfh->size = NFS2_FHSIZE;
> > > > - args->version = 2;
> > > > - }
> > > > -
> > > > -
> > > > - memcpy(mntfh->data, data->root.data, mntfh-
> > > > >size);
> > > > - if (mntfh->size < sizeof(mntfh->data))
> > > > - memset(mntfh->data + mntfh->size, 0,
> > > > - sizeof(mntfh->data) - mntfh-
> > > > >size);
> > > > -
> > > > - /*
> > > > - * Translate to nfs_parsed_mount_data, which
> > > > nfs_fill_super
> > > > - * can deal with.
> > > > - */
> > > > - args->flags = data->flags &
> > > > NFS_MOUNT_FLAGMASK;
> > > > - args->flags |= extra_flags;
> > > > - args->rsize = data->rsize;
> > > > - args->wsize = data->wsize;
> > > > - args->timeo = data->timeo;
> > > > - args->retrans = data->retrans;
> > > > - args->acregmin = data->acregmin;
> > > > - args->acregmax = data->acregmax;
> > > > - args->acdirmin = data->acdirmin;
> > > > - args->acdirmax = data->acdirmax;
> > > > - args->need_mount = false;
> > > > -
> > > > - memcpy(sap, &data->addr, sizeof(data->addr));
> > > > - args->nfs_server.addrlen = sizeof(data->addr);
> > > > - args->nfs_server.port = ntohs(data-
> > > > >addr.sin_port);
> > > > - if (sap->sa_family != AF_INET ||
> > > > - !nfs_verify_server_address(sap))
> > > > - goto out_no_address;
> > > > -
> > > > - if (!(data->flags & NFS_MOUNT_TCP))
> > > > - args->nfs_server.protocol =
> > > > XPRT_TRANSPORT_UDP;
> > > > - /* N.B. caller will free nfs_server.hostname in
> > > > all
> > > > cases */
> > > > - args->nfs_server.hostname = kstrdup(data-
> > > > >hostname,
> > > > GFP_KERNEL);
> > > > - args->namlen = data->namlen;
> > > > - args->bsize = data->bsize;
> > > > -
> > > > - if (data->flags & NFS_MOUNT_SECFLAVOUR)
> > > > - args->selected_flavor = data-
> > > > >pseudoflavor;
> > > > - else
> > > > - args->selected_flavor = RPC_AUTH_UNIX;
> > > > - if (!args->nfs_server.hostname)
> > > > - goto out_nomem;
> > > > -
> > > > - if (!(data->flags & NFS_MOUNT_NONLM))
> > > > - args->flags &= ~(NFS_MOUNT_LOCAL_FLOCK|
> > > > - NFS_MOUNT_LOCAL_FCNTL)
> > > > ;
> > > > - else
> > > > - args->flags |= (NFS_MOUNT_LOCAL_FLOCK|
> > > > - NFS_MOUNT_LOCAL_FCNTL);
> > > > - /*
> > > > - * The legacy version 6 binary mount data from
> > > > userspace has a
> > > > - * field used only to transport selinux
> > > > information
> > > > into the
> > > > - * the kernel. To continue to support that
> > > > functionality we
> > > > - * have a touch of selinux knowledge here in
> > > > the NFS
> > > > code. The
> > > > - * userspace code converted context=blah to
> > > > just blah
> > > > so we are
> > > > - * converting back to the full string selinux
> > > > understands.
> > > > - */
> > > > - if (data->context[0]){
> > > > -#ifdef CONFIG_SECURITY_SELINUX
> > > > - int rc;
> > > > - data->context[NFS_MAX_CONTEXT_LEN] =
> > > > '\0';
> > > > - rc = security_add_mnt_opt("context",
> > > > data-
> > > > > context,
> > > > - strlen(data->context),
> > > > &args-
> > > > > lsm_opts);
> > > > - if (rc)
> > > > - return rc;
> > > > -#else
> > > > - return -EINVAL;
> > > > -#endif
> > > > - }
> > > > -
> > > > - break;
> > > > - default:
> > > > - return NFS_TEXT_DATA;
> > > > - }
> > > > -
> > > > - return 0;
> > > > -
> > > > -out_no_data:
> > > > - dfprintk(MOUNT, "NFS: mount program didn't pass any
> > > > mount
> > > > data\n");
> > > > - return -EINVAL;
> > > > -
> > > > -out_no_v3:
> > > > - dfprintk(MOUNT, "NFS: nfs_mount_data version %d does
> > > > not
> > > > support v3\n",
> > > > - data->version);
> > > > - return -EINVAL;
> > > > -
> > > > -out_no_sec:
> > > > - dfprintk(MOUNT, "NFS: nfs_mount_data version supports
> > > > only
> > > > AUTH_SYS\n");
> > > > - return -EINVAL;
> > > > -
> > > > -out_nomem:
> > > > - dfprintk(MOUNT, "NFS: not enough memory to handle mount
> > > > options\n");
> > > > - return -ENOMEM;
> > > > -
> > > > -out_no_address:
> > > > - dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > > > address\n");
> > > > - return -EINVAL;
> > > > -
> > > > -out_invalid_fh:
> > > > - dfprintk(MOUNT, "NFS: invalid root filehandle\n");
> > > > - return -EINVAL;
> > > > -}
> > > > -
> > > > -#if IS_ENABLED(CONFIG_NFS_V4)
> > > > -static int nfs_validate_mount_data(struct file_system_type
> > > > *fs_type,
> > > > - void *options,
> > > > - struct nfs_parsed_mount_data
> > > > *args,
> > > > - struct nfs_fh *mntfh,
> > > > - const char *dev_name)
> > > > -{
> > > > - if (fs_type == &nfs_fs_type)
> > > > - return nfs23_validate_mount_data(options, args,
> > > > mntfh,
> > > > dev_name);
> > > > - return nfs4_validate_mount_data(options, args,
> > > > dev_name);
> > > > -}
> > > > -#else
> > > > -static int nfs_validate_mount_data(struct file_system_type
> > > > *fs_type,
> > > > - void *options,
> > > > - struct nfs_parsed_mount_data
> > > > *args,
> > > > - struct nfs_fh *mntfh,
> > > > - const char *dev_name)
> > > > -{
> > > > - return nfs23_validate_mount_data(options, args, mntfh,
> > > > dev_name);
> > > > -}
> > > > -#endif
> > > > -
> > > > -static int nfs_validate_text_mount_data(void *options,
> > > > - struct
> > > > nfs_parsed_mount_data
> > > > *args,
> > > > - const char *dev_name)
> > > > -{
> > > > - int port = 0;
> > > > - int max_namelen = PAGE_SIZE;
> > > > - int max_pathlen = NFS_MAXPATHLEN;
> > > > - struct sockaddr *sap = (struct sockaddr *)&args-
> > > > > nfs_server.address;
> > > > -
> > > > - if (nfs_parse_mount_options((char *)options, args) ==
> > > > 0)
> > > > - return -EINVAL;
> > > > -
> > > > - if (!nfs_verify_server_address(sap))
> > > > - goto out_no_address;
> > > > -
> > > > - if (args->version == 4) {
> > > > -#if IS_ENABLED(CONFIG_NFS_V4)
> > > > - if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_RDMA)
> > > > - port = NFS_RDMA_PORT;
> > > > - else
> > > > - port = NFS_PORT;
> > > > - max_namelen = NFS4_MAXNAMLEN;
> > > > - max_pathlen = NFS4_MAXPATHLEN;
> > > > - nfs_validate_transport_protocol(args);
> > > > - if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_UDP)
> > > > - goto out_invalid_transport_udp;
> > > > - nfs4_validate_mount_flags(args);
> > > > -#else
> > > > - goto out_v4_not_compiled;
> > > > -#endif /* CONFIG_NFS_V4 */
> > > > - } else {
> > > > - nfs_set_mount_transport_protocol(args);
> > > > - if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_RDMA)
> > > > - port = NFS_RDMA_PORT;
> > > > - }
> > > > -
> > > > - nfs_set_port(sap, &args->nfs_server.port, port);
> > > > -
> > > > - return nfs_parse_devname(dev_name,
> > > > - &args->nfs_server.hostname,
> > > > - max_namelen,
> > > > - &args-
> > > > >nfs_server.export_path,
> > > > - max_pathlen);
> > > > -
> > > > -#if !IS_ENABLED(CONFIG_NFS_V4)
> > > > -out_v4_not_compiled:
> > > > - dfprintk(MOUNT, "NFS: NFSv4 is not compiled into
> > > > kernel\n");
> > > > - return -EPROTONOSUPPORT;
> > > > -#else
> > > > -out_invalid_transport_udp:
> > > > - dfprintk(MOUNT, "NFSv4: Unsupported transport protocol
> > > > udp\n");
> > > > - return -EINVAL;
> > > > -#endif /* !CONFIG_NFS_V4 */
> > > > -
> > > > -out_no_address:
> > > > - dfprintk(MOUNT, "NFS: mount program didn't pass remote
> > > > address\n");
> > > > - return -EINVAL;
> > > > -}
> > > > -
> > > > #define NFS_REMOUNT_CMP_FLAGMASK ~(NFS_MOUNT_INTR \
> > > > | NFS_MOUNT_SECURE \
> > > > | NFS_MOUNT_TCP \
> > > > @@ -2719,113 +1415,6 @@ nfs_prepared_mount(struct
> > > > file_system_type
> > > > *fs_type, int flags,
> > > >
> > > > #if IS_ENABLED(CONFIG_NFS_V4)
> > > >
> > > > -static void nfs4_validate_mount_flags(struct
> > > > nfs_parsed_mount_data
> > > > *args)
> > > > -{
> > > > - args->flags &=
> > > > ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3|
> > > > - NFS_MOUNT_LOCAL_FLOCK|NFS_MOUNT_LOCAL_
> > > > FCNTL);
> > > > -}
> > > > -
> > > > -/*
> > > > - * Validate NFSv4 mount options
> > > > - */
> > > > -static int nfs4_validate_mount_data(void *options,
> > > > - struct
> > > > nfs_parsed_mount_data *args,
> > > > - const char *dev_name)
> > > > -{
> > > > - struct sockaddr *sap = (struct sockaddr *)&args-
> > > > > nfs_server.address;
> > > > - struct nfs4_mount_data *data = (struct nfs4_mount_data
> > > > *)options;
> > > > - char *c;
> > > > -
> > > > - if (data == NULL)
> > > > - goto out_no_data;
> > > > -
> > > > - args->version = 4;
> > > > -
> > > > - switch (data->version) {
> > > > - case 1:
> > > > - if (data->host_addrlen > sizeof(args-
> > > > > nfs_server.address))
> > > > - goto out_no_address;
> > > > - if (data->host_addrlen == 0)
> > > > - goto out_no_address;
> > > > - args->nfs_server.addrlen = data->host_addrlen;
> > > > - if (copy_from_user(sap, data->host_addr, data-
> > > > > host_addrlen))
> > > > - return -EFAULT;
> > > > - if (!nfs_verify_server_address(sap))
> > > > - goto out_no_address;
> > > > - args->nfs_server.port = ntohs(((struct
> > > > sockaddr_in
> > > > *)sap)->sin_port);
> > > > -
> > > > - if (data->auth_flavourlen) {
> > > > - rpc_authflavor_t pseudoflavor;
> > > > - if (data->auth_flavourlen > 1)
> > > > - goto out_inval_auth;
> > > > - if (copy_from_user(&pseudoflavor,
> > > > - data->auth_flavours,
> > > > - sizeof(pseudoflavor)
> > > > ))
> > > > - return -EFAULT;
> > > > - args->selected_flavor = pseudoflavor;
> > > > - } else
> > > > - args->selected_flavor = RPC_AUTH_UNIX;
> > > > -
> > > > - c = strndup_user(data->hostname.data,
> > > > NFS4_MAXNAMLEN);
> > > > - if (IS_ERR(c))
> > > > - return PTR_ERR(c);
> > > > - args->nfs_server.hostname = c;
> > > > -
> > > > - c = strndup_user(data->mnt_path.data,
> > > > NFS4_MAXPATHLEN);
> > > > - if (IS_ERR(c))
> > > > - return PTR_ERR(c);
> > > > - args->nfs_server.export_path = c;
> > > > - dfprintk(MOUNT, "NFS: MNTPATH: '%s'\n", c);
> > > > -
> > > > - c = strndup_user(data->client_addr.data, 16);
> > > > - if (IS_ERR(c))
> > > > - return PTR_ERR(c);
> > > > - args->client_address = c;
> > > > -
> > > > - /*
> > > > - * Translate to nfs_parsed_mount_data, which
> > > > nfs4_fill_super
> > > > - * can deal with.
> > > > - */
> > > > -
> > > > - args->flags = data->flags &
> > > > NFS4_MOUNT_FLAGMASK;
> > > > - args->rsize = data->rsize;
> > > > - args->wsize = data->wsize;
> > > > - args->timeo = data->timeo;
> > > > - args->retrans = data->retrans;
> > > > - args->acregmin = data->acregmin;
> > > > - args->acregmax = data->acregmax;
> > > > - args->acdirmin = data->acdirmin;
> > > > - args->acdirmax = data->acdirmax;
> > > > - args->nfs_server.protocol = data->proto;
> > > > - nfs_validate_transport_protocol(args);
> > > > - if (args->nfs_server.protocol ==
> > > > XPRT_TRANSPORT_UDP)
> > > > - goto out_invalid_transport_udp;
> > > > -
> > > > - break;
> > > > - default:
> > > > - return NFS_TEXT_DATA;
> > > > - }
> > > > -
> > > > - return 0;
> > > > -
> > > > -out_no_data:
> > > > - dfprintk(MOUNT, "NFS4: mount program didn't pass any
> > > > mount
> > > > data\n");
> > > > - return -EINVAL;
> > > > -
> > > > -out_inval_auth:
> > > > - dfprintk(MOUNT, "NFS4: Invalid number of RPC auth
> > > > flavours
> > > > %d\n",
> > > > - data->auth_flavourlen);
> > > > - return -EINVAL;
> > > > -
> > > > -out_no_address:
> > > > - dfprintk(MOUNT, "NFS4: mount program didn't pass remote
> > > > address\n");
> > > > - return -EINVAL;
> > > > -
> > > > -out_invalid_transport_udp:
> > > > - dfprintk(MOUNT, "NFSv4: Unsupported transport protocol
> > > > udp\n");
> > > > - return -EINVAL;
> > > > -}
> > > > -
> > > > /*
> > > > * NFS v4 module parameters need to stay in the
> > > > * NFS client for backwards compatibility
> > > > --
> > > > 2.17.2
> > > >
> > >
> > > --
> > > Chuck Lever
> > > [email protected]
> > >
> > >
> > >
> > --
> > Trond Myklebust
> > Linux NFS client maintainer, Hammerspace
> > [email protected]
> >
> >
--
Trond Myklebust
Linux NFS client maintainer, Hammerspace
[email protected]