devicetree_reserved.h 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*
  2. * Copyright (c) 2021, Carlo Caione <ccaione@baylibre.com>
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Generate memory regions and sections from reserved-memory nodes.
  7. */
  8. #include <devicetree.h>
  9. /* Reserved memory node */
  10. #define _NODE_RESERVED DT_INST(0, reserved_memory)
  11. /* Unquoted region label */
  12. #define _DT_LABEL_TOKEN(res) DT_STRING_TOKEN(res, label)
  13. /* _start and _end section symbols */
  14. #define _DT_RESERVED_PREFIX(res) UTIL_CAT(__, _DT_LABEL_TOKEN(res))
  15. #define _DT_RESERVED_START(res) UTIL_CAT(_DT_RESERVED_PREFIX(res), _start)
  16. #define _DT_RESERVED_END(res) UTIL_CAT(_DT_RESERVED_PREFIX(res), _end)
  17. /* Declare a reserved memory region */
  18. #define _RESERVED_REGION_DECLARE(res) DT_STRING_TOKEN(res, label) (rw) : \
  19. ORIGIN = DT_REG_ADDR(res), \
  20. LENGTH = DT_REG_SIZE(res)
  21. /* Declare a reserved memory section */
  22. #define _RESERVED_SECTION_DECLARE(res) SECTION_DATA_PROLOGUE(_DT_LABEL_TOKEN(res), ,) \
  23. { \
  24. _DT_RESERVED_START(res) = .; \
  25. KEEP(*(._DT_LABEL_TOKEN(res))) \
  26. KEEP(*(._DT_LABEL_TOKEN(res).*)) \
  27. _DT_RESERVED_END(res) = \
  28. _DT_RESERVED_START(res) + DT_REG_SIZE(res); \
  29. } GROUP_LINK_IN(_DT_LABEL_TOKEN(res))
  30. /* Declare reserved memory linker symbols */
  31. #define _RESERVED_SYMBOL_DECLARE(res) extern char _DT_RESERVED_START(res)[]; \
  32. extern char _DT_RESERVED_END(res)[];
  33. /* Apply a macro to a reserved memory region */
  34. #define _RESERVED_REGION_APPLY(f) \
  35. COND_CODE_1(IS_ENABLED(UTIL_CAT(_NODE_RESERVED, _EXISTS)), \
  36. (DT_FOREACH_CHILD(_NODE_RESERVED, f)), ())
  37. /**
  38. * @brief Generate region definitions for all the reserved memory regions
  39. */
  40. #define LINKER_DT_RESERVED_MEM_REGIONS() _RESERVED_REGION_APPLY(_RESERVED_REGION_DECLARE)
  41. /**
  42. * @brief Generate section definitions for all the reserved memory regions
  43. */
  44. #define LINKER_DT_RESERVED_MEM_SECTIONS() _RESERVED_REGION_APPLY(_RESERVED_SECTION_DECLARE)
  45. /**
  46. * @brief Generate linker script symbols for all the reserved memory regions
  47. */
  48. #define LINKER_DT_RESERVED_MEM_SYMBOLS() _RESERVED_REGION_APPLY(_RESERVED_SYMBOL_DECLARE)
  49. /**
  50. * @brief Get the pointer to the reserved-memory region
  51. *
  52. * Example devicetree fragment:
  53. *
  54. * reserved: reserved-memory {
  55. * compatible = "reserved-memory";
  56. * ...
  57. * n: node {
  58. * reg = <0x42000000 0x1000>;
  59. * };
  60. * };
  61. *
  62. * Example usage:
  63. *
  64. * LINKER_DT_RESERVED_MEM_GET_PTR(DT_NODELABEL(n)) // (uint8_t *) 0x42000000
  65. *
  66. * @param node_id node identifier
  67. * @return pointer to the beginning of the reserved-memory region
  68. */
  69. #define LINKER_DT_RESERVED_MEM_GET_PTR(node_id) _DT_RESERVED_START(node_id)
  70. /**
  71. * @brief Get the size of the reserved-memory region
  72. *
  73. * Example devicetree fragment:
  74. *
  75. * reserved: reserved-memory {
  76. * compatible = "reserved-memory";
  77. * ...
  78. * n: node {
  79. * reg = <0x42000000 0x1000>;
  80. * };
  81. * };
  82. *
  83. * Example usage:
  84. *
  85. * LINKER_DT_RESERVED_MEM_GET_SIZE(DT_NODELABEL(n)) // 0x1000
  86. *
  87. * @param node_id node identifier
  88. * @return the size of the reserved-memory region
  89. */
  90. #define LINKER_DT_RESERVED_MEM_GET_SIZE(node_id) DT_REG_SIZE(node_id)
  91. /**
  92. * @brief Get the pointer to the reserved-memory region from a memory-reserved
  93. * phandle
  94. *
  95. * Example devicetree fragment:
  96. *
  97. * reserved: reserved-memory {
  98. * compatible = "reserved-memory";
  99. * ...
  100. * res0: res {
  101. * reg = <0x42000000 0x1000>;
  102. * label = "res0";
  103. * };
  104. * };
  105. *
  106. * n: node {
  107. * memory-region = <&res0>;
  108. * };
  109. *
  110. * Example usage:
  111. *
  112. * LINKER_DT_RESERVED_MEM_GET_PTR_BY_PHANDLE(DT_NODELABEL(n), \
  113. * memory_region) // (uint8_t *) 0x42000000
  114. *
  115. * @param node_id node identifier
  116. * @param ph phandle to reserved-memory region
  117. *
  118. * @return pointer to the beginning of the reserved-memory region
  119. */
  120. #define LINKER_DT_RESERVED_MEM_GET_PTR_BY_PHANDLE(node_id, ph) \
  121. LINKER_DT_RESERVED_MEM_GET_PTR(DT_PHANDLE(node_id, ph))
  122. /**
  123. * @brief Get the size of the reserved-memory region from a memory-reserved
  124. * phandle
  125. *
  126. * Example devicetree fragment:
  127. *
  128. * reserved: reserved-memory {
  129. * compatible = "reserved-memory";
  130. * ...
  131. * res0: res {
  132. * reg = <0x42000000 0x1000>;
  133. * label = "res0";
  134. * };
  135. * };
  136. *
  137. * n: node {
  138. * memory-region = <&res0>;
  139. * };
  140. *
  141. * Example usage:
  142. *
  143. * LINKER_DT_RESERVED_MEM_GET_SIZE_BY_PHANDLE(DT_NODELABEL(n), \
  144. * memory_region) // (uint8_t *) 0x42000000
  145. *
  146. * @param node_id node identifier
  147. * @param ph phandle to reserved-memory region
  148. *
  149. * @return size of the reserved-memory region
  150. */
  151. #define LINKER_DT_RESERVED_MEM_GET_SIZE_BY_PHANDLE(node_id, ph) \
  152. LINKER_DT_RESERVED_MEM_GET_SIZE(DT_PHANDLE(node_id, ph))