ceph: new device mount syntax
Old mount device syntax (source) has the following problems: - mounts to the same cluster but with different fsnames and/or creds have identical device string which can confuse xfstests. - Userspace mount helper tool resolves monitor addresses and fill in mon addrs automatically, but that means the device shown in /proc/mounts is different than what was used for mounting. New device syntax is as follows: cephuser@fsid.mycephfs2=/path Note, there is no "monitor address" in the device string. That gets passed in as mount option. This keeps the device string same when monitor addresses change (on remounts). Also note that the userspace mount helper tool is backward compatible. I.e., the mount helper will fallback to using old syntax after trying to mount with the new syntax. [ idryomov: drop CEPH_MON_ADDR_MNTOPT_DELIM ] Signed-off-by: Venky Shankar <vshankar@redhat.com> Reviewed-by: Jeff Layton <jlayton@kernel.org> Signed-off-by: Ilya Dryomov <idryomov@gmail.com>
This commit is contained in:
parent
4153c7fc93
commit
7b19b4db5a
141
fs/ceph/super.c
141
fs/ceph/super.c
@ -146,6 +146,7 @@ enum {
|
|||||||
Opt_mds_namespace,
|
Opt_mds_namespace,
|
||||||
Opt_recover_session,
|
Opt_recover_session,
|
||||||
Opt_source,
|
Opt_source,
|
||||||
|
Opt_mon_addr,
|
||||||
/* string args above */
|
/* string args above */
|
||||||
Opt_dirstat,
|
Opt_dirstat,
|
||||||
Opt_rbytes,
|
Opt_rbytes,
|
||||||
@ -197,6 +198,7 @@ static const struct fs_parameter_spec ceph_mount_parameters[] = {
|
|||||||
fsparam_u32 ("rsize", Opt_rsize),
|
fsparam_u32 ("rsize", Opt_rsize),
|
||||||
fsparam_string ("snapdirname", Opt_snapdirname),
|
fsparam_string ("snapdirname", Opt_snapdirname),
|
||||||
fsparam_string ("source", Opt_source),
|
fsparam_string ("source", Opt_source),
|
||||||
|
fsparam_string ("mon_addr", Opt_mon_addr),
|
||||||
fsparam_u32 ("wsize", Opt_wsize),
|
fsparam_u32 ("wsize", Opt_wsize),
|
||||||
fsparam_flag_no ("wsync", Opt_wsync),
|
fsparam_flag_no ("wsync", Opt_wsync),
|
||||||
{}
|
{}
|
||||||
@ -228,9 +230,92 @@ static void canonicalize_path(char *path)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Parse the source parameter. Distinguish the server list from the path.
|
* Check if the mds namespace in ceph_mount_options matches
|
||||||
|
* the passed in namespace string. First time match (when
|
||||||
|
* ->mds_namespace is NULL) is treated specially, since
|
||||||
|
* ->mds_namespace needs to be initialized by the caller.
|
||||||
|
*/
|
||||||
|
static int namespace_equals(struct ceph_mount_options *fsopt,
|
||||||
|
const char *namespace, size_t len)
|
||||||
|
{
|
||||||
|
return !(fsopt->mds_namespace &&
|
||||||
|
(strlen(fsopt->mds_namespace) != len ||
|
||||||
|
strncmp(fsopt->mds_namespace, namespace, len)));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ceph_parse_old_source(const char *dev_name, const char *dev_name_end,
|
||||||
|
struct fs_context *fc)
|
||||||
|
{
|
||||||
|
int r;
|
||||||
|
struct ceph_parse_opts_ctx *pctx = fc->fs_private;
|
||||||
|
struct ceph_mount_options *fsopt = pctx->opts;
|
||||||
|
|
||||||
|
if (*dev_name_end != ':')
|
||||||
|
return invalfc(fc, "separator ':' missing in source");
|
||||||
|
|
||||||
|
r = ceph_parse_mon_ips(dev_name, dev_name_end - dev_name,
|
||||||
|
pctx->copts, fc->log.log, ',');
|
||||||
|
if (r)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
fsopt->new_dev_syntax = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ceph_parse_new_source(const char *dev_name, const char *dev_name_end,
|
||||||
|
struct fs_context *fc)
|
||||||
|
{
|
||||||
|
size_t len;
|
||||||
|
struct ceph_fsid fsid;
|
||||||
|
struct ceph_parse_opts_ctx *pctx = fc->fs_private;
|
||||||
|
struct ceph_mount_options *fsopt = pctx->opts;
|
||||||
|
char *fsid_start, *fs_name_start;
|
||||||
|
|
||||||
|
if (*dev_name_end != '=') {
|
||||||
|
dout("separator '=' missing in source");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
fsid_start = strchr(dev_name, '@');
|
||||||
|
if (!fsid_start)
|
||||||
|
return invalfc(fc, "missing cluster fsid");
|
||||||
|
++fsid_start; /* start of cluster fsid */
|
||||||
|
|
||||||
|
fs_name_start = strchr(fsid_start, '.');
|
||||||
|
if (!fs_name_start)
|
||||||
|
return invalfc(fc, "missing file system name");
|
||||||
|
|
||||||
|
if (ceph_parse_fsid(fsid_start, &fsid))
|
||||||
|
return invalfc(fc, "Invalid FSID");
|
||||||
|
|
||||||
|
++fs_name_start; /* start of file system name */
|
||||||
|
len = dev_name_end - fs_name_start;
|
||||||
|
|
||||||
|
if (!namespace_equals(fsopt, fs_name_start, len))
|
||||||
|
return invalfc(fc, "Mismatching mds_namespace");
|
||||||
|
kfree(fsopt->mds_namespace);
|
||||||
|
fsopt->mds_namespace = kstrndup(fs_name_start, len, GFP_KERNEL);
|
||||||
|
if (!fsopt->mds_namespace)
|
||||||
|
return -ENOMEM;
|
||||||
|
dout("file system (mds namespace) '%s'\n", fsopt->mds_namespace);
|
||||||
|
|
||||||
|
fsopt->new_dev_syntax = true;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Parse the source parameter for new device format. Distinguish the device
|
||||||
|
* spec from the path. Try parsing new device format and fallback to old
|
||||||
|
* format if needed.
|
||||||
*
|
*
|
||||||
* The source will look like:
|
* New device syntax will looks like:
|
||||||
|
* <device_spec>=/<path>
|
||||||
|
* where
|
||||||
|
* <device_spec> is name@fsid.fsname
|
||||||
|
* <path> is optional, but if present must begin with '/'
|
||||||
|
* (monitor addresses are passed via mount option)
|
||||||
|
*
|
||||||
|
* Old device syntax is:
|
||||||
* <server_spec>[,<server_spec>...]:[<path>]
|
* <server_spec>[,<server_spec>...]:[<path>]
|
||||||
* where
|
* where
|
||||||
* <server_spec> is <ip>[:<port>]
|
* <server_spec> is <ip>[:<port>]
|
||||||
@ -263,24 +348,44 @@ static int ceph_parse_source(struct fs_parameter *param, struct fs_context *fc)
|
|||||||
dev_name_end = dev_name + strlen(dev_name);
|
dev_name_end = dev_name + strlen(dev_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_name_end--; /* back up to ':' separator */
|
dev_name_end--; /* back up to separator */
|
||||||
if (dev_name_end < dev_name || *dev_name_end != ':')
|
if (dev_name_end < dev_name)
|
||||||
return invalfc(fc, "No path or : separator in source");
|
return invalfc(fc, "Path missing in source");
|
||||||
|
|
||||||
dout("device name '%.*s'\n", (int)(dev_name_end - dev_name), dev_name);
|
dout("device name '%.*s'\n", (int)(dev_name_end - dev_name), dev_name);
|
||||||
if (fsopt->server_path)
|
if (fsopt->server_path)
|
||||||
dout("server path '%s'\n", fsopt->server_path);
|
dout("server path '%s'\n", fsopt->server_path);
|
||||||
|
|
||||||
ret = ceph_parse_mon_ips(param->string, dev_name_end - dev_name,
|
dout("trying new device syntax");
|
||||||
pctx->copts, fc->log.log, ',');
|
ret = ceph_parse_new_source(dev_name, dev_name_end, fc);
|
||||||
if (ret)
|
if (ret) {
|
||||||
return ret;
|
if (ret != -EINVAL)
|
||||||
|
return ret;
|
||||||
|
dout("trying old device syntax");
|
||||||
|
ret = ceph_parse_old_source(dev_name, dev_name_end, fc);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
fc->source = param->string;
|
fc->source = param->string;
|
||||||
param->string = NULL;
|
param->string = NULL;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ceph_parse_mon_addr(struct fs_parameter *param,
|
||||||
|
struct fs_context *fc)
|
||||||
|
{
|
||||||
|
struct ceph_parse_opts_ctx *pctx = fc->fs_private;
|
||||||
|
struct ceph_mount_options *fsopt = pctx->opts;
|
||||||
|
|
||||||
|
kfree(fsopt->mon_addr);
|
||||||
|
fsopt->mon_addr = param->string;
|
||||||
|
param->string = NULL;
|
||||||
|
|
||||||
|
return ceph_parse_mon_ips(fsopt->mon_addr, strlen(fsopt->mon_addr),
|
||||||
|
pctx->copts, fc->log.log, '/');
|
||||||
|
}
|
||||||
|
|
||||||
static int ceph_parse_mount_param(struct fs_context *fc,
|
static int ceph_parse_mount_param(struct fs_context *fc,
|
||||||
struct fs_parameter *param)
|
struct fs_parameter *param)
|
||||||
{
|
{
|
||||||
@ -306,6 +411,8 @@ static int ceph_parse_mount_param(struct fs_context *fc,
|
|||||||
param->string = NULL;
|
param->string = NULL;
|
||||||
break;
|
break;
|
||||||
case Opt_mds_namespace:
|
case Opt_mds_namespace:
|
||||||
|
if (!namespace_equals(fsopt, param->string, strlen(param->string)))
|
||||||
|
return invalfc(fc, "Mismatching mds_namespace");
|
||||||
kfree(fsopt->mds_namespace);
|
kfree(fsopt->mds_namespace);
|
||||||
fsopt->mds_namespace = param->string;
|
fsopt->mds_namespace = param->string;
|
||||||
param->string = NULL;
|
param->string = NULL;
|
||||||
@ -323,6 +430,8 @@ static int ceph_parse_mount_param(struct fs_context *fc,
|
|||||||
if (fc->source)
|
if (fc->source)
|
||||||
return invalfc(fc, "Multiple sources specified");
|
return invalfc(fc, "Multiple sources specified");
|
||||||
return ceph_parse_source(param, fc);
|
return ceph_parse_source(param, fc);
|
||||||
|
case Opt_mon_addr:
|
||||||
|
return ceph_parse_mon_addr(param, fc);
|
||||||
case Opt_wsize:
|
case Opt_wsize:
|
||||||
if (result.uint_32 < PAGE_SIZE ||
|
if (result.uint_32 < PAGE_SIZE ||
|
||||||
result.uint_32 > CEPH_MAX_WRITE_SIZE)
|
result.uint_32 > CEPH_MAX_WRITE_SIZE)
|
||||||
@ -474,6 +583,7 @@ static void destroy_mount_options(struct ceph_mount_options *args)
|
|||||||
kfree(args->mds_namespace);
|
kfree(args->mds_namespace);
|
||||||
kfree(args->server_path);
|
kfree(args->server_path);
|
||||||
kfree(args->fscache_uniq);
|
kfree(args->fscache_uniq);
|
||||||
|
kfree(args->mon_addr);
|
||||||
kfree(args);
|
kfree(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -517,6 +627,10 @@ static int compare_mount_options(struct ceph_mount_options *new_fsopt,
|
|||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
ret = strcmp_null(fsopt1->mon_addr, fsopt2->mon_addr);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
return ceph_compare_options(new_opt, fsc->client);
|
return ceph_compare_options(new_opt, fsc->client);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -572,9 +686,13 @@ static int ceph_show_options(struct seq_file *m, struct dentry *root)
|
|||||||
if ((fsopt->flags & CEPH_MOUNT_OPT_NOCOPYFROM) == 0)
|
if ((fsopt->flags & CEPH_MOUNT_OPT_NOCOPYFROM) == 0)
|
||||||
seq_puts(m, ",copyfrom");
|
seq_puts(m, ",copyfrom");
|
||||||
|
|
||||||
if (fsopt->mds_namespace)
|
/* dump mds_namespace when old device syntax is in use */
|
||||||
|
if (fsopt->mds_namespace && !fsopt->new_dev_syntax)
|
||||||
seq_show_option(m, "mds_namespace", fsopt->mds_namespace);
|
seq_show_option(m, "mds_namespace", fsopt->mds_namespace);
|
||||||
|
|
||||||
|
if (fsopt->mon_addr)
|
||||||
|
seq_printf(m, ",mon_addr=%s", fsopt->mon_addr);
|
||||||
|
|
||||||
if (fsopt->flags & CEPH_MOUNT_OPT_CLEANRECOVER)
|
if (fsopt->flags & CEPH_MOUNT_OPT_CLEANRECOVER)
|
||||||
seq_show_option(m, "recover_session", "clean");
|
seq_show_option(m, "recover_session", "clean");
|
||||||
|
|
||||||
@ -1060,6 +1178,7 @@ static int ceph_setup_bdi(struct super_block *sb, struct ceph_fs_client *fsc)
|
|||||||
static int ceph_get_tree(struct fs_context *fc)
|
static int ceph_get_tree(struct fs_context *fc)
|
||||||
{
|
{
|
||||||
struct ceph_parse_opts_ctx *pctx = fc->fs_private;
|
struct ceph_parse_opts_ctx *pctx = fc->fs_private;
|
||||||
|
struct ceph_mount_options *fsopt = pctx->opts;
|
||||||
struct super_block *sb;
|
struct super_block *sb;
|
||||||
struct ceph_fs_client *fsc;
|
struct ceph_fs_client *fsc;
|
||||||
struct dentry *res;
|
struct dentry *res;
|
||||||
@ -1071,6 +1190,8 @@ static int ceph_get_tree(struct fs_context *fc)
|
|||||||
|
|
||||||
if (!fc->source)
|
if (!fc->source)
|
||||||
return invalfc(fc, "No source");
|
return invalfc(fc, "No source");
|
||||||
|
if (fsopt->new_dev_syntax && !fsopt->mon_addr)
|
||||||
|
return invalfc(fc, "No monitor address");
|
||||||
|
|
||||||
/* create client (which we may/may not use) */
|
/* create client (which we may/may not use) */
|
||||||
fsc = create_fs_client(pctx->opts, pctx->copts);
|
fsc = create_fs_client(pctx->opts, pctx->copts);
|
||||||
|
@ -89,6 +89,8 @@ struct ceph_mount_options {
|
|||||||
unsigned int max_readdir; /* max readdir result (entries) */
|
unsigned int max_readdir; /* max readdir result (entries) */
|
||||||
unsigned int max_readdir_bytes; /* max readdir result (bytes) */
|
unsigned int max_readdir_bytes; /* max readdir result (bytes) */
|
||||||
|
|
||||||
|
bool new_dev_syntax;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* everything above this point can be memcmp'd; everything below
|
* everything above this point can be memcmp'd; everything below
|
||||||
* is handled in compare_mount_options()
|
* is handled in compare_mount_options()
|
||||||
@ -98,6 +100,7 @@ struct ceph_mount_options {
|
|||||||
char *mds_namespace; /* default NULL */
|
char *mds_namespace; /* default NULL */
|
||||||
char *server_path; /* default NULL (means "/") */
|
char *server_path; /* default NULL (means "/") */
|
||||||
char *fscache_uniq; /* default NULL */
|
char *fscache_uniq; /* default NULL */
|
||||||
|
char *mon_addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ceph_fs_client {
|
struct ceph_fs_client {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user