zephyr_module.cmake 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. # SPDX-License-Identifier: Apache-2.0
  2. # This cmake file provides functionality to import CMakeLists.txt and Kconfig
  3. # files for Zephyr modules into Zephyr build system.
  4. #
  5. # CMakeLists.txt and Kconfig files can reside directly in the module or in a
  6. # MODULE_EXT_ROOT.
  7. # The `<module>/zephyr/module.yml` file specifies whether the build files are
  8. # located in the module or in a MODULE_EXT_ROOT.
  9. #
  10. # A list of Zephyr modules can be provided to the build system using:
  11. # -DZEPHYR_MODULES=<module-path>[;<additional-module(s)-path>]
  12. #
  13. # It looks for: <module>/zephyr/module.yml or
  14. # <module>/zephyr/CMakeLists.txt
  15. # to load the module into Zephyr build system.
  16. # If west is available, it uses `west list` to obtain a list of projects to
  17. # search for zephyr/module.yml
  18. #
  19. # If the module.yml file specifies that build files are located in a
  20. # MODULE_EXT_ROOT then the variables:
  21. # - `ZEPHYR_<MODULE_NAME>_CMAKE_DIR` is used for inclusion of the CMakeLists.txt
  22. # - `ZEPHYR_<MODULE_NAME>_KCONFIG` is used for inclusion of the Kconfig
  23. # files into the build system.
  24. if(ZEPHYR_MODULES)
  25. set(ZEPHYR_MODULES_ARG "--modules" ${ZEPHYR_MODULES})
  26. endif()
  27. if(ZEPHYR_EXTRA_MODULES)
  28. set(ZEPHYR_EXTRA_MODULES_ARG "--extra-modules" ${ZEPHYR_EXTRA_MODULES})
  29. endif()
  30. set(KCONFIG_MODULES_FILE ${KCONFIG_BINARY_DIR}/Kconfig.modules)
  31. set(ZEPHYR_SETTINGS_FILE ${CMAKE_BINARY_DIR}/zephyr_settings.txt)
  32. if(WEST)
  33. set(WEST_ARG "--zephyr-base" ${ZEPHYR_BASE})
  34. endif()
  35. if(WEST OR ZEPHYR_MODULES)
  36. # Zephyr module uses west, so only call it if west is installed or
  37. # ZEPHYR_MODULES was provided as argument to CMake.
  38. execute_process(
  39. COMMAND
  40. ${PYTHON_EXECUTABLE} ${ZEPHYR_BASE}/scripts/zephyr_module.py
  41. ${WEST_ARG}
  42. ${ZEPHYR_MODULES_ARG}
  43. ${ZEPHYR_EXTRA_MODULES_ARG}
  44. --kconfig-out ${KCONFIG_MODULES_FILE}
  45. --cmake-out ${CMAKE_BINARY_DIR}/zephyr_modules.txt
  46. --settings-out ${ZEPHYR_SETTINGS_FILE}
  47. WORKING_DIRECTORY ${ZEPHYR_BASE}
  48. ERROR_VARIABLE
  49. zephyr_module_error_text
  50. RESULT_VARIABLE
  51. zephyr_module_return
  52. )
  53. if(${zephyr_module_return})
  54. message(FATAL_ERROR "${zephyr_module_error_text}")
  55. endif()
  56. if(EXISTS ${ZEPHYR_SETTINGS_FILE})
  57. file(STRINGS ${ZEPHYR_SETTINGS_FILE} ZEPHYR_SETTINGS_TXT ENCODING UTF-8 REGEX "^[^#]")
  58. foreach(setting ${ZEPHYR_SETTINGS_TXT})
  59. # Match <key>:<value> for each line of file, each corresponding to
  60. # a setting. The use of quotes is required due to CMake not supporting
  61. # lazy regexes (it supports greedy only).
  62. string(REGEX REPLACE "\"(.*)\":\".*\"" "\\1" key ${setting})
  63. string(REGEX REPLACE "\".*\":\"(.*)\"" "\\1" value ${setting})
  64. # MODULE_EXT_ROOT is process order which means module roots processed
  65. # later wins. To ensure ZEPHYR_BASE stays first, and command line settings
  66. # are processed last, we insert at position 1.
  67. if ("${key}" STREQUAL "MODULE_EXT_ROOT")
  68. list(INSERT ${key} 1 ${value})
  69. else()
  70. list(APPEND ${key} ${value})
  71. endif()
  72. endforeach()
  73. endif()
  74. foreach(root ${MODULE_EXT_ROOT})
  75. if(NOT EXISTS ${root})
  76. message(FATAL_ERROR "No `modules.cmake` found in module root `${root}`.")
  77. endif()
  78. include(${root}/modules/modules.cmake)
  79. endforeach()
  80. if(EXISTS ${CMAKE_BINARY_DIR}/zephyr_modules.txt)
  81. file(STRINGS ${CMAKE_BINARY_DIR}/zephyr_modules.txt ZEPHYR_MODULES_TXT
  82. ENCODING UTF-8)
  83. set(ZEPHYR_MODULE_NAMES)
  84. foreach(module ${ZEPHYR_MODULES_TXT})
  85. # Match "<name>":"<path>" for each line of file, each corresponding to
  86. # one module. The use of quotes is required due to CMake not supporting
  87. # lazy regexes (it supports greedy only).
  88. string(CONFIGURE ${module} module)
  89. string(REGEX REPLACE "\"(.*)\":\".*\":\".*\"" "\\1" module_name ${module})
  90. string(REGEX REPLACE "\".*\":\"(.*)\":\".*\"" "\\1" module_path ${module})
  91. string(REGEX REPLACE "\".*\":\".*\":\"(.*)\"" "\\1" cmake_path ${module})
  92. list(APPEND ZEPHYR_MODULE_NAMES ${module_name})
  93. zephyr_string(SANITIZE TOUPPER MODULE_NAME_UPPER ${module_name})
  94. if(NOT ${MODULE_NAME_UPPER} STREQUAL CURRENT)
  95. set(ZEPHYR_${MODULE_NAME_UPPER}_MODULE_DIR ${module_path})
  96. set(ZEPHYR_${MODULE_NAME_UPPER}_CMAKE_DIR ${cmake_path})
  97. else()
  98. message(FATAL_ERROR "Found Zephyr module named: ${module_name}\n\
  99. ${MODULE_NAME_UPPER} is a restricted name for Zephyr modules as it is used for \
  100. \${ZEPHYR_${MODULE_NAME_UPPER}_MODULE_DIR} CMake variable.")
  101. endif()
  102. endforeach()
  103. endif()
  104. else()
  105. file(WRITE ${KCONFIG_MODULES_FILE}
  106. "# No west and no modules\n"
  107. )
  108. endif()
  109. if(DEFINED ZEPHYR_MODULE_NAMES)
  110. list(REMOVE_DUPLICATES ZEPHYR_MODULE_NAMES)
  111. endif()