Re: [PATCH 4/4] kconfig: add savedefconfig

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On 25.7.2010 23:40, Sam Ravnborg wrote:
> From 5edffcc6890a7dbd43b8da9c453bee794e81e7c7 Mon Sep 17 00:00:00 2001
> From: Sam Ravnborg <sam@xxxxxxxxxxxx>
> Date: Sun, 25 Jul 2010 23:18:47 +0200
> Subject: [PATCH 4/4] kconfig: add savedefconfig
> MIME-Version: 1.0
> Content-Type: text/plain; charset=utf-8
> Content-Transfer-Encoding: 8bit
> 
> savedefconfig is used to save a minimal configuration
> that is useful for a defconfig.

General comment: I like this more than the "replace defconfigs with
Kconfig files" approach -- the .config format is efficient and easy to
read, we have tools to edit it (make *config), the issue was the size of
the defconfigs and the updates to them, which your patches address now.



> Signed-off-by: Sam Ravnborg <sam@xxxxxxxxxxxx>
> Cc: Uwe Kleine-König <u.kleine-koenig@xxxxxxxxxxxxxx>
> Cc: Stephen Rothwell <sfr@xxxxxxxxxxxxxxxx>
> ---
>  scripts/kconfig/Makefile    |    7 +++-
>  scripts/kconfig/conf.c      |   20 ++++++++-
>  scripts/kconfig/confdata.c  |   94 ++++++++++++++++++++++++++++++++++++++++++
>  scripts/kconfig/lkc.h       |    2 +
>  scripts/kconfig/lkc_proto.h |    1 +
>  scripts/kconfig/symbol.c    |   96 +++++++++++++++++++++++++++++++++++++++++++
>  6 files changed, 217 insertions(+), 3 deletions(-)
> 
> diff --git a/scripts/kconfig/Makefile b/scripts/kconfig/Makefile
> index 366711a..aeb04c2 100644
> --- a/scripts/kconfig/Makefile
> +++ b/scripts/kconfig/Makefile
> @@ -95,7 +95,8 @@ update-po-config: $(obj)/kxgettext $(obj)/gconf.glade.h
>  	$(Q)rm -f arch/um/Kconfig.arch
>  	$(Q)rm -f $(obj)/config.pot
>  
> -PHONY += randconfig allyesconfig allnoconfig allmodconfig alldefconfig defconfig
> +PHONY += randconfig allyesconfig allnoconfig allmodconfig alldefconfig
> +PHONY += defconfig savedefconfig
>  
>  randconfig: $(obj)/conf
>  	$< -r $(Kconfig)
> @@ -112,6 +113,9 @@ allmodconfig: $(obj)/conf
>  alldefconfig: $(obj)/conf
>  	$< -f $(Kconfig)
>  
> +savedefconfig: $(obj)/conf
> +	$< -M defconfig $(Kconfig)
> +

Out of curiosity, do you have any mnemonics for -f and -M, or are these
just random letters that were free? :) Maybe we should add long options
like --defconfig, --allmodconfig for each of the targets and use them in
the Makefile. The meaning of the single-letter options is not obvious
sometimes. scripts/genksyms already uses getopt_long() so we would not
regress in portability.


>  defconfig: $(obj)/conf
>  ifeq ($(KBUILD_DEFCONFIG),)
>  	$< -d $(Kconfig)
> @@ -140,6 +144,7 @@ help:
>  	@echo  '  allmodconfig	  - New config selecting modules when possible'
>  	@echo  '  allyesconfig	  - New config where all options are accepted with yes'
>  	@echo  '  allnoconfig	  - New config where all options are answered with no'
> +	@echo  '  savedefconfig   - Save current config as ./defconfig (minimal config)'
>  
>  # lxdialog stuff
>  check-lxdialog  := $(srctree)/$(src)/lxdialog/check-lxdialog.sh
> diff --git a/scripts/kconfig/conf.c b/scripts/kconfig/conf.c
> index 2b4775e..b1a903b 100644
> --- a/scripts/kconfig/conf.c
> +++ b/scripts/kconfig/conf.c
> @@ -28,7 +28,8 @@ enum {
>  	set_mod,
>  	set_no,
>  	set_default,
> -	set_random
> +	set_random,
> +	save_defconfig,
>  } input_mode = ask_all;
>  char *defconfig_file;
>  
> @@ -440,7 +441,7 @@ int main(int ac, char **av)
>  	bindtextdomain(PACKAGE, LOCALEDIR);
>  	textdomain(PACKAGE);
>  
> -	while ((opt = getopt(ac, av, "osdD:nmyfrh")) != -1) {
> +	while ((opt = getopt(ac, av, "osdD:nmM:yfrh")) != -1) {
>  		switch (opt) {
>  		case 'o':
>  			input_mode = ask_silent;
> @@ -485,6 +486,10 @@ int main(int ac, char **av)
>  			input_mode = set_random;
>  			break;
>  		}
> +		case 'M':
> +			input_mode = save_defconfig;
> +			defconfig_file = optarg;
> +			break;
>  		case 'h':
>  			printf(_("See README for usage info\n"));
>  			exit(0);
> @@ -553,6 +558,9 @@ int main(int ac, char **av)
>  		else if (!stat("all.config", &tmpstat))
>  			conf_read_simple("all.config", S_DEF_USER);
>  		break;
> +	case save_defconfig:
> +		conf_read(NULL);
> +		break;
>  	default:
>  		break;
>  	}
> @@ -601,6 +609,8 @@ int main(int ac, char **av)
>  			check_conf(&rootmenu);
>  		} while (conf_cnt);
>  		break;
> +	case save_defconfig:
> +		break;
>  	}
>  
>  	if (sync_kconfig) {
> @@ -615,6 +625,12 @@ int main(int ac, char **av)
>  			fprintf(stderr, _("\n*** Error during update of the kernel configuration.\n\n"));
>  			return 1;
>  		}
> +	} else if (input_mode == save_defconfig) {
> +		if (conf_write_defconfig(defconfig_file)) {
> +			fprintf(stderr, _("\n*** Error during writing of mini config to %s.\n\n"),
> +			        defconfig_file);
> +			return 1;
> +		}
>  	} else {
>  		if (conf_write(NULL)) {
>  			fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
> diff --git a/scripts/kconfig/confdata.c b/scripts/kconfig/confdata.c
> index c4dec80..76c4e23 100644
> --- a/scripts/kconfig/confdata.c
> +++ b/scripts/kconfig/confdata.c
> @@ -396,6 +396,100 @@ int conf_read(const char *name)
>  	return 0;
>  }
>  
> +/*
> + * Write out a minimal config.
> + * All values that has default values are skipped as this is redundant.
> + */
> +int conf_write_defconfig(const char *filename)
> +{
> +	struct symbol *sym;
> +	const char *str;
> +	FILE *out;
> +	int i, l;
> +
> +	out = fopen(filename, "w");
> +	if (!out)
> +		return 1;
> +
> +	sym_clear_all_valid();
> +
> +	for_all_symbols(i, sym) {
> +		sym_calc_value(sym);
> +		if (!(sym->flags & SYMBOL_WRITE) || !sym->name)
> +			goto next_symbol;
> +		/* If we cannot change the symbol - skip */
> +		if (!sym_is_changable(sym))
> +			goto next_symbol;
> +		/* If symbol equals to default value - skip */
> +		if (strcmp(sym_get_string_value(sym), sym_get_string_default(sym)) == 0)
> +			goto next_symbol;
> +
> +		/* choice symbols does not have a value - skip */
> +		if (sym_is_choice(sym))
> +			goto next_symbol;
> +		/*
> +		 * If symbol is a choice value and equals to the
> +		 * default for a choice - skip.
> +		 * But only if value equal to "y".
> +		 */
> +		if (sym_is_choice_value(sym)) {
> +			struct symbol *cs;
> +			struct symbol *ds;
> +
> +			cs = prop_get_symbol(sym_get_choice_prop(sym));
> +			ds = sym_choice_default(cs);
> +			if (sym == ds) {
> +				if ((sym->type == S_BOOLEAN ||
> +				     sym->type == S_TRISTATE) &&
> +				     sym_get_tristate_value(sym) == yes)
> +					goto next_symbol;
> +			}
> +		}
> +		switch (sym->type) {

This duplicates the switch statement in conf_write(), we should move it
into a separate function and call it from conf_write() and here.


> +		case S_BOOLEAN:
> +		case S_TRISTATE:
> +			switch (sym_get_tristate_value(sym)) {
> +			case no:
> +				fprintf(out, "# CONFIG_%s is not set\n", sym->name);
> +				break;
> +			case yes:
> +				fprintf(out, "CONFIG_%s=y\n", sym->name);
> +				break;
> +			case mod:
> +				fprintf(out, "CONFIG_%s=m\n", sym->name);
> +				break;
> +			}
> +			break;
> +		case S_STRING:
> +			str = sym_get_string_value(sym);
> +			fprintf(out, "CONFIG_%s=\"", sym->name);
> +			while (1) {
> +				l = strcspn(str, "\"\\");
> +				if (l) {
> +					fwrite(str, l, 1, out);
> +					str += l;
> +				}
> +				if (!*str)
> +					break;
> +				fprintf(out, "\\%c", *str++);
> +			}
> +			fputs("\"\n", out);
> +			break;
> +		case S_HEX:
> +		case S_INT:
> +			str = sym_get_string_value(sym);
> +			fprintf(out, "CONFIG_%s=%s\n", sym->name, str);
> +			break;
> +		default:
> +			break;
> +		}
> +next_symbol:
> +		;
> +	}

The gotos should be replaced by simple continue statements if there is
nothing to do after the label.

(maybe there are more issues, I didn't look closely into it. But I like
the approach.)

Michal

> +	fclose(out);
> +	return 0;
> +}
> +
>  int conf_write(const char *name)
>  {
>  	FILE *out;
> diff --git a/scripts/kconfig/lkc.h b/scripts/kconfig/lkc.h
> index ce6549c..76db065 100644
> --- a/scripts/kconfig/lkc.h
> +++ b/scripts/kconfig/lkc.h
> @@ -126,6 +126,8 @@ void sym_init(void);
>  void sym_clear_all_valid(void);
>  void sym_set_all_changed(void);
>  void sym_set_changed(struct symbol *sym);
> +struct symbol *sym_choice_default(struct symbol *sym);
> +const char *sym_get_string_default(struct symbol *sym);
>  struct symbol *sym_check_deps(struct symbol *sym);
>  struct property *prop_alloc(enum prop_type type, struct symbol *sym);
>  struct symbol *prop_get_symbol(struct property *prop);
> diff --git a/scripts/kconfig/lkc_proto.h b/scripts/kconfig/lkc_proto.h
> index 7cadcad..e30d5f2 100644
> --- a/scripts/kconfig/lkc_proto.h
> +++ b/scripts/kconfig/lkc_proto.h
> @@ -4,6 +4,7 @@ P(conf_parse,void,(const char *name));
>  P(conf_read,int,(const char *name));
>  P(conf_read_simple,int,(const char *name, int));
>  P(conf_write,int,(const char *name));
> +P(conf_write_defconfig,int,(const char *name));
>  P(conf_write_autoconf,int,(void));
>  P(conf_get_changed,bool,(void));
>  P(conf_set_changed_callback, void,(void (*fn)(void)));
> diff --git a/scripts/kconfig/symbol.c b/scripts/kconfig/symbol.c
> index 0ea9c46..0bda646 100644
> --- a/scripts/kconfig/symbol.c
> +++ b/scripts/kconfig/symbol.c
> @@ -216,6 +216,41 @@ static void sym_calc_visibility(struct symbol *sym)
>  	}
>  }
>  
> +/*
> + * Find the default isymbol for a choice.
> + * First try the default values for the choice symbol
> + * Next locate the first visible choice value
> + * Return NULL if none was found
> + */
> +struct symbol *sym_choice_default(struct symbol *sym)
> +{
> +	struct property *prop;
> +	struct symbol *def_sym;
> +	struct expr *e;
> +
> +	/* any of the defaults visible? */
> +	for_all_defaults(sym, prop) {
> +		prop->visible.tri = expr_calc_value(prop->visible.expr);
> +		if (prop->visible.tri == no)
> +			continue;
> +		def_sym = prop_get_symbol(prop);
> +		sym_calc_visibility(def_sym);
> +		if (def_sym->visible != no)
> +			return def_sym;
> +	}
> +
> +	/* just get the first visible value */
> +	prop = sym_get_choice_prop(sym);
> +	expr_list_for_each_sym(prop->expr, e, def_sym) {
> +		sym_calc_visibility(def_sym);
> +		if (def_sym->visible != no)
> +			return def_sym;
> +	}
> +
> +	/* failed to locate any defaults */
> +	return NULL;
> +}
> +
>  static struct symbol *sym_calc_choice(struct symbol *sym)
>  {
>  	struct symbol *def_sym;
> @@ -646,6 +681,67 @@ const char *sym_get_string_value(struct symbol *sym)
>  	return (const char *)sym->curr.val;
>  }
>  
> +/*
> + * Find the default value associated to a symbol.
> + * For tristate symbol hande the modules=n case
> + * in which case "m" bocomes "y".
> + * If the symbol does not have any default fallback
> + * to the fixed default values.
> + */
> +const char *sym_get_string_default(struct symbol *sym)
> +{
> +	struct property *prop;
> +	struct symbol *ds;
> +	tristate val;
> +
> +	prop = sym_get_default_prop(sym);
> +	if (!prop)
> +		goto default_value;
> +	ds = prop_get_symbol(prop);
> +	if (!ds)
> +		goto default_value;
> +
> +	switch (sym->type) {
> +	case S_BOOLEAN:
> +	case S_TRISTATE:
> +		val = sym_get_tristate_value(ds);
> +		switch (val) {
> +		case no:
> +			return "n";
> +		case yes:
> +			return "y";
> +		case mod:
> +			if (sym_is_choice_value(sym))
> +				return "m";
> +			sym_calc_value(modules_sym);
> +			if (modules_sym->curr.tri == no)
> +				return "y";
> +			else
> +				return "m";
> +		}
> +		break;
> +	default:
> +		return (const char *)ds->curr.val;
> +	}
> +
> +default_value:
> +
> +	switch (sym->type) {
> +	case S_BOOLEAN:
> +	case S_TRISTATE:
> +		return "n";
> +	case S_INT:
> +	case S_HEX:
> +		return "0";
> +	case S_STRING:
> +		return "";
> +	case S_OTHER:
> +	case S_UNKNOWN:
> +		break;
> +	}
> +	return "";
> +}
> +
>  bool sym_is_changable(struct symbol *sym)
>  {
>  	return sym->visible > sym->rev_dep.tri;

--
To unsubscribe from this list: send the line "unsubscribe linux-kbuild" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Linux&nblp;USB Development]     [Linux Media]     [Video for Linux]     [Linux Audio Users]     [Yosemite Secrets]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux