| #!/usr/bin/env python |
| # |
| # Copyright (C) 2014, Masahiro Yamada <yamada.m@jp.panasonic.com> |
| # |
| # SPDX-License-Identifier: GPL-2.0+ |
| # |
| |
| """ |
| A wrapper script to adjust Kconfig for U-Boot |
| |
| The biggest difference between Linux Kernel and U-Boot in terms of the |
| board configuration is that U-Boot has to configure multiple boot images |
| per board: Normal, SPL, TPL. |
| We need to expand the functions of Kconfig to handle multiple boot |
| images. |
| |
| Instead of touching various parts under the scripts/kconfig/ directory, |
| pushing necessary adjustments into this single script would be better |
| for code maintainance. All the make targets related to the configuration |
| (make %config) should be invoked via this script. |
| |
| Let's see what is different from the original Kconfig. |
| |
| - config, menuconfig, etc. |
| |
| The commands 'make config', 'make menuconfig', etc. are used to create |
| or modify the .config file, which stores configs for Normal boot image. |
| |
| The location of the one for SPL, TPL image is spl/.config, tpl/.config, |
| respectively. Use 'make spl/config', 'make spl/menuconfig', etc. |
| to create or modify the spl/.config file, which contains configs |
| for SPL image. |
| Do likewise for the tpl/.config file. |
| The generic syntax for SPL, TPL configuration is |
| 'make <target_image>/<config_command>'. |
| |
| - silentoldconfig |
| |
| The command 'make silentoldconfig' updates .config, if necessary, and |
| additionally updates include/generated/autoconf.h and files under |
| include/configs/ directory. In U-Boot, it should do the same things for |
| SPL, TPL images for boards supporting them. |
| Depending on whether CONFIG_SPL, CONFIG_TPL is defined or not, |
| 'make silentoldconfig' iterates three times at most changing the target |
| directory. |
| |
| To sum up, 'make silentoldconfig' possibly updates |
| - .config, include/generated/autoconf.h, include/config/* |
| - spl/.config, spl/include/generated/autoconf.h, spl/include/config/* |
| (in case CONFIG_SPL=y) |
| - tpl/.config, tpl/include/generated/autoconf.h, tpl/include/config/* |
| (in case CONFIG_TPL=y) |
| |
| - defconfig, <board>_defconfig |
| |
| The command 'make <board>_defconfig' creates a new .config based on the |
| file configs/<board>_defconfig. The command 'make defconfig' is the same |
| but the difference is it uses the file specified with KBUILD_DEFCONFIG |
| environment. |
| |
| We need to create .config, spl/.config, tpl/.config for boards where SPL |
| and TPL images are supported. One possible solution for that is to have |
| multiple defconfig files per board, but it would produce duplication |
| among the defconfigs. |
| The approach chosen here is to expand the feature and support |
| conditional definition in defconfig, that is, each line in defconfig |
| files has the form of: |
| <condition>:<macro definition> |
| |
| The '<condition>:' prefix specifies which image the line is valid for. |
| The '<condition>:' is one of: |
| None - the line is valid only for Normal image |
| S: - the line is valid only for SPL image |
| T: - the line is valid only for TPL image |
| ST: - the line is valid for SPL and TPL images |
| +S: - the line is valid for Normal and SPL images |
| +T: - the line is valid for Normal and TPL images |
| +ST: - the line is valid for Normal, SPL and SPL images |
| |
| So, if neither CONFIG_SPL nor CONFIG_TPL is defined, the defconfig file |
| has no '<condition>:' part and therefore has the same form of that of |
| Linux Kernel. |
| |
| In U-Boot, for example, a defconfig file can be written like this: |
| |
| CONFIG_FOO=100 |
| S:CONFIG_FOO=200 |
| T:CONFIG_FOO=300 |
| ST:CONFIG_BAR=y |
| +S:CONFIG_BAZ=y |
| +T:CONFIG_QUX=y |
| +ST:CONFIG_QUUX=y |
| |
| The defconfig above is parsed by this script and internally divided into |
| three temporary defconfig files. |
| |
| - Temporary defconfig for Normal image |
| CONFIG_FOO=100 |
| CONFIG_BAZ=y |
| CONFIG_QUX=y |
| CONFIG_QUUX=y |
| |
| - Temporary defconfig for SPL image |
| CONFIG_FOO=200 |
| CONFIG_BAR=y |
| CONFIG_BAZ=y |
| CONFIG_QUUX=y |
| |
| - Temporary defconfig for TPL image |
| CONFIG_FOO=300 |
| CONFIG_BAR=y |
| CONFIG_QUX=y |
| CONFIG_QUUX=y |
| |
| They are passed to scripts/kconfig/conf, each is used for generating |
| .config, spl/.config, tpl/.config, respectively. |
| |
| - savedefconfig |
| |
| This is the reverse operation of 'make defconfig'. |
| If neither CONFIG_SPL nor CONFIG_TPL is defined in the .config file, |
| it works as 'make savedefconfig' in Linux Kernel: create the minimal set |
| of config based on the .config and save it into 'defconfig' file. |
| |
| If CONFIG_SPL or CONFIG_TPL is defined, the common lines among .config, |
| spl/.config, tpl/.config are coalesced together and output to the file |
| 'defconfig' in the form like: |
| |
| CONFIG_FOO=100 |
| S:CONFIG_FOO=200 |
| T:CONFIG_FOO=300 |
| ST:CONFIG_BAR=y |
| +S:CONFIG_BAZ=y |
| +T:CONFIG_QUX=y |
| +ST:CONFIG_QUUX=y |
| |
| This can be used as an input of 'make <board>_defconfig' command. |
| """ |
| |
| import errno |
| import os |
| import re |
| import subprocess |
| import sys |
| |
| # Constant variables |
| SUB_IMAGES = ('spl', 'tpl') |
| IMAGES = ('',) + SUB_IMAGES |
| SYMBOL_MAP = {'': '+', 'spl': 'S', 'tpl': 'T'} |
| PATTERN_SYMBOL = re.compile(r'(\+?)(S?)(T?):(.*)') |
| |
| # Environment variables (should be defined in the top Makefile) |
| # .get('key', 'default_value') method is useful for standalone testing. |
| MAKE = os.environ.get('MAKE', 'make') |
| srctree = os.environ.get('srctree', '.') |
| KCONFIG_CONFIG = os.environ.get('KCONFIG_CONFIG', '.config') |
| |
| # Useful shorthand |
| build = '%s -f %s/scripts/Makefile.build obj=scripts/kconfig %%s' % (MAKE, srctree) |
| autoconf = '%s -f %s/scripts/Makefile.autoconf obj=%%s %%s' % (MAKE, srctree) |
| |
| ### helper functions ### |
| def mkdirs(*dirs): |
| """Make directories ignoring 'File exists' error.""" |
| for d in dirs: |
| try: |
| os.makedirs(d) |
| except OSError as exception: |
| # Ignore 'File exists' error |
| if exception.errno != errno.EEXIST: |
| raise |
| |
| def rmfiles(*files): |
| """Remove files ignoring 'No such file or directory' error.""" |
| for f in files: |
| try: |
| os.remove(f) |
| except OSError as exception: |
| # Ignore 'No such file or directory' error |
| if exception.errno != errno.ENOENT: |
| raise |
| |
| def rmdirs(*dirs): |
| """Remove directories ignoring 'No such file or directory' |
| and 'Directory not empty' error. |
| """ |
| for d in dirs: |
| try: |
| os.rmdir(d) |
| except OSError as exception: |
| # Ignore 'No such file or directory' |
| # and 'Directory not empty' error |
| if exception.errno != errno.ENOENT and \ |
| exception.errno != errno.ENOTEMPTY: |
| raise |
| |
| def run_command(command, callback_on_error=None): |
| """Run the given command in a sub-shell (and exit if it fails). |
| |
| Arguments: |
| command: A string of the command |
| callback_on_error: Callback handler invoked just before exit |
| when the command fails (Default=None) |
| """ |
| retcode = subprocess.call(command, shell=True) |
| if retcode: |
| if callback_on_error: |
| callback_on_error() |
| sys.exit("'%s' Failed" % command) |
| |
| def run_make_config(cmd, objdir, callback_on_error=None): |
| """Run the make command in a sub-shell (and exit if it fails). |
| |
| Arguments: |
| cmd: Make target such as 'config', 'menuconfig', 'defconfig', etc. |
| objdir: Target directory where the make command is run. |
| Typically '', 'spl', 'tpl' for Normal, SPL, TPL image, |
| respectively. |
| callback_on_error: Callback handler invoked just before exit |
| when the command fails (Default=None) |
| """ |
| # Linux expects defconfig files in arch/$(SRCARCH)/configs/ directory, |
| # but U-Boot puts them in configs/ directory. |
| # Give SRCARCH=.. to fake scripts/kconfig/Makefile. |
| options = 'SRCARCH=.. KCONFIG_OBJDIR=%s' % objdir |
| if objdir: |
| options += ' KCONFIG_CONFIG=%s/%s' % (objdir, KCONFIG_CONFIG) |
| mkdirs(objdir) |
| run_command(build % cmd + ' ' + options, callback_on_error) |
| |
| def get_enabled_subimages(ignore_error=False): |
| """Parse .config file to detect if CONFIG_SPL, CONFIG_TPL is enabled |
| and return a tuple of enabled subimages. |
| |
| Arguments: |
| ignore_error: Specify the behavior when '.config' is not found; |
| Raise an exception if this flag is False. |
| Return a null tuple if this flag is True. |
| |
| Returns: |
| A tuple of enabled subimages as follows: |
| () if neither CONFIG_SPL nor CONFIG_TPL is defined |
| ('spl',) if CONFIG_SPL is defined but CONFIG_TPL is not |
| ('spl', 'tpl') if both CONFIG_SPL and CONFIG_TPL are defined |
| """ |
| enabled = () |
| match_patterns = [ (img, 'CONFIG_' + img.upper() + '=y\n') |
| for img in SUB_IMAGES ] |
| try: |
| f = open(KCONFIG_CONFIG) |
| except IOError as exception: |
| if not ignore_error or exception.errno != errno.ENOENT: |
| raise |
| return enabled |
| with f: |
| for line in f: |
| for img, pattern in match_patterns: |
| if line == pattern: |
| enabled += (img,) |
| return enabled |
| |
| def do_silentoldconfig(cmd): |
| """Run 'make silentoldconfig' for all the enabled images. |
| |
| Arguments: |
| cmd: should always be a string 'silentoldconfig' |
| """ |
| run_make_config(cmd, '') |
| subimages = get_enabled_subimages() |
| for obj in subimages: |
| mkdirs(os.path.join(obj, 'include', 'config'), |
| os.path.join(obj, 'include', 'generated')) |
| run_make_config(cmd, obj) |
| remove_auto_conf = lambda : rmfiles('include/config/auto.conf') |
| # If the following part failed, include/config/auto.conf should be deleted |
| # so 'make silentoldconfig' will be re-run on the next build. |
| run_command(autoconf % |
| ('include', 'include/autoconf.mk include/autoconf.mk.dep'), |
| remove_auto_conf) |
| # include/config.h has been updated after 'make silentoldconfig'. |
| # We need to touch include/config/auto.conf so it gets newer |
| # than include/config.h. |
| # Otherwise, 'make silentoldconfig' would be invoked twice. |
| os.utime('include/config/auto.conf', None) |
| for obj in subimages: |
| run_command(autoconf % (obj + '/include', |
| obj + '/include/autoconf.mk'), |
| remove_auto_conf) |
| |
| def do_tmp_defconfig(output_lines, img): |
| """Helper function for do_board_defconfig(). |
| |
| Write the defconfig contents into a file '.tmp_defconfig' and |
| invoke 'make .tmp_defconfig'. |
| |
| Arguments: |
| output_lines: A sequence of defconfig lines of each image |
| img: Target image. Typically '', 'spl', 'tpl' for |
| Normal, SPL, TPL images, respectively. |
| """ |
| TMP_DEFCONFIG = '.tmp_defconfig' |
| TMP_DIRS = ('arch', 'configs') |
| defconfig_path = os.path.join('configs', TMP_DEFCONFIG) |
| mkdirs(*TMP_DIRS) |
| with open(defconfig_path, 'w') as f: |
| f.write(''.join(output_lines[img])) |
| cleanup = lambda: (rmfiles(defconfig_path), rmdirs(*TMP_DIRS)) |
| run_make_config(TMP_DEFCONFIG, img, cleanup) |
| cleanup() |
| |
| def do_board_defconfig(cmd): |
| """Run 'make <board>_defconfig'. |
| |
| Arguments: |
| cmd: should be a string '<board>_defconfig' |
| """ |
| defconfig_path = os.path.join(srctree, 'configs', cmd) |
| output_lines = dict([ (img, []) for img in IMAGES ]) |
| with open(defconfig_path) as f: |
| for line in f: |
| m = PATTERN_SYMBOL.match(line) |
| if m: |
| for idx, img in enumerate(IMAGES): |
| if m.group(idx + 1): |
| output_lines[img].append(m.group(4) + '\n') |
| continue |
| output_lines[''].append(line) |
| do_tmp_defconfig(output_lines, '') |
| for img in get_enabled_subimages(): |
| do_tmp_defconfig(output_lines, img) |
| |
| def do_defconfig(cmd): |
| """Run 'make defconfig'. |
| |
| Arguments: |
| cmd: should always be a string 'defconfig' |
| """ |
| KBUILD_DEFCONFIG = os.environ['KBUILD_DEFCONFIG'] |
| print "*** Default configuration is based on '%s'" % KBUILD_DEFCONFIG |
| do_board_defconfig(KBUILD_DEFCONFIG) |
| |
| def do_savedefconfig(cmd): |
| """Run 'make savedefconfig'. |
| |
| Arguments: |
| cmd: should always be a string 'savedefconfig' |
| """ |
| DEFCONFIG = 'defconfig' |
| # Continue even if '.config' does not exist |
| subimages = get_enabled_subimages(True) |
| run_make_config(cmd, '') |
| output_lines = [] |
| prefix = {} |
| with open(DEFCONFIG) as f: |
| for line in f: |
| output_lines.append(line) |
| prefix[line] = '+' |
| for img in subimages: |
| run_make_config(cmd, img) |
| unmatched_lines = [] |
| with open(DEFCONFIG) as f: |
| for line in f: |
| if line in output_lines: |
| index = output_lines.index(line) |
| output_lines[index:index] = unmatched_lines |
| unmatched_lines = [] |
| prefix[line] += SYMBOL_MAP[img] |
| else: |
| ummatched_lines.append(line) |
| prefix[line] = SYMBOL_MAP[img] |
| with open(DEFCONFIG, 'w') as f: |
| for line in output_lines: |
| if prefix[line] == '+': |
| f.write(line) |
| else: |
| f.write(prefix[line] + ':' + line) |
| |
| def do_others(cmd): |
| """Run the make command other than 'silentoldconfig', 'defconfig', |
| '<board>_defconfig' and 'savedefconfig'. |
| |
| Arguments: |
| cmd: Make target in the form of '<target_image>/<config_command>' |
| The field '<target_image>/' is typically empty, 'spl/', 'tpl/' |
| for Normal, SPL, TPL images, respectively. |
| The field '<config_command>' is make target such as 'config', |
| 'menuconfig', etc. |
| """ |
| objdir, _, cmd = cmd.rpartition('/') |
| run_make_config(cmd, objdir) |
| |
| cmd_list = {'silentoldconfig': do_silentoldconfig, |
| 'defconfig': do_defconfig, |
| 'savedefconfig': do_savedefconfig} |
| |
| def main(): |
| cmd = sys.argv[1] |
| if cmd.endswith('_defconfig'): |
| do_board_defconfig(cmd) |
| else: |
| func = cmd_list.get(cmd, do_others) |
| func(cmd) |
| |
| if __name__ == '__main__': |
| main() |