Skip to content

Commit 51e158c

Browse files
committed
param: hand arguments after -- straight to init
The kernel passes any args it doesn't need through to init, except it assumes anything containing '.' belongs to the kernel (for a module). This change means all users can clearly distinguish which arguments are for init. For example, the kernel uses debug ("dee-bug") to mean log everything to the console, where systemd uses the debug from the Scandinavian "day-boog" meaning "fail to boot". If a future versions uses argv[] instead of reading /proc/cmdline, this confusion will be avoided. eg: test 'FOO="this is --foo"' -- 'systemd.debug="true true true"' Gives: argv[0] = '/debug-init' argv[1] = 'test' argv[2] = 'systemd.debug=true true true' envp[0] = 'HOME=/' envp[1] = 'TERM=linux' envp[2] = 'FOO=this is --foo' Signed-off-by: Rusty Russell <[email protected]>
1 parent 2ee41e6 commit 51e158c

File tree

4 files changed

+53
-19
lines changed

4 files changed

+53
-19
lines changed

include/linux/moduleparam.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -321,7 +321,7 @@ extern bool parameq(const char *name1, const char *name2);
321321
extern bool parameqn(const char *name1, const char *name2, size_t n);
322322

323323
/* Called on module insert or kernel boot */
324-
extern int parse_args(const char *name,
324+
extern char *parse_args(const char *name,
325325
char *args,
326326
const struct kernel_param *params,
327327
unsigned num,

init/main.c

+29-4
Original file line numberDiff line numberDiff line change
@@ -252,6 +252,27 @@ static int __init repair_env_string(char *param, char *val, const char *unused)
252252
return 0;
253253
}
254254

255+
/* Anything after -- gets handed straight to init. */
256+
static int __init set_init_arg(char *param, char *val, const char *unused)
257+
{
258+
unsigned int i;
259+
260+
if (panic_later)
261+
return 0;
262+
263+
repair_env_string(param, val, unused);
264+
265+
for (i = 0; argv_init[i]; i++) {
266+
if (i == MAX_INIT_ARGS) {
267+
panic_later = "init";
268+
panic_param = param;
269+
return 0;
270+
}
271+
}
272+
argv_init[i] = param;
273+
return 0;
274+
}
275+
255276
/*
256277
* Unknown boot options get handed to init, unless they look like
257278
* unused parameters (modprobe will find them in /proc/cmdline).
@@ -478,7 +499,7 @@ static void __init mm_init(void)
478499

479500
asmlinkage void __init start_kernel(void)
480501
{
481-
char * command_line;
502+
char * command_line, *after_dashes;
482503
extern const struct kernel_param __start___param[], __stop___param[];
483504

484505
/*
@@ -519,9 +540,13 @@ asmlinkage void __init start_kernel(void)
519540

520541
pr_notice("Kernel command line: %s\n", boot_command_line);
521542
parse_early_param();
522-
parse_args("Booting kernel", static_command_line, __start___param,
523-
__stop___param - __start___param,
524-
-1, -1, &unknown_bootoption);
543+
after_dashes = parse_args("Booting kernel",
544+
static_command_line, __start___param,
545+
__stop___param - __start___param,
546+
-1, -1, &unknown_bootoption);
547+
if (after_dashes)
548+
parse_args("Setting init args", after_dashes, NULL, 0, -1, -1,
549+
set_init_arg);
525550

526551
jump_label_init();
527552

kernel/module.c

+9-3
Original file line numberDiff line numberDiff line change
@@ -3193,6 +3193,7 @@ static int load_module(struct load_info *info, const char __user *uargs,
31933193
{
31943194
struct module *mod;
31953195
long err;
3196+
char *after_dashes;
31963197

31973198
err = module_sig_check(info);
31983199
if (err)
@@ -3277,10 +3278,15 @@ static int load_module(struct load_info *info, const char __user *uargs,
32773278
goto ddebug_cleanup;
32783279

32793280
/* Module is ready to execute: parsing args may do that. */
3280-
err = parse_args(mod->name, mod->args, mod->kp, mod->num_kp,
3281-
-32768, 32767, unknown_module_param_cb);
3282-
if (err < 0)
3281+
after_dashes = parse_args(mod->name, mod->args, mod->kp, mod->num_kp,
3282+
-32768, 32767, unknown_module_param_cb);
3283+
if (IS_ERR(after_dashes)) {
3284+
err = PTR_ERR(after_dashes);
32833285
goto bug_cleanup;
3286+
} else if (after_dashes) {
3287+
pr_warn("%s: parameters '%s' after `--' ignored\n",
3288+
mod->name, after_dashes);
3289+
}
32843290

32853291
/* Link in to syfs. */
32863292
err = mod_sysfs_setup(mod, info, mod->kp, mod->num_kp);

kernel/params.c

+14-11
Original file line numberDiff line numberDiff line change
@@ -177,13 +177,13 @@ static char *next_arg(char *args, char **param, char **val)
177177
}
178178

179179
/* Args looks like "foo=bar,bar2 baz=fuz wiz". */
180-
int parse_args(const char *doing,
181-
char *args,
182-
const struct kernel_param *params,
183-
unsigned num,
184-
s16 min_level,
185-
s16 max_level,
186-
int (*unknown)(char *param, char *val, const char *doing))
180+
char *parse_args(const char *doing,
181+
char *args,
182+
const struct kernel_param *params,
183+
unsigned num,
184+
s16 min_level,
185+
s16 max_level,
186+
int (*unknown)(char *param, char *val, const char *doing))
187187
{
188188
char *param, *val;
189189

@@ -198,6 +198,9 @@ int parse_args(const char *doing,
198198
int irq_was_disabled;
199199

200200
args = next_arg(args, &param, &val);
201+
/* Stop at -- */
202+
if (!val && strcmp(param, "--") == 0)
203+
return args;
201204
irq_was_disabled = irqs_disabled();
202205
ret = parse_one(param, val, doing, params, num,
203206
min_level, max_level, unknown);
@@ -208,22 +211,22 @@ int parse_args(const char *doing,
208211
switch (ret) {
209212
case -ENOENT:
210213
pr_err("%s: Unknown parameter `%s'\n", doing, param);
211-
return ret;
214+
return ERR_PTR(ret);
212215
case -ENOSPC:
213216
pr_err("%s: `%s' too large for parameter `%s'\n",
214217
doing, val ?: "", param);
215-
return ret;
218+
return ERR_PTR(ret);
216219
case 0:
217220
break;
218221
default:
219222
pr_err("%s: `%s' invalid for parameter `%s'\n",
220223
doing, val ?: "", param);
221-
return ret;
224+
return ERR_PTR(ret);
222225
}
223226
}
224227

225228
/* All parsed OK. */
226-
return 0;
229+
return NULL;
227230
}
228231

229232
/* Lazy bastard, eh? */

0 commit comments

Comments
 (0)