dts.cmake 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. # SPDX-License-Identifier: Apache-2.0
  2. file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/include/generated)
  3. # Zephyr code can configure itself based on a KConfig'uration with the
  4. # header file autoconf.h. There exists an analogous file devicetree_unfixed.h
  5. # that allows configuration based on information encoded in DTS.
  6. #
  7. # Here we call on dtc, the gcc preprocessor and
  8. # scripts/dts/gen_defines.py to generate various DT-related files at
  9. # CMake configure-time.
  10. #
  11. # See the Devicetree user guide in the Zephyr documentation for details.
  12. set(GEN_DEFINES_SCRIPT ${ZEPHYR_BASE}/scripts/dts/gen_defines.py)
  13. set(ZEPHYR_DTS ${PROJECT_BINARY_DIR}/zephyr.dts)
  14. # This contains the edtlib.EDT object created from zephyr.dts in Python's
  15. # pickle data marshalling format (https://docs.python.org/3/library/pickle.html)
  16. #
  17. # Its existence is an implementation detail used to speed up further
  18. # use of the devicetree by processes that run later on in the build,
  19. # and should not be made part of the documentation.
  20. set(EDT_PICKLE ${PROJECT_BINARY_DIR}/edt.pickle)
  21. set(DEVICETREE_UNFIXED_H ${PROJECT_BINARY_DIR}/include/generated/devicetree_unfixed.h)
  22. set(DEVICE_EXTERN_H ${PROJECT_BINARY_DIR}/include/generated/device_extern.h)
  23. set(DTS_POST_CPP ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.tmp)
  24. # The location of a list of known vendor prefixes.
  25. # This is relative to each element of DTS_ROOT.
  26. set(VENDOR_PREFIXES dts/bindings/vendor-prefixes.txt)
  27. # Devicetree in CMake.
  28. set(DTS_CMAKE_SCRIPT ${ZEPHYR_BASE}/scripts/dts/gen_dts_cmake.py)
  29. set(DTS_CMAKE ${PROJECT_BINARY_DIR}/dts.cmake)
  30. set_ifndef(DTS_SOURCE ${BOARD_DIR}/${BOARD}.dts)
  31. zephyr_file(APPLICATION_ROOT DTS_ROOT)
  32. # 'DTS_ROOT' is a list of directories where a directory tree with DT
  33. # files may be found. It always includes the application directory,
  34. # the board directory, and ${ZEPHYR_BASE}.
  35. list(APPEND
  36. DTS_ROOT
  37. ${APPLICATION_SOURCE_DIR}
  38. ${BOARD_DIR}
  39. ${SHIELD_DIRS}
  40. ${ZEPHYR_BASE}
  41. )
  42. list(REMOVE_DUPLICATES
  43. DTS_ROOT
  44. )
  45. # TODO: What to do about non-posix platforms where NOT CONFIG_HAS_DTS (xtensa)?
  46. # Drop support for NOT CONFIG_HAS_DTS perhaps?
  47. if(EXISTS ${DTS_SOURCE})
  48. set(SUPPORTS_DTS 1)
  49. if(BOARD_REVISION AND EXISTS ${BOARD_DIR}/${BOARD}_${BOARD_REVISION_STRING}.overlay)
  50. list(APPEND DTS_SOURCE ${BOARD_DIR}/${BOARD}_${BOARD_REVISION_STRING}.overlay)
  51. endif()
  52. else()
  53. set(SUPPORTS_DTS 0)
  54. endif()
  55. set(dts_files
  56. ${DTS_SOURCE}
  57. ${shield_dts_files}
  58. )
  59. if(SUPPORTS_DTS)
  60. if(DTC_OVERLAY_FILE)
  61. # Convert from space-separated files into file list
  62. string(REPLACE " " ";" DTC_OVERLAY_FILE_RAW_LIST "${DTC_OVERLAY_FILE}")
  63. foreach(file ${DTC_OVERLAY_FILE_RAW_LIST})
  64. file(TO_CMAKE_PATH "${file}" cmake_path_file)
  65. list(APPEND DTC_OVERLAY_FILE_AS_LIST ${cmake_path_file})
  66. endforeach()
  67. list(APPEND
  68. dts_files
  69. ${DTC_OVERLAY_FILE_AS_LIST}
  70. )
  71. endif()
  72. set(i 0)
  73. unset(DTC_INCLUDE_FLAG_FOR_DTS)
  74. foreach(dts_file ${dts_files})
  75. list(APPEND DTC_INCLUDE_FLAG_FOR_DTS
  76. -include ${dts_file})
  77. if(i EQUAL 0)
  78. message(STATUS "Found BOARD.dts: ${dts_file}")
  79. else()
  80. message(STATUS "Found devicetree overlay: ${dts_file}")
  81. endif()
  82. math(EXPR i "${i}+1")
  83. endforeach()
  84. unset(DTS_ROOT_SYSTEM_INCLUDE_DIRS)
  85. unset(DTS_ROOT_BINDINGS)
  86. foreach(dts_root ${DTS_ROOT})
  87. foreach(dts_root_path
  88. include
  89. dts/common
  90. dts/${ARCH}
  91. dts
  92. )
  93. get_filename_component(full_path ${dts_root}/${dts_root_path} REALPATH)
  94. if(EXISTS ${full_path})
  95. list(APPEND
  96. DTS_ROOT_SYSTEM_INCLUDE_DIRS
  97. -isystem ${full_path}
  98. )
  99. endif()
  100. endforeach()
  101. set(bindings_path ${dts_root}/dts/bindings)
  102. if(EXISTS ${bindings_path})
  103. list(APPEND
  104. DTS_ROOT_BINDINGS
  105. ${bindings_path}
  106. )
  107. endif()
  108. set(vendor_prefixes ${dts_root}/${VENDOR_PREFIXES})
  109. if(EXISTS ${vendor_prefixes})
  110. list(APPEND EXTRA_GEN_DEFINES_ARGS --vendor-prefixes ${vendor_prefixes})
  111. endif()
  112. endforeach()
  113. # Cache the location of the root bindings so they can be used by
  114. # scripts which use the build directory.
  115. set(CACHED_DTS_ROOT_BINDINGS ${DTS_ROOT_BINDINGS} CACHE INTERNAL
  116. "DT bindings root directories")
  117. if(NOT DEFINED CMAKE_DTS_PREPROCESSOR)
  118. set(CMAKE_DTS_PREPROCESSOR ${CMAKE_C_COMPILER})
  119. endif()
  120. # TODO: Cut down on CMake configuration time by avoiding
  121. # regeneration of devicetree_unfixed.h on every configure. How
  122. # challenging is this? What are the dts dependencies? We run the
  123. # preprocessor, and it seems to be including all kinds of
  124. # directories with who-knows how many header files.
  125. # Run the C preprocessor on an empty C source file that has one or
  126. # more DTS source files -include'd into it to create the
  127. # intermediary file *.dts.pre.tmp. Also, generate a dependency file
  128. # so that changes to DT sources are detected.
  129. execute_process(
  130. COMMAND ${CMAKE_DTS_PREPROCESSOR}
  131. -x assembler-with-cpp
  132. -nostdinc
  133. ${DTS_ROOT_SYSTEM_INCLUDE_DIRS}
  134. ${DTC_INCLUDE_FLAG_FOR_DTS} # include the DTS source and overlays
  135. ${NOSYSDEF_CFLAG}
  136. -D__DTS__
  137. ${DTS_EXTRA_CPPFLAGS}
  138. -P
  139. -E # Stop after preprocessing
  140. -MD # Generate a dependency file as a side-effect
  141. -MF ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.d
  142. -o ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.tmp
  143. ${ZEPHYR_BASE}/misc/empty_file.c
  144. WORKING_DIRECTORY ${APPLICATION_SOURCE_DIR}
  145. RESULT_VARIABLE ret
  146. )
  147. if(NOT "${ret}" STREQUAL "0")
  148. message(FATAL_ERROR "command failed with return code: ${ret}")
  149. endif()
  150. # Parse the generated dependency file to find the DT sources that
  151. # were included and then add them to the list of files that trigger
  152. # a re-run of CMake.
  153. toolchain_parse_make_rule(
  154. ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.d
  155. include_files # Output parameter
  156. )
  157. set_property(DIRECTORY APPEND PROPERTY
  158. CMAKE_CONFIGURE_DEPENDS
  159. ${include_files}
  160. ${GEN_DEFINES_SCRIPT}
  161. ${DTS_CMAKE_SCRIPT}
  162. )
  163. #
  164. # Run the C devicetree compiler on *.dts.pre.tmp, just to catch any
  165. # warnings/errors from it. dtlib and edtlib parse the devicetree files
  166. # themselves, so we don't rely on the C compiler otherwise.
  167. #
  168. if(DTC)
  169. set(DTC_WARN_UNIT_ADDR_IF_ENABLED "")
  170. check_dtc_flag("-Wunique_unit_address_if_enabled" check)
  171. if (check)
  172. set(DTC_WARN_UNIT_ADDR_IF_ENABLED "-Wunique_unit_address_if_enabled")
  173. endif()
  174. set(DTC_NO_WARN_UNIT_ADDR "")
  175. check_dtc_flag("-Wno-unique_unit_address" check)
  176. if (check)
  177. set(DTC_NO_WARN_UNIT_ADDR "-Wno-unique_unit_address")
  178. endif()
  179. set(VALID_EXTRA_DTC_FLAGS "")
  180. foreach(extra_opt ${EXTRA_DTC_FLAGS})
  181. check_dtc_flag(${extra_opt} check)
  182. if (check)
  183. list(APPEND VALID_EXTRA_DTC_FLAGS ${extra_opt})
  184. endif()
  185. endforeach()
  186. set(EXTRA_DTC_FLAGS ${VALID_EXTRA_DTC_FLAGS})
  187. execute_process(
  188. COMMAND ${DTC}
  189. -O dts
  190. -o - # Write output to stdout, which we discard below
  191. -b 0
  192. -E unit_address_vs_reg
  193. ${DTC_NO_WARN_UNIT_ADDR}
  194. ${DTC_WARN_UNIT_ADDR_IF_ENABLED}
  195. ${EXTRA_DTC_FLAGS} # User settable
  196. ${BOARD}.dts.pre.tmp
  197. OUTPUT_QUIET # Discard stdout
  198. WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
  199. RESULT_VARIABLE ret
  200. )
  201. if(NOT "${ret}" STREQUAL "0")
  202. message(FATAL_ERROR "command failed with return code: ${ret}")
  203. endif()
  204. endif(DTC)
  205. #
  206. # Run gen_defines.py to create a header file, zephyr.dts, and edt.pickle.
  207. #
  208. string(REPLACE ";" " " EXTRA_DTC_FLAGS_RAW "${EXTRA_DTC_FLAGS}")
  209. set(CMD_EXTRACT ${PYTHON_EXECUTABLE} ${GEN_DEFINES_SCRIPT}
  210. --dts ${BOARD}.dts.pre.tmp
  211. --dtc-flags '${EXTRA_DTC_FLAGS_RAW}'
  212. --bindings-dirs ${DTS_ROOT_BINDINGS}
  213. --header-out ${DEVICETREE_UNFIXED_H}
  214. --device-header-out ${DEVICE_EXTERN_H}
  215. --dts-out ${ZEPHYR_DTS} # As a debugging aid
  216. --edt-pickle-out ${EDT_PICKLE}
  217. ${EXTRA_GEN_DEFINES_ARGS}
  218. )
  219. execute_process(
  220. COMMAND ${CMD_EXTRACT}
  221. WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
  222. RESULT_VARIABLE ret
  223. )
  224. if(NOT "${ret}" STREQUAL "0")
  225. message(FATAL_ERROR "gen_defines.py failed with return code: ${ret}")
  226. else()
  227. message(STATUS "Generated zephyr.dts: ${ZEPHYR_DTS}")
  228. message(STATUS "Generated devicetree_unfixed.h: ${DEVICETREE_UNFIXED_H}")
  229. message(STATUS "Generated device_extern.h: ${DEVICE_EXTERN_H}")
  230. endif()
  231. execute_process(
  232. COMMAND ${PYTHON_EXECUTABLE} ${DTS_CMAKE_SCRIPT}
  233. --edt-pickle ${EDT_PICKLE}
  234. --cmake-out ${DTS_CMAKE}
  235. WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
  236. RESULT_VARIABLE ret
  237. )
  238. if(NOT "${ret}" STREQUAL "0")
  239. message(FATAL_ERROR "gen_dts_cmake.py failed with return code: ${ret}")
  240. else()
  241. message(STATUS "Including generated dts.cmake file: ${DTS_CMAKE}")
  242. include(${DTS_CMAKE})
  243. endif()
  244. else()
  245. file(WRITE ${DEVICETREE_UNFIXED_H} "/* WARNING. THIS FILE IS AUTO-GENERATED. DO NOT MODIFY! */")
  246. file(WRITE ${DEVICE_EXTERN_H} "/* WARNING. THIS FILE IS AUTO-GENERATED. DO NOT MODIFY! */")
  247. endif(SUPPORTS_DTS)