kconfig.cmake 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. # SPDX-License-Identifier: Apache-2.0
  2. # Folders needed for conf/mconf files (kconfig has no method of redirecting all output files).
  3. # conf/mconf needs to be run from a different directory because of: GH-3408
  4. file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/kconfig/include/generated)
  5. file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/kconfig/include/config)
  6. # Support multiple SOC_ROOT, remove ZEPHYR_BASE as that is always sourced.
  7. set(kconfig_soc_root ${SOC_ROOT})
  8. list(REMOVE_ITEM kconfig_soc_root ${ZEPHYR_BASE})
  9. set(OPERATION WRITE)
  10. foreach(root ${kconfig_soc_root})
  11. file(${OPERATION} ${KCONFIG_BINARY_DIR}/Kconfig.soc.defconfig
  12. "osource \"${root}/soc/$(ARCH)/*/Kconfig.defconfig\"\n"
  13. )
  14. file(${OPERATION} ${KCONFIG_BINARY_DIR}/Kconfig.soc
  15. "osource \"${root}/soc/$(ARCH)/*/Kconfig.soc\"\n"
  16. )
  17. file(${OPERATION} ${KCONFIG_BINARY_DIR}/Kconfig.soc.arch
  18. "osource \"${root}/soc/$(ARCH)/Kconfig\"\n"
  19. "osource \"${root}/soc/$(ARCH)/*/Kconfig\"\n"
  20. )
  21. set(OPERATION APPEND)
  22. endforeach()
  23. # Support multiple shields in BOARD_ROOT, remove ZEPHYR_BASE as that is always sourced.
  24. set(kconfig_board_root ${BOARD_ROOT})
  25. list(REMOVE_ITEM kconfig_board_root ${ZEPHYR_BASE})
  26. set(OPERATION WRITE)
  27. foreach(root ${kconfig_board_root})
  28. file(${OPERATION} ${KCONFIG_BINARY_DIR}/Kconfig.shield.defconfig
  29. "osource \"${root}/boards/shields/*/Kconfig.defconfig\"\n"
  30. )
  31. file(${OPERATION} ${KCONFIG_BINARY_DIR}/Kconfig.shield
  32. "osource \"${root}/boards/shields/*/Kconfig.shield\"\n"
  33. )
  34. set(OPERATION APPEND)
  35. endforeach()
  36. if(KCONFIG_ROOT)
  37. zephyr_file(APPLICATION_ROOT KCONFIG_ROOT)
  38. # KCONFIG_ROOT has either been specified as a CMake variable or is
  39. # already in the CMakeCache.txt. This has precedence.
  40. elseif(EXISTS ${APPLICATION_SOURCE_DIR}/Kconfig)
  41. set(KCONFIG_ROOT ${APPLICATION_SOURCE_DIR}/Kconfig)
  42. else()
  43. set(KCONFIG_ROOT ${ZEPHYR_BASE}/Kconfig)
  44. endif()
  45. set(BOARD_DEFCONFIG ${BOARD_DIR}/${BOARD}_defconfig)
  46. set(DOTCONFIG ${PROJECT_BINARY_DIR}/.config)
  47. set(PARSED_KCONFIG_SOURCES_TXT ${PROJECT_BINARY_DIR}/kconfig/sources.txt)
  48. if(CONF_FILE)
  49. string(REPLACE " " ";" CONF_FILE_AS_LIST "${CONF_FILE}")
  50. endif()
  51. if(OVERLAY_CONFIG)
  52. string(REPLACE " " ";" OVERLAY_CONFIG_AS_LIST "${OVERLAY_CONFIG}")
  53. endif()
  54. if((DEFINED BOARD_REVISION) AND EXISTS ${BOARD_DIR}/${BOARD}_${BOARD_REVISION_STRING}.conf)
  55. list(INSERT CONF_FILE_AS_LIST 0 ${BOARD_DIR}/${BOARD}_${BOARD_REVISION_STRING}.conf)
  56. endif()
  57. # DTS_ROOT_BINDINGS is a semicolon separated list, this causes
  58. # problems when invoking kconfig_target since semicolon is a special
  59. # character in the C shell, so we make it into a question-mark
  60. # separated list instead.
  61. string(REPLACE ";" "?" DTS_ROOT_BINDINGS "${DTS_ROOT_BINDINGS}")
  62. # Export each `ZEPHYR_<module>_MODULE_DIR` to Kconfig.
  63. # This allows Kconfig files to refer relative from a modules root as:
  64. # source "$(ZEPHYR_FOO_MODULE_DIR)/Kconfig"
  65. foreach(module_name ${ZEPHYR_MODULE_NAMES})
  66. zephyr_string(SANITIZE TOUPPER MODULE_NAME_UPPER ${module_name})
  67. list(APPEND
  68. ZEPHYR_KCONFIG_MODULES_DIR
  69. "ZEPHYR_${MODULE_NAME_UPPER}_MODULE_DIR=${ZEPHYR_${MODULE_NAME_UPPER}_MODULE_DIR}"
  70. )
  71. if(ZEPHYR_${MODULE_NAME_UPPER}_KCONFIG)
  72. list(APPEND
  73. ZEPHYR_KCONFIG_MODULES_DIR
  74. "ZEPHYR_${MODULE_NAME_UPPER}_KCONFIG=${ZEPHYR_${MODULE_NAME_UPPER}_KCONFIG}"
  75. )
  76. endif()
  77. endforeach()
  78. # A list of common environment settings used when invoking Kconfig during CMake
  79. # configure time or menuconfig and related build target.
  80. string(REPLACE ";" "\\\;" SHIELD_AS_LIST_ESCAPED "${SHIELD_AS_LIST}")
  81. set(COMMON_KCONFIG_ENV_SETTINGS
  82. PYTHON_EXECUTABLE=${PYTHON_EXECUTABLE}
  83. srctree=${ZEPHYR_BASE}
  84. KERNELVERSION=${KERNELVERSION}
  85. KCONFIG_CONFIG=${DOTCONFIG}
  86. # Set environment variables so that Kconfig can prune Kconfig source
  87. # files for other architectures
  88. ARCH=${ARCH}
  89. ARCH_DIR=${ARCH_DIR}
  90. BOARD_DIR=${BOARD_DIR}
  91. KCONFIG_BINARY_DIR=${KCONFIG_BINARY_DIR}
  92. TOOLCHAIN_KCONFIG_DIR=${TOOLCHAIN_KCONFIG_DIR}
  93. TOOLCHAIN_HAS_NEWLIB=$<IF:$<BOOL:${TOOLCHAIN_HAS_NEWLIB}>,y,n>
  94. EDT_PICKLE=${EDT_PICKLE}
  95. # Export all Zephyr modules to Kconfig
  96. ${ZEPHYR_KCONFIG_MODULES_DIR}
  97. )
  98. # Allow out-of-tree users to add their own Kconfig python frontend
  99. # targets by appending targets to the CMake list
  100. # 'EXTRA_KCONFIG_TARGETS' and setting variables named
  101. # 'EXTRA_KCONFIG_TARGET_COMMAND_FOR_<target>'
  102. #
  103. # e.g.
  104. # cmake -DEXTRA_KCONFIG_TARGETS=cli
  105. # -DEXTRA_KCONFIG_TARGET_COMMAND_FOR_cli=cli_kconfig_frontend.py
  106. set(EXTRA_KCONFIG_TARGET_COMMAND_FOR_menuconfig
  107. ${ZEPHYR_BASE}/scripts/kconfig/menuconfig.py
  108. )
  109. set(EXTRA_KCONFIG_TARGET_COMMAND_FOR_guiconfig
  110. ${ZEPHYR_BASE}/scripts/kconfig/guiconfig.py
  111. )
  112. set(EXTRA_KCONFIG_TARGET_COMMAND_FOR_hardenconfig
  113. ${ZEPHYR_BASE}/scripts/kconfig/hardenconfig.py
  114. )
  115. foreach(kconfig_target
  116. menuconfig
  117. guiconfig
  118. hardenconfig
  119. ${EXTRA_KCONFIG_TARGETS}
  120. )
  121. add_custom_target(
  122. ${kconfig_target}
  123. ${CMAKE_COMMAND} -E env
  124. ZEPHYR_BASE=${ZEPHYR_BASE}
  125. ZEPHYR_TOOLCHAIN_VARIANT=${ZEPHYR_TOOLCHAIN_VARIANT}
  126. ${COMMON_KCONFIG_ENV_SETTINGS}
  127. "SHIELD_AS_LIST=${SHIELD_AS_LIST_ESCAPED}"
  128. DTS_POST_CPP=${DTS_POST_CPP}
  129. DTS_ROOT_BINDINGS=${DTS_ROOT_BINDINGS}
  130. ${PYTHON_EXECUTABLE}
  131. ${EXTRA_KCONFIG_TARGET_COMMAND_FOR_${kconfig_target}}
  132. ${KCONFIG_ROOT}
  133. WORKING_DIRECTORY ${PROJECT_BINARY_DIR}/kconfig
  134. USES_TERMINAL
  135. COMMAND_EXPAND_LISTS
  136. )
  137. endforeach()
  138. # Support assigning Kconfig symbols on the command-line with CMake
  139. # cache variables prefixed with 'CONFIG_'. This feature is
  140. # experimental and undocumented until it has undergone more
  141. # user-testing.
  142. unset(EXTRA_KCONFIG_OPTIONS)
  143. get_cmake_property(cache_variable_names CACHE_VARIABLES)
  144. foreach (name ${cache_variable_names})
  145. if("${name}" MATCHES "^CONFIG_")
  146. # When a cache variable starts with 'CONFIG_', it is assumed to be
  147. # a Kconfig symbol assignment from the CMake command line.
  148. set(EXTRA_KCONFIG_OPTIONS
  149. "${EXTRA_KCONFIG_OPTIONS}\n${name}=${${name}}"
  150. )
  151. endif()
  152. endforeach()
  153. if(EXTRA_KCONFIG_OPTIONS)
  154. set(EXTRA_KCONFIG_OPTIONS_FILE ${PROJECT_BINARY_DIR}/misc/generated/extra_kconfig_options.conf)
  155. file(WRITE
  156. ${EXTRA_KCONFIG_OPTIONS_FILE}
  157. ${EXTRA_KCONFIG_OPTIONS}
  158. )
  159. endif()
  160. # Bring in extra configuration files dropped in by the user or anyone else;
  161. # make sure they are set at the end so we can override any other setting
  162. file(GLOB config_files ${APPLICATION_BINARY_DIR}/*.conf)
  163. list(SORT config_files)
  164. set(
  165. merge_config_files
  166. ${BOARD_DEFCONFIG}
  167. ${CONF_FILE_AS_LIST}
  168. ${shield_conf_files}
  169. ${OVERLAY_CONFIG_AS_LIST}
  170. ${EXTRA_KCONFIG_OPTIONS_FILE}
  171. ${config_files}
  172. )
  173. # Create a list of absolute paths to the .config sources from
  174. # merge_config_files, which is a mix of absolute and relative paths.
  175. set(merge_config_files_with_absolute_paths "")
  176. foreach(f ${merge_config_files})
  177. if(IS_ABSOLUTE ${f})
  178. set(path ${f})
  179. else()
  180. set(path ${APPLICATION_SOURCE_DIR}/${f})
  181. endif()
  182. list(APPEND merge_config_files_with_absolute_paths ${path})
  183. endforeach()
  184. foreach(f ${merge_config_files_with_absolute_paths})
  185. if(NOT EXISTS ${f} OR IS_DIRECTORY ${f})
  186. message(FATAL_ERROR "File not found: ${f}")
  187. endif()
  188. endforeach()
  189. # Calculate a checksum of merge_config_files to determine if we need
  190. # to re-generate .config
  191. set(merge_config_files_checksum "")
  192. foreach(f ${merge_config_files_with_absolute_paths})
  193. file(MD5 ${f} checksum)
  194. set(merge_config_files_checksum "${merge_config_files_checksum}${checksum}")
  195. endforeach()
  196. # Create a new .config if it does not exists, or if the checksum of
  197. # the dependencies has changed
  198. set(merge_config_files_checksum_file ${PROJECT_BINARY_DIR}/.cmake.dotconfig.checksum)
  199. set(CREATE_NEW_DOTCONFIG 1)
  200. # Check if the checksum file exists too before trying to open it, though it
  201. # should under normal circumstances
  202. if(EXISTS ${DOTCONFIG} AND EXISTS ${merge_config_files_checksum_file})
  203. # Read out what the checksum was previously
  204. file(READ
  205. ${merge_config_files_checksum_file}
  206. merge_config_files_checksum_prev
  207. )
  208. if(
  209. ${merge_config_files_checksum} STREQUAL
  210. ${merge_config_files_checksum_prev}
  211. )
  212. # Checksum is the same as before
  213. set(CREATE_NEW_DOTCONFIG 0)
  214. endif()
  215. endif()
  216. if(CREATE_NEW_DOTCONFIG)
  217. set(input_configs_are_handwritten --handwritten-input-configs)
  218. set(input_configs ${merge_config_files})
  219. else()
  220. set(input_configs ${DOTCONFIG})
  221. endif()
  222. execute_process(
  223. COMMAND ${CMAKE_COMMAND} -E env
  224. ${COMMON_KCONFIG_ENV_SETTINGS}
  225. SHIELD_AS_LIST=${SHIELD_AS_LIST_ESCAPED}
  226. ${PYTHON_EXECUTABLE}
  227. ${ZEPHYR_BASE}/scripts/kconfig/kconfig.py
  228. --zephyr-base=${ZEPHYR_BASE}
  229. ${input_configs_are_handwritten}
  230. ${KCONFIG_ROOT}
  231. ${DOTCONFIG}
  232. ${AUTOCONF_H}
  233. ${PARSED_KCONFIG_SOURCES_TXT}
  234. ${input_configs}
  235. WORKING_DIRECTORY ${APPLICATION_SOURCE_DIR}
  236. # The working directory is set to the app dir such that the user
  237. # can use relative paths in CONF_FILE, e.g. CONF_FILE=nrf5.conf
  238. RESULT_VARIABLE ret
  239. )
  240. if(NOT "${ret}" STREQUAL "0")
  241. message(FATAL_ERROR "command failed with return code: ${ret}")
  242. endif()
  243. if(CREATE_NEW_DOTCONFIG)
  244. # Write the new configuration fragment checksum. Only do this if kconfig.py
  245. # succeeds, to avoid marking zephyr/.config as up-to-date when it hasn't been
  246. # regenerated.
  247. file(WRITE ${merge_config_files_checksum_file}
  248. ${merge_config_files_checksum})
  249. endif()
  250. # Read out the list of 'Kconfig' sources that were used by the engine.
  251. file(STRINGS ${PARSED_KCONFIG_SOURCES_TXT} PARSED_KCONFIG_SOURCES_LIST)
  252. # Force CMAKE configure when the Kconfig sources or configuration files changes.
  253. foreach(kconfig_input
  254. ${merge_config_files}
  255. ${DOTCONFIG}
  256. ${PARSED_KCONFIG_SOURCES_LIST}
  257. )
  258. set_property(DIRECTORY APPEND PROPERTY CMAKE_CONFIGURE_DEPENDS ${kconfig_input})
  259. endforeach()
  260. add_custom_target(config-twister DEPENDS ${DOTCONFIG})
  261. # Remove the CLI Kconfig symbols from the namespace and
  262. # CMakeCache.txt. If the symbols end up in DOTCONFIG they will be
  263. # re-introduced to the namespace through 'import_kconfig'.
  264. foreach (name ${cache_variable_names})
  265. if("${name}" MATCHES "^CONFIG_")
  266. unset(${name})
  267. unset(${name} CACHE)
  268. endif()
  269. endforeach()
  270. # Parse the lines prefixed with CONFIG_ in the .config file from Kconfig
  271. import_kconfig(CONFIG_ ${DOTCONFIG})
  272. # Re-introduce the CLI Kconfig symbols that survived
  273. foreach (name ${cache_variable_names})
  274. if("${name}" MATCHES "^CONFIG_")
  275. if(DEFINED ${name})
  276. set(${name} ${${name}} CACHE STRING "")
  277. endif()
  278. endif()
  279. endforeach()