mirror of https://gitee.com/openkylin/linux.git
kconfig: fix new choices being skipped upon config update
Running `oldconfig' after any of the following configuration change: either trivial addition, such as: config A bool "A" choice prompt "Choice ?" depends on A config CHOICE_B bool "Choice B" config CHOICE_C bool "Choice C" endchoice or more tricky change: OLD KCONFIG | NEW KCONFIG | | config A | bool "A" | choice | choice prompt "Choice ?" | prompt "Choice ?" | config CHOICE_C | config CHOICE_C bool "Choice C" | bool "Choice C" | config CHOICE_D | config CHOICE_D bool "Choice D" | bool "Choice D" endchoice | | config CHOICE_E | bool "Choice E" | depends on A | endchoice will not cause the choice to be considered as NEW, and thus not be asked. The cause of this behavior is that choice's novelty are computed statically right after the saved configuration has been read. At this point, the new dependency's value is still unknown and asserted to be `no'. Moreover, no update to this decision is made afterward. Correct this by dynamically evaluating a choice's novelty, and removing the static evaluation. Reported-and-tested-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de> Signed-off-by: Arnaud Lacombe <lacombar@gmail.com> Signed-off-by: Michal Marek <mmarek@suse.cz>
This commit is contained in:
parent
dcd6c92267
commit
5d09598d48
|
@ -344,10 +344,8 @@ int conf_read_simple(const char *name, int def)
|
||||||
|
|
||||||
int conf_read(const char *name)
|
int conf_read(const char *name)
|
||||||
{
|
{
|
||||||
struct symbol *sym, *choice_sym;
|
struct symbol *sym;
|
||||||
struct property *prop;
|
int i;
|
||||||
struct expr *e;
|
|
||||||
int i, flags;
|
|
||||||
|
|
||||||
sym_set_change_count(0);
|
sym_set_change_count(0);
|
||||||
|
|
||||||
|
@ -357,7 +355,7 @@ int conf_read(const char *name)
|
||||||
for_all_symbols(i, sym) {
|
for_all_symbols(i, sym) {
|
||||||
sym_calc_value(sym);
|
sym_calc_value(sym);
|
||||||
if (sym_is_choice(sym) || (sym->flags & SYMBOL_AUTO))
|
if (sym_is_choice(sym) || (sym->flags & SYMBOL_AUTO))
|
||||||
goto sym_ok;
|
continue;
|
||||||
if (sym_has_value(sym) && (sym->flags & SYMBOL_WRITE)) {
|
if (sym_has_value(sym) && (sym->flags & SYMBOL_WRITE)) {
|
||||||
/* check that calculated value agrees with saved value */
|
/* check that calculated value agrees with saved value */
|
||||||
switch (sym->type) {
|
switch (sym->type) {
|
||||||
|
@ -366,30 +364,18 @@ int conf_read(const char *name)
|
||||||
if (sym->def[S_DEF_USER].tri != sym_get_tristate_value(sym))
|
if (sym->def[S_DEF_USER].tri != sym_get_tristate_value(sym))
|
||||||
break;
|
break;
|
||||||
if (!sym_is_choice(sym))
|
if (!sym_is_choice(sym))
|
||||||
goto sym_ok;
|
continue;
|
||||||
/* fall through */
|
/* fall through */
|
||||||
default:
|
default:
|
||||||
if (!strcmp(sym->curr.val, sym->def[S_DEF_USER].val))
|
if (!strcmp(sym->curr.val, sym->def[S_DEF_USER].val))
|
||||||
goto sym_ok;
|
continue;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else if (!sym_has_value(sym) && !(sym->flags & SYMBOL_WRITE))
|
} else if (!sym_has_value(sym) && !(sym->flags & SYMBOL_WRITE))
|
||||||
/* no previous value and not saved */
|
/* no previous value and not saved */
|
||||||
goto sym_ok;
|
continue;
|
||||||
conf_unsaved++;
|
conf_unsaved++;
|
||||||
/* maybe print value in verbose mode... */
|
/* maybe print value in verbose mode... */
|
||||||
sym_ok:
|
|
||||||
if (!sym_is_choice(sym))
|
|
||||||
continue;
|
|
||||||
/* The choice symbol only has a set value (and thus is not new)
|
|
||||||
* if all its visible childs have values.
|
|
||||||
*/
|
|
||||||
prop = sym_get_choice_prop(sym);
|
|
||||||
flags = sym->flags;
|
|
||||||
expr_list_for_each_sym(prop->expr, e, choice_sym)
|
|
||||||
if (choice_sym->visible != no)
|
|
||||||
flags &= choice_sym->flags;
|
|
||||||
sym->flags &= flags | ~SYMBOL_DEF_USER;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for_all_symbols(i, sym) {
|
for_all_symbols(i, sym) {
|
||||||
|
|
|
@ -262,11 +262,18 @@ static struct symbol *sym_calc_choice(struct symbol *sym)
|
||||||
struct symbol *def_sym;
|
struct symbol *def_sym;
|
||||||
struct property *prop;
|
struct property *prop;
|
||||||
struct expr *e;
|
struct expr *e;
|
||||||
|
int flags;
|
||||||
|
|
||||||
/* first calculate all choice values' visibilities */
|
/* first calculate all choice values' visibilities */
|
||||||
|
flags = sym->flags;
|
||||||
prop = sym_get_choice_prop(sym);
|
prop = sym_get_choice_prop(sym);
|
||||||
expr_list_for_each_sym(prop->expr, e, def_sym)
|
expr_list_for_each_sym(prop->expr, e, def_sym) {
|
||||||
sym_calc_visibility(def_sym);
|
sym_calc_visibility(def_sym);
|
||||||
|
if (def_sym->visible != no)
|
||||||
|
flags &= def_sym->flags;
|
||||||
|
}
|
||||||
|
|
||||||
|
sym->flags &= flags | ~SYMBOL_DEF_USER;
|
||||||
|
|
||||||
/* is the user choice visible? */
|
/* is the user choice visible? */
|
||||||
def_sym = sym->def[S_DEF_USER].val;
|
def_sym = sym->def[S_DEF_USER].val;
|
||||||
|
|
Loading…
Reference in New Issue