Re: [PATCH] kconfig: warn on dead default
From: Sam Ravnborg
Date: Sat Jun 06 2026 - 10:24:52 EST
Hi Julian.
On Sat, Jun 06, 2026 at 03:00:08PM +0100, Julian Braha wrote:
> The dead default check was originally introduced with kconfirm:
> https://lore.kernel.org/all/6ec4df6d-1445-48ca-8f54-1d1a83c4716d@xxxxxxxxx/
>
> While I'm still working on that tool, it's not yet ready for inclusion
> into the tree. I am currently waiting for common distro packagers to
> package the parsing library before submitting the next RFC iteration.
>
> However, the dead default check is more impactful than the other checks:
> all 4 dead defaults that were detected should not have been dead and could
> cause misconfiguration bugs. But fortunately, these were just for kunit
> tests. The 3 patches to fix them have all since been merged:
> commit aef656a0e6c0 ("powerpc: fix dead default for GUEST_STATE_BUFFER_TEST")
> commit 30cc5e2ad826 ("s390/Kconfig: Cleanup defaults for selftests")
> commit df75430515c3 ("drm: fix dead default for DRM_TTM_KUNIT_TEST")
>
> We can actually check for dead defaults while evaluating Kconfig, which
> should be even more effective at preventing future instances than keeping
> it in a static checker.
>
> Note that this patch will only trigger a warning when the default values
> are different, in other words, pure duplicate defaults won't cause a
> warning, as they are simply redundant.
A drive-by comment below.
>
> Signed-off-by: Julian Braha <julianbraha@xxxxxxxxx>
> ---
> scripts/kconfig/menu.c | 22 +++++++++-
> .../kconfig/tests/warn_dead_default/Kconfig | 40 +++++++++++++++++++
> .../tests/warn_dead_default/__init__.py | 8 ++++
> .../tests/warn_dead_default/expected_stderr | 4 ++
> 4 files changed, 73 insertions(+), 1 deletion(-)
> create mode 100644 scripts/kconfig/tests/warn_dead_default/Kconfig
> create mode 100644 scripts/kconfig/tests/warn_dead_default/__init__.py
> create mode 100644 scripts/kconfig/tests/warn_dead_default/expected_stderr
>
> diff --git a/scripts/kconfig/menu.c b/scripts/kconfig/menu.c
> index b2d8d4e11e07..8c280292f9cd 100644
> --- a/scripts/kconfig/menu.c
> +++ b/scripts/kconfig/menu.c
> @@ -242,13 +242,33 @@ static int menu_validate_number(struct symbol *sym, struct symbol *sym2)
>
> static void sym_check_prop(struct symbol *sym)
> {
> - struct property *prop;
> + struct property *prev, *prop;
> struct symbol *sym2;
> char *use;
>
> for (prop = sym->prop; prop; prop = prop->next) {
This part iterates through all properties of sym.
> switch (prop->type) {
> case P_DEFAULT:
> + for_all_defaults(sym, prev) {
And this part iterates through all properties of sym and will match only
P_DEFAULT.
I looks redundant with the two loops were we for the latter knows it is
a P_DEFAULT.
> + if (prev == prop)
> + break;
> + if (expr_is_yes(prev->visible.expr)) {
> + if (!expr_eq(prev->expr, prop->expr))
> + prop_warn(prop,
> + "default for '%s' is unreachable: earlier default at %s:%d is unconditional",
> + sym->name ? sym->name : "<choice>",
> + prev->filename, prev->lineno);
> + break;
> + }
> + if (expr_eq(prev->visible.expr, prop->visible.expr)) {
> + if (!expr_eq(prev->expr, prop->expr))
> + prop_warn(prop,
> + "default for '%s' has the same condition as the earlier default at %s:%d",
> + sym->name ? sym->name : "<choice>",
> + prev->filename, prev->lineno);
> + break;
> + }
> + }
> if ((sym->type == S_STRING || sym->type == S_INT || sym->type == S_HEX) &&
> prop->expr->type != E_SYMBOL)
> prop_warn(prop,
Sam