[PATCH] of: Support CONFIG_CMDLINE_EXTEND config option

Rob Herring robherring2 at gmail.com
Wed Jan 11 13:38:23 EST 2012


On 01/09/2012 06:54 PM, Doug Anderson wrote:
> The old logic assumes CMDLINE_FROM_BOOTLOADER vs. CMDLINE_FORCE and
> ignores CMDLINE_EXTEND.  Here's the old logic:
> 
> - CONFIG_CMDLINE_FORCE=true
>     CONFIG_CMDLINE
> - dt bootargs=non-empty:
>     dt bootargs
> - dt bootargs=empty, @data is non-empty string
>     @data is left unchanged
> - dt bootargs=empty, @data is empty string
>     CONFIG_CMDLINE (or "" if that's not defined)
> 
> The old logic would also not honor CONFIG_CMDLINE_FORCE if there was no
> "chosen" attribute in the device tree.
> 
> The new logic is now documented in of_fdt.h and is copied here for
> reference:
> 
> - CONFIG_CMDLINE_FORCE=true
>     CONFIG_CMDLINE
> - CONFIG_CMDLINE_EXTEND=true
>     CONFIG_CMDLINE + dt bootargs (even if dt bootargs are empty)
> - CMDLINE_FROM_BOOTLOADER=true, dt bootargs=non-empty:
>     dt bootargs
> - CMDLINE_FROM_BOOTLOADER=true, dt bootargs=empty, @data is non-empty string
>     @data is left unchanged
> - CMDLINE_FROM_BOOTLOADER=true, dt bootargs=empty, @data is empty string
>     CONFIG_CMDLINE (or "" if that's not defined)
> 
> Signed-off-by: Doug Anderson <dianders at chromium.org>
> CC: devicetree-discuss at lists-ozlabs.org
> CC: Grant Likely <grant.likely at secretlab.ca>
> CC: Benjamin Herrenschmidt <benh at kernel.crashing.org>
> ---

Acked-by: Rob Herring <rob.herring at calxeda.com>

I'll apply, but would like Ben's ack first.

Rob

>  drivers/of/fdt.c       |   56 ++++++++++++++++++++++++++++++++++-------------
>  include/linux/of_fdt.h |   19 ++++++++++++++++
>  2 files changed, 59 insertions(+), 16 deletions(-)
> 
> diff --git a/drivers/of/fdt.c b/drivers/of/fdt.c
> index ea2bd1b..577f3a9 100644
> --- a/drivers/of/fdt.c
> +++ b/drivers/of/fdt.c
> @@ -664,6 +664,29 @@ int __init early_init_dt_scan_memory(unsigned long node, const char *uname,
>  	return 0;
>  }
>  
> +/*
> + * Convert configs to something easy to use in C code
> + */
> +#if defined(CONFIG_CMDLINE_FORCE)
> +static const int overwrite_incoming_cmdline = 1;
> +static const int read_dt_cmdline;
> +static const int concat_cmdline;
> +#elif defined(CONFIG_CMDLINE_EXTEND)
> +static const int overwrite_incoming_cmdline = 1;
> +static const int read_dt_cmdline = 1;
> +static const int concat_cmdline = 1;
> +#else /* CMDLINE_FROM_BOOTLOADER */
> +static const int overwrite_incoming_cmdline;
> +static const int read_dt_cmdline = 1;
> +static const int concat_cmdline;
> +#endif
> +
> +#ifdef CONFIG_CMDLINE
> +static const char *config_cmdline = CONFIG_CMDLINE;
> +#else
> +static const char *config_cmdline = "";
> +#endif
> +
>  int __init early_init_dt_scan_chosen(unsigned long node, const char *uname,
>  				     int depth, void *data)
>  {
> @@ -672,28 +695,29 @@ int __init early_init_dt_scan_chosen(unsigned long node, const char *uname,
>  
>  	pr_debug("search \"chosen\", depth: %d, uname: %s\n", depth, uname);
>  
> +	/* Make sure cmdline default is set early to handle case of no chosen */
> +	if (data && (overwrite_incoming_cmdline || !((char *)data)[0]))
> +		strlcpy(data, config_cmdline, COMMAND_LINE_SIZE);
> +
>  	if (depth != 1 || !data ||
>  	    (strcmp(uname, "chosen") != 0 && strcmp(uname, "chosen at 0") != 0))
>  		return 0;
>  
>  	early_init_dt_check_for_initrd(node);
>  
> -	/* Retrieve command line */
> -	p = of_get_flat_dt_prop(node, "bootargs", &l);
> -	if (p != NULL && l > 0)
> -		strlcpy(data, p, min((int)l, COMMAND_LINE_SIZE));
> -
> -	/*
> -	 * CONFIG_CMDLINE is meant to be a default in case nothing else
> -	 * managed to set the command line, unless CONFIG_CMDLINE_FORCE
> -	 * is set in which case we override whatever was found earlier.
> -	 */
> -#ifdef CONFIG_CMDLINE
> -#ifndef CONFIG_CMDLINE_FORCE
> -	if (!((char *)data)[0])
> -#endif
> -		strlcpy(data, CONFIG_CMDLINE, COMMAND_LINE_SIZE);
> -#endif /* CONFIG_CMDLINE */
> +	/* Retrieve command line unless forcing */
> +	if (read_dt_cmdline) {
> +		p = of_get_flat_dt_prop(node, "bootargs", &l);
> +		if (p != NULL && l > 0) {
> +			if (concat_cmdline) {
> +				strlcat(data, " ", COMMAND_LINE_SIZE);
> +				strlcat(data, p, min_t(int, (int)l,
> +						       COMMAND_LINE_SIZE));
> +			} else
> +				strlcpy(data, p, min_t(int, (int)l,
> +						       COMMAND_LINE_SIZE));
> +		}
> +	}
>  
>  	pr_debug("Command line is: %s\n", (char*)data);
>  
> diff --git a/include/linux/of_fdt.h b/include/linux/of_fdt.h
> index ed136ad..346d6c7 100644
> --- a/include/linux/of_fdt.h
> +++ b/include/linux/of_fdt.h
> @@ -91,6 +91,25 @@ extern int of_flat_dt_is_compatible(unsigned long node, const char *name);
>  extern int of_flat_dt_match(unsigned long node, const char *const *matches);
>  extern unsigned long of_get_flat_dt_root(void);
>  
> +/*
> + * early_init_dt_scan_chosen - scan the device tree for ramdisk and bootargs
> + *
> + * The boot arguments will be placed into the memory pointed to by @data.
> + * That memory should be COMMAND_LINE_SIZE big and initialized to be a valid
> + * (possibly empty) string.  Logic for what will be in @data after this
> + * function finishes:
> + *
> + * - CONFIG_CMDLINE_FORCE=true
> + *     CONFIG_CMDLINE
> + * - CONFIG_CMDLINE_EXTEND=true
> + *     CONFIG_CMDLINE + dt bootargs (even if dt bootargs are empty)
> + * - CMDLINE_FROM_BOOTLOADER=true, dt bootargs=non-empty:
> + *     dt bootargs
> + * - CMDLINE_FROM_BOOTLOADER=true, dt bootargs=empty, @data is non-empty string
> + *     @data is left unchanged
> + * - CMDLINE_FROM_BOOTLOADER=true, dt bootargs=empty, @data is empty string
> + *     CONFIG_CMDLINE (or "" if that's not defined)
> + */
>  extern int early_init_dt_scan_chosen(unsigned long node, const char *uname,
>  				     int depth, void *data);
>  extern void early_init_dt_check_for_initrd(unsigned long node);



More information about the devicetree-discuss mailing list