1*4882a593Smuzhiyun#!/bin/sh 2*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0 3*4882a593Smuzhiyun# 4*4882a593Smuzhiyun# merge_config.sh - Takes a list of config fragment values, and merges 5*4882a593Smuzhiyun# them one by one. Provides warnings on overridden values, and specified 6*4882a593Smuzhiyun# values that did not make it to the resulting .config file (due to missed 7*4882a593Smuzhiyun# dependencies or config symbol removal). 8*4882a593Smuzhiyun# 9*4882a593Smuzhiyun# Portions reused from kconf_check and generate_cfg: 10*4882a593Smuzhiyun# http://git.yoctoproject.org/cgit/cgit.cgi/yocto-kernel-tools/tree/tools/kconf_check 11*4882a593Smuzhiyun# http://git.yoctoproject.org/cgit/cgit.cgi/yocto-kernel-tools/tree/tools/generate_cfg 12*4882a593Smuzhiyun# 13*4882a593Smuzhiyun# Copyright (c) 2009-2010 Wind River Systems, Inc. 14*4882a593Smuzhiyun# Copyright 2011 Linaro 15*4882a593Smuzhiyun 16*4882a593Smuzhiyunset -e 17*4882a593Smuzhiyun 18*4882a593Smuzhiyunclean_up() { 19*4882a593Smuzhiyun rm -f $TMP_FILE 20*4882a593Smuzhiyun rm -f $MERGE_FILE 21*4882a593Smuzhiyun} 22*4882a593Smuzhiyun 23*4882a593Smuzhiyunusage() { 24*4882a593Smuzhiyun echo "Usage: $0 [OPTIONS] [CONFIG [...]]" 25*4882a593Smuzhiyun echo " -h display this help text" 26*4882a593Smuzhiyun echo " -m only merge the fragments, do not execute the make command" 27*4882a593Smuzhiyun echo " -n use allnoconfig instead of alldefconfig" 28*4882a593Smuzhiyun echo " -r list redundant entries when merging fragments" 29*4882a593Smuzhiyun echo " -y make builtin have precedence over modules" 30*4882a593Smuzhiyun echo " -O dir to put generated output files. Consider setting \$KCONFIG_CONFIG instead." 31*4882a593Smuzhiyun echo " -s strict mode. Fail if the fragment redefines any value." 32*4882a593Smuzhiyun echo 33*4882a593Smuzhiyun echo "Used prefix: '$CONFIG_PREFIX'. You can redefine it with \$CONFIG_ environment variable." 34*4882a593Smuzhiyun} 35*4882a593Smuzhiyun 36*4882a593SmuzhiyunRUNMAKE=true 37*4882a593SmuzhiyunALLTARGET=alldefconfig 38*4882a593SmuzhiyunWARNREDUN=false 39*4882a593SmuzhiyunBUILTIN=false 40*4882a593SmuzhiyunOUTPUT=. 41*4882a593SmuzhiyunSTRICT=false 42*4882a593SmuzhiyunCONFIG_PREFIX=${CONFIG_-CONFIG_} 43*4882a593Smuzhiyun 44*4882a593Smuzhiyunwhile true; do 45*4882a593Smuzhiyun case $1 in 46*4882a593Smuzhiyun "-n") 47*4882a593Smuzhiyun ALLTARGET=allnoconfig 48*4882a593Smuzhiyun shift 49*4882a593Smuzhiyun continue 50*4882a593Smuzhiyun ;; 51*4882a593Smuzhiyun "-m") 52*4882a593Smuzhiyun RUNMAKE=false 53*4882a593Smuzhiyun shift 54*4882a593Smuzhiyun continue 55*4882a593Smuzhiyun ;; 56*4882a593Smuzhiyun "-h") 57*4882a593Smuzhiyun usage 58*4882a593Smuzhiyun exit 59*4882a593Smuzhiyun ;; 60*4882a593Smuzhiyun "-r") 61*4882a593Smuzhiyun WARNREDUN=true 62*4882a593Smuzhiyun shift 63*4882a593Smuzhiyun continue 64*4882a593Smuzhiyun ;; 65*4882a593Smuzhiyun "-y") 66*4882a593Smuzhiyun BUILTIN=true 67*4882a593Smuzhiyun shift 68*4882a593Smuzhiyun continue 69*4882a593Smuzhiyun ;; 70*4882a593Smuzhiyun "-O") 71*4882a593Smuzhiyun if [ -d $2 ];then 72*4882a593Smuzhiyun OUTPUT=$(echo $2 | sed 's/\/*$//') 73*4882a593Smuzhiyun else 74*4882a593Smuzhiyun echo "output directory $2 does not exist" 1>&2 75*4882a593Smuzhiyun exit 1 76*4882a593Smuzhiyun fi 77*4882a593Smuzhiyun shift 2 78*4882a593Smuzhiyun continue 79*4882a593Smuzhiyun ;; 80*4882a593Smuzhiyun "-s") 81*4882a593Smuzhiyun STRICT=true 82*4882a593Smuzhiyun shift 83*4882a593Smuzhiyun continue 84*4882a593Smuzhiyun ;; 85*4882a593Smuzhiyun *) 86*4882a593Smuzhiyun break 87*4882a593Smuzhiyun ;; 88*4882a593Smuzhiyun esac 89*4882a593Smuzhiyundone 90*4882a593Smuzhiyun 91*4882a593Smuzhiyunif [ "$#" -lt 1 ] ; then 92*4882a593Smuzhiyun usage 93*4882a593Smuzhiyun exit 94*4882a593Smuzhiyunfi 95*4882a593Smuzhiyun 96*4882a593Smuzhiyunif [ -z "$KCONFIG_CONFIG" ]; then 97*4882a593Smuzhiyun if [ "$OUTPUT" != . ]; then 98*4882a593Smuzhiyun KCONFIG_CONFIG=$(readlink -m -- "$OUTPUT/.config") 99*4882a593Smuzhiyun else 100*4882a593Smuzhiyun KCONFIG_CONFIG=.config 101*4882a593Smuzhiyun fi 102*4882a593Smuzhiyunfi 103*4882a593Smuzhiyun 104*4882a593SmuzhiyunINITFILE=$1 105*4882a593Smuzhiyunshift; 106*4882a593Smuzhiyun 107*4882a593Smuzhiyunif [ ! -r "$INITFILE" ]; then 108*4882a593Smuzhiyun echo "The base file '$INITFILE' does not exist. Exit." >&2 109*4882a593Smuzhiyun exit 1 110*4882a593Smuzhiyunfi 111*4882a593Smuzhiyun 112*4882a593SmuzhiyunMERGE_LIST=$* 113*4882a593SmuzhiyunSED_CONFIG_EXP1="s/^\(${CONFIG_PREFIX}[a-zA-Z0-9_]*\)=.*/\1/p" 114*4882a593SmuzhiyunSED_CONFIG_EXP2="s/^# \(${CONFIG_PREFIX}[a-zA-Z0-9_]*\) is not set$/\1/p" 115*4882a593Smuzhiyun 116*4882a593SmuzhiyunTMP_FILE=$(mktemp ./.tmp.config.XXXXXXXXXX) 117*4882a593SmuzhiyunMERGE_FILE=$(mktemp ./.merge_tmp.config.XXXXXXXXXX) 118*4882a593Smuzhiyun 119*4882a593Smuzhiyunecho "Using $INITFILE as base" 120*4882a593Smuzhiyun 121*4882a593Smuzhiyuntrap clean_up EXIT 122*4882a593Smuzhiyun 123*4882a593Smuzhiyuncat $INITFILE > $TMP_FILE 124*4882a593Smuzhiyun 125*4882a593Smuzhiyun# Merge files, printing warnings on overridden values 126*4882a593Smuzhiyunfor ORIG_MERGE_FILE in $MERGE_LIST ; do 127*4882a593Smuzhiyun echo "Merging $ORIG_MERGE_FILE" 128*4882a593Smuzhiyun if [ ! -r "$ORIG_MERGE_FILE" ]; then 129*4882a593Smuzhiyun echo "The merge file '$ORIG_MERGE_FILE' does not exist. Exit." >&2 130*4882a593Smuzhiyun exit 1 131*4882a593Smuzhiyun fi 132*4882a593Smuzhiyun cat $ORIG_MERGE_FILE > $MERGE_FILE 133*4882a593Smuzhiyun CFG_LIST=$(sed -n -e "$SED_CONFIG_EXP1" -e "$SED_CONFIG_EXP2" $MERGE_FILE) 134*4882a593Smuzhiyun 135*4882a593Smuzhiyun for CFG in $CFG_LIST ; do 136*4882a593Smuzhiyun grep -q -w $CFG $TMP_FILE || continue 137*4882a593Smuzhiyun PREV_VAL=$(grep -w $CFG $TMP_FILE) 138*4882a593Smuzhiyun NEW_VAL=$(grep -w $CFG $MERGE_FILE) 139*4882a593Smuzhiyun BUILTIN_FLAG=false 140*4882a593Smuzhiyun if [ "$BUILTIN" = "true" ] && [ "${NEW_VAL#CONFIG_*=}" = "m" ] && [ "${PREV_VAL#CONFIG_*=}" = "y" ]; then 141*4882a593Smuzhiyun echo Previous value: $PREV_VAL 142*4882a593Smuzhiyun echo New value: $NEW_VAL 143*4882a593Smuzhiyun echo -y passed, will not demote y to m 144*4882a593Smuzhiyun echo 145*4882a593Smuzhiyun BUILTIN_FLAG=true 146*4882a593Smuzhiyun elif [ "x$PREV_VAL" != "x$NEW_VAL" ] ; then 147*4882a593Smuzhiyun echo Value of $CFG is redefined by fragment $ORIG_MERGE_FILE: 148*4882a593Smuzhiyun echo Previous value: $PREV_VAL 149*4882a593Smuzhiyun echo New value: $NEW_VAL 150*4882a593Smuzhiyun echo 151*4882a593Smuzhiyun if [ "$STRICT" = "true" ]; then 152*4882a593Smuzhiyun STRICT_MODE_VIOLATED=true 153*4882a593Smuzhiyun fi 154*4882a593Smuzhiyun elif [ "$WARNREDUN" = "true" ]; then 155*4882a593Smuzhiyun echo Value of $CFG is redundant by fragment $ORIG_MERGE_FILE: 156*4882a593Smuzhiyun fi 157*4882a593Smuzhiyun if [ "$BUILTIN_FLAG" = "false" ]; then 158*4882a593Smuzhiyun sed -i "/$CFG[ =]/d" $TMP_FILE 159*4882a593Smuzhiyun else 160*4882a593Smuzhiyun sed -i "/$CFG[ =]/d" $MERGE_FILE 161*4882a593Smuzhiyun fi 162*4882a593Smuzhiyun done 163*4882a593Smuzhiyun cat $MERGE_FILE >> $TMP_FILE 164*4882a593Smuzhiyundone 165*4882a593Smuzhiyun 166*4882a593Smuzhiyunif [ "$STRICT_MODE_VIOLATED" = "true" ]; then 167*4882a593Smuzhiyun echo "The fragment redefined a value and strict mode had been passed." 168*4882a593Smuzhiyun exit 1 169*4882a593Smuzhiyunfi 170*4882a593Smuzhiyun 171*4882a593Smuzhiyunif [ "$RUNMAKE" = "false" ]; then 172*4882a593Smuzhiyun cp -T -- "$TMP_FILE" "$KCONFIG_CONFIG" 173*4882a593Smuzhiyun echo "#" 174*4882a593Smuzhiyun echo "# merged configuration written to $KCONFIG_CONFIG (needs make)" 175*4882a593Smuzhiyun echo "#" 176*4882a593Smuzhiyun exit 177*4882a593Smuzhiyunfi 178*4882a593Smuzhiyun 179*4882a593Smuzhiyun# If we have an output dir, setup the O= argument, otherwise leave 180*4882a593Smuzhiyun# it blank, since O=. will create an unnecessary ./source softlink 181*4882a593SmuzhiyunOUTPUT_ARG="" 182*4882a593Smuzhiyunif [ "$OUTPUT" != "." ] ; then 183*4882a593Smuzhiyun OUTPUT_ARG="O=$OUTPUT" 184*4882a593Smuzhiyunfi 185*4882a593Smuzhiyun 186*4882a593Smuzhiyun 187*4882a593Smuzhiyun# Use the merged file as the starting point for: 188*4882a593Smuzhiyun# alldefconfig: Fills in any missing symbols with Kconfig default 189*4882a593Smuzhiyun# allnoconfig: Fills in any missing symbols with # CONFIG_* is not set 190*4882a593Smuzhiyunmake KCONFIG_ALLCONFIG=$TMP_FILE $OUTPUT_ARG $ALLTARGET 191*4882a593Smuzhiyun 192*4882a593Smuzhiyun 193*4882a593Smuzhiyun# Check all specified config values took (might have missed-dependency issues) 194*4882a593Smuzhiyunfor CFG in $(sed -n -e "$SED_CONFIG_EXP1" -e "$SED_CONFIG_EXP2" $TMP_FILE); do 195*4882a593Smuzhiyun 196*4882a593Smuzhiyun REQUESTED_VAL=$(grep -w -e "$CFG" $TMP_FILE) 197*4882a593Smuzhiyun ACTUAL_VAL=$(grep -w -e "$CFG" "$KCONFIG_CONFIG" || true) 198*4882a593Smuzhiyun if [ "x$REQUESTED_VAL" != "x$ACTUAL_VAL" ] ; then 199*4882a593Smuzhiyun echo "Value requested for $CFG not in final .config" 200*4882a593Smuzhiyun echo "Requested value: $REQUESTED_VAL" 201*4882a593Smuzhiyun echo "Actual value: $ACTUAL_VAL" 202*4882a593Smuzhiyun echo "" 203*4882a593Smuzhiyun fi 204*4882a593Smuzhiyundone 205