9p: convert to the new mount API
Convert 9p to the new mount API. This patch consolidates all parsing into fs/9p/v9fs.c, which stores all results into a filesystem context which can be passed to the various transports as needed. Some of the parsing helper functions such as get_cache_mode() have been eliminated in favor of using the new mount API's enum param type, for simplicity. Signed-off-by: Eric Sandeen <sandeen@redhat.com> Message-ID: <20251010214222.1347785-5-sandeen@redhat.com> [ Dominique: handled source explicitly as per follow-up discussion ] Signed-off-by: Dominique Martinet <asmadeus@codewreck.org>master
parent
075e8bd412
commit
1f3e4142c0
568
fs/9p/v9fs.c
568
fs/9p/v9fs.c
|
|
@ -13,7 +13,8 @@
|
|||
#include <linux/fs.h>
|
||||
#include <linux/sched.h>
|
||||
#include <linux/cred.h>
|
||||
#include <linux/parser.h>
|
||||
#include <linux/fs_parser.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/seq_file.h>
|
||||
#include <net/9p/9p.h>
|
||||
|
|
@ -33,6 +34,10 @@ struct kmem_cache *v9fs_inode_cache;
|
|||
*/
|
||||
|
||||
enum {
|
||||
/* Mount-point source, we need to handle this explicitly because
|
||||
* the code below accepts unknown args and the vfs layer only handles
|
||||
* source if we rejected it as EINVAL */
|
||||
Opt_source,
|
||||
/* Options that take integer arguments */
|
||||
Opt_debug, Opt_dfltuid, Opt_dfltgid, Opt_afid,
|
||||
/* String options */
|
||||
|
|
@ -43,55 +48,81 @@ enum {
|
|||
Opt_access, Opt_posixacl,
|
||||
/* Lock timeout option */
|
||||
Opt_locktimeout,
|
||||
/* Error token */
|
||||
Opt_err
|
||||
|
||||
/* Client options */
|
||||
Opt_msize, Opt_trans, Opt_legacy, Opt_version,
|
||||
|
||||
/* fd transport options */
|
||||
/* Options that take integer arguments */
|
||||
Opt_rfdno, Opt_wfdno,
|
||||
/* Options that take no arguments */
|
||||
|
||||
/* rdma transport options */
|
||||
/* Options that take integer arguments */
|
||||
Opt_rq_depth, Opt_sq_depth, Opt_timeout,
|
||||
|
||||
/* Options for both fd and rdma transports */
|
||||
Opt_port, Opt_privport,
|
||||
};
|
||||
|
||||
static const match_table_t tokens = {
|
||||
{Opt_debug, "debug=%x"},
|
||||
{Opt_dfltuid, "dfltuid=%u"},
|
||||
{Opt_dfltgid, "dfltgid=%u"},
|
||||
{Opt_afid, "afid=%u"},
|
||||
{Opt_uname, "uname=%s"},
|
||||
{Opt_remotename, "aname=%s"},
|
||||
{Opt_nodevmap, "nodevmap"},
|
||||
{Opt_noxattr, "noxattr"},
|
||||
{Opt_directio, "directio"},
|
||||
{Opt_ignoreqv, "ignoreqv"},
|
||||
{Opt_cache, "cache=%s"},
|
||||
{Opt_cachetag, "cachetag=%s"},
|
||||
{Opt_access, "access=%s"},
|
||||
{Opt_posixacl, "posixacl"},
|
||||
{Opt_locktimeout, "locktimeout=%u"},
|
||||
{Opt_err, NULL}
|
||||
static const struct constant_table p9_versions[] = {
|
||||
{ "9p2000", p9_proto_legacy },
|
||||
{ "9p2000.u", p9_proto_2000u },
|
||||
{ "9p2000.L", p9_proto_2000L },
|
||||
{}
|
||||
};
|
||||
|
||||
/* Interpret mount options for cache mode */
|
||||
static int get_cache_mode(char *s)
|
||||
{
|
||||
int version = -EINVAL;
|
||||
static const struct constant_table p9_cache_mode[] = {
|
||||
{ "loose", CACHE_SC_LOOSE },
|
||||
{ "fscache", CACHE_SC_FSCACHE },
|
||||
{ "mmap", CACHE_SC_MMAP },
|
||||
{ "readahead", CACHE_SC_READAHEAD },
|
||||
{ "none", CACHE_SC_NONE },
|
||||
{}
|
||||
};
|
||||
|
||||
if (!strcmp(s, "loose")) {
|
||||
version = CACHE_SC_LOOSE;
|
||||
p9_debug(P9_DEBUG_9P, "Cache mode: loose\n");
|
||||
} else if (!strcmp(s, "fscache")) {
|
||||
version = CACHE_SC_FSCACHE;
|
||||
p9_debug(P9_DEBUG_9P, "Cache mode: fscache\n");
|
||||
} else if (!strcmp(s, "mmap")) {
|
||||
version = CACHE_SC_MMAP;
|
||||
p9_debug(P9_DEBUG_9P, "Cache mode: mmap\n");
|
||||
} else if (!strcmp(s, "readahead")) {
|
||||
version = CACHE_SC_READAHEAD;
|
||||
p9_debug(P9_DEBUG_9P, "Cache mode: readahead\n");
|
||||
} else if (!strcmp(s, "none")) {
|
||||
version = CACHE_SC_NONE;
|
||||
p9_debug(P9_DEBUG_9P, "Cache mode: none\n");
|
||||
} else if (kstrtoint(s, 0, &version) != 0) {
|
||||
version = -EINVAL;
|
||||
pr_info("Unknown Cache mode or invalid value %s\n", s);
|
||||
}
|
||||
return version;
|
||||
}
|
||||
/*
|
||||
* This structure contains all parameters used for the core code,
|
||||
* the client, and all the transports.
|
||||
*/
|
||||
const struct fs_parameter_spec v9fs_param_spec[] = {
|
||||
fsparam_string ("source", Opt_source),
|
||||
fsparam_u32hex ("debug", Opt_debug),
|
||||
fsparam_uid ("dfltuid", Opt_dfltuid),
|
||||
fsparam_gid ("dfltgid", Opt_dfltgid),
|
||||
fsparam_u32 ("afid", Opt_afid),
|
||||
fsparam_string ("uname", Opt_uname),
|
||||
fsparam_string ("aname", Opt_remotename),
|
||||
fsparam_flag ("nodevmap", Opt_nodevmap),
|
||||
fsparam_flag ("noxattr", Opt_noxattr),
|
||||
fsparam_flag ("directio", Opt_directio),
|
||||
fsparam_flag ("ignoreqv", Opt_ignoreqv),
|
||||
fsparam_enum ("cache", Opt_cache, p9_cache_mode),
|
||||
fsparam_string ("cachetag", Opt_cachetag),
|
||||
fsparam_string ("access", Opt_access),
|
||||
fsparam_flag ("posixacl", Opt_posixacl),
|
||||
fsparam_u32 ("locktimeout", Opt_locktimeout),
|
||||
|
||||
/* client options */
|
||||
fsparam_u32 ("msize", Opt_msize),
|
||||
fsparam_flag ("noextend", Opt_legacy),
|
||||
fsparam_string ("trans", Opt_trans),
|
||||
fsparam_enum ("version", Opt_version, p9_versions),
|
||||
|
||||
/* fd transport options */
|
||||
fsparam_u32 ("rfdno", Opt_rfdno),
|
||||
fsparam_u32 ("wfdno", Opt_wfdno),
|
||||
|
||||
/* rdma transport options */
|
||||
fsparam_u32 ("sq", Opt_sq_depth),
|
||||
fsparam_u32 ("rq", Opt_rq_depth),
|
||||
fsparam_u32 ("timeout", Opt_timeout),
|
||||
|
||||
/* fd and rdma transprt options */
|
||||
fsparam_u32 ("port", Opt_port),
|
||||
fsparam_flag ("privport", Opt_privport),
|
||||
{}
|
||||
};
|
||||
|
||||
/*
|
||||
* Display the mount options in /proc/mounts.
|
||||
|
|
@ -153,267 +184,252 @@ int v9fs_show_options(struct seq_file *m, struct dentry *root)
|
|||
}
|
||||
|
||||
/**
|
||||
* v9fs_parse_options - parse mount options into session structure
|
||||
* @v9ses: existing v9fs session information
|
||||
* @opts: The mount option string
|
||||
* v9fs_parse_param - parse a mount option into the filesystem context
|
||||
* @fc: the filesystem context
|
||||
* @param: the parameter to parse
|
||||
*
|
||||
* Return 0 upon success, -ERRNO upon failure.
|
||||
*/
|
||||
|
||||
static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts)
|
||||
int v9fs_parse_param(struct fs_context *fc, struct fs_parameter *param)
|
||||
{
|
||||
char *options, *tmp_options;
|
||||
substring_t args[MAX_OPT_ARGS];
|
||||
char *p;
|
||||
int option = 0;
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
struct fs_parse_result result;
|
||||
char *s;
|
||||
int ret = 0;
|
||||
int r;
|
||||
int opt;
|
||||
struct p9_client_opts *clnt = &ctx->client_opts;
|
||||
struct p9_fd_opts *fd_opts = &ctx->fd_opts;
|
||||
struct p9_rdma_opts *rdma_opts = &ctx->rdma_opts;
|
||||
struct p9_session_opts *session_opts = &ctx->session_opts;
|
||||
|
||||
/* setup defaults */
|
||||
v9ses->afid = ~0;
|
||||
v9ses->debug = 0;
|
||||
v9ses->cache = CACHE_NONE;
|
||||
#ifdef CONFIG_9P_FSCACHE
|
||||
v9ses->cachetag = NULL;
|
||||
#endif
|
||||
v9ses->session_lock_timeout = P9_LOCK_TIMEOUT;
|
||||
opt = fs_parse(fc, v9fs_param_spec, param, &result);
|
||||
if (opt < 0) {
|
||||
/*
|
||||
* We might like to report bad mount options here, but
|
||||
* traditionally 9p has ignored unknown mount options
|
||||
*/
|
||||
if (opt == -ENOPARAM)
|
||||
return 0;
|
||||
|
||||
if (!opts)
|
||||
return 0;
|
||||
|
||||
tmp_options = kstrdup(opts, GFP_KERNEL);
|
||||
if (!tmp_options) {
|
||||
ret = -ENOMEM;
|
||||
goto fail_option_alloc;
|
||||
return opt;
|
||||
}
|
||||
options = tmp_options;
|
||||
|
||||
while ((p = strsep(&options, ",")) != NULL) {
|
||||
int token, r;
|
||||
|
||||
if (!*p)
|
||||
continue;
|
||||
|
||||
token = match_token(p, tokens, args);
|
||||
switch (token) {
|
||||
case Opt_debug:
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
ret = r;
|
||||
} else {
|
||||
v9ses->debug = option;
|
||||
#ifdef CONFIG_NET_9P_DEBUG
|
||||
p9_debug_level = option;
|
||||
#endif
|
||||
}
|
||||
break;
|
||||
|
||||
case Opt_dfltuid:
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
ret = r;
|
||||
continue;
|
||||
}
|
||||
v9ses->dfltuid = make_kuid(current_user_ns(), option);
|
||||
if (!uid_valid(v9ses->dfltuid)) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"uid field, but not a uid?\n");
|
||||
ret = -EINVAL;
|
||||
}
|
||||
break;
|
||||
case Opt_dfltgid:
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
ret = r;
|
||||
continue;
|
||||
}
|
||||
v9ses->dfltgid = make_kgid(current_user_ns(), option);
|
||||
if (!gid_valid(v9ses->dfltgid)) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"gid field, but not a gid?\n");
|
||||
ret = -EINVAL;
|
||||
}
|
||||
break;
|
||||
case Opt_afid:
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
ret = r;
|
||||
} else {
|
||||
v9ses->afid = option;
|
||||
}
|
||||
break;
|
||||
case Opt_uname:
|
||||
kfree(v9ses->uname);
|
||||
v9ses->uname = match_strdup(&args[0]);
|
||||
if (!v9ses->uname) {
|
||||
ret = -ENOMEM;
|
||||
goto free_and_return;
|
||||
}
|
||||
break;
|
||||
case Opt_remotename:
|
||||
kfree(v9ses->aname);
|
||||
v9ses->aname = match_strdup(&args[0]);
|
||||
if (!v9ses->aname) {
|
||||
ret = -ENOMEM;
|
||||
goto free_and_return;
|
||||
}
|
||||
break;
|
||||
case Opt_nodevmap:
|
||||
v9ses->nodev = 1;
|
||||
break;
|
||||
case Opt_noxattr:
|
||||
v9ses->flags |= V9FS_NO_XATTR;
|
||||
break;
|
||||
case Opt_directio:
|
||||
v9ses->flags |= V9FS_DIRECT_IO;
|
||||
break;
|
||||
case Opt_ignoreqv:
|
||||
v9ses->flags |= V9FS_IGNORE_QV;
|
||||
break;
|
||||
case Opt_cachetag:
|
||||
#ifdef CONFIG_9P_FSCACHE
|
||||
kfree(v9ses->cachetag);
|
||||
v9ses->cachetag = match_strdup(&args[0]);
|
||||
if (!v9ses->cachetag) {
|
||||
ret = -ENOMEM;
|
||||
goto free_and_return;
|
||||
}
|
||||
#endif
|
||||
break;
|
||||
case Opt_cache:
|
||||
s = match_strdup(&args[0]);
|
||||
if (!s) {
|
||||
ret = -ENOMEM;
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"problem allocating copy of cache arg\n");
|
||||
goto free_and_return;
|
||||
}
|
||||
r = get_cache_mode(s);
|
||||
if (r < 0)
|
||||
ret = r;
|
||||
else
|
||||
v9ses->cache = r;
|
||||
|
||||
kfree(s);
|
||||
break;
|
||||
|
||||
case Opt_access:
|
||||
s = match_strdup(&args[0]);
|
||||
if (!s) {
|
||||
ret = -ENOMEM;
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"problem allocating copy of access arg\n");
|
||||
goto free_and_return;
|
||||
}
|
||||
|
||||
v9ses->flags &= ~V9FS_ACCESS_MASK;
|
||||
if (strcmp(s, "user") == 0)
|
||||
v9ses->flags |= V9FS_ACCESS_USER;
|
||||
else if (strcmp(s, "any") == 0)
|
||||
v9ses->flags |= V9FS_ACCESS_ANY;
|
||||
else if (strcmp(s, "client") == 0) {
|
||||
v9ses->flags |= V9FS_ACCESS_CLIENT;
|
||||
} else {
|
||||
uid_t uid;
|
||||
|
||||
v9ses->flags |= V9FS_ACCESS_SINGLE;
|
||||
r = kstrtouint(s, 10, &uid);
|
||||
if (r) {
|
||||
ret = r;
|
||||
pr_info("Unknown access argument %s: %d\n",
|
||||
s, r);
|
||||
kfree(s);
|
||||
continue;
|
||||
}
|
||||
v9ses->uid = make_kuid(current_user_ns(), uid);
|
||||
if (!uid_valid(v9ses->uid)) {
|
||||
ret = -EINVAL;
|
||||
pr_info("Unknown uid %s\n", s);
|
||||
}
|
||||
}
|
||||
|
||||
kfree(s);
|
||||
break;
|
||||
|
||||
case Opt_posixacl:
|
||||
#ifdef CONFIG_9P_FS_POSIX_ACL
|
||||
v9ses->flags |= V9FS_POSIX_ACL;
|
||||
#else
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"Not defined CONFIG_9P_FS_POSIX_ACL. Ignoring posixacl option\n");
|
||||
#endif
|
||||
break;
|
||||
|
||||
case Opt_locktimeout:
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
ret = r;
|
||||
continue;
|
||||
}
|
||||
if (option < 1) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"locktimeout must be a greater than zero integer.\n");
|
||||
ret = -EINVAL;
|
||||
continue;
|
||||
}
|
||||
v9ses->session_lock_timeout = (long)option * HZ;
|
||||
break;
|
||||
|
||||
default:
|
||||
continue;
|
||||
switch (opt) {
|
||||
case Opt_source:
|
||||
if (fc->source) {
|
||||
pr_info("p9: multiple sources not supported\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
fc->source = param->string;
|
||||
param->string = NULL;
|
||||
break;
|
||||
case Opt_debug:
|
||||
session_opts->debug = result.uint_32;
|
||||
#ifdef CONFIG_NET_9P_DEBUG
|
||||
p9_debug_level = result.uint_32;
|
||||
#endif
|
||||
break;
|
||||
|
||||
case Opt_dfltuid:
|
||||
session_opts->dfltuid = result.uid;
|
||||
break;
|
||||
case Opt_dfltgid:
|
||||
session_opts->dfltgid = result.gid;
|
||||
break;
|
||||
case Opt_afid:
|
||||
session_opts->afid = result.uint_32;
|
||||
break;
|
||||
case Opt_uname:
|
||||
kfree(session_opts->uname);
|
||||
session_opts->uname = param->string;
|
||||
param->string = NULL;
|
||||
break;
|
||||
case Opt_remotename:
|
||||
kfree(session_opts->aname);
|
||||
session_opts->aname = param->string;
|
||||
param->string = NULL;
|
||||
break;
|
||||
case Opt_nodevmap:
|
||||
session_opts->nodev = 1;
|
||||
break;
|
||||
case Opt_noxattr:
|
||||
session_opts->flags |= V9FS_NO_XATTR;
|
||||
break;
|
||||
case Opt_directio:
|
||||
session_opts->flags |= V9FS_DIRECT_IO;
|
||||
break;
|
||||
case Opt_ignoreqv:
|
||||
session_opts->flags |= V9FS_IGNORE_QV;
|
||||
break;
|
||||
case Opt_cachetag:
|
||||
#ifdef CONFIG_9P_FSCACHE
|
||||
kfree(session_opts->cachetag);
|
||||
session_opts->cachetag = param->string;
|
||||
param->string = NULL;
|
||||
#endif
|
||||
break;
|
||||
case Opt_cache:
|
||||
session_opts->cache = result.uint_32;
|
||||
p9_debug(P9_DEBUG_9P, "Cache mode: %s\n", param->string);
|
||||
break;
|
||||
case Opt_access:
|
||||
s = param->string;
|
||||
session_opts->flags &= ~V9FS_ACCESS_MASK;
|
||||
if (strcmp(s, "user") == 0) {
|
||||
session_opts->flags |= V9FS_ACCESS_USER;
|
||||
} else if (strcmp(s, "any") == 0) {
|
||||
session_opts->flags |= V9FS_ACCESS_ANY;
|
||||
} else if (strcmp(s, "client") == 0) {
|
||||
session_opts->flags |= V9FS_ACCESS_CLIENT;
|
||||
} else {
|
||||
uid_t uid;
|
||||
|
||||
session_opts->flags |= V9FS_ACCESS_SINGLE;
|
||||
r = kstrtouint(s, 10, &uid);
|
||||
if (r) {
|
||||
pr_info("Unknown access argument %s: %d\n",
|
||||
param->string, r);
|
||||
return r;
|
||||
}
|
||||
session_opts->uid = make_kuid(current_user_ns(), uid);
|
||||
if (!uid_valid(session_opts->uid)) {
|
||||
pr_info("Unknown uid %s\n", s);
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case Opt_posixacl:
|
||||
#ifdef CONFIG_9P_FS_POSIX_ACL
|
||||
session_opts->flags |= V9FS_POSIX_ACL;
|
||||
#else
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"Not defined CONFIG_9P_FS_POSIX_ACL. Ignoring posixacl option\n");
|
||||
#endif
|
||||
break;
|
||||
|
||||
case Opt_locktimeout:
|
||||
if (result.uint_32 < 1) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"locktimeout must be a greater than zero integer.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
session_opts->session_lock_timeout = (long)result.uint_32 * HZ;
|
||||
break;
|
||||
|
||||
/* Options for client */
|
||||
case Opt_msize:
|
||||
if (result.uint_32 < 4096) {
|
||||
p9_debug(P9_DEBUG_ERROR, "msize should be at least 4k\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
if (result.uint_32 > INT_MAX) {
|
||||
p9_debug(P9_DEBUG_ERROR, "msize too big\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
clnt->msize = result.uint_32;
|
||||
break;
|
||||
case Opt_trans:
|
||||
v9fs_put_trans(clnt->trans_mod);
|
||||
clnt->trans_mod = v9fs_get_trans_by_name(param->string);
|
||||
if (!clnt->trans_mod) {
|
||||
pr_info("Could not find request transport: %s\n",
|
||||
param->string);
|
||||
return -EINVAL;
|
||||
}
|
||||
break;
|
||||
case Opt_legacy:
|
||||
clnt->proto_version = p9_proto_legacy;
|
||||
break;
|
||||
case Opt_version:
|
||||
clnt->proto_version = result.uint_32;
|
||||
p9_debug(P9_DEBUG_9P, "Protocol version: %s\n", param->string);
|
||||
break;
|
||||
/* Options for fd transport */
|
||||
case Opt_rfdno:
|
||||
fd_opts->rfd = result.uint_32;
|
||||
break;
|
||||
case Opt_wfdno:
|
||||
fd_opts->wfd = result.uint_32;
|
||||
break;
|
||||
/* Options for rdma transport */
|
||||
case Opt_sq_depth:
|
||||
rdma_opts->sq_depth = result.uint_32;
|
||||
break;
|
||||
case Opt_rq_depth:
|
||||
rdma_opts->rq_depth = result.uint_32;
|
||||
break;
|
||||
case Opt_timeout:
|
||||
rdma_opts->timeout = result.uint_32;
|
||||
break;
|
||||
/* Options for both fd and rdma transports */
|
||||
case Opt_port:
|
||||
fd_opts->port = result.uint_32;
|
||||
rdma_opts->port = result.uint_32;
|
||||
break;
|
||||
case Opt_privport:
|
||||
fd_opts->privport = true;
|
||||
rdma_opts->port = true;
|
||||
break;
|
||||
}
|
||||
|
||||
free_and_return:
|
||||
kfree(tmp_options);
|
||||
fail_option_alloc:
|
||||
return ret;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void v9fs_apply_options(struct v9fs_session_info *v9ses,
|
||||
struct fs_context *fc)
|
||||
{
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
|
||||
v9ses->debug = ctx->session_opts.debug;
|
||||
v9ses->dfltuid = ctx->session_opts.dfltuid;
|
||||
v9ses->dfltgid = ctx->session_opts.dfltgid;
|
||||
v9ses->afid = ctx->session_opts.afid;
|
||||
v9ses->uname = ctx->session_opts.uname;
|
||||
ctx->session_opts.uname = NULL;
|
||||
v9ses->aname = ctx->session_opts.aname;
|
||||
ctx->session_opts.aname = NULL;
|
||||
v9ses->nodev = ctx->session_opts.nodev;
|
||||
/*
|
||||
* Note that we must |= flags here as session_init already
|
||||
* set basic flags. This adds in flags from parsed options.
|
||||
*/
|
||||
v9ses->flags |= ctx->session_opts.flags;
|
||||
#ifdef CONFIG_9P_FSCACHE
|
||||
v9ses->cachetag = ctx->session_opts.cachetag;
|
||||
ctx->session_opts.cachetag = NULL;
|
||||
#endif
|
||||
v9ses->cache = ctx->session_opts.cache;
|
||||
v9ses->uid = ctx->session_opts.uid;
|
||||
v9ses->session_lock_timeout = ctx->session_opts.session_lock_timeout;
|
||||
}
|
||||
|
||||
/**
|
||||
* v9fs_session_init - initialize session
|
||||
* @v9ses: session information structure
|
||||
* @dev_name: device being mounted
|
||||
* @data: options
|
||||
* @fc: the filesystem mount context
|
||||
*
|
||||
*/
|
||||
|
||||
struct p9_fid *v9fs_session_init(struct v9fs_session_info *v9ses,
|
||||
const char *dev_name, char *data)
|
||||
struct fs_context *fc)
|
||||
{
|
||||
struct p9_fid *fid;
|
||||
int rc = -ENOMEM;
|
||||
|
||||
v9ses->uname = kstrdup(V9FS_DEFUSER, GFP_KERNEL);
|
||||
if (!v9ses->uname)
|
||||
goto err_names;
|
||||
|
||||
v9ses->aname = kstrdup(V9FS_DEFANAME, GFP_KERNEL);
|
||||
if (!v9ses->aname)
|
||||
goto err_names;
|
||||
init_rwsem(&v9ses->rename_sem);
|
||||
|
||||
v9ses->uid = INVALID_UID;
|
||||
v9ses->dfltuid = V9FS_DEFUID;
|
||||
v9ses->dfltgid = V9FS_DEFGID;
|
||||
|
||||
v9ses->clnt = p9_client_create(dev_name, data);
|
||||
v9ses->clnt = p9_client_create(fc);
|
||||
if (IS_ERR(v9ses->clnt)) {
|
||||
rc = PTR_ERR(v9ses->clnt);
|
||||
p9_debug(P9_DEBUG_ERROR, "problem initializing 9p client\n");
|
||||
goto err_names;
|
||||
}
|
||||
|
||||
/*
|
||||
* Initialize flags on the real v9ses. v9fs_apply_options below
|
||||
* will |= the additional flags from parsed options.
|
||||
*/
|
||||
v9ses->flags = V9FS_ACCESS_USER;
|
||||
|
||||
if (p9_is_proto_dotl(v9ses->clnt)) {
|
||||
|
|
@ -423,9 +439,7 @@ struct p9_fid *v9fs_session_init(struct v9fs_session_info *v9ses,
|
|||
v9ses->flags |= V9FS_PROTO_2000U;
|
||||
}
|
||||
|
||||
rc = v9fs_parse_options(v9ses, data);
|
||||
if (rc < 0)
|
||||
goto err_clnt;
|
||||
v9fs_apply_options(v9ses, fc);
|
||||
|
||||
v9ses->maxdata = v9ses->clnt->msize - P9_IOHDRSZ;
|
||||
|
||||
|
|
@ -471,7 +485,7 @@ struct p9_fid *v9fs_session_init(struct v9fs_session_info *v9ses,
|
|||
#ifdef CONFIG_9P_FSCACHE
|
||||
/* register the session for caching */
|
||||
if (v9ses->cache & CACHE_FSCACHE) {
|
||||
rc = v9fs_cache_session_get_cookie(v9ses, dev_name);
|
||||
rc = v9fs_cache_session_get_cookie(v9ses, fc->source);
|
||||
if (rc < 0)
|
||||
goto err_clnt;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -10,6 +10,9 @@
|
|||
|
||||
#include <linux/backing-dev.h>
|
||||
#include <linux/netfs.h>
|
||||
#include <linux/fs_parser.h>
|
||||
#include <net/9p/client.h>
|
||||
#include <net/9p/transport.h>
|
||||
|
||||
/**
|
||||
* enum p9_session_flags - option flags for each 9P session
|
||||
|
|
@ -163,11 +166,13 @@ static inline struct fscache_volume *v9fs_session_cache(struct v9fs_session_info
|
|||
#endif
|
||||
}
|
||||
|
||||
extern const struct fs_parameter_spec v9fs_param_spec[];
|
||||
|
||||
extern int v9fs_parse_param(struct fs_context *fc, struct fs_parameter *param);
|
||||
extern int v9fs_show_options(struct seq_file *m, struct dentry *root);
|
||||
|
||||
struct p9_fid *v9fs_session_init(struct v9fs_session_info *v9ses,
|
||||
const char *dev_name, char *data);
|
||||
struct fs_context *fc);
|
||||
extern void v9fs_session_close(struct v9fs_session_info *v9ses);
|
||||
extern void v9fs_session_cancel(struct v9fs_session_info *v9ses);
|
||||
extern void v9fs_session_begin_cancel(struct v9fs_session_info *v9ses);
|
||||
|
|
|
|||
|
|
@ -19,6 +19,7 @@
|
|||
#include <linux/statfs.h>
|
||||
#include <linux/magic.h>
|
||||
#include <linux/fscache.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <net/9p/9p.h>
|
||||
#include <net/9p/client.h>
|
||||
|
||||
|
|
@ -30,32 +31,10 @@
|
|||
|
||||
static const struct super_operations v9fs_super_ops, v9fs_super_ops_dotl;
|
||||
|
||||
/**
|
||||
* v9fs_set_super - set the superblock
|
||||
* @s: super block
|
||||
* @data: file system specific data
|
||||
*
|
||||
*/
|
||||
|
||||
static int v9fs_set_super(struct super_block *s, void *data)
|
||||
{
|
||||
s->s_fs_info = data;
|
||||
return set_anon_super(s, data);
|
||||
}
|
||||
|
||||
/**
|
||||
* v9fs_fill_super - populate superblock with info
|
||||
* @sb: superblock
|
||||
* @v9ses: session information
|
||||
* @flags: flags propagated from v9fs_mount()
|
||||
*
|
||||
*/
|
||||
|
||||
static int
|
||||
v9fs_fill_super(struct super_block *sb, struct v9fs_session_info *v9ses,
|
||||
int flags)
|
||||
static int v9fs_fill_super(struct super_block *sb)
|
||||
{
|
||||
int ret;
|
||||
struct v9fs_session_info *v9ses = v9ses = sb->s_fs_info;
|
||||
|
||||
sb->s_maxbytes = MAX_LFS_FILESIZE;
|
||||
sb->s_blocksize_bits = fls(v9ses->maxdata - 1);
|
||||
|
|
@ -95,16 +74,12 @@ v9fs_fill_super(struct super_block *sb, struct v9fs_session_info *v9ses,
|
|||
}
|
||||
|
||||
/**
|
||||
* v9fs_mount - mount a superblock
|
||||
* @fs_type: file system type
|
||||
* @flags: mount flags
|
||||
* @dev_name: device name that was mounted
|
||||
* @data: mount options
|
||||
* v9fs_get_tree - create the mountable root and superblock
|
||||
* @fc: the filesystem context
|
||||
*
|
||||
*/
|
||||
|
||||
static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags,
|
||||
const char *dev_name, void *data)
|
||||
static int v9fs_get_tree(struct fs_context *fc)
|
||||
{
|
||||
struct super_block *sb = NULL;
|
||||
struct inode *inode = NULL;
|
||||
|
|
@ -117,20 +92,21 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags,
|
|||
|
||||
v9ses = kzalloc(sizeof(struct v9fs_session_info), GFP_KERNEL);
|
||||
if (!v9ses)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
return -ENOMEM;
|
||||
|
||||
fid = v9fs_session_init(v9ses, dev_name, data);
|
||||
fid = v9fs_session_init(v9ses, fc);
|
||||
if (IS_ERR(fid)) {
|
||||
retval = PTR_ERR(fid);
|
||||
goto free_session;
|
||||
}
|
||||
|
||||
sb = sget(fs_type, NULL, v9fs_set_super, flags, v9ses);
|
||||
fc->s_fs_info = v9ses;
|
||||
sb = sget_fc(fc, NULL, set_anon_super_fc);
|
||||
if (IS_ERR(sb)) {
|
||||
retval = PTR_ERR(sb);
|
||||
goto clunk_fid;
|
||||
}
|
||||
retval = v9fs_fill_super(sb, v9ses, flags);
|
||||
retval = v9fs_fill_super(sb);
|
||||
if (retval)
|
||||
goto release_sb;
|
||||
|
||||
|
|
@ -159,14 +135,15 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags,
|
|||
v9fs_fid_add(root, &fid);
|
||||
|
||||
p9_debug(P9_DEBUG_VFS, " simple set mount, return 0\n");
|
||||
return dget(sb->s_root);
|
||||
fc->root = dget(sb->s_root);
|
||||
return 0;
|
||||
|
||||
clunk_fid:
|
||||
p9_fid_put(fid);
|
||||
v9fs_session_close(v9ses);
|
||||
free_session:
|
||||
kfree(v9ses);
|
||||
return ERR_PTR(retval);
|
||||
return retval;
|
||||
|
||||
release_sb:
|
||||
/*
|
||||
|
|
@ -177,7 +154,7 @@ release_sb:
|
|||
*/
|
||||
p9_fid_put(fid);
|
||||
deactivate_locked_super(sb);
|
||||
return ERR_PTR(retval);
|
||||
return retval;
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
@ -303,11 +280,86 @@ static const struct super_operations v9fs_super_ops_dotl = {
|
|||
.write_inode = v9fs_write_inode_dotl,
|
||||
};
|
||||
|
||||
static void v9fs_free_fc(struct fs_context *fc)
|
||||
{
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
|
||||
if (!ctx)
|
||||
return;
|
||||
|
||||
/* These should be NULL by now but guard against leaks */
|
||||
kfree(ctx->session_opts.uname);
|
||||
kfree(ctx->session_opts.aname);
|
||||
#ifdef CONFIG_9P_FSCACHE
|
||||
kfree(ctx->session_opts.cachetag);
|
||||
#endif
|
||||
if (ctx->client_opts.trans_mod)
|
||||
v9fs_put_trans(ctx->client_opts.trans_mod);
|
||||
kfree(ctx);
|
||||
}
|
||||
|
||||
static const struct fs_context_operations v9fs_context_ops = {
|
||||
.parse_param = v9fs_parse_param,
|
||||
.get_tree = v9fs_get_tree,
|
||||
.free = v9fs_free_fc,
|
||||
};
|
||||
|
||||
static int v9fs_init_fs_context(struct fs_context *fc)
|
||||
{
|
||||
struct v9fs_context *ctx;
|
||||
|
||||
ctx = kzalloc(sizeof(*ctx), GFP_KERNEL);
|
||||
if (!ctx)
|
||||
return -ENOMEM;
|
||||
|
||||
/* initialize core options */
|
||||
ctx->session_opts.afid = ~0;
|
||||
ctx->session_opts.cache = CACHE_NONE;
|
||||
ctx->session_opts.session_lock_timeout = P9_LOCK_TIMEOUT;
|
||||
ctx->session_opts.uname = kstrdup(V9FS_DEFUSER, GFP_KERNEL);
|
||||
if (!ctx->session_opts.uname)
|
||||
goto error;
|
||||
|
||||
ctx->session_opts.aname = kstrdup(V9FS_DEFANAME, GFP_KERNEL);
|
||||
if (!ctx->session_opts.aname)
|
||||
goto error;
|
||||
|
||||
ctx->session_opts.uid = INVALID_UID;
|
||||
ctx->session_opts.dfltuid = V9FS_DEFUID;
|
||||
ctx->session_opts.dfltgid = V9FS_DEFGID;
|
||||
|
||||
/* initialize client options */
|
||||
ctx->client_opts.proto_version = p9_proto_2000L;
|
||||
ctx->client_opts.msize = DEFAULT_MSIZE;
|
||||
|
||||
/* initialize fd transport options */
|
||||
ctx->fd_opts.port = P9_FD_PORT;
|
||||
ctx->fd_opts.rfd = ~0;
|
||||
ctx->fd_opts.wfd = ~0;
|
||||
ctx->fd_opts.privport = false;
|
||||
|
||||
/* initialize rdma transport options */
|
||||
ctx->rdma_opts.port = P9_RDMA_PORT;
|
||||
ctx->rdma_opts.sq_depth = P9_RDMA_SQ_DEPTH;
|
||||
ctx->rdma_opts.rq_depth = P9_RDMA_RQ_DEPTH;
|
||||
ctx->rdma_opts.timeout = P9_RDMA_TIMEOUT;
|
||||
ctx->rdma_opts.privport = false;
|
||||
|
||||
fc->ops = &v9fs_context_ops;
|
||||
fc->fs_private = ctx;
|
||||
|
||||
return 0;
|
||||
error:
|
||||
fc->need_free = 1;
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
struct file_system_type v9fs_fs_type = {
|
||||
.name = "9p",
|
||||
.mount = v9fs_mount,
|
||||
.kill_sb = v9fs_kill_super,
|
||||
.owner = THIS_MODULE,
|
||||
.fs_flags = FS_RENAME_DOES_D_MOVE,
|
||||
.init_fs_context = v9fs_init_fs_context,
|
||||
.parameters = v9fs_param_spec,
|
||||
};
|
||||
MODULE_ALIAS_FS("9p");
|
||||
|
|
|
|||
|
|
@ -279,7 +279,7 @@ int p9_client_rename(struct p9_fid *fid, struct p9_fid *newdirfid,
|
|||
const char *name);
|
||||
int p9_client_renameat(struct p9_fid *olddirfid, const char *old_name,
|
||||
struct p9_fid *newdirfid, const char *new_name);
|
||||
struct p9_client *p9_client_create(const char *dev_name, char *options);
|
||||
struct p9_client *p9_client_create(struct fs_context *fc);
|
||||
void p9_client_destroy(struct p9_client *clnt);
|
||||
void p9_client_disconnect(struct p9_client *clnt);
|
||||
void p9_client_begin_disconnect(struct p9_client *clnt);
|
||||
|
|
|
|||
|
|
@ -57,7 +57,7 @@ struct p9_trans_module {
|
|||
bool supports_vmalloc; /* can work with vmalloc'd buffers */
|
||||
struct module *owner;
|
||||
int (*create)(struct p9_client *client,
|
||||
const char *devname, char *args);
|
||||
struct fs_context *fc);
|
||||
void (*close)(struct p9_client *client);
|
||||
int (*request)(struct p9_client *client, struct p9_req_t *req);
|
||||
int (*cancel)(struct p9_client *client, struct p9_req_t *req);
|
||||
|
|
|
|||
148
net/9p/client.c
148
net/9p/client.c
|
|
@ -20,8 +20,8 @@
|
|||
#include <linux/uio.h>
|
||||
#include <linux/netfs.h>
|
||||
#include <net/9p/9p.h>
|
||||
#include <linux/parser.h>
|
||||
#include <linux/seq_file.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <net/9p/client.h>
|
||||
#include <net/9p/transport.h>
|
||||
#include "protocol.h"
|
||||
|
|
@ -33,22 +33,6 @@
|
|||
* - a little lazy - parse all client options
|
||||
*/
|
||||
|
||||
enum {
|
||||
Opt_msize,
|
||||
Opt_trans,
|
||||
Opt_legacy,
|
||||
Opt_version,
|
||||
Opt_err,
|
||||
};
|
||||
|
||||
static const match_table_t tokens = {
|
||||
{Opt_msize, "msize=%u"},
|
||||
{Opt_legacy, "noextend"},
|
||||
{Opt_trans, "trans=%s"},
|
||||
{Opt_version, "version=%s"},
|
||||
{Opt_err, NULL},
|
||||
};
|
||||
|
||||
inline int p9_is_proto_dotl(struct p9_client *clnt)
|
||||
{
|
||||
return clnt->proto_version == p9_proto_2000L;
|
||||
|
|
@ -97,124 +81,16 @@ static int safe_errno(int err)
|
|||
return err;
|
||||
}
|
||||
|
||||
/* Interpret mount option for protocol version */
|
||||
static int get_protocol_version(char *s)
|
||||
static int apply_client_options(struct p9_client *clnt, struct fs_context *fc)
|
||||
{
|
||||
int version = -EINVAL;
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
|
||||
if (!strcmp(s, "9p2000")) {
|
||||
version = p9_proto_legacy;
|
||||
p9_debug(P9_DEBUG_9P, "Protocol version: Legacy\n");
|
||||
} else if (!strcmp(s, "9p2000.u")) {
|
||||
version = p9_proto_2000u;
|
||||
p9_debug(P9_DEBUG_9P, "Protocol version: 9P2000.u\n");
|
||||
} else if (!strcmp(s, "9p2000.L")) {
|
||||
version = p9_proto_2000L;
|
||||
p9_debug(P9_DEBUG_9P, "Protocol version: 9P2000.L\n");
|
||||
} else {
|
||||
pr_info("Unknown protocol version %s\n", s);
|
||||
}
|
||||
clnt->msize = ctx->client_opts.msize;
|
||||
clnt->trans_mod = ctx->client_opts.trans_mod;
|
||||
ctx->client_opts.trans_mod = NULL;
|
||||
clnt->proto_version = ctx->client_opts.proto_version;
|
||||
|
||||
return version;
|
||||
}
|
||||
|
||||
/**
|
||||
* parse_opts - parse mount options into client structure
|
||||
* @opts: options string passed from mount
|
||||
* @clnt: existing v9fs client information
|
||||
*
|
||||
* Return 0 upon success, -ERRNO upon failure
|
||||
*/
|
||||
|
||||
static int parse_opts(char *opts, struct p9_client *clnt)
|
||||
{
|
||||
char *options, *tmp_options;
|
||||
char *p;
|
||||
substring_t args[MAX_OPT_ARGS];
|
||||
int option;
|
||||
char *s;
|
||||
int ret = 0;
|
||||
|
||||
clnt->proto_version = p9_proto_2000L;
|
||||
clnt->msize = DEFAULT_MSIZE;
|
||||
|
||||
if (!opts)
|
||||
return 0;
|
||||
|
||||
tmp_options = kstrdup(opts, GFP_KERNEL);
|
||||
if (!tmp_options)
|
||||
return -ENOMEM;
|
||||
options = tmp_options;
|
||||
|
||||
while ((p = strsep(&options, ",")) != NULL) {
|
||||
int token, r;
|
||||
|
||||
if (!*p)
|
||||
continue;
|
||||
token = match_token(p, tokens, args);
|
||||
switch (token) {
|
||||
case Opt_msize:
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
ret = r;
|
||||
continue;
|
||||
}
|
||||
if (option < 4096) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"msize should be at least 4k\n");
|
||||
ret = -EINVAL;
|
||||
continue;
|
||||
}
|
||||
clnt->msize = option;
|
||||
break;
|
||||
case Opt_trans:
|
||||
s = match_strdup(&args[0]);
|
||||
if (!s) {
|
||||
ret = -ENOMEM;
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"problem allocating copy of trans arg\n");
|
||||
goto free_and_return;
|
||||
}
|
||||
|
||||
v9fs_put_trans(clnt->trans_mod);
|
||||
clnt->trans_mod = v9fs_get_trans_by_name(s);
|
||||
if (!clnt->trans_mod) {
|
||||
pr_info("Could not find request transport: %s\n",
|
||||
s);
|
||||
ret = -EINVAL;
|
||||
}
|
||||
kfree(s);
|
||||
break;
|
||||
case Opt_legacy:
|
||||
clnt->proto_version = p9_proto_legacy;
|
||||
break;
|
||||
case Opt_version:
|
||||
s = match_strdup(&args[0]);
|
||||
if (!s) {
|
||||
ret = -ENOMEM;
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"problem allocating copy of version arg\n");
|
||||
goto free_and_return;
|
||||
}
|
||||
r = get_protocol_version(s);
|
||||
if (r < 0)
|
||||
ret = r;
|
||||
else
|
||||
clnt->proto_version = r;
|
||||
kfree(s);
|
||||
break;
|
||||
default:
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
free_and_return:
|
||||
if (ret)
|
||||
v9fs_put_trans(clnt->trans_mod);
|
||||
kfree(tmp_options);
|
||||
return ret;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int p9_fcall_init(struct p9_client *c, struct p9_fcall *fc,
|
||||
|
|
@ -975,7 +851,7 @@ error:
|
|||
return err;
|
||||
}
|
||||
|
||||
struct p9_client *p9_client_create(const char *dev_name, char *options)
|
||||
struct p9_client *p9_client_create(struct fs_context *fc)
|
||||
{
|
||||
int err;
|
||||
static atomic_t seqno = ATOMIC_INIT(0);
|
||||
|
|
@ -998,8 +874,8 @@ struct p9_client *p9_client_create(const char *dev_name, char *options)
|
|||
idr_init(&clnt->fids);
|
||||
idr_init(&clnt->reqs);
|
||||
|
||||
err = parse_opts(options, clnt);
|
||||
if (err < 0)
|
||||
err = apply_client_options(clnt, fc);
|
||||
if (err)
|
||||
goto free_client;
|
||||
|
||||
if (!clnt->trans_mod)
|
||||
|
|
@ -1015,7 +891,7 @@ struct p9_client *p9_client_create(const char *dev_name, char *options)
|
|||
p9_debug(P9_DEBUG_MUX, "clnt %p trans %p msize %d protocol %d\n",
|
||||
clnt, clnt->trans_mod, clnt->msize, clnt->proto_version);
|
||||
|
||||
err = clnt->trans_mod->create(clnt, dev_name, options);
|
||||
err = clnt->trans_mod->create(clnt, fc);
|
||||
if (err)
|
||||
goto put_trans;
|
||||
|
||||
|
|
|
|||
|
|
@ -16,7 +16,6 @@
|
|||
#include <linux/moduleparam.h>
|
||||
#include <net/9p/9p.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/parser.h>
|
||||
#include <net/9p/client.h>
|
||||
#include <net/9p/transport.h>
|
||||
#include <linux/list.h>
|
||||
|
|
@ -171,6 +170,7 @@ void v9fs_put_trans(struct p9_trans_module *m)
|
|||
if (m)
|
||||
module_put(m->owner);
|
||||
}
|
||||
EXPORT_SYMBOL(v9fs_put_trans);
|
||||
|
||||
/**
|
||||
* init_p9 - Initialize module
|
||||
|
|
|
|||
|
|
@ -22,7 +22,7 @@
|
|||
#include <linux/uaccess.h>
|
||||
#include <linux/inet.h>
|
||||
#include <linux/file.h>
|
||||
#include <linux/parser.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/seq_file.h>
|
||||
#include <net/9p/9p.h>
|
||||
|
|
@ -37,26 +37,6 @@
|
|||
static struct p9_trans_module p9_tcp_trans;
|
||||
static struct p9_trans_module p9_fd_trans;
|
||||
|
||||
/*
|
||||
* Option Parsing (code inspired by NFS code)
|
||||
* - a little lazy - parse all fd-transport options
|
||||
*/
|
||||
|
||||
enum {
|
||||
/* Options that take integer arguments */
|
||||
Opt_port, Opt_rfdno, Opt_wfdno, Opt_err,
|
||||
/* Options that take no arguments */
|
||||
Opt_privport,
|
||||
};
|
||||
|
||||
static const match_table_t tokens = {
|
||||
{Opt_port, "port=%u"},
|
||||
{Opt_rfdno, "rfdno=%u"},
|
||||
{Opt_wfdno, "wfdno=%u"},
|
||||
{Opt_privport, "privport"},
|
||||
{Opt_err, NULL},
|
||||
};
|
||||
|
||||
enum {
|
||||
Rworksched = 1, /* read work scheduled or running */
|
||||
Rpending = 2, /* can read */
|
||||
|
|
@ -737,73 +717,6 @@ static int p9_fd_show_options(struct seq_file *m, struct p9_client *clnt)
|
|||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* parse_opts - parse mount options into p9_fd_opts structure
|
||||
* @params: options string passed from mount
|
||||
* @opts: fd transport-specific structure to parse options into
|
||||
*
|
||||
* Returns 0 upon success, -ERRNO upon failure
|
||||
*/
|
||||
|
||||
static int parse_opts(char *params, struct p9_fd_opts *opts)
|
||||
{
|
||||
char *p;
|
||||
substring_t args[MAX_OPT_ARGS];
|
||||
int option;
|
||||
char *options, *tmp_options;
|
||||
|
||||
opts->port = P9_FD_PORT;
|
||||
opts->rfd = ~0;
|
||||
opts->wfd = ~0;
|
||||
opts->privport = false;
|
||||
|
||||
if (!params)
|
||||
return 0;
|
||||
|
||||
tmp_options = kstrdup(params, GFP_KERNEL);
|
||||
if (!tmp_options) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"failed to allocate copy of option string\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
options = tmp_options;
|
||||
|
||||
while ((p = strsep(&options, ",")) != NULL) {
|
||||
int token;
|
||||
int r;
|
||||
if (!*p)
|
||||
continue;
|
||||
token = match_token(p, tokens, args);
|
||||
if ((token != Opt_err) && (token != Opt_privport)) {
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
continue;
|
||||
}
|
||||
}
|
||||
switch (token) {
|
||||
case Opt_port:
|
||||
opts->port = option;
|
||||
break;
|
||||
case Opt_rfdno:
|
||||
opts->rfd = option;
|
||||
break;
|
||||
case Opt_wfdno:
|
||||
opts->wfd = option;
|
||||
break;
|
||||
case Opt_privport:
|
||||
opts->privport = true;
|
||||
break;
|
||||
default:
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
kfree(tmp_options);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int p9_fd_open(struct p9_client *client, int rfd, int wfd)
|
||||
{
|
||||
struct p9_trans_fd *ts = kzalloc(sizeof(struct p9_trans_fd),
|
||||
|
|
@ -958,17 +871,18 @@ static int p9_bind_privport(struct socket *sock)
|
|||
}
|
||||
|
||||
static int
|
||||
p9_fd_create_tcp(struct p9_client *client, const char *addr, char *args)
|
||||
p9_fd_create_tcp(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
const char *addr = fc->source;
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
int err;
|
||||
char port_str[6];
|
||||
struct socket *csocket;
|
||||
struct sockaddr_storage stor = { 0 };
|
||||
struct p9_fd_opts opts;
|
||||
|
||||
err = parse_opts(args, &opts);
|
||||
if (err < 0)
|
||||
return err;
|
||||
/* opts are already parsed in context */
|
||||
opts = ctx->fd_opts;
|
||||
|
||||
if (!addr)
|
||||
return -EINVAL;
|
||||
|
|
@ -1015,8 +929,9 @@ p9_fd_create_tcp(struct p9_client *client, const char *addr, char *args)
|
|||
}
|
||||
|
||||
static int
|
||||
p9_fd_create_unix(struct p9_client *client, const char *addr, char *args)
|
||||
p9_fd_create_unix(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
const char *addr = fc->source;
|
||||
int err;
|
||||
struct socket *csocket;
|
||||
struct sockaddr_un sun_server;
|
||||
|
|
@ -1055,14 +970,12 @@ p9_fd_create_unix(struct p9_client *client, const char *addr, char *args)
|
|||
}
|
||||
|
||||
static int
|
||||
p9_fd_create(struct p9_client *client, const char *addr, char *args)
|
||||
p9_fd_create(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
struct p9_fd_opts opts = ctx->fd_opts;
|
||||
int err;
|
||||
struct p9_fd_opts opts;
|
||||
|
||||
err = parse_opts(args, &opts);
|
||||
if (err < 0)
|
||||
return err;
|
||||
client->trans_opts.fd.rfd = opts.rfd;
|
||||
client->trans_opts.fd.wfd = opts.wfd;
|
||||
|
||||
|
|
|
|||
|
|
@ -22,7 +22,7 @@
|
|||
#include <linux/uaccess.h>
|
||||
#include <linux/inet.h>
|
||||
#include <linux/file.h>
|
||||
#include <linux/parser.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <linux/semaphore.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/seq_file.h>
|
||||
|
|
@ -106,26 +106,6 @@ struct p9_rdma_context {
|
|||
};
|
||||
};
|
||||
|
||||
/*
|
||||
* Option Parsing (code inspired by NFS code)
|
||||
*/
|
||||
enum {
|
||||
/* Options that take integer arguments */
|
||||
Opt_port, Opt_rq_depth, Opt_sq_depth, Opt_timeout,
|
||||
/* Options that take no argument */
|
||||
Opt_privport,
|
||||
Opt_err,
|
||||
};
|
||||
|
||||
static match_table_t tokens = {
|
||||
{Opt_port, "port=%u"},
|
||||
{Opt_sq_depth, "sq=%u"},
|
||||
{Opt_rq_depth, "rq=%u"},
|
||||
{Opt_timeout, "timeout=%u"},
|
||||
{Opt_privport, "privport"},
|
||||
{Opt_err, NULL},
|
||||
};
|
||||
|
||||
static int p9_rdma_show_options(struct seq_file *m, struct p9_client *clnt)
|
||||
{
|
||||
struct p9_trans_rdma *rdma = clnt->trans;
|
||||
|
|
@ -143,77 +123,6 @@ static int p9_rdma_show_options(struct seq_file *m, struct p9_client *clnt)
|
|||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* parse_opts - parse mount options into rdma options structure
|
||||
* @params: options string passed from mount
|
||||
* @opts: rdma transport-specific structure to parse options into
|
||||
*
|
||||
* Returns 0 upon success, -ERRNO upon failure
|
||||
*/
|
||||
static int parse_opts(char *params, struct p9_rdma_opts *opts)
|
||||
{
|
||||
char *p;
|
||||
substring_t args[MAX_OPT_ARGS];
|
||||
int option;
|
||||
char *options, *tmp_options;
|
||||
|
||||
opts->port = P9_RDMA_PORT;
|
||||
opts->sq_depth = P9_RDMA_SQ_DEPTH;
|
||||
opts->rq_depth = P9_RDMA_RQ_DEPTH;
|
||||
opts->timeout = P9_RDMA_TIMEOUT;
|
||||
opts->privport = false;
|
||||
|
||||
if (!params)
|
||||
return 0;
|
||||
|
||||
tmp_options = kstrdup(params, GFP_KERNEL);
|
||||
if (!tmp_options) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"failed to allocate copy of option string\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
options = tmp_options;
|
||||
|
||||
while ((p = strsep(&options, ",")) != NULL) {
|
||||
int token;
|
||||
int r;
|
||||
if (!*p)
|
||||
continue;
|
||||
token = match_token(p, tokens, args);
|
||||
if ((token != Opt_err) && (token != Opt_privport)) {
|
||||
r = match_int(&args[0], &option);
|
||||
if (r < 0) {
|
||||
p9_debug(P9_DEBUG_ERROR,
|
||||
"integer field, but no integer?\n");
|
||||
continue;
|
||||
}
|
||||
}
|
||||
switch (token) {
|
||||
case Opt_port:
|
||||
opts->port = option;
|
||||
break;
|
||||
case Opt_sq_depth:
|
||||
opts->sq_depth = option;
|
||||
break;
|
||||
case Opt_rq_depth:
|
||||
opts->rq_depth = option;
|
||||
break;
|
||||
case Opt_timeout:
|
||||
opts->timeout = option;
|
||||
break;
|
||||
case Opt_privport:
|
||||
opts->privport = true;
|
||||
break;
|
||||
default:
|
||||
continue;
|
||||
}
|
||||
}
|
||||
/* RQ must be at least as large as the SQ */
|
||||
opts->rq_depth = max(opts->rq_depth, opts->sq_depth);
|
||||
kfree(tmp_options);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
p9_cm_event_handler(struct rdma_cm_id *id, struct rdma_cm_event *event)
|
||||
{
|
||||
|
|
@ -607,14 +516,15 @@ static int p9_rdma_bind_privport(struct p9_trans_rdma *rdma)
|
|||
/**
|
||||
* rdma_create_trans - Transport method for creating a transport instance
|
||||
* @client: client instance
|
||||
* @addr: IP address string
|
||||
* @args: Mount options string
|
||||
* @fc: The filesystem context
|
||||
*/
|
||||
static int
|
||||
rdma_create_trans(struct p9_client *client, const char *addr, char *args)
|
||||
rdma_create_trans(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
const char *addr = fc->source;
|
||||
struct v9fs_context *ctx = fc->fs_private;
|
||||
struct p9_rdma_opts opts = ctx->rdma_opts;
|
||||
int err;
|
||||
struct p9_rdma_opts opts;
|
||||
struct p9_trans_rdma *rdma;
|
||||
struct rdma_conn_param conn_param;
|
||||
struct ib_qp_init_attr qp_attr;
|
||||
|
|
@ -622,10 +532,8 @@ rdma_create_trans(struct p9_client *client, const char *addr, char *args)
|
|||
if (addr == NULL)
|
||||
return -EINVAL;
|
||||
|
||||
/* Parse the transport specific mount options */
|
||||
err = parse_opts(args, &opts);
|
||||
if (err < 0)
|
||||
return err;
|
||||
/* options are already parsed, in the fs context */
|
||||
opts = ctx->rdma_opts;
|
||||
|
||||
/* Create and initialize the RDMA transport structure */
|
||||
rdma = alloc_rdma(&opts);
|
||||
|
|
|
|||
|
|
@ -27,6 +27,7 @@
|
|||
#include <linux/cleanup.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <linux/usb/composite.h>
|
||||
#include <linux/usb/func_utils.h>
|
||||
|
||||
|
|
@ -376,8 +377,9 @@ out:
|
|||
return ret;
|
||||
}
|
||||
|
||||
static int p9_usbg_create(struct p9_client *client, const char *devname, char *args)
|
||||
static int p9_usbg_create(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
const char *devname = fc->source;
|
||||
struct f_usb9pfs_dev *dev;
|
||||
struct f_usb9pfs *usb9pfs;
|
||||
int ret = -ENOENT;
|
||||
|
|
|
|||
|
|
@ -26,7 +26,7 @@
|
|||
#include <linux/highmem.h>
|
||||
#include <linux/slab.h>
|
||||
#include <net/9p/9p.h>
|
||||
#include <linux/parser.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <net/9p/client.h>
|
||||
#include <net/9p/transport.h>
|
||||
#include <linux/scatterlist.h>
|
||||
|
|
@ -679,8 +679,7 @@ fail:
|
|||
/**
|
||||
* p9_virtio_create - allocate a new virtio channel
|
||||
* @client: client instance invoking this transport
|
||||
* @devname: string identifying the channel to connect to (unused)
|
||||
* @args: args passed from sys_mount() for per-transport options (unused)
|
||||
* @fc: the filesystem context
|
||||
*
|
||||
* This sets up a transport channel for 9p communication. Right now
|
||||
* we only match the first available channel, but eventually we could look up
|
||||
|
|
@ -691,8 +690,9 @@ fail:
|
|||
*/
|
||||
|
||||
static int
|
||||
p9_virtio_create(struct p9_client *client, const char *devname, char *args)
|
||||
p9_virtio_create(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
const char *devname = fc->source;
|
||||
struct virtio_chan *chan;
|
||||
int ret = -ENOENT;
|
||||
int found = 0;
|
||||
|
|
|
|||
|
|
@ -15,6 +15,7 @@
|
|||
|
||||
#include <linux/module.h>
|
||||
#include <linux/spinlock.h>
|
||||
#include <linux/fs_context.h>
|
||||
#include <net/9p/9p.h>
|
||||
#include <net/9p/client.h>
|
||||
#include <net/9p/transport.h>
|
||||
|
|
@ -66,8 +67,9 @@ static int p9_xen_cancel(struct p9_client *client, struct p9_req_t *req)
|
|||
return 1;
|
||||
}
|
||||
|
||||
static int p9_xen_create(struct p9_client *client, const char *addr, char *args)
|
||||
static int p9_xen_create(struct p9_client *client, struct fs_context *fc)
|
||||
{
|
||||
const char *addr = fc->source;
|
||||
struct xen_9pfs_front_priv *priv;
|
||||
|
||||
if (addr == NULL)
|
||||
|
|
|
|||
Loading…
Reference in New Issue