modules: support tests/samples/boards in modules

Generate options for sanitycheck to run tests and samples in modules.
Use the --sanitycheck-out <file> to generate a file that can be supplied
to sanitycheck on the commandline which will add additional testroots
and boards if the module does contain out of tree boards.

the module.yaml file now accepts the following:

samples:
  - path/to/samples
tests:
  - path/to/tests
boards:
  - path/to/boards

Signed-off-by: Anas Nashif <anas.nashif@intel.com>
This commit is contained in:
Anas Nashif 2019-12-11 10:13:23 -05:00 committed by Carles Cufí
parent f95a7faa58
commit 286a9eda84

View file

@ -10,6 +10,11 @@ used as project list.
Include file is generated for Kconfig using --kconfig-out. Include file is generated for Kconfig using --kconfig-out.
A <name>:<path> text file is generated for use with CMake using --cmake-out. A <name>:<path> text file is generated for use with CMake using --cmake-out.
Using --sanitycheck-out <filename> an argument file for sanitycheck script will
be generated which would point to test and sample roots available in modules
that can be included during a sanitycheck run. This allows testing code
maintained in modules in addition to what is available in the main Zephyr tree.
''' '''
import argparse import argparse
@ -30,7 +35,7 @@ METADATA_SCHEMA = '''
type: map type: map
mapping: mapping:
build: build:
required: true required: false
type: map type: map
mapping: mapping:
cmake: cmake:
@ -39,6 +44,21 @@ mapping:
kconfig: kconfig:
required: false required: false
type: str type: str
tests:
required: false
type: seq
sequence:
- type: str
samples:
required: false
type: seq
sequence:
- type: str
boards:
required: false
type: seq
sequence:
- type: str
''' '''
schema = yaml.safe_load(METADATA_SCHEMA) schema = yaml.safe_load(METADATA_SCHEMA)
@ -55,12 +75,10 @@ def validate_setting(setting, module_path, filename=None):
return True return True
def process_module(module, cmake_out=None, kconfig_out=None): def process_module(module):
cmake_setting = None
kconfig_setting = None
module_path = PurePath(module) module_path = PurePath(module)
module_yml = module_path.joinpath('zephyr/module.yml') module_yml = module_path.joinpath('zephyr/module.yml')
if Path(module_yml).is_file(): if Path(module_yml).is_file():
with Path(module_yml).open('r') as f: with Path(module_yml).open('r') as f:
meta = yaml.safe_load(f.read()) meta = yaml.safe_load(f.read())
@ -72,49 +90,83 @@ def process_module(module, cmake_out=None, kconfig_out=None):
sys.exit('ERROR: Malformed "build" section in file: {}\n{}' sys.exit('ERROR: Malformed "build" section in file: {}\n{}'
.format(module_yml.as_posix(), e)) .format(module_yml.as_posix(), e))
section = meta.get('build', dict()) return meta
cmake_setting = section.get('cmake', None)
if not validate_setting(cmake_setting, module, 'CMakeLists.txt'):
sys.exit('ERROR: "cmake" key in {} has folder value "{}" which '
'does not contain a CMakeLists.txt file.'
.format(module_yml.as_posix(), cmake_setting))
kconfig_setting = section.get('kconfig', None) return None
if not validate_setting(kconfig_setting, module):
sys.exit('ERROR: "kconfig" key in {} has value "{}" which does '
'not point to a valid Kconfig file.'
.format(module_yml.as_posix(), kconfig_setting))
cmake_path = module_path.joinpath(cmake_setting or 'zephyr')
cmake_file = cmake_path.joinpath('CMakeLists.txt')
if Path(cmake_file).is_file() and cmake_out is not None:
cmake_out.write('\"{}\":\"{}\"\n'
.format(module_path.name,
Path(cmake_path).resolve().as_posix()))
kconfig_file = module_path.joinpath(kconfig_setting or 'zephyr/Kconfig') def process_cmake(module, meta):
if Path(kconfig_file).is_file() and kconfig_out is not None: section = meta.get('build', dict())
kconfig_out.write('osource "{}"\n\n' module_path = PurePath(module)
.format(Path(kconfig_file).resolve().as_posix())) module_yml = module_path.joinpath('zephyr/module.yml')
cmake_setting = section.get('cmake', None)
if not validate_setting(cmake_setting, module, 'CMakeLists.txt'):
sys.exit('ERROR: "cmake" key in {} has folder value "{}" which '
'does not contain a CMakeLists.txt file.'
.format(module_yml.as_posix(), cmake_setting))
cmake_path = os.path.join(module, cmake_setting or 'zephyr')
cmake_file = os.path.join(cmake_path, 'CMakeLists.txt')
if os.path.isfile(cmake_file):
return('\"{}\":\"{}\"\n'
.format(module_path.name, Path(cmake_path).resolve().as_posix()))
else:
return ""
def process_kconfig(module, meta):
section = meta.get('build', dict())
module_path = PurePath(module)
module_yml = module_path.joinpath('zephyr/module.yml')
kconfig_setting = section.get('kconfig', None)
if not validate_setting(kconfig_setting, module):
sys.exit('ERROR: "kconfig" key in {} has value "{}" which does '
'not point to a valid Kconfig file.'
.format(module_yml, kconfig_setting))
kconfig_file = os.path.join(module, kconfig_setting or 'zephyr/Kconfig')
if os.path.isfile(kconfig_file):
return 'osource "{}"\n\n'.format(Path(kconfig_file).resolve().as_posix())
else:
return ""
def process_sanitycheck(module, meta):
out = ""
tests = meta.get('tests', [])
samples = meta.get('samples', [])
boards = meta.get('boards', [])
for pth in tests + samples:
if pth:
dir = os.path.join(module, pth)
out += '-T\n{}\n'.format(PurePath(os.path.abspath(dir)).as_posix())
for pth in boards:
if pth:
dir = os.path.join(module, pth)
out += '--board-root\n{}\n'.format(PurePath(os.path.abspath(dir)).as_posix())
return out
def main(): def main():
kconfig_out_file = None
cmake_out_file = None
parser = argparse.ArgumentParser(description=''' parser = argparse.ArgumentParser(description='''
Process a list of projects and create Kconfig / CMake include files for Process a list of projects and create Kconfig / CMake include files for
projects which are also a Zephyr module''') projects which are also a Zephyr module''')
parser.add_argument('--kconfig-out', parser.add_argument('--kconfig-out',
help='File to write with resulting KConfig import' help="""File to write with resulting KConfig import
'statements.') statements.""")
parser.add_argument('--sanitycheck-out',
help="""File to write with resulting sanitycheck parameters.""")
parser.add_argument('--cmake-out', parser.add_argument('--cmake-out',
help='File to write with resulting <name>:<path>' help="""File to write with resulting <name>:<path>
'values to use for including in CMake') values to use for including in CMake""")
parser.add_argument('-m', '--modules', nargs='+', parser.add_argument('-m', '--modules', nargs='+',
help='List of modules to parse instead of using `west' help="""List of modules to parse instead of using `west
'list`') list`""")
parser.add_argument('-x', '--extra-modules', nargs='+', parser.add_argument('-x', '--extra-modules', nargs='+',
help='List of extra modules to parse') help='List of extra modules to parse')
parser.add_argument('-w', '--west-path', default='west', parser.add_argument('-w', '--west-path', default='west',
@ -145,23 +197,33 @@ def main():
if args.extra_modules is not None: if args.extra_modules is not None:
projects += args.extra_modules projects += args.extra_modules
kconfig = ""
cmake = ""
sanitycheck = ""
for project in projects:
# Avoid including Zephyr base project as module.
if project == os.environ.get('ZEPHYR_BASE'):
continue
meta = process_module(project)
if meta:
kconfig += process_kconfig(project, meta)
cmake += process_cmake(project, meta)
sanitycheck += process_sanitycheck(project, meta)
if args.kconfig_out: if args.kconfig_out:
kconfig_out_file = open(args.kconfig_out, 'w', encoding="utf-8") with open(args.kconfig_out, 'w', encoding="utf-8") as fp:
fp.write(kconfig)
if args.cmake_out: if args.cmake_out:
cmake_out_file = open(args.cmake_out, 'w', encoding="utf-8") with open(args.cmake_out, 'w', encoding="utf-8") as fp:
fp.write(cmake)
try:
for project in projects:
# Avoid including Zephyr base project as module.
if project != os.environ.get('ZEPHYR_BASE'):
process_module(project, cmake_out_file, kconfig_out_file)
finally:
if args.kconfig_out:
kconfig_out_file.close()
if args.cmake_out:
cmake_out_file.close()
if args.sanitycheck_out:
with open(args.sanitycheck_out, 'w', encoding="utf-8") as fp:
fp.write(sanitycheck)
if __name__ == "__main__": if __name__ == "__main__":
main() main()