gen_dts_cmake.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. #!/usr/bin/env python3
  2. # Copyright (c) 2021 Nordic Semiconductor ASA
  3. # SPDX-License-Identifier: Apache-2.0
  4. '''
  5. This script uses edtlib and the devicetree data in the build directory
  6. to generate a CMake file which contains devicetree data.
  7. That data can then be used in the rest of the build system.
  8. The generated CMake file looks like this:
  9. add_custom_target(devicetree_target)
  10. set_target_properties(devicetree_target PROPERTIES
  11. "DT_PROP|/soc|compatible" "vnd,soc;")
  12. ...
  13. It defines a special CMake target, and saves various values in the
  14. devicetree as CMake target properties.
  15. Be careful:
  16. "Property" here can refer to a CMake target property or a
  17. DTS property. DTS property values are stored inside
  18. CMake target properties, along with other devicetree data.
  19. The build system includes this generated file early on, so
  20. devicetree values can be used at CMake processing time.
  21. Accss is not done directly, but with Zephyr CMake extension APIs,
  22. like this:
  23. # sets 'compat' to "vnd,soc" in CMake
  24. dt_prop(compat PATH "/soc" PROPERTY compatible INDEX 0)
  25. This is analogous to how DTS values are encoded as C macros,
  26. which can be read in source code using C APIs like
  27. DT_PROP(node_id, foo) from devicetree.h.
  28. '''
  29. import argparse
  30. import os
  31. import pickle
  32. import sys
  33. sys.path.append(os.path.join(os.path.dirname(__file__), 'python-devicetree',
  34. 'src'))
  35. def parse_args():
  36. # Returns parsed command-line arguments
  37. parser = argparse.ArgumentParser()
  38. parser.add_argument("--cmake-out", required=True,
  39. help="path to write the CMake property file")
  40. parser.add_argument("--edt-pickle", required=True,
  41. help="path to read the pickled edtlib.EDT object from")
  42. return parser.parse_args()
  43. def main():
  44. args = parse_args()
  45. with open(args.edt_pickle, 'rb') as f:
  46. edt = pickle.load(f)
  47. # In what looks like an undocumented implementation detail, CMake
  48. # target properties are stored in a C++ standard library map whose
  49. # keys and values are each arbitrary strings, so we can use
  50. # whatever we want as target property names.
  51. #
  52. # We therefore use '|' as a field separator character below within
  53. # because it's not a valid character in DTS node paths or property
  54. # names. This lets us store the "real" paths and property names
  55. # without conversion to lowercase-and-underscores like we have to
  56. # do in C.
  57. #
  58. # If CMake adds restrictions on target property names later, we
  59. # can just tweak the generated file to use a more restrictive
  60. # property encoding, perhaps reusing the same style documented in
  61. # macros.bnf for C macros.
  62. cmake_props = []
  63. for node in edt.chosen_nodes:
  64. path = edt.chosen_nodes[node].path
  65. cmake_props.append(f'"DT_CHOSEN|{node}" "{path}"')
  66. for node in edt.nodes:
  67. cmake_props.append(f'"DT_NODE|{node.path}" TRUE')
  68. for label in node.labels:
  69. cmake_props.append(f'"DT_NODELABEL|{label}" "{node.path}"')
  70. for item in node.props:
  71. # We currently do not support phandles for edt -> cmake conversion.
  72. if "phandle" not in node.props[item].type:
  73. if "array" in node.props[item].type:
  74. # Convert array to CMake list
  75. cmake_value = ''
  76. for val in node.props[item].val:
  77. cmake_value = f'{cmake_value}{val};'
  78. else:
  79. cmake_value = node.props[item].val
  80. # Encode node's property 'item' as a CMake target property
  81. # with a name like 'DT_PROP|<path>|<property>'.
  82. cmake_prop = f'DT_PROP|{node.path}|{item}'
  83. cmake_props.append(f'"{cmake_prop}" "{cmake_value}"')
  84. if node.regs is not None:
  85. cmake_props.append(f'"DT_REG|{node.path}|NUM" "{len(node.regs)}"')
  86. cmake_addr = ''
  87. cmake_size = ''
  88. for reg in node.regs:
  89. if reg.addr is None:
  90. cmake_addr = f'{cmake_addr}NONE;'
  91. else:
  92. cmake_addr = f'{cmake_addr}{hex(reg.addr)};'
  93. if reg.size is None:
  94. cmake_size = f'{cmake_size}NONE;'
  95. else:
  96. cmake_size = f'{cmake_size}{hex(reg.size)};'
  97. cmake_props.append(f'"DT_REG|{node.path}|ADDR" "{cmake_addr}"')
  98. cmake_props.append(f'"DT_REG|{node.path}|SIZE" "{cmake_size}"')
  99. with open(args.cmake_out, "w", encoding="utf-8") as cmake_file:
  100. print('add_custom_target(devicetree_target)', file=cmake_file)
  101. print(file=cmake_file)
  102. for prop in cmake_props:
  103. print(
  104. f'set_target_properties(devicetree_target PROPERTIES {prop})',
  105. file=cmake_file
  106. )
  107. if __name__ == "__main__":
  108. main()