Hi J, On Wed, Jan 26, 2011 at 05:46:03PM +0100, Jean-Christophe PLAGNIOL-VILLARD wrote: > for now just support static boot command support > > Signed-off-by: Jean-Christophe PLAGNIOL-VILLARD <plagnioj@xxxxxxxxxxxx> > --- > arch/arm/lib/barebox.lds.S | 4 + > arch/blackfin/boards/ipe337/barebox.lds.S | 4 + > arch/ppc/boards/pcm030/barebox.lds.S | 4 + > arch/sandbox/board/barebox.lds.S | 4 + > arch/sandbox/lib/barebox.lds.S | 4 + > arch/x86/lib/barebox.lds.S | 9 ++- > common/Kconfig | 11 ++ > common/Makefile | 1 + > common/params.c | 158 +++++++++++++++++++++++++++++ > common/startup.c | 74 +++++++++++++- > include/asm-generic/barebox.lds.h | 2 + > include/init.h | 28 +++++ > 12 files changed, 301 insertions(+), 2 deletions(-) > create mode 100644 common/params.c Except for the inlined comments I'm generally ok with this patch. The more interesting part will be how the calling convention for barebox as a second stage loader is and how you preserve the registers used for commandline/atag passing during startup. An idea which comes to my mind is that we could introduce a CONFIG_BAREBOX_SECOND_STAGE which switches to a completely different startup process. This startup process could then assume that sdram is already initialized and that we do not have to call board_init_lowlevel. Another interesting thing is how will barebox behave if called as a second stage loader but without valid atags? > > > +config BOOT_CMDLINE > + bool "barebox boot command string" > + help This is useful if you intend to use barebox as a second stage bootloader and want to pass kernel like command line parameters to barebox. Otherwise say no here. > + > +config CMDLINE > + string "Default barebox command string" > + depends on BOOT_CMDLINE > + default "" > + help > + > config GLOB > bool > prompt "hush globbing support" > diff --git a/common/Makefile b/common/Makefile > index 98c9d36..94816d5 100644 > --- a/common/Makefile > +++ b/common/Makefile > @@ -16,6 +16,7 @@ obj-$(CONFIG_DIGEST) += digest.o > obj-y += env.o > obj-$(CONFIG_CMD_BOOTM) += image.o > obj-y += startup.o > +obj-$(CONFIG_BOOT_CMDLINE) += params.o > obj-y += misc.o > obj-y += memsize.o > obj-$(CONFIG_MENU) += menu.o > diff --git a/common/params.c b/common/params.c > new file mode 100644 > index 0000000..ea35401 > --- /dev/null > +++ b/common/params.c > @@ -0,0 +1,158 @@ > +/* Helpers for initial module or kernel cmdline parsing > + Copyright (C) 2001 Rusty Russell. > + > + This program is free software; you can redistribute it and/or modify > + it under the terms of the GNU General Public License as published by > + the Free Software Foundation; either version 2 of the License, or > + (at your option) any later version. > + > + This program is distributed in the hope that it will be useful, > + but WITHOUT ANY WARRANTY; without even the implied warranty of > + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > + GNU General Public License for more details. > + > + You should have received a copy of the GNU General Public License > + along with this program; if not, write to the Free Software > + Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA > +*/ > +#include <malloc.h> > +#include <linux/kernel.h> > +#include <linux/string.h> > +#include <errno.h> > +#include <linux/err.h> > +#include <linux/ctype.h> > + > +#if 0 > +#define DEBUGP printk > +#else > +#define DEBUGP(fmt, a...) > +#endif Please use the regular 'debug' function here. > + > +/* This just allows us to keep track of which parameters are kmalloced. */ > +struct kmalloced_param { > + struct list_head list; > + char val[]; > +}; > +static LIST_HEAD(kmalloced_params); This seems unused. > + > +static inline char dash2underscore(char c) > +{ > + if (c == '-') > + return '_'; > + return c; > +} > + > +static inline int parameq(const char *input, const char *paramname) > +{ > + unsigned int i; > + for (i = 0; dash2underscore(input[i]) == paramname[i]; i++) > + if (input[i] == '\0') > + return 1; > + return 0; > +} > + > +static int parse_one(char *param, > + char *val, > + int (*handle_unknown)(char *param, char *val)) > +{ > + if (handle_unknown) { > + DEBUGP("Unknown argument: calling %p\n", handle_unknown); > + return handle_unknown(param, val); > + } > + > + DEBUGP("Unknown argument `%s'\n", param); > + return -ENOENT; > +} > + > +/* You can use " around spaces, but can't escape ". */ > +/* Hyphens and underscores equivalent in parameter names. */ > +static char *next_arg(char *args, char **param, char **val) > +{ > + unsigned int i, equals = 0; > + int in_quote = 0, quoted = 0; > + char *next; > + > + if (*args == '"') { > + args++; > + in_quote = 1; > + quoted = 1; > + } > + > + for (i = 0; args[i]; i++) { > + if (isspace(args[i]) && !in_quote) > + break; > + if (equals == 0) { > + if (args[i] == '=') > + equals = i; > + } > + if (args[i] == '"') > + in_quote = !in_quote; > + } > + > + *param = args; > + if (!equals) > + *val = NULL; > + else { > + args[equals] = '\0'; > + *val = args + equals + 1; > + > + /* Don't include quotes in value. */ > + if (**val == '"') { > + (*val)++; > + if (args[i-1] == '"') > + args[i-1] = '\0'; > + } > + if (quoted && args[i-1] == '"') > + args[i-1] = '\0'; > + } > + > + if (args[i]) { > + args[i] = '\0'; > + next = args + i + 1; > + } else > + next = args + i; > + > + /* Chew up trailing spaces. */ > + return skip_spaces(next); > +} > + > +/* Args looks like "foo=bar,bar2 baz=fuz wiz". */ > +int parse_args(const char *name, > + char *args, > + int (*unknown)(char *param, char *val)) > +{ > + char *param, *val; > + > + DEBUGP("Parsing ARGS: %s\n", args); > + > + /* Chew leading spaces */ > + args = skip_spaces(args); > + > + while (*args) { > + int ret; > + > + args = next_arg(args, ¶m, &val); > + ret = parse_one(param, val, unknown); > + switch (ret) { > + case -ENOENT: > + printk(KERN_ERR "%s: Unknown parameter `%s'\n", > + name, param); > + return ret; > + case -ENOSPC: > + printk(KERN_ERR > + "%s: `%s' too large for parameter `%s'\n", > + name, val ?: "", param); > + return ret; > + case 0: > + break; > + default: > + printk(KERN_ERR > + "%s: `%s' invalid for parameter `%s'\n", > + name, val ?: "", param); > + return ret; > + } > + } > + > + /* All parsed OK. */ > + return 0; > +} > diff --git a/common/startup.c b/common/startup.c > index aa76cb7..98b4aab 100644 > --- a/common/startup.c > +++ b/common/startup.c > @@ -41,6 +41,7 @@ > #include <reloc.h> > #include <asm-generic/memory_layout.h> > #include <asm/sections.h> > +#include <asm/setup.h> > > extern initcall_t __barebox_initcalls_start[], __barebox_early_initcalls_end[], > __barebox_initcalls_end[]; > @@ -103,6 +104,15 @@ static int register_default_env(void) > device_initcall(register_default_env); > #endif > > +#ifdef CONFIG_BOOT_CMDLINE > +static int __init bootmode(char *mode) > +{ > + printf("boot=%s\n", mode); > + return 0; > +} > +__setup("boot=", bootmode); > +#endif > + > static int mount_root(void) > { > mount("none", "ramfs", "/"); > @@ -112,6 +122,67 @@ static int mount_root(void) > } > fs_initcall(mount_root); > > +#ifdef CONFIG_BOOT_CMDLINE > +extern const struct obs_kernel_param __setup_start[], __setup_end[]; > + > +static int __init obsolete_checksetup(char *line) > +{ > + const struct obs_kernel_param *p; > + > + p = __setup_start; > + do { > + int n = strlen(p->str); > + if (!strncmp(line, p->str, n)) { > + if (!p->setup_func) { > + printk(KERN_WARNING "Parameter %s is obsolete," > + " ignored\n", p->str); > + return 1; > + } else if (p->setup_func(line + n)) > + return 1; > + } > + p++; > + } while (p < __setup_end); > + > + return 0; > +} > + > +/* > + * Unknown boot options get handed to init, unless they look like > + * unused parameters (modprobe will find them in /proc/cmdline). > + */ > +static int __init unknown_bootoption(char *param, char *val) > +{ > + /* Change NUL term back to "=", to make "param" the whole string. */ > + if (val) { > + /* param=val or param="val"? */ > + if (val == param+strlen(param)+1) > + val[-1] = '='; > + else if (val == param+strlen(param)+2) { > + val[-2] = '='; > + memmove(val-1, val, strlen(val)+1); > + val--; > + } else > + BUG(); > + } > + > + /* Handle obsolete-style parameters */ > + obsolete_checksetup(param); > + return 0; > +} > + > +void __init parse_param(void) > +{ > + static __initdata char tmp_cmdline[COMMAND_LINE_SIZE]; > + > + strlcpy(tmp_cmdline, CONFIG_CMDLINE, COMMAND_LINE_SIZE); > + parse_args("Booting Barebox", tmp_cmdline, unknown_bootoption); > +} > +#else > +static void inline parse_param(void) > +{ > +} > +#endif > + > void start_barebox (void) > { > initcall_t *initcall; > @@ -128,6 +199,8 @@ void start_barebox (void) > init_data_ptr = &__early_init_data_begin; > #endif /* CONFIG_HAS_EARLY_INIT */ > > + parse_param(); > + > for (initcall = __barebox_initcalls_start; > initcall < __barebox_initcalls_end; initcall++) { > PUTHEX_LL(*initcall); > @@ -180,4 +253,3 @@ void shutdown_barebox(void) > arch_shutdown(); > #endif > } > - > diff --git a/include/asm-generic/barebox.lds.h b/include/asm-generic/barebox.lds.h > index fc141a4..18eab31 100644 > --- a/include/asm-generic/barebox.lds.h > +++ b/include/asm-generic/barebox.lds.h > @@ -7,6 +7,8 @@ > #define PRE_IMAGE > #endif > > +#define INIT_SETUP KEEP(*(.init_setup)) > + > #define INITCALLS \ > KEEP(*(.initcall.0)) \ > KEEP(*(.initcall.1)) \ > diff --git a/include/init.h b/include/init.h > index 2f4fac1..53148fa 100644 > --- a/include/init.h > +++ b/include/init.h > @@ -44,6 +44,34 @@ typedef int (*initcall_t)(void); > */ > #define __bare_init __section(.text_bare_init.text) > > +#define __initconst __section(.rodata) > + > +struct obs_kernel_param { > + const char *str; > + int (*setup_func)(char *); > +}; > + > +/* > + * Only for really core code. See moduleparam.h for the normal way. > + * > + * Force the alignment so the compiler doesn't space elements of the > + * obs_kernel_param "array" too far apart in .init.setup. > + */ > +#define __setup_param(str, unique_id, fn) \ > + static const char __setup_str_##unique_id[] __initconst \ > + __aligned(1) = str; \ > + static struct obs_kernel_param __setup_##unique_id \ > + __used __section(.init_setup) \ > + __attribute__((aligned((sizeof(long))))) \ > + = { __setup_str_##unique_id, fn} > + > +#define __setup(str, fn) \ > + __setup_param(str, fn, fn) > + > +/* Called on module insert or kernel boot */ > +extern int parse_args(const char *name, > + char *args, > + int (*unknown)(char *param, char *val)); > #endif > > #endif /* _INIT_H */ > -- > 1.7.2.3 > > > _______________________________________________ > barebox mailing list > barebox@xxxxxxxxxxxxxxxxxxx > http://lists.infradead.org/mailman/listinfo/barebox > -- Pengutronix e.K. | | Industrial Linux Solutions | http://www.pengutronix.de/ | Peiner Str. 6-8, 31137 Hildesheim, Germany | Phone: +49-5121-206917-0 | Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 | _______________________________________________ barebox mailing list barebox@xxxxxxxxxxxxxxxxxxx http://lists.infradead.org/mailman/listinfo/barebox