dc85edd2e9
This is joint work with Kumar Gala (see signed-off-by). This supports a new devicetree macro syntax coming. It's not really worth mixing up the old and the new generation scripts into one file, because: - we aim to remove support for the old macros at some point, so it will be cleaner to start fresh with a new script based on the old one that only generates the new syntax - it will avoid regressions to leave the existing code alone while we're moving users to the new names Keep the existing script by moving it to gen_legacy_defines.py and changing a few comments and strings around. It's responsible for generating: - devicetree.conf: only needed by deprecated kconfigfunctions - devicetree_legacy_unfixed.h: "old" devicetree_unfixed.h macros Put a new gen_defines.py in its place. It generates: - zephyr.dts - devicetree_unfixed.h in the new syntax Include devicetree_legacy_unfixed.h from devicetree.h so no DT users are affected by this change. Signed-off-by: Martí Bolívar <marti.bolivar@nordicsemi.no> Signed-off-by: Kumar Gala <kumar.gala@linaro.org>
253 lines
7.8 KiB
CMake
253 lines
7.8 KiB
CMake
# 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, and a similar
|
|
# file with legacy contents called devicetree_unfixed_legacy.h.
|
|
#
|
|
# Here we call on dtc, the gcc preprocessor,
|
|
# scripts/dts/gen_defines.py, and scripts/dts/gen_legacy_defines.py to
|
|
# generate various DT-related files at CMake configure-time.
|
|
#
|
|
# The devicetree.conf file is still needed by some deprecated
|
|
# functions in kconfigfunctions.py.
|
|
#
|
|
# See the Devicetree user guide in the Zephyr documentation for details.
|
|
set(ZEPHYR_DTS ${PROJECT_BINARY_DIR}/zephyr.dts)
|
|
set(DEVICETREE_UNFIXED_H ${PROJECT_BINARY_DIR}/include/generated/devicetree_unfixed.h)
|
|
set(DEVICETREE_UNFIXED_LEGACY_H ${PROJECT_BINARY_DIR}/include/generated/devicetree_legacy_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 "Found BOARD.dts: ${dts_file}")
|
|
else()
|
|
message(STATUS "Found devicetree overlay: ${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 header file and zephyr.dts.
|
|
#
|
|
|
|
set(CMD_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}
|
|
--header-out ${DEVICETREE_UNFIXED_H}
|
|
--dts-out ${ZEPHYR_DTS} # As a debugging aid
|
|
)
|
|
|
|
#
|
|
# Run gen_legacy_defines.py to create a header file with legacy contents
|
|
# and a .conf file.
|
|
#
|
|
|
|
set(CMD_LEGACY_EXTRACT ${PYTHON_EXECUTABLE} ${ZEPHYR_BASE}/scripts/dts/gen_legacy_defines.py
|
|
--dts ${BOARD}.dts.pre.tmp
|
|
--dtc-flags '${EXTRA_DTC_FLAGS}'
|
|
--bindings-dirs ${DTS_ROOT_BINDINGS}
|
|
--header-out ${DEVICETREE_UNFIXED_LEGACY_H}
|
|
--conf-out ${DEVICETREE_CONF}
|
|
)
|
|
|
|
execute_process(
|
|
COMMAND ${CMD_EXTRACT}
|
|
WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
|
|
RESULT_VARIABLE ret
|
|
)
|
|
if(NOT "${ret}" STREQUAL "0")
|
|
message(FATAL_ERROR "gen_defines.py failed with return code: ${ret}")
|
|
else()
|
|
message(STATUS "Generated zephyr.dts: ${ZEPHYR_DTS}")
|
|
message(STATUS "Generated devicetree_unfixed.h: ${DEVICETREE_UNFIXED_H}")
|
|
endif()
|
|
|
|
execute_process(
|
|
COMMAND ${CMD_LEGACY_EXTRACT}
|
|
WORKING_DIRECTORY ${PROJECT_BINARY_DIR}
|
|
RESULT_VARIABLE ret
|
|
)
|
|
if(NOT "${ret}" STREQUAL "0")
|
|
message(FATAL_ERROR "gen_legacy_defines.py 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! */")
|
|
file(WRITE ${DEVICETREE_UNFIXED_LEGACY_H} "/* WARNING. THIS FILE IS AUTO-GENERATED. DO NOT MODIFY! */")
|
|
endif(SUPPORTS_DTS)
|