common.cmake 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. set(STM32_SUPPORTED_FAMILIES_LONG_NAME
  2. STM32F0 STM32F1 STM32F2 STM32F3 STM32F4 STM32F7
  3. STM32G0 STM32G4
  4. STM32H7_M4 STM32H7_M7
  5. STM32L0 STM32L1 STM32L4 STM32L5
  6. STM32WB_M4 STM32WL_M4 STM32WL_M0PLUS )
  7. foreach(FAMILY ${STM32_SUPPORTED_FAMILIES_LONG_NAME})
  8. # append short names (F0, F1, H7_M4, ...) to STM32_SUPPORTED_FAMILIES_SHORT_NAME
  9. string(REGEX MATCH "^STM32([FGHLW][0-9BL])_?(M0PLUS|M4|M7)?" FAMILY ${FAMILY})
  10. list(APPEND STM32_SUPPORTED_FAMILIES_SHORT_NAME ${CMAKE_MATCH_1})
  11. endforeach()
  12. list(REMOVE_DUPLICATES STM32_SUPPORTED_FAMILIES_SHORT_NAME)
  13. if(NOT STM32_TOOLCHAIN_PATH)
  14. if(DEFINED ENV{STM32_TOOLCHAIN_PATH})
  15. message(STATUS "Detected toolchain path STM32_TOOLCHAIN_PATH in environmental variables: ")
  16. message(STATUS "$ENV{STM32_TOOLCHAIN_PATH}")
  17. set(STM32_TOOLCHAIN_PATH $ENV{STM32_TOOLCHAIN_PATH})
  18. endif()
  19. if(NOT CMAKE_C_COMPILER)
  20. set(STM32_TOOLCHAIN_PATH "/usr")
  21. message(STATUS "No STM32_TOOLCHAIN_PATH specified, using default: " ${STM32_TOOLCHAIN_PATH})
  22. else()
  23. # keep only directory of compiler
  24. get_filename_component(STM32_TOOLCHAIN_PATH ${CMAKE_C_COMPILER} DIRECTORY)
  25. # remove the last /bin directory
  26. get_filename_component(STM32_TOOLCHAIN_PATH ${STM32_TOOLCHAIN_PATH} DIRECTORY)
  27. endif()
  28. file(TO_CMAKE_PATH "${STM32_TOOLCHAIN_PATH}" STM32_TOOLCHAIN_PATH)
  29. endif()
  30. if(NOT STM32_TARGET_TRIPLET)
  31. set(STM32_TARGET_TRIPLET "arm-none-eabi")
  32. message(STATUS "No STM32_TARGET_TRIPLET specified, using default: " ${STM32_TARGET_TRIPLET})
  33. endif()
  34. set(CMAKE_SYSTEM_NAME Generic)
  35. set(CMAKE_SYSTEM_PROCESSOR arm)
  36. set(TOOLCHAIN_SYSROOT "${STM32_TOOLCHAIN_PATH}/${STM32_TARGET_TRIPLET}")
  37. set(TOOLCHAIN_BIN_PATH "${STM32_TOOLCHAIN_PATH}/bin")
  38. set(TOOLCHAIN_INC_PATH "${STM32_TOOLCHAIN_PATH}/${STM32_TARGET_TRIPLET}/include")
  39. set(TOOLCHAIN_LIB_PATH "${STM32_TOOLCHAIN_PATH}/${STM32_TARGET_TRIPLET}/lib")
  40. find_program(CMAKE_OBJCOPY NAMES ${STM32_TARGET_TRIPLET}-objcopy PATHS ${TOOLCHAIN_BIN_PATH})
  41. find_program(CMAKE_OBJDUMP NAMES ${STM32_TARGET_TRIPLET}-objdump PATHS ${TOOLCHAIN_BIN_PATH})
  42. find_program(CMAKE_SIZE NAMES ${STM32_TARGET_TRIPLET}-size PATHS ${TOOLCHAIN_BIN_PATH})
  43. find_program(CMAKE_DEBUGGER NAMES ${STM32_TARGET_TRIPLET}-gdb PATHS ${TOOLCHAIN_BIN_PATH})
  44. find_program(CMAKE_CPPFILT NAMES ${STM32_TARGET_TRIPLET}-c++filt PATHS ${TOOLCHAIN_BIN_PATH})
  45. function(stm32_print_size_of_target TARGET)
  46. add_custom_target(${TARGET}_always_display_size
  47. ALL COMMAND ${CMAKE_SIZE} ${TARGET}${CMAKE_EXECUTABLE_SUFFIX_C}
  48. COMMENT "Target Sizes: "
  49. DEPENDS ${TARGET}
  50. )
  51. endfunction()
  52. function(stm32_generate_binary_file TARGET)
  53. add_custom_command(
  54. TARGET ${TARGET}
  55. POST_BUILD
  56. COMMAND ${CMAKE_OBJCOPY} -O binary ${TARGET}${CMAKE_EXECUTABLE_SUFFIX_C} ${TARGET}.bin
  57. BYPRODUCTS ${TARGET}.bin
  58. COMMENT "Generating binary file ${CMAKE_PROJECT_NAME}.bin"
  59. )
  60. endfunction()
  61. function(stm32_generate_hex_file TARGET)
  62. add_custom_command(
  63. TARGET ${TARGET}
  64. POST_BUILD
  65. COMMAND ${CMAKE_OBJCOPY} -O ihex ${TARGET}${CMAKE_EXECUTABLE_SUFFIX_C} ${TARGET}.hex
  66. BYPRODUCTS ${TARGET}.hex
  67. COMMENT "Generating hex file ${CMAKE_PROJECT_NAME}.hex"
  68. )
  69. endfunction()
  70. # This function takes FAMILY (e.g. L4) and DEVICE (e.g. L496VG) to output TYPE (e.g. L496xx)
  71. function(stm32_get_chip_type FAMILY DEVICE TYPE)
  72. set(INDEX 0)
  73. foreach(C_TYPE ${STM32_${FAMILY}_TYPES})
  74. list(GET STM32_${FAMILY}_TYPE_MATCH ${INDEX} REGEXP)
  75. if(${DEVICE} MATCHES ${REGEXP})
  76. set(RESULT_TYPE ${C_TYPE})
  77. endif()
  78. math(EXPR INDEX "${INDEX}+1")
  79. endforeach()
  80. if(NOT RESULT_TYPE)
  81. message(FATAL_ERROR "Invalid/unsupported device: ${DEVICE}")
  82. endif()
  83. set(${TYPE} ${RESULT_TYPE} PARENT_SCOPE)
  84. endfunction()
  85. function(stm32_get_chip_info CHIP)
  86. set(ARG_OPTIONS "")
  87. set(ARG_SINGLE FAMILY DEVICE TYPE)
  88. set(ARG_MULTIPLE "")
  89. cmake_parse_arguments(PARSE_ARGV 1 ARG "${ARG_OPTIONS}" "${ARG_SINGLE}" "${ARG_MULTIPLE}")
  90. string(TOUPPER ${CHIP} CHIP)
  91. string(REGEX MATCH "^STM32([FGHLW][0-9BL])([0-9A-Z][0-9M][A-Z][0-9A-Z]).*$" CHIP ${CHIP})
  92. if((NOT CMAKE_MATCH_1) OR (NOT CMAKE_MATCH_2))
  93. message(FATAL_ERROR "Unknown chip ${CHIP}")
  94. endif()
  95. set(STM32_FAMILY ${CMAKE_MATCH_1})
  96. set(STM32_DEVICE "${CMAKE_MATCH_1}${CMAKE_MATCH_2}")
  97. if(NOT (${STM32_FAMILY} IN_LIST STM32_SUPPORTED_FAMILIES_SHORT_NAME))
  98. message(FATAL_ERROR "Unsupported family ${STM32_FAMILY} for device ${CHIP}")
  99. endif()
  100. stm32_get_chip_type(${STM32_FAMILY} ${STM32_DEVICE} STM32_TYPE)
  101. if(ARG_FAMILY)
  102. set(${ARG_FAMILY} ${STM32_FAMILY} PARENT_SCOPE)
  103. endif()
  104. if(ARG_DEVICE)
  105. set(${ARG_DEVICE} ${STM32_DEVICE} PARENT_SCOPE)
  106. endif()
  107. if(ARG_TYPE)
  108. set(${ARG_TYPE} ${STM32_TYPE} PARENT_SCOPE)
  109. endif()
  110. endfunction()
  111. function(stm32_get_cores CORES)
  112. set(ARG_OPTIONS "")
  113. set(ARG_SINGLE CHIP FAMILY DEVICE)
  114. set(ARG_MULTIPLE "")
  115. cmake_parse_arguments(PARSE_ARGV 1 ARG "${ARG_OPTIONS}" "${ARG_SINGLE}" "${ARG_MULTIPLE}")
  116. if(ARG_CHIP)
  117. # TODO: I don't get why stm32_get_chip_info is called in stm32_get_cores
  118. stm32_get_chip_info(${ARG_CHIP} FAMILY ARG_FAMILY TYPE ARG_TYPE DEVICE ARG_DEVICE)
  119. elseif(ARG_FAMILY AND ARG_DEVICE)
  120. # TODO: I don't get why stm32_get_chip_type is called in stm32_get_cores
  121. stm32_get_chip_type(${ARG_FAMILY} ${ARG_DEVICE} ARG_TYPE)
  122. elseif(ARG_FAMILY)
  123. if(${ARG_FAMILY} STREQUAL "H7")
  124. set(${CORES} M7 M4 PARENT_SCOPE)
  125. elseif(${ARG_FAMILY} STREQUAL "WB")
  126. set(${CORES} M4 PARENT_SCOPE)
  127. elseif(${ARG_FAMILY} STREQUAL "WL")
  128. set(${CORES} M4 M0PLUS PARENT_SCOPE)
  129. else()
  130. set(${CORES} "" PARENT_SCOPE)
  131. endif()
  132. return()
  133. else()
  134. message(FATAL_ERROR "Either CHIP or FAMILY or FAMILY/DEVICE should be specified for stm32_get_cores()")
  135. endif()
  136. # TODO following is the only part really used by FindCMSIS. Maybe a cleanup is needed
  137. if(${ARG_FAMILY} STREQUAL "H7")
  138. stm32h7_get_device_cores(${ARG_DEVICE} ${ARG_TYPE} CORE_LIST)
  139. elseif(${ARG_FAMILY} STREQUAL "WB")
  140. # note STM32WB have an M0 core but in current state of the art it runs ST stacks and is not needed/allowed to build for customer
  141. set(CORE_LIST M4)
  142. elseif(${ARG_FAMILY} STREQUAL "WL")
  143. stm32wl_get_device_cores(${ARG_DEVICE} ${ARG_TYPE} CORE_LIST)
  144. endif()
  145. set(${CORES} "${CORE_LIST}" PARENT_SCOPE)
  146. endfunction()
  147. function(stm32_get_memory_info)
  148. set(ARG_OPTIONS FLASH RAM CCRAM STACK HEAP)
  149. set(ARG_SINGLE CHIP FAMILY DEVICE CORE SIZE ORIGIN)
  150. set(ARG_MULTIPLE "")
  151. cmake_parse_arguments(INFO "${ARG_OPTIONS}" "${ARG_SINGLE}" "${ARG_MULTIPLE}" ${ARGN})
  152. if((NOT INFO_CHIP) AND ((NOT INFO_FAMILY) OR (NOT INFO_DEVICE)))
  153. message(FATAL_ERROR "Either CHIP or FAMILY/DEVICE is required for stm32_get_memory_info()")
  154. endif()
  155. if(INFO_CHIP)
  156. stm32_get_chip_info(${INFO_CHIP} FAMILY INFO_FAMILY TYPE INFO_TYPE DEVICE INFO_DEVICE)
  157. else()
  158. stm32_get_chip_type(${INFO_FAMILY} ${INFO_DEVICE} INFO_TYPE)
  159. endif()
  160. string(REGEX REPLACE "^[FGHLW][0-9BL][0-9A-Z][0-9M].([3468BCDEFGHIYZ])$" "\\1" SIZE_CODE ${INFO_DEVICE})
  161. if(SIZE_CODE STREQUAL "3")
  162. set(FLASH "8K")
  163. elseif(SIZE_CODE STREQUAL "4")
  164. set(FLASH "16K")
  165. elseif(SIZE_CODE STREQUAL "6")
  166. set(FLASH "32K")
  167. elseif(SIZE_CODE STREQUAL "8")
  168. set(FLASH "64K")
  169. elseif(SIZE_CODE STREQUAL "B")
  170. set(FLASH "128K")
  171. elseif(SIZE_CODE STREQUAL "C")
  172. # Note there is a problem with STM32WB15CC (320kB flash)
  173. set(FLASH "256K")
  174. elseif(SIZE_CODE STREQUAL "D")
  175. set(FLASH "384K")
  176. elseif(SIZE_CODE STREQUAL "E")
  177. set(FLASH "512K")
  178. elseif(SIZE_CODE STREQUAL "F")
  179. set(FLASH "768K")
  180. elseif(SIZE_CODE STREQUAL "G")
  181. set(FLASH "1024K")
  182. elseif(SIZE_CODE STREQUAL "H")
  183. set(FLASH "1536K")
  184. elseif(SIZE_CODE STREQUAL "I")
  185. set(FLASH "2048K")
  186. elseif(SIZE_CODE STREQUAL "Y")
  187. set(FLASH "640K")
  188. elseif(SIZE_CODE STREQUAL "Z")
  189. set(FLASH "192K")
  190. else()
  191. set(FLASH "16K")
  192. message(WARNING "Unknow flash size for device ${DEVICE}. Set to ${FLASH}")
  193. endif()
  194. list(FIND STM32_${INFO_FAMILY}_TYPES ${INFO_TYPE} TYPE_INDEX)
  195. list(GET STM32_${INFO_FAMILY}_RAM_SIZES ${TYPE_INDEX} RAM)
  196. list(GET STM32_${INFO_FAMILY}_CCRAM_SIZES ${TYPE_INDEX} CCRAM)
  197. set(FLASH_ORIGIN 0x8000000)
  198. set(RAM_ORIGIN 0x20000000)
  199. set(CCRAM_ORIGIN 0x10000000)
  200. if(FAMILY STREQUAL "F1")
  201. stm32f1_get_memory_info(${INFO_DEVICE} ${INFO_TYPE} FLASH RAM)
  202. elseif(FAMILY STREQUAL "L1")
  203. stm32l1_get_memory_info(${INFO_DEVICE} ${INFO_TYPE} FLASH RAM)
  204. elseif(FAMILY STREQUAL "F2")
  205. stm32f2_get_memory_info(${INFO_DEVICE} ${INFO_TYPE} FLASH RAM)
  206. elseif(FAMILY STREQUAL "F3")
  207. stm32f3_get_memory_info(${INFO_DEVICE} ${INFO_TYPE} FLASH RAM)
  208. elseif(FAMILY STREQUAL "H7")
  209. stm32h7_get_memory_info(${INFO_DEVICE} ${INFO_TYPE} "${INFO_CORE}" RAM FLASH_ORIGIN RAM_ORIGIN TWO_FLASH_BANKS)
  210. if(TWO_FLASH_BANKS)
  211. string(REGEX MATCH "([0-9]+)K" FLASH_KB ${FLASH})
  212. math(EXPR FLASH_KB "${CMAKE_MATCH_1} / 2")
  213. set(FLASH "${FLASH_KB}K")
  214. endif()
  215. endif()
  216. if(INFO_FLASH)
  217. set(SIZE ${FLASH})
  218. set(ORIGIN ${FLASH_ORIGIN})
  219. elseif(INFO_RAM)
  220. set(SIZE ${RAM})
  221. set(ORIGIN ${RAM_ORIGIN})
  222. elseif(INFO_CCRAM)
  223. set(SIZE ${CCRAM})
  224. set(ORIGIN ${CCRAM_ORIGIN})
  225. elseif(INFO_STACK)
  226. if (RAM STREQUAL "2K")
  227. set(SIZE 0x200)
  228. else()
  229. set(SIZE 0x400)
  230. endif()
  231. set(ORIGIN ${RAM_ORIGIN}) #TODO: Real stack pointer?
  232. elseif(INFO_HEAP)
  233. if (RAM STREQUAL "2K")
  234. set(SIZE 0x100)
  235. else()
  236. set(SIZE 0x200)
  237. endif()
  238. set(ORIGIN ${RAM_ORIGIN}) #TODO: Real heap pointer?
  239. endif()
  240. if(INFO_SIZE)
  241. set(${INFO_SIZE} ${SIZE} PARENT_SCOPE)
  242. endif()
  243. if(INFO_ORIGIN)
  244. set(${INFO_ORIGIN} ${ORIGIN} PARENT_SCOPE)
  245. endif()
  246. endfunction()
  247. function(stm32_add_linker_script TARGET VISIBILITY SCRIPT)
  248. get_filename_component(SCRIPT "${SCRIPT}" ABSOLUTE)
  249. target_link_options(${TARGET} ${VISIBILITY} -T "${SCRIPT}")
  250. get_target_property(TARGET_TYPE ${TARGET} TYPE)
  251. if(TARGET_TYPE STREQUAL "INTERFACE_LIBRARY")
  252. set(INTERFACE_PREFIX "INTERFACE_")
  253. endif()
  254. get_target_property(LINK_DEPENDS ${TARGET} ${INTERFACE_PREFIX}LINK_DEPENDS)
  255. if(LINK_DEPENDS)
  256. list(APPEND LINK_DEPENDS "${SCRIPT}")
  257. else()
  258. set(LINK_DEPENDS "${SCRIPT}")
  259. endif()
  260. set_target_properties(${TARGET} PROPERTIES ${INTERFACE_PREFIX}LINK_DEPENDS "${LINK_DEPENDS}")
  261. endfunction()
  262. if(NOT (TARGET STM32::NoSys))
  263. add_library(STM32::NoSys INTERFACE IMPORTED)
  264. target_compile_options(STM32::NoSys INTERFACE $<$<C_COMPILER_ID:GNU>:--specs=nosys.specs>)
  265. target_link_options(STM32::NoSys INTERFACE $<$<C_COMPILER_ID:GNU>:--specs=nosys.specs>)
  266. endif()
  267. if(NOT (TARGET STM32::Nano))
  268. add_library(STM32::Nano INTERFACE IMPORTED)
  269. target_compile_options(STM32::Nano INTERFACE $<$<C_COMPILER_ID:GNU>:--specs=nano.specs>)
  270. target_link_options(STM32::Nano INTERFACE $<$<C_COMPILER_ID:GNU>:--specs=nano.specs>)
  271. endif()
  272. if(NOT (TARGET STM32::Nano::FloatPrint))
  273. add_library(STM32::Nano::FloatPrint INTERFACE IMPORTED)
  274. target_link_options(STM32::Nano::FloatPrint INTERFACE
  275. $<$<C_COMPILER_ID:GNU>:-Wl,--undefined,_printf_float>
  276. )
  277. endif()
  278. if(NOT (TARGET STM32::Nano::FloatScan))
  279. add_library(STM32::Nano::FloatScan INTERFACE IMPORTED)
  280. target_link_options(STM32::Nano::FloatPrint INTERFACE
  281. $<$<C_COMPILER_ID:GNU>:-Wl,--undefined,_scanf_float>
  282. )
  283. endif()
  284. include(stm32/utilities)
  285. include(stm32/f0)
  286. include(stm32/f1)
  287. include(stm32/f2)
  288. include(stm32/f3)
  289. include(stm32/f4)
  290. include(stm32/f7)
  291. include(stm32/g0)
  292. include(stm32/g4)
  293. include(stm32/h7)
  294. include(stm32/l0)
  295. include(stm32/l1)
  296. include(stm32/l4)
  297. include(stm32/l5)
  298. include(stm32/wb)
  299. include(stm32/wl)