cmake.py 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. """
  2. Utils for CMake
  3. Author: https://github.com/klivelinux
  4. """
  5. import os
  6. import sys
  7. import re
  8. import utils
  9. import rtconfig
  10. from utils import _make_path_relative
  11. from collections import defaultdict
  12. def GenerateCFiles(env, project, project_name):
  13. """
  14. Generate CMakeLists.txt files
  15. """
  16. info = utils.ProjectInfo(env)
  17. PROJECT_NAME = project_name if project_name != "project" else "rtthread"
  18. tool_path_conv = defaultdict(lambda : {"name":"", "path": ""})
  19. tool_path_conv_helper = lambda tool: {"name": tool, "path": os.path.join(rtconfig.EXEC_PATH, tool).replace('\\', "/")}
  20. tool_path_conv["CMAKE_C_COMPILER"] = tool_path_conv_helper(rtconfig.CC)
  21. if 'CXX' in dir(rtconfig):
  22. tool_path_conv["CMAKE_CXX_COMPILER"] = tool_path_conv_helper(rtconfig.CXX)
  23. tool_path_conv["CMAKE_ASM_COMPILER"] = tool_path_conv_helper(rtconfig.AS)
  24. tool_path_conv["CMAKE_AR"] = tool_path_conv_helper(rtconfig.AR)
  25. tool_path_conv["CMAKE_LINKER"] = tool_path_conv_helper(rtconfig.LINK)
  26. if rtconfig.PLATFORM in ['gcc']:
  27. tool_path_conv["CMAKE_SIZE"] = tool_path_conv_helper(rtconfig.SIZE)
  28. tool_path_conv["CMAKE_OBJDUMP"] = tool_path_conv_helper(rtconfig.OBJDUMP)
  29. tool_path_conv["CMAKE_OBJCOPY"] = tool_path_conv_helper(rtconfig.OBJCPY)
  30. elif rtconfig.PLATFORM in ['armcc', 'armclang']:
  31. tool_path_conv["CMAKE_FROMELF"] = tool_path_conv_helper(rtconfig.FROMELF)
  32. CC = tool_path_conv["CMAKE_C_COMPILER"]["path"]
  33. CXX = tool_path_conv["CMAKE_CXX_COMPILER"]["path"]
  34. AS = tool_path_conv["CMAKE_ASM_COMPILER"]["path"]
  35. AR = tool_path_conv["CMAKE_AR"]["path"]
  36. LINK = tool_path_conv["CMAKE_LINKER"]["path"]
  37. SIZE = tool_path_conv["CMAKE_SIZE"]["path"]
  38. OBJDUMP = tool_path_conv["CMAKE_OBJDUMP"]["path"]
  39. OBJCOPY = tool_path_conv["CMAKE_OBJCOPY"]["path"]
  40. FROMELF = tool_path_conv["CMAKE_FROMELF"]["path"]
  41. CFLAGS = rtconfig.CFLAGS.replace('\\', "/").replace('\"', "\\\"")
  42. if 'CXXFLAGS' in dir(rtconfig):
  43. CXXFLAGS = rtconfig.CXXFLAGS.replace('\\', "/").replace('\"', "\\\"")
  44. else:
  45. CXXFLAGS = CFLAGS
  46. AFLAGS = rtconfig.AFLAGS.replace('\\', "/").replace('\"', "\\\"")
  47. LFLAGS = env['LINKFLAGS'].replace('\\', "/").replace('\"', "\\\"")
  48. POST_ACTION = rtconfig.POST_ACTION
  49. # replace the tool name with the cmake variable
  50. for cmake_var, each_tool in tool_path_conv.items():
  51. tool_name = each_tool['name']
  52. if tool_name == "": continue
  53. if "win32" in sys.platform:
  54. while f"{tool_name}.exe" in POST_ACTION: # find the tool with `.exe` suffix first
  55. POST_ACTION = POST_ACTION.replace(tool_name, "string_to_replace")
  56. while tool_name in POST_ACTION:
  57. POST_ACTION = POST_ACTION.replace(tool_name, "string_to_replace")
  58. while "string_to_replace" in POST_ACTION:
  59. POST_ACTION = POST_ACTION.replace("string_to_replace", f"${{{cmake_var}}}")
  60. # replace the `$TARGET` with `${CMAKE_PROJECT_NAME}.elf`
  61. while "$TARGET" in POST_ACTION:
  62. POST_ACTION = POST_ACTION.replace("$TARGET", "${CMAKE_PROJECT_NAME}.elf")
  63. # add COMMAAND before each command
  64. POST_ACTION = POST_ACTION.split('\n')
  65. POST_ACTION = [each_line.strip() for each_line in POST_ACTION]
  66. POST_ACTION = [f"\tCOMMAND {each_line}" for each_line in POST_ACTION if each_line != '']
  67. POST_ACTION = "\n".join(POST_ACTION)
  68. if "win32" in sys.platform:
  69. CC += ".exe"
  70. if CXX != '':
  71. CXX += ".exe"
  72. AS += ".exe"
  73. AR += ".exe"
  74. LINK += ".exe"
  75. if rtconfig.PLATFORM in ['gcc']:
  76. SIZE += ".exe"
  77. OBJDUMP += ".exe"
  78. OBJCOPY += ".exe"
  79. elif rtconfig.PLATFORM in ['armcc', 'armclang']:
  80. FROMELF += ".exe"
  81. if not os.path.exists(CC) or not os.path.exists(AS) or not os.path.exists(AR) or not os.path.exists(LINK):
  82. print("'Cannot found toolchain directory, please check RTT_CC and RTT_EXEC_PATH'")
  83. sys.exit(-1)
  84. with open("CMakeLists.txt", "w") as cm_file:
  85. cm_file.write("CMAKE_MINIMUM_REQUIRED(VERSION 3.10)\n\n")
  86. cm_file.write("SET(CMAKE_SYSTEM_NAME Generic)\n")
  87. cm_file.write("SET(CMAKE_SYSTEM_PROCESSOR " + rtconfig.CPU +")\n")
  88. cm_file.write("#SET(CMAKE_VERBOSE_MAKEFILE ON)\n\n")
  89. cm_file.write("SET(CMAKE_EXPORT_COMPILE_COMMANDS ON)\n\n")
  90. cm_file.write("SET(CMAKE_C_COMPILER \""+ CC + "\")\n")
  91. cm_file.write("SET(CMAKE_ASM_COMPILER \""+ AS + "\")\n")
  92. cm_file.write("SET(CMAKE_C_FLAGS \""+ CFLAGS + "\")\n")
  93. cm_file.write("SET(CMAKE_ASM_FLAGS \""+ AFLAGS + "\")\n")
  94. cm_file.write("SET(CMAKE_C_COMPILER_WORKS TRUE)\n\n")
  95. if CXX != '':
  96. cm_file.write("SET(CMAKE_CXX_COMPILER \""+ CXX + "\")\n")
  97. cm_file.write("SET(CMAKE_CXX_FLAGS \""+ CXXFLAGS + "\")\n")
  98. cm_file.write("SET(CMAKE_CXX_COMPILER_WORKS TRUE)\n\n")
  99. if rtconfig.PLATFORM in ['gcc']:
  100. cm_file.write("SET(CMAKE_OBJCOPY \""+ OBJCOPY + "\")\n")
  101. cm_file.write("SET(CMAKE_SIZE \""+ SIZE + "\")\n\n")
  102. elif rtconfig.PLATFORM in ['armcc', 'armclang']:
  103. cm_file.write("SET(CMAKE_FROMELF \""+ FROMELF + "\")\n\n")
  104. LINKER_FLAGS = ''
  105. LINKER_LIBS = ''
  106. if rtconfig.PLATFORM in ['gcc']:
  107. LINKER_FLAGS += '-T'
  108. elif rtconfig.PLATFORM in ['armcc', 'armclang']:
  109. LINKER_FLAGS += '--scatter'
  110. for group in project:
  111. if 'LIBPATH' in group.keys():
  112. for f in group['LIBPATH']:
  113. LINKER_LIBS += ' --userlibpath ' + f.replace("\\", "/")
  114. for group in project:
  115. if 'LIBS' in group.keys():
  116. for f in group['LIBS']:
  117. LINKER_LIBS += ' ' + f.replace("\\", "/") + '.lib'
  118. cm_file.write("SET(CMAKE_EXE_LINKER_FLAGS \""+ re.sub(LINKER_FLAGS + '(\s*)', LINKER_FLAGS + ' ${CMAKE_SOURCE_DIR}/', LFLAGS) + LINKER_LIBS + "\")\n\n")
  119. # get the c/cpp standard version from compilation flags
  120. # not support the version with alphabet in `-std` param yet
  121. pattern = re.compile('-std=[\w+]+')
  122. c_standard = 11
  123. if '-std=' in CFLAGS:
  124. c_standard = re.search(pattern, CFLAGS).group(0)
  125. c_standard = "".join([each for each in c_standard if each.isdigit()])
  126. else:
  127. print(f"Cannot find the param of the c standard in build flag, set to default {c_standard}")
  128. cm_file.write(f"SET(CMAKE_C_STANDARD {c_standard})\n")
  129. if CXX != '':
  130. cpp_standard = 17
  131. if '-std=' in CXXFLAGS:
  132. cpp_standard = re.search(pattern, CXXFLAGS).group(0)
  133. cpp_standard = "".join([each for each in cpp_standard if each.isdigit()])
  134. else:
  135. print(f"Cannot find the param of the cpp standard in build flag, set to default {cpp_standard}")
  136. cm_file.write(f"SET(CMAKE_CXX_STANDARD {cpp_standard})\n")
  137. cm_file.write('\n')
  138. cm_file.write(f"PROJECT({PROJECT_NAME} C {'CXX' if CXX != '' else ''} ASM)\n")
  139. cm_file.write('\n')
  140. cm_file.write("INCLUDE_DIRECTORIES(\n")
  141. for i in info['CPPPATH']:
  142. # use relative path
  143. path = _make_path_relative(os.getcwd(), i)
  144. cm_file.write( "\t" + path.replace("\\", "/") + "\n")
  145. cm_file.write(")\n\n")
  146. cm_file.write("ADD_DEFINITIONS(\n")
  147. for i in info['CPPDEFINES']:
  148. cm_file.write("\t-D" + i + "\n")
  149. cm_file.write(")\n\n")
  150. cm_file.write("SET(PROJECT_SOURCES\n")
  151. for group in project:
  152. for f in group['src']:
  153. # use relative path
  154. path = _make_path_relative(os.getcwd(), os.path.normpath(f.rfile().abspath))
  155. cm_file.write( "\t" + path.replace("\\", "/") + "\n" )
  156. cm_file.write(")\n\n")
  157. if rtconfig.PLATFORM in ['gcc']:
  158. cm_file.write("LINK_DIRECTORIES(\n")
  159. for group in project:
  160. if 'LIBPATH' in group.keys():
  161. for f in group['LIBPATH']:
  162. cm_file.write( "\t"+ f.replace("\\", "/") + "\n" )
  163. cm_file.write(")\n\n")
  164. cm_file.write("LINK_LIBRARIES(\n")
  165. for group in project:
  166. if 'LIBS' in group.keys():
  167. for f in group['LIBS']:
  168. cm_file.write( "\t"+ "{}\n".format(f.replace("\\", "/")))
  169. cm_file.write(")\n\n")
  170. cm_file.write("ADD_EXECUTABLE(${CMAKE_PROJECT_NAME}.elf ${PROJECT_SOURCES})\n")
  171. cm_file.write("ADD_CUSTOM_COMMAND(TARGET ${CMAKE_PROJECT_NAME}.elf POST_BUILD \n" + POST_ACTION + '\n)\n')
  172. # auto inclue `custom.cmake` for user custom settings
  173. custom_cmake = \
  174. '''
  175. # if custom.cmake is exist, add it
  176. if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/custom.cmake)
  177. include(${CMAKE_CURRENT_SOURCE_DIR}/custom.cmake)
  178. endif()
  179. '''
  180. custom_cmake = custom_cmake.split('\n')
  181. custom_cmake = [each.strip() for each in custom_cmake]
  182. custom_cmake = "\n".join(custom_cmake)
  183. cm_file.write(custom_cmake)
  184. return
  185. def CMakeProject(env, project, project_name):
  186. print('Update setting files for CMakeLists.txt...')
  187. GenerateCFiles(env, project, project_name)
  188. print('Done!')
  189. return