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:
parent
f95a7faa58
commit
286a9eda84
|
@ -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()
|
||||||
|
|
Loading…
Reference in a new issue