/* * Copyright (c) 2021, Commonwealth Scientific and Industrial Research * Organisation (CSIRO) ABN 41 687 119 230. * * SPDX-License-Identifier: Apache-2.0 * * Generate memory regions from devicetree nodes. */ #ifndef ZEPHYR_INCLUDE_LINKER_DEVICETREE_REGIONS_H_ #define ZEPHYR_INCLUDE_LINKER_DEVICETREE_REGIONS_H_ #include #include #include /** * @brief Get the linker memory-region name in a token form * * This attempts to use the zephyr,memory-region property (with * non-alphanumeric characters replaced with underscores) returning a token. * * Example devicetree fragment: * * @code{.dts} * / { * soc { * sram1: memory@2000000 { * zephyr,memory-region = "MY_NAME"; * }; * sram2: memory@2001000 { * zephyr,memory-region = "MY@OTHER@NAME"; * }; * }; * }; * @endcode * * Example usage: * * @code{.c} * LINKER_DT_NODE_REGION_NAME_TOKEN(DT_NODELABEL(sram1)) // MY_NAME * LINKER_DT_NODE_REGION_NAME_TOKEN(DT_NODELABEL(sram2)) // MY_OTHER_NAME * @endcode * * @param node_id node identifier * @return the name of the memory memory region the node will generate */ #define LINKER_DT_NODE_REGION_NAME_TOKEN(node_id) \ DT_STRING_TOKEN(node_id, zephyr_memory_region) /** * @brief Get the linker memory-region name * * This attempts to use the zephyr,memory-region property (with * non-alphanumeric characters replaced with underscores). * * Example devicetree fragment: * * @code{.dts} * / { * soc { * sram1: memory@2000000 { * zephyr,memory-region = "MY_NAME"; * }; * sram2: memory@2001000 { * zephyr,memory-region = "MY@OTHER@NAME"; * }; * }; * }; * @endcode * * Example usage: * * @code{.c} * LINKER_DT_NODE_REGION_NAME(DT_NODELABEL(sram1)) // "MY_NAME" * LINKER_DT_NODE_REGION_NAME(DT_NODELABEL(sram2)) // "MY_OTHER_NAME" * @endcode * * @param node_id node identifier * @return the name of the memory memory region the node will generate */ #define LINKER_DT_NODE_REGION_NAME(node_id) \ STRINGIFY(LINKER_DT_NODE_REGION_NAME_TOKEN(node_id)) #define _DT_MEMORY_REGION_FLAGS_TOKEN(n) DT_STRING_TOKEN(n, zephyr_memory_region_flags) #define _DT_MEMORY_REGION_FLAGS_UNQUOTED(n) DT_STRING_UNQUOTED(n, zephyr_memory_region_flags) #define _LINKER_L_PAREN ( #define _LINKER_R_PAREN ) #define _LINKER_ENCLOSE_PAREN(x) _LINKER_L_PAREN x _LINKER_R_PAREN #define _LINKER_IS_EMPTY_TOKEN_ 1 #define _LINKER_IS_EMPTY_TOKEN_EXPAND(x) _LINKER_IS_EMPTY_TOKEN_##x #define _LINKER_IS_EMPTY_TOKEN(x) _LINKER_IS_EMPTY_TOKEN_EXPAND(x) /** * @brief Get the linker memory-region flags with parentheses. * * This attempts to return the zephyr,memory-region-flags property * with parentheses. * Return empty string if not set the property. * * Example devicetree fragment: * * @code{.dts} * / { * soc { * rx: memory@2000000 { * zephyr,memory-region = "READ_EXEC"; * zephyr,memory-region-flags = "rx"; * }; * rx_not_w: memory@2001000 { * zephyr,memory-region = "READ_EXEC_NOT_WRITE"; * zephyr,memory-region-flags = "rx!w"; * }; * no_flags: memory@2001000 { * zephyr,memory-region = "NO_FLAGS"; * }; * }; * }; * @endcode * * Example usage: * * @code{.c} * LINKER_DT_NODE_REGION_FLAGS(DT_NODELABEL(rx)) // (rx) * LINKER_DT_NODE_REGION_FLAGS(DT_NODELABEL(rx_not_w)) // (rx!w) * LINKER_DT_NODE_REGION_FLAGS(DT_NODELABEL(no_flags)) // [flags will not be specified] * @endcode * * @param node_id node identifier * @return the value of the memory region flag specified in the device tree * enclosed in parentheses. */ #define LINKER_DT_NODE_REGION_FLAGS(node_id) \ COND_CODE_1(DT_NODE_HAS_PROP(node_id, zephyr_memory_region_flags), \ (COND_CODE_1(_LINKER_IS_EMPTY_TOKEN(_DT_MEMORY_REGION_FLAGS_TOKEN(node_id)), \ (), \ (_LINKER_ENCLOSE_PAREN( \ _DT_MEMORY_REGION_FLAGS_UNQUOTED(node_id)) \ ))), \ (_LINKER_ENCLOSE_PAREN(rw))) /** @cond INTERNAL_HIDDEN */ #define _DT_COMPATIBLE zephyr_memory_region #define _DT_SECTION_PREFIX(node_id) UTIL_CAT(__, LINKER_DT_NODE_REGION_NAME_TOKEN(node_id)) #define _DT_SECTION_START(node_id) UTIL_CAT(_DT_SECTION_PREFIX(node_id), _start) #define _DT_SECTION_END(node_id) UTIL_CAT(_DT_SECTION_PREFIX(node_id), _end) #define _DT_SECTION_SIZE(node_id) UTIL_CAT(_DT_SECTION_PREFIX(node_id), _size) #define _DT_SECTION_LOAD(node_id) UTIL_CAT(_DT_SECTION_PREFIX(node_id), _load_start) /** * @brief Declare a memory region * * Example devicetree fragment: * * @code{.dts} * test_sram: sram@20010000 { * compatible = "zephyr,memory-region", "mmio-sram"; * reg = < 0x20010000 0x1000 >; * zephyr,memory-region = "FOOBAR"; * zephyr,memory-region-flags = "rw"; * }; * @endcode * * will result in: * * @code{.unparsed} * FOOBAR (rw) : ORIGIN = (0x20010000), LENGTH = (0x1000) * @endcode * * @param node_id devicetree node identifier * @param attr region attributes */ #define _REGION_DECLARE(node_id) \ LINKER_DT_NODE_REGION_NAME_TOKEN(node_id) \ LINKER_DT_NODE_REGION_FLAGS(node_id) \ : ORIGIN = DT_REG_ADDR(node_id), \ LENGTH = DT_REG_SIZE(node_id) /** * @brief Declare a memory section from the device tree nodes with * compatible 'zephyr,memory-region' * * Example devicetree fragment: * * @code{.dts} * test_sram: sram@20010000 { * compatible = "zephyr,memory-region", "mmio-sram"; * reg = < 0x20010000 0x1000 >; * zephyr,memory-region = "FOOBAR"; * }; * @endcode * * will result in: * * @code{.unparsed} * FOOBAR (NOLOAD) : * { * __FOOBAR_start = .; * KEEP(*(FOOBAR)) * KEEP(*(FOOBAR.*)) * __FOOBAR_end = .; * } > FOOBAR * __FOOBAR_size = __FOOBAR_end - __FOOBAR_start; * __FOOBAR_load_start = LOADADDR(FOOBAR); * @endcode * * @param node_id devicetree node identifier */ #define _SECTION_DECLARE(node_id) \ LINKER_DT_NODE_REGION_NAME_TOKEN(node_id) (NOLOAD) : \ { \ _DT_SECTION_START(node_id) = .; \ KEEP(*(LINKER_DT_NODE_REGION_NAME_TOKEN(node_id))) \ KEEP(*(LINKER_DT_NODE_REGION_NAME_TOKEN(node_id).*)) \ _DT_SECTION_END(node_id) = .; \ } > LINKER_DT_NODE_REGION_NAME_TOKEN(node_id) \ _DT_SECTION_SIZE(node_id) = _DT_SECTION_END(node_id) - _DT_SECTION_START(node_id); \ _DT_SECTION_LOAD(node_id) = LOADADDR(LINKER_DT_NODE_REGION_NAME_TOKEN(node_id)); /** @endcond */ /** * @brief Generate linker memory regions from the device tree nodes with * compatible 'zephyr,memory-region' * * Note: for now we do not deal with MEMORY attributes since those are * optional, not actually used by Zephyr and they will likely conflict with the * MPU configuration. */ #define LINKER_DT_REGIONS() \ DT_FOREACH_STATUS_OKAY(_DT_COMPATIBLE, _REGION_DECLARE) /** * @brief Generate linker memory sections from the device tree nodes with * compatible 'zephyr,memory-region' */ #define LINKER_DT_SECTIONS() \ DT_FOREACH_STATUS_OKAY(_DT_COMPATIBLE, _SECTION_DECLARE) #endif /* ZEPHYR_INCLUDE_LINKER_DEVICETREE_REGIONS_H_ */