xen_commandline_t saved_cmdline;
static const char __initconst opt_builtin_cmdline[] = CONFIG_CMDLINE;
-static void __init assign_integer_param(
+static int __init assign_integer_param(
const struct kernel_param *param, uint64_t val)
{
switch ( param->len )
{
case sizeof(uint8_t):
- *(uint8_t *)param->var = val;
+ if ( val > UINT8_MAX && val < (uint64_t)INT8_MIN )
+ return -EOVERFLOW;
+ *(uint8_t *)param->par.var = val;
break;
case sizeof(uint16_t):
- *(uint16_t *)param->var = val;
+ if ( val > UINT16_MAX && val < (uint64_t)INT16_MIN )
+ return -EOVERFLOW;
+ *(uint16_t *)param->par.var = val;
break;
case sizeof(uint32_t):
- *(uint32_t *)param->var = val;
+ if ( val > UINT32_MAX && val < (uint64_t)INT32_MIN )
+ return -EOVERFLOW;
+ *(uint32_t *)param->par.var = val;
break;
case sizeof(uint64_t):
- *(uint64_t *)param->var = val;
+ *(uint64_t *)param->par.var = val;
break;
default:
BUG();
}
+
+ return 0;
}
static void __init _cmdline_parse(const char *cmdline)
{
char opt[128], *optval, *optkey, *q;
- const char *p = cmdline;
+ const char *p = cmdline, *key;
const struct kernel_param *param;
- int bool_assert;
+ int rc;
+ bool bool_assert, found;
for ( ; ; )
{
}
/* Boolean parameters can be inverted with 'no-' prefix. */
+ key = optkey;
bool_assert = !!strncmp("no-", optkey, 3);
if ( !bool_assert )
optkey += 3;
+ rc = 0;
+ found = false;
for ( param = __setup_start; param < __setup_end; param++ )
{
+ int rctmp;
+ const char *s;
+
if ( strcmp(param->name, optkey) )
{
if ( param->type == OPT_CUSTOM && q &&
strlen(param->name) == q + 1 - opt &&
!strncmp(param->name, opt, q + 1 - opt) )
{
+ found = true;
optval[-1] = '=';
- ((void (*)(const char *))param->var)(q);
+ rctmp = param->par.func(q);
optval[-1] = '\0';
+ if ( !rc )
+ rc = rctmp;
}
continue;
}
+ rctmp = 0;
+ found = true;
switch ( param->type )
{
case OPT_STR:
- strlcpy(param->var, optval, param->len);
+ strlcpy(param->par.var, optval, param->len);
break;
case OPT_UINT:
- assign_integer_param(
+ rctmp = assign_integer_param(
param,
- simple_strtoll(optval, NULL, 0));
+ simple_strtoll(optval, &s, 0));
+ if ( *s )
+ rctmp = -EINVAL;
break;
case OPT_BOOL:
- if ( !parse_bool(*optval ? optval : "yes", NULL) )
+ rctmp = *optval ? parse_bool(optval, NULL) : 0;
+ if ( rctmp < 0 )
+ break;
+ if ( !rctmp )
bool_assert = !bool_assert;
+ rctmp = 0;
assign_integer_param(param, bool_assert);
break;
case OPT_SIZE:
- assign_integer_param(
+ rctmp = assign_integer_param(
param,
- parse_size_and_unit(optval, NULL));
+ parse_size_and_unit(optval, &s));
+ if ( *s )
+ rctmp = -EINVAL;
break;
case OPT_CUSTOM:
+ rctmp = -EINVAL;
if ( !bool_assert )
{
if ( *optval )
safe_strcpy(opt, "no");
optval = opt;
}
- ((void (*)(const char *))param->var)(optval);
+ rctmp = param->par.func(optval);
break;
default:
BUG();
break;
}
+
+ if ( !rc )
+ rc = rctmp;
}
+
+ if ( rc )
+ printk("parameter \"%s\" has invalid value \"%s\", rc=%d!\n",
+ key, optval, rc);
+ if ( !found )
+ printk("parameter \"%s\" unknown!\n", key);
}
}
OPT_CUSTOM
} type;
unsigned int len;
- void *var;
+ union {
+ void *var;
+ int (*func)(const char *);
+ } par;
};
extern const struct kernel_param __setup_start[], __setup_end[];
#define custom_param(_name, _var) \
__setup_str __setup_str_##_var[] = _name; \
- __kparam __setup_##_var = { __setup_str_##_var, OPT_CUSTOM, 0, _var }
+ __kparam __setup_##_var = \
+ { .name = __setup_str_##_var, \
+ .type = OPT_CUSTOM, \
+ .par.func = _var }
#define boolean_param(_name, _var) \
__setup_str __setup_str_##_var[] = _name; \
__kparam __setup_##_var = \
- { __setup_str_##_var, OPT_BOOL, sizeof(_var), &_var }
+ { .name = __setup_str_##_var, \
+ .type = OPT_BOOL, \
+ .len = sizeof(_var), \
+ .par.var = &_var }
#define integer_param(_name, _var) \
__setup_str __setup_str_##_var[] = _name; \
__kparam __setup_##_var = \
- { __setup_str_##_var, OPT_UINT, sizeof(_var), &_var }
+ { .name = __setup_str_##_var, \
+ .type = OPT_UINT, \
+ .len = sizeof(_var), \
+ .par.var = &_var }
#define size_param(_name, _var) \
__setup_str __setup_str_##_var[] = _name; \
__kparam __setup_##_var = \
- { __setup_str_##_var, OPT_SIZE, sizeof(_var), &_var }
+ { .name = __setup_str_##_var, \
+ .type = OPT_SIZE, \
+ .len = sizeof(_var), \
+ .par.var = &_var }
#define string_param(_name, _var) \
__setup_str __setup_str_##_var[] = _name; \
__kparam __setup_##_var = \
- { __setup_str_##_var, OPT_STR, sizeof(_var), &_var }
+ { .name = __setup_str_##_var, \
+ .type = OPT_STR, \
+ .len = sizeof(_var), \
+ .par.var = &_var }
#endif /* __ASSEMBLY__ */