1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

dist/tools/kconfiglib: introduce own genconfig.py tool

This tool unifies multiple functionalities needed by the Kconfig
integration into one tool.
This commit is contained in:
Leandro Lanzieri 2020-08-07 09:27:29 +02:00
parent 574676b150
commit 9c10580ba4
No known key found for this signature in database
GPG Key ID: 13559905E2EBEAA5
4 changed files with 175 additions and 9 deletions

View File

@ -1,5 +1,3 @@
genconfig.py
kconfiglib.py
menuconfig.py
merge_config.py
__pycache__

View File

@ -7,9 +7,7 @@ include $(RIOTBASE)/pkg/pkg.mk
all:
$(Q)cp $(PKG_SOURCE_DIR)/kconfiglib.py $(PKG_SOURCE_DIR)/menuconfig.py \
$(PKG_SOURCE_DIR)/genconfig.py $(PKG_SOURCE_DIR)/examples/merge_config.py \
.
remove:
$(Q)$(RM) -r $(PKG_SOURCE_DIR) kconfiglib.py menuconfig.py genconfig.py \
merge_config.py
$(Q)$(RM) -r $(PKG_SOURCE_DIR) kconfiglib.py menuconfig.py

173
dist/tools/kconfiglib/genconfig.py vendored Executable file
View File

@ -0,0 +1,173 @@
#!/usr/bin/env python3
# Copyright (c) 2018-2019, Ulf Magnusson
# 2020 HAW Hamburg
# SPDX-License-Identifier: ISC
"""
This script is used to merge multiple configuration sources and generate
different outputs related to Kconfig:
- Generate a header file with #defines from the configuration, matching the
format of include/generated/autoconf.h in the Linux kernel.
- Write the configuration output as a .config file. See --config-out.
- The --sync-deps, --file-list, and --env-list options generate information that
can be used to avoid needless rebuilds/reconfigurations.
Before writing a header or configuration file, Kconfiglib compares the old
contents of the file against the new contents. If there's no change, the write
is skipped. This avoids updating file metadata like the modification time, and
might save work depending on your build setup.
A custom header string can be inserted at the beginning of generated
configuration and header files by setting the KCONFIG_CONFIG_HEADER and
KCONFIG_AUTOHEADER_HEADER environment variables, respectively. The string is
not automatically made a comment (this is by design, to allow anything to be
added), and no trailing newline is added, so add '/* */', '#', and newlines as
appropriate.
"""
import argparse
import logging
import os
import kconfiglib
DEFAULT_SYNC_DEPS_PATH = "deps/"
class NoConfigurationFile(Exception):
"""
Raised when an operation that requires a configuration input file is
executed but the file is not specified.
"""
pass
def merge_configs(kconf, configs=[]):
# Enable warnings for assignments to undefined symbols
kconf.warn_assign_undef = True
# (This script uses alldefconfig as the base. Other starting states could be
# set up here as well. The approach in examples/allnoconfig_simpler.py could
# provide an allnoconfig starting state for example.)
# Disable warnings generated for multiple assignments to the same symbol within
# a (set of) configuration files. Assigning a symbol multiple times might be
# done intentionally when merging configuration files.
kconf.warn_assign_override = False
kconf.warn_assign_redun = False
# Create a merged configuration by loading the fragments with replace=False.
for config in configs:
logging.debug(kconf.load_config(config, replace=False))
def main():
parser = argparse.ArgumentParser(
formatter_class=argparse.RawDescriptionHelpFormatter,
description=__doc__)
parser.add_argument(
"--header-path",
metavar="HEADER_FILE",
help="""
Path to write the generated header file to. If not specified the header file is
not written.
""")
parser.add_argument(
"--config-out",
metavar="CONFIG_FILE",
help="""
Write the configuration to CONFIG_FILE. If not specified the file is not
written.
""")
parser.add_argument(
"--kconfig-filename",
metavar="KCONFIG_FILENAME",
nargs="?",
default="Kconfig",
help="Top-level Kconfig file (default: Kconfig)")
parser.add_argument(
"--sync-deps",
metavar="OUTPUT_DIR",
nargs="?",
const=DEFAULT_SYNC_DEPS_PATH,
help="""
Enable generation of symbol dependency information for incremental builds,
optionally specifying the output directory (default: {}). See the docstring of
Kconfig.sync_deps() in Kconfiglib for more information.
""".format(DEFAULT_SYNC_DEPS_PATH))
parser.add_argument(
"--file-list",
metavar="FILE_LIST_FILE",
help="""
Write a makefile listing all the Kconfig files used, and adding them as
dependencies of HEADER_FILE. The paths are absolute. Files appear in the order
they're 'source'd.
""")
parser.add_argument(
"--env-list",
metavar="ENV_LIST_FILE",
help="""
Write a list of all environment variables referenced in Kconfig files to
ENV_LIST_FILE, with one variable per line. Each line has the format NAME=VALUE.
Only environment variables referenced with the preprocessor $(VAR) syntax are
included, and not variables referenced with the older $VAR syntax (which is
only supported for backwards compatibility).
""")
parser.add_argument(
"-d", "--debug",
action="store_true",
help="Enable debug messages")
parser.add_argument(
"--config-sources",
metavar="CONFIG_SOURCES",
nargs='*',
help="List of configuration files to merge and apply. May be empty.")
args = parser.parse_args()
log_level = logging.DEBUG if args.debug else logging.ERROR
logging.basicConfig(format='[genconfig.py]:%(levelname)s-%(message)s',
level=log_level)
kconf = kconfiglib.Kconfig(args.kconfig_filename)
merge_configs(kconf, args.config_sources)
if args.config_out is not None:
logging.debug(kconf.write_config(args.config_out, save_old=False))
if args.header_path is not None:
logging.debug(kconf.write_autoconf(args.header_path))
if args.sync_deps is not None:
logging.debug("Incremental build header files generated at '{}'".format(args.sync_deps))
kconf.sync_deps(args.sync_deps)
if args.file_list is not None:
if args.config_out is None:
raise NoConfigurationFile("Can't generate Kconfig dependency file without configuration file")
logging.debug("Kconfig dependencies written to '{}'".format(args.file_list))
with open(args.file_list, "w", encoding="utf-8") as f:
f.write("{}: \\\n".format(args.config_out))
for path in kconf.kconfig_filenames:
f.write(" {} \\\n".format(os.path.abspath(path)))
if args.env_list is not None:
logging.debug("Kconfig environmental variables written to '{}'".format(args.env_list))
with open(args.env_list, "w", encoding="utf-8") as f:
for env_var in kconf.env_vars:
f.write("{}={}\n".format(env_var, os.environ[env_var]))
if __name__ == "__main__":
main()

View File

@ -1,8 +1,7 @@
# Define tools to use
MENUCONFIG ?= $(RIOTTOOLS)/kconfiglib/riot_menuconfig.py
BASE_MENUCONFIG ?= $(RIOTTOOLS)/kconfiglib/menuconfig.py
GENCONFIG ?= $(RIOTTOOLS)/kconfiglib/genconfig.py
MERGECONFIG ?= $(RIOTTOOLS)/kconfiglib/merge_config.py
GENCONFIG := $(RIOTTOOLS)/kconfiglib/genconfig.py
$(BASE_MENUCONFIG):
@echo "[INFO] Kconfiglib not found - getting it"
@ -10,5 +9,3 @@ $(BASE_MENUCONFIG):
@echo "[INFO] Kconfiglib downloaded"
$(GENCONFIG): $(BASE_MENUCONFIG)
$(MERGECONFIG): $(BASE_MENUCONFIG)