#!/bin/sh # SPDX-License-Identifier: GPL-2.0 # # merge_config.sh - Takes a list of config fragment values, and merges # them one by one. Provides warnings on overridden values, and specified # values that did not make it to the resulting .config file (due to missed # dependencies or config symbol removal). # # Portions reused from kconf_check and generate_cfg: # http://git.yoctoproject.org/cgit/cgit.cgi/yocto-kernel-tools/tree/tools/kconf_check # http://git.yoctoproject.org/cgit/cgit.cgi/yocto-kernel-tools/tree/tools/generate_cfg # # Copyright (c) 2009-2010 Wind River Systems, Inc. # Copyright 2011 Linaro set -e clean_up() { rm -f "$TMP_FILE" rm -f "$TMP_FILE.new" } usage() { echo "Usage: $0 [OPTIONS] [CONFIG [...]]" echo " -h display this help text" echo " -m only merge the fragments, do not execute the make command" echo " -n use allnoconfig instead of alldefconfig" echo " -r list redundant entries when merging fragments" echo " -y make builtin have precedence over modules" echo " -O dir to put generated output files. Consider setting \$KCONFIG_CONFIG instead." echo " -s strict mode. Fail if the fragment redefines any value." echo " -Q disable warning messages for overridden options." echo echo "Used prefix: '$CONFIG_PREFIX'. You can redefine it with \$CONFIG_ environment variable." } RUNMAKE=true ALLTARGET=alldefconfig WARNREDUN=false BUILTIN=false OUTPUT=. STRICT=false CONFIG_PREFIX=${CONFIG_-CONFIG_} WARNOVERRIDE=echo if [ -z "$AWK" ]; then AWK=awk fi while true; do case $1 in "-n") ALLTARGET=allnoconfig shift continue ;; "-m") RUNMAKE=false shift continue ;; "-h") usage exit ;; "-r") WARNREDUN=true shift continue ;; "-y") BUILTIN=true shift continue ;; "-O") if [ -d $2 ];then OUTPUT=$(echo $2 | sed 's/\/*$//') else echo "output directory $2 does not exist" 1>&2 exit 1 fi shift 2 continue ;; "-s") STRICT=true shift continue ;; "-Q") WARNOVERRIDE=true shift continue ;; *) break ;; esac done if [ "$#" -lt 1 ] ; then usage exit fi if [ -z "$KCONFIG_CONFIG" ]; then if [ "$OUTPUT" != . ]; then KCONFIG_CONFIG=$(readlink -m -- "$OUTPUT/.config") else KCONFIG_CONFIG=.config fi fi INITFILE=$1 shift; if [ ! -r "$INITFILE" ]; then echo "The base file '$INITFILE' does not exist. Creating one..." >&2 touch "$INITFILE" fi MERGE_LIST=$* TMP_FILE=$(mktemp ./.tmp.config.XXXXXXXXXX) echo "Using $INITFILE as base" trap clean_up EXIT cat $INITFILE > $TMP_FILE PROCESSED_FILES="" # Merge files, printing warnings on overridden values for ORIG_MERGE_FILE in $MERGE_LIST ; do echo "Merging $ORIG_MERGE_FILE" if [ ! -r "$ORIG_MERGE_FILE" ]; then echo "The merge file '$ORIG_MERGE_FILE' does not exist. Exit." >&2 exit 1 fi # Check for duplicate input files case " $PROCESSED_FILES " in *" $ORIG_MERGE_FILE "*) ${WARNOVERRIDE} "WARNING: Input file provided multiple times: $ORIG_MERGE_FILE" ;; esac # Use awk for single-pass processing instead of per-symbol grep/sed if ! "$AWK" -v prefix="$CONFIG_PREFIX" \ -v warnoverride="$WARNOVERRIDE" \ -v strict="$STRICT" \ -v outfile="$TMP_FILE.new" \ -v builtin="$BUILTIN" \ -v warnredun="$WARNREDUN" ' BEGIN { strict_violated = 0 cfg_regex = "^" prefix "[a-zA-Z0-9_]+" notset_regex = "^# " prefix "[a-zA-Z0-9_]+ is not set$" } # Extract config name from a line, returns "" if not a config line function get_cfg(line) { if (match(line, cfg_regex)) { return substr(line, RSTART, RLENGTH) } else if (match(line, notset_regex)) { # Extract CONFIG_FOO from "# CONFIG_FOO is not set" sub(/^# /, "", line) sub(/ is not set$/, "", line) return line } return "" } function warn_builtin(cfg, prev, new) { if (warnoverride == "true") return print cfg ": -y passed, will not demote y to m" print "Previous value: " prev print "New value: " new print "" } function warn_redefined(cfg, prev, new) { if (warnoverride == "true") return print "Value of " cfg " is redefined by fragment " mergefile ":" print "Previous value: " prev print "New value: " new print "" } function warn_redundant(cfg) { if (warnredun != "true" || warnoverride == "true") return print "Value of " cfg " is redundant by fragment " mergefile ":" } # First pass: read merge file, store all lines and index FILENAME == ARGV[1] { mergefile = FILENAME merge_lines[FNR] = $0 merge_total = FNR cfg = get_cfg($0) if (cfg != "") { merge_cfg[cfg] = $0 merge_cfg_line[cfg] = FNR } next } # Second pass: process base file (TMP_FILE) FILENAME == ARGV[2] { cfg = get_cfg($0) # Not a config or not in merge file - keep it if (cfg == "" || !(cfg in merge_cfg)) { print $0 >> outfile next } prev_val = $0 new_val = merge_cfg[cfg] # BUILTIN: do not demote y to m if (builtin == "true" && new_val ~ /=m$/ && prev_val ~ /=y$/) { warn_builtin(cfg, prev_val, new_val) print $0 >> outfile skip_merge[merge_cfg_line[cfg]] = 1 next } # Values equal - redundant if (prev_val == new_val) { warn_redundant(cfg) next } # "=n" is the same as "is not set" if (prev_val ~ /=n$/ && new_val ~ / is not set$/) { print $0 >> outfile next } # Values differ - redefined warn_redefined(cfg, prev_val, new_val) if (strict == "true") { strict_violated = 1 } } END { # Newline in case base file lacks trailing newline print "" >> outfile # Append merge file, skipping lines marked for builtin preservation for (i = 1; i <= merge_total; i++) { if (!(i in skip_merge)) { print merge_lines[i] >> outfile } } if (strict_violated) { exit 1 } }' \ "$ORIG_MERGE_FILE" "$TMP_FILE"; then # awk exited non-zero, strict mode was violated STRICT_MODE_VIOLATED=true fi mv "$TMP_FILE.new" "$TMP_FILE" PROCESSED_FILES="$PROCESSED_FILES $ORIG_MERGE_FILE" done if [ "$STRICT_MODE_VIOLATED" = "true" ]; then echo "The fragment redefined a value and strict mode had been passed." exit 1 fi if [ "$RUNMAKE" = "false" ]; then cp -T -- "$TMP_FILE" "$KCONFIG_CONFIG" echo "#" echo "# merged configuration written to $KCONFIG_CONFIG (needs make)" echo "#" exit fi # If we have an output dir, setup the O= argument, otherwise leave # it blank, since O=. will create an unnecessary ./source softlink OUTPUT_ARG="" if [ "$OUTPUT" != "." ] ; then OUTPUT_ARG="O=$OUTPUT" fi # Use the merged file as the starting point for: # alldefconfig: Fills in any missing symbols with Kconfig default # allnoconfig: Fills in any missing symbols with # CONFIG_* is not set make KCONFIG_ALLCONFIG=$TMP_FILE $OUTPUT_ARG $ALLTARGET # Check all specified config values took effect (might have missed-dependency issues) if ! "$AWK" -v prefix="$CONFIG_PREFIX" \ -v warnoverride="$WARNOVERRIDE" \ -v strict="$STRICT" \ -v warnredun="$WARNREDUN" ' BEGIN { strict_violated = 0 cfg_regex = "^" prefix "[a-zA-Z0-9_]+" notset_regex = "^# " prefix "[a-zA-Z0-9_]+ is not set$" } # Extract config name from a line, returns "" if not a config line function get_cfg(line) { if (match(line, cfg_regex)) { return substr(line, RSTART, RLENGTH) } else if (match(line, notset_regex)) { # Extract CONFIG_FOO from "# CONFIG_FOO is not set" sub(/^# /, "", line) sub(/ is not set$/, "", line) return line } return "" } function warn_mismatch(cfg, merged, final) { if (warnredun == "true") return if (final == "" && !(merged ~ / is not set$/ || merged ~ /=n$/)) { print "WARNING: Value requested for " cfg " not in final .config" print "Requested value: " merged print "Actual value: " final } else if (final == "" && merged ~ / is not set$/) { # not set, pass } else if (merged == "" && final != "") { print "WARNING: " cfg " not in merged config but added in final .config:" print "Requested value: " merged print "Actual value: " final } else { print "WARNING: " cfg " differs:" print "Requested value: " merged print "Actual value: " final } } # First pass: read effective config file, store all lines FILENAME == ARGV[1] { cfg = get_cfg($0) if (cfg != "") { config_cfg[cfg] = $0 } next } # Second pass: process merged config and compare against effective config { cfg = get_cfg($0) if (cfg == "") next # strip trailing comment sub(/[[:space:]]+#.*/, "", $0) merged_val = $0 final_val = config_cfg[cfg] if (merged_val == final_val) next if (merged_val ~ /=n$/ && final_val ~ / is not set$/) next if (merged_val ~ /=n$/ && final_val == "") next warn_mismatch(cfg, merged_val, final_val) if (strict == "true") { strict_violated = 1 } } END { if (strict_violated) { exit 1 } }' \ "$KCONFIG_CONFIG" "$TMP_FILE"; then # awk exited non-zero, strict mode was violated STRICT_MODE_VIOLATED=true fi if [ "$STRICT" = "true" ] && [ "$STRICT_MODE_VIOLATED" = "true" ]; then echo "Requested and effective config differ" exit 1 fi