build_helpers.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. # Copyright 2018 (c) Foundries.io.
  2. #
  3. # SPDX-License-Identifier: Apache-2.0
  4. '''Common definitions for building Zephyr applications.
  5. This provides some default settings and convenience wrappers for
  6. building Zephyr applications needed by multiple commands.
  7. See build.py for the build command itself.
  8. '''
  9. import zcmake
  10. import os
  11. from pathlib import Path
  12. from west import log
  13. from west.configuration import config
  14. from west.util import escapes_directory
  15. DEFAULT_BUILD_DIR = 'build'
  16. '''Name of the default Zephyr build directory.'''
  17. DEFAULT_CMAKE_GENERATOR = 'Ninja'
  18. '''Name of the default CMake generator.'''
  19. FIND_BUILD_DIR_DESCRIPTION = '''\
  20. If the build directory is not given, the default is {}/ unless the
  21. build.dir-fmt configuration variable is set. The current directory is
  22. checked after that. If either is a Zephyr build directory, it is used.
  23. '''.format(DEFAULT_BUILD_DIR)
  24. def _resolve_build_dir(fmt, guess, cwd, **kwargs):
  25. # Remove any None values, we do not want 'None' as a string
  26. kwargs = {k: v for k, v in kwargs.items() if v is not None}
  27. # Check if source_dir is below cwd first
  28. source_dir = kwargs.get('source_dir')
  29. if source_dir:
  30. if escapes_directory(cwd, source_dir):
  31. kwargs['source_dir'] = os.path.relpath(source_dir, cwd)
  32. else:
  33. # no meaningful relative path possible
  34. kwargs['source_dir'] = ''
  35. try:
  36. return fmt.format(**kwargs)
  37. except KeyError:
  38. if not guess:
  39. return None
  40. # Guess the build folder by iterating through all sub-folders from the
  41. # root of the format string and trying to resolve. If resolving fails,
  42. # proceed to iterate over subfolders only if there is a single folder
  43. # present on each iteration.
  44. parts = Path(fmt).parts
  45. b = Path('.')
  46. for p in parts:
  47. # default to cwd in the first iteration
  48. curr = b
  49. b = b.joinpath(p)
  50. try:
  51. # if fmt is an absolute path, the first iteration will always
  52. # resolve '/'
  53. b = Path(str(b).format(**kwargs))
  54. except KeyError:
  55. # Missing key, check sub-folders and match if a single one exists
  56. while True:
  57. if not curr.exists():
  58. return None
  59. dirs = [f for f in curr.iterdir() if f.is_dir()]
  60. if len(dirs) != 1:
  61. return None
  62. curr = dirs[0]
  63. if is_zephyr_build(str(curr)):
  64. return str(curr)
  65. return str(b)
  66. def find_build_dir(dir, guess=False, **kwargs):
  67. '''Heuristic for finding a build directory.
  68. The default build directory is computed by reading the build.dir-fmt
  69. configuration option, defaulting to DEFAULT_BUILD_DIR if not set. It might
  70. be None if the build.dir-fmt configuration option is set but cannot be
  71. resolved.
  72. If the given argument is truthy, it is returned. Otherwise, if
  73. the default build folder is a build directory, it is returned.
  74. Next, if the current working directory is a build directory, it is
  75. returned. Finally, the default build directory is returned (may be None).
  76. '''
  77. if dir:
  78. build_dir = dir
  79. else:
  80. cwd = os.getcwd()
  81. default = config.get('build', 'dir-fmt', fallback=DEFAULT_BUILD_DIR)
  82. default = _resolve_build_dir(default, guess, cwd, **kwargs)
  83. log.dbg('config dir-fmt: {}'.format(default), level=log.VERBOSE_EXTREME)
  84. if default and is_zephyr_build(default):
  85. build_dir = default
  86. elif is_zephyr_build(cwd):
  87. build_dir = cwd
  88. else:
  89. build_dir = default
  90. log.dbg('build dir: {}'.format(build_dir), level=log.VERBOSE_EXTREME)
  91. if build_dir:
  92. return os.path.abspath(build_dir)
  93. else:
  94. return None
  95. def is_zephyr_build(path):
  96. '''Return true if and only if `path` appears to be a valid Zephyr
  97. build directory.
  98. "Valid" means the given path is a directory which contains a CMake
  99. cache with a 'ZEPHYR_BASE' or 'ZEPHYR_TOOLCHAIN_VARIANT' variable.
  100. (The check for ZEPHYR_BASE introduced sometime after Zephyr 2.4 to
  101. fix https://github.com/zephyrproject-rtos/zephyr/issues/28876; we
  102. keep support for the second variable around for compatibility with
  103. versions 2.2 and earlier, which didn't have ZEPHYR_BASE in cache.
  104. The cached ZEPHYR_BASE was added in
  105. https://github.com/zephyrproject-rtos/zephyr/pull/23054.)
  106. '''
  107. try:
  108. cache = zcmake.CMakeCache.from_build_dir(path)
  109. except FileNotFoundError:
  110. cache = {}
  111. if 'ZEPHYR_BASE' in cache or 'ZEPHYR_TOOLCHAIN_VARIANT' in cache:
  112. log.dbg(f'{path} is a zephyr build directory',
  113. level=log.VERBOSE_EXTREME)
  114. return True
  115. log.dbg(f'{path} is NOT a valid zephyr build directory',
  116. level=log.VERBOSE_EXTREME)
  117. return False