NFS: Add a small buffer in nfs_fs_context to avoid string dup
Add a small buffer in nfs_fs_context to avoid string duplication when parsing numbers. Also make the parsing function wrapper place the parsed integer directly in the appropriate nfs_fs_context struct member. Signed-off-by: David Howells <dhowells@redhat.com> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk> Signed-off-by: Anna Schumaker <Anna.Schumaker@Netapp.com>
This commit is contained in:
parent
cbd071b5da
commit
48be8a66cf
|
@ -468,27 +468,38 @@ static int nfs_get_option_str(substring_t args[], char **option)
|
||||||
return !*option;
|
return !*option;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nfs_get_option_ul(substring_t args[], unsigned long *option)
|
static int nfs_get_option_ui(struct nfs_fs_context *ctx,
|
||||||
|
substring_t args[], unsigned int *option)
|
||||||
{
|
{
|
||||||
int rc;
|
match_strlcpy(ctx->buf, args, sizeof(ctx->buf));
|
||||||
char *string;
|
return kstrtouint(ctx->buf, 10, option);
|
||||||
|
|
||||||
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,
|
static int nfs_get_option_ui_bound(struct nfs_fs_context *ctx,
|
||||||
unsigned long l_bound, unsigned long u_bound)
|
substring_t args[], unsigned int *option,
|
||||||
|
unsigned int l_bound, unsigned u_bound)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = nfs_get_option_ul(args, option);
|
match_strlcpy(ctx->buf, args, sizeof(ctx->buf));
|
||||||
if (ret != 0)
|
ret = kstrtouint(ctx->buf, 10, option);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
if (*option < l_bound || *option > u_bound)
|
||||||
|
return -ERANGE;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int nfs_get_option_us_bound(struct nfs_fs_context *ctx,
|
||||||
|
substring_t args[], unsigned short *option,
|
||||||
|
unsigned short l_bound,
|
||||||
|
unsigned short u_bound)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
match_strlcpy(ctx->buf, args, sizeof(ctx->buf));
|
||||||
|
ret = kstrtou16(ctx->buf, 10, option);
|
||||||
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
if (*option < l_bound || *option > u_bound)
|
if (*option < l_bound || *option > u_bound)
|
||||||
return -ERANGE;
|
return -ERANGE;
|
||||||
|
@ -501,7 +512,6 @@ static int nfs_get_option_ul_bound(substring_t args[], unsigned long *option,
|
||||||
static int nfs_fs_context_parse_option(struct nfs_fs_context *ctx, char *p)
|
static int nfs_fs_context_parse_option(struct nfs_fs_context *ctx, char *p)
|
||||||
{
|
{
|
||||||
substring_t args[MAX_OPT_ARGS];
|
substring_t args[MAX_OPT_ARGS];
|
||||||
unsigned long option;
|
|
||||||
char *string;
|
char *string;
|
||||||
int token, rc;
|
int token, rc;
|
||||||
|
|
||||||
|
@ -609,86 +619,70 @@ static int nfs_fs_context_parse_option(struct nfs_fs_context *ctx, char *p)
|
||||||
* options that take numeric values
|
* options that take numeric values
|
||||||
*/
|
*/
|
||||||
case Opt_port:
|
case Opt_port:
|
||||||
if (nfs_get_option_ul(args, &option) ||
|
if (nfs_get_option_ui_bound(ctx, args, &ctx->nfs_server.port,
|
||||||
option > USHRT_MAX)
|
0, USHRT_MAX))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->nfs_server.port = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_rsize:
|
case Opt_rsize:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->rsize))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->rsize = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_wsize:
|
case Opt_wsize:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->wsize))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->wsize = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_bsize:
|
case Opt_bsize:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->bsize))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->bsize = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_timeo:
|
case Opt_timeo:
|
||||||
if (nfs_get_option_ul_bound(args, &option, 1, INT_MAX))
|
if (nfs_get_option_ui_bound(ctx, args, &ctx->timeo, 1, INT_MAX))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->timeo = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_retrans:
|
case Opt_retrans:
|
||||||
if (nfs_get_option_ul_bound(args, &option, 0, INT_MAX))
|
if (nfs_get_option_ui_bound(ctx, args, &ctx->retrans, 0, INT_MAX))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->retrans = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_acregmin:
|
case Opt_acregmin:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->acregmin))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->acregmin = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_acregmax:
|
case Opt_acregmax:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->acregmax))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->acregmax = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_acdirmin:
|
case Opt_acdirmin:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->acdirmin))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->acdirmin = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_acdirmax:
|
case Opt_acdirmax:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->acdirmax))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->acdirmax = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_actimeo:
|
case Opt_actimeo:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->acdirmax))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->acregmin = ctx->acregmax =
|
ctx->acregmin = ctx->acregmax =
|
||||||
ctx->acdirmin = ctx->acdirmax = option;
|
ctx->acdirmin = ctx->acdirmax;
|
||||||
break;
|
break;
|
||||||
case Opt_namelen:
|
case Opt_namelen:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui(ctx, args, &ctx->namlen))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->namlen = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_mountport:
|
case Opt_mountport:
|
||||||
if (nfs_get_option_ul(args, &option) ||
|
if (nfs_get_option_ui_bound(ctx, args, &ctx->mount_server.port,
|
||||||
option > USHRT_MAX)
|
0, USHRT_MAX))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->mount_server.port = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_mountvers:
|
case Opt_mountvers:
|
||||||
if (nfs_get_option_ul(args, &option) ||
|
if (nfs_get_option_ui_bound(ctx, args, &ctx->mount_server.version,
|
||||||
option < NFS_MNT_VERSION ||
|
NFS_MNT_VERSION, NFS_MNT3_VERSION))
|
||||||
option > NFS_MNT3_VERSION)
|
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->mount_server.version = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_minorversion:
|
case Opt_minorversion:
|
||||||
if (nfs_get_option_ul(args, &option))
|
if (nfs_get_option_ui_bound(ctx, args, &ctx->minorversion,
|
||||||
|
0, NFS4_MAX_MINOR_VERSION))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
if (option > NFS4_MAX_MINOR_VERSION)
|
|
||||||
goto out_invalid_value;
|
|
||||||
ctx->minorversion = option;
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -820,9 +814,9 @@ static int nfs_fs_context_parse_option(struct nfs_fs_context *ctx, char *p)
|
||||||
goto out_invalid_address;
|
goto out_invalid_address;
|
||||||
break;
|
break;
|
||||||
case Opt_nconnect:
|
case Opt_nconnect:
|
||||||
if (nfs_get_option_ul_bound(args, &option, 1, NFS_MAX_CONNECTIONS))
|
if (nfs_get_option_us_bound(ctx, args, &ctx->nfs_server.nconnect,
|
||||||
|
1, NFS_MAX_CONNECTIONS))
|
||||||
goto out_invalid_value;
|
goto out_invalid_value;
|
||||||
ctx->nfs_server.nconnect = option;
|
|
||||||
break;
|
break;
|
||||||
case Opt_lookupcache:
|
case Opt_lookupcache:
|
||||||
string = match_strdup(args);
|
string = match_strdup(args);
|
||||||
|
|
|
@ -122,6 +122,8 @@ struct nfs_fs_context {
|
||||||
|
|
||||||
void *lsm_opts;
|
void *lsm_opts;
|
||||||
struct net *net;
|
struct net *net;
|
||||||
|
|
||||||
|
char buf[32]; /* Parse buffer */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* mount_clnt.c */
|
/* mount_clnt.c */
|
||||||
|
|
Loading…
Reference in New Issue