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:
David Howells 2019-12-10 07:31:09 -05:00 committed by Anna Schumaker
parent cbd071b5da
commit 48be8a66cf
2 changed files with 50 additions and 54 deletions

View File

@ -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);

View File

@ -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 */