# SPDX-License-Identifier: Apache-2.0 file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/include/generated) # Zephyr code can configure itself based on a KConfig'uration with the # header file autoconf.h. There exists an analogous file devicetree_unfixed.h # that allows configuration based on information encoded in DTS. # # Here we call on dtc, the gcc preprocessor, and # scripts/dts/gen_defines.py to generate this header file at # CMake configure-time. # # See ~/zephyr/doc/dts set(DEVICETREE_UNFIXED_H ${PROJECT_BINARY_DIR}/include/generated/devicetree_unfixed.h) set(DEVICETREE_CONF ${PROJECT_BINARY_DIR}/include/generated/devicetree.conf) set(DTS_POST_CPP ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.tmp) set_ifndef(DTS_SOURCE ${BOARD_DIR}/${BOARD}.dts) if(DEFINED DTS_COMMON_OVERLAYS) # TODO: remove this warning in version 1.16 message(FATAL_ERROR "DTS_COMMON_OVERLAYS is no longer supported. Use DTC_OVERLAY_FILE instead.") endif() # 'DTS_ROOT' is a list of directories where a directory tree with DT # files may be found. It always includes the application directory, # the board directory, and ${ZEPHYR_BASE}. list(APPEND DTS_ROOT ${APPLICATION_SOURCE_DIR} ${BOARD_DIR} ${ZEPHYR_BASE} ) list(REMOVE_DUPLICATES DTS_ROOT ) list(REMOVE_DUPLICATES DTS_ROOT) set(dts_files ${DTS_SOURCE} ${shield_dts_files} ) # TODO: What to do about non-posix platforms where NOT CONFIG_HAS_DTS (xtensa)? # Drop support for NOT CONFIG_HAS_DTS perhaps? if(EXISTS ${DTS_SOURCE}) set(SUPPORTS_DTS 1) else() set(SUPPORTS_DTS 0) endif() if(SUPPORTS_DTS) if(DTC_OVERLAY_FILE) # Convert from space-separated files into file list string(REPLACE " " ";" DTC_OVERLAY_FILE_AS_LIST ${DTC_OVERLAY_FILE}) list(APPEND dts_files ${DTC_OVERLAY_FILE_AS_LIST} ) endif() set(i 0) unset(DTC_INCLUDE_FLAG_FOR_DTS) foreach(dts_file ${dts_files}) list(APPEND DTC_INCLUDE_FLAG_FOR_DTS -include ${dts_file}) if(i EQUAL 0) message(STATUS "Loading ${dts_file} as base") else() message(STATUS "Overlaying ${dts_file}") endif() math(EXPR i "${i}+1") endforeach() unset(DTS_ROOT_SYSTEM_INCLUDE_DIRS) foreach(dts_root ${DTS_ROOT}) foreach(dts_root_path include dts/common dts/${ARCH} dts ) set(full_path ${dts_root}/${dts_root_path}) if(EXISTS ${full_path}) list(APPEND DTS_ROOT_SYSTEM_INCLUDE_DIRS -isystem ${full_path} ) endif() endforeach() endforeach() unset(DTS_ROOT_BINDINGS) foreach(dts_root ${DTS_ROOT}) set(full_path ${dts_root}/dts/bindings) if(EXISTS ${full_path}) list(APPEND DTS_ROOT_BINDINGS ${full_path} ) endif() endforeach() # Cache the location of the root bindings so they can be used by # scripts which use the build directory. set(CACHED_DTS_ROOT_BINDINGS ${DTS_ROOT_BINDINGS} CACHE INTERNAL "DT bindings root directories") # TODO: Cut down on CMake configuration time by avoiding # regeneration of devicetree_unfixed.h on every configure. How # challenging is this? What are the dts dependencies? We run the # preprocessor, and it seems to be including all kinds of # directories with who-knows how many header files. # Run the C preprocessor on an empty C source file that has one or # more DTS source files -include'd into it to create the # intermediary file *.dts.pre.tmp. Also, generate a dependency file # so that changes to DT sources are detected. execute_process( COMMAND ${CMAKE_C_COMPILER} -x assembler-with-cpp -nostdinc ${DTS_ROOT_SYSTEM_INCLUDE_DIRS} ${DTC_INCLUDE_FLAG_FOR_DTS} # include the DTS source and overlays ${NOSYSDEF_CFLAG} -D__DTS__ -P -E # Stop after preprocessing -MD # Generate a dependency file as a side-effect -MF ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.d -o ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.tmp ${ZEPHYR_BASE}/misc/empty_file.c WORKING_DIRECTORY ${APPLICATION_SOURCE_DIR} RESULT_VARIABLE ret ) if(NOT "${ret}" STREQUAL "0") message(FATAL_ERROR "command failed with return code: ${ret}") endif() # Parse the generated dependency file to find the DT sources that # were included and then add them to the list of files that trigger # a re-run of CMake. toolchain_parse_make_rule( ${PROJECT_BINARY_DIR}/${BOARD}.dts.pre.d include_files # Output parameter ) set_property(DIRECTORY APPEND PROPERTY CMAKE_CONFIGURE_DEPENDS ${include_files} ) # # Run the C devicetree compiler on *.dts.pre.tmp, just to catch any # warnings/errors from it. dtlib and edtlib parse the devicetree files # themselves, so we don't rely on the C compiler otherwise. # if(DTC) set(DTC_WARN_UNIT_ADDR_IF_ENABLED "") check_dtc_flag("-Wunique_unit_address_if_enabled" check) if (check) set(DTC_WARN_UNIT_ADDR_IF_ENABLED "-Wunique_unit_address_if_enabled") endif() set(DTC_NO_WARN_UNIT_ADDR "") check_dtc_flag("-Wno-unique_unit_address" check) if (check) set(DTC_NO_WARN_UNIT_ADDR "-Wno-unique_unit_address") endif() execute_process( COMMAND ${DTC} -O dts -o - # Write output to stdout, which we discard below -b 0 -E unit_address_vs_reg ${DTC_NO_WARN_UNIT_ADDR} ${DTC_WARN_UNIT_ADDR_IF_ENABLED} ${EXTRA_DTC_FLAGS} # User settable ${BOARD}.dts.pre.tmp OUTPUT_QUIET # Discard stdout WORKING_DIRECTORY ${PROJECT_BINARY_DIR} RESULT_VARIABLE ret ) if(NOT "${ret}" STREQUAL "0") message(FATAL_ERROR "command failed with return code: ${ret}") endif() endif(DTC) # # Run gen_defines.py to create a .conf file and a header file # set(CMD_NEW_EXTRACT ${PYTHON_EXECUTABLE} ${ZEPHYR_BASE}/scripts/dts/gen_defines.py --dts ${BOARD}.dts.pre.tmp --dtc-flags '${EXTRA_DTC_FLAGS}' --bindings-dirs ${DTS_ROOT_BINDINGS} --conf-out ${DEVICETREE_CONF} --header-out ${DEVICETREE_UNFIXED_H} --dts-out ${PROJECT_BINARY_DIR}/zephyr.dts # As a debugging aid ) execute_process( COMMAND ${CMD_NEW_EXTRACT} WORKING_DIRECTORY ${PROJECT_BINARY_DIR} RESULT_VARIABLE ret ) if(NOT "${ret}" STREQUAL "0") message(FATAL_ERROR "new extractor failed with return code: ${ret}") endif() # A file that used to be generated by 'dtc'. zephyr.dts is the new # equivalent. Will be removed in Zephyr 2.3. file(WRITE ${PROJECT_BINARY_DIR}/${BOARD}.dts_compiled "See zephyr.dts for the final merged devicetree.") else() file(WRITE ${DEVICETREE_UNFIXED_H} "/* WARNING. THIS FILE IS AUTO-GENERATED. DO NOT MODIFY! */") endif(SUPPORTS_DTS)