FindCMSIS.cmake 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. if(NOT CMSIS_FIND_COMPONENTS)
  2. foreach(FAMILY ${STM32_SUPPORTED_FAMILIES})
  3. list(APPEND CMSIS_FIND_COMPONENTS STM32${FAMILY})
  4. endforeach()
  5. endif()
  6. if(STM32H7 IN_LIST CMSIS_FIND_COMPONENTS)
  7. list(REMOVE_ITEM CMSIS_FIND_COMPONENTS STM32H7)
  8. list(APPEND CMSIS_FIND_COMPONENTS STM32H7_M7 STM32H7_M4)
  9. endif()
  10. list(REMOVE_DUPLICATES CMSIS_FIND_COMPONENTS)
  11. include(stm32/devices)
  12. function(cmsis_generate_default_linker_script FAMILY DEVICE CORE)
  13. if(CORE)
  14. set(CORE_C "::${CORE}")
  15. set(CORE_U "_${CORE}")
  16. endif()
  17. set(OUTPUT_LD_FILE "${CMAKE_CURRENT_BINARY_DIR}/${DEVICE}${CORE_U}.ld")
  18. stm32_get_memory_info(FAMILY ${FAMILY} DEVICE ${DEVICE} CORE ${CORE} FLASH SIZE FLASH_SIZE ORIGIN FLASH_ORIGIN)
  19. stm32_get_memory_info(FAMILY ${FAMILY} DEVICE ${DEVICE} CORE ${CORE} RAM SIZE RAM_SIZE ORIGIN RAM_ORIGIN)
  20. stm32_get_memory_info(FAMILY ${FAMILY} DEVICE ${DEVICE} CORE ${CORE} CCRAM SIZE CCRAM_SIZE ORIGIN CCRAM_ORIGIN)
  21. stm32_get_memory_info(FAMILY ${FAMILY} DEVICE ${DEVICE} CORE ${CORE} HEAP SIZE HEAP_SIZE)
  22. stm32_get_memory_info(FAMILY ${FAMILY} DEVICE ${DEVICE} CORE ${CORE} STACK SIZE STACK_SIZE)
  23. add_custom_command(OUTPUT "${OUTPUT_LD_FILE}"
  24. COMMAND ${CMAKE_COMMAND}
  25. -DFLASH_ORIGIN="${FLASH_ORIGIN}"
  26. -DRAM_ORIGIN="${RAM_ORIGIN}"
  27. -DCCRAM_ORIGIN="${CCRAM_ORIGIN}"
  28. -DFLASH_SIZE="${FLASH_SIZE}"
  29. -DRAM_SIZE="${RAM_SIZE}"
  30. -DCCRAM_SIZE="${CCRAM_SIZE}"
  31. -DSTACK_SIZE="${STACK_SIZE}"
  32. -DHEAP_SIZE="${HEAP_SIZE}"
  33. -DLINKER_SCRIPT="${OUTPUT_LD_FILE}"
  34. -P "${STM32_CMAKE_DIR}/stm32/linker_ld.cmake"
  35. )
  36. add_custom_target(CMSIS_LD_${DEVICE}${CORE_U} DEPENDS "${OUTPUT_LD_FILE}")
  37. add_dependencies(CMSIS::STM32::${DEVICE}${CORE_C} CMSIS_LD_${DEVICE}${CORE_U})
  38. stm32_add_linker_script(CMSIS::STM32::${DEVICE}${CORE_C} INTERFACE "${OUTPUT_LD_FILE}")
  39. endfunction()
  40. foreach(COMP ${CMSIS_FIND_COMPONENTS})
  41. string(TOLOWER ${COMP} COMP_L)
  42. string(TOUPPER ${COMP} COMP)
  43. string(REGEX MATCH "^STM32([A-Z][0-9])([0-9A-Z][0-9][A-Z][0-9A-Z])?_?(M[47])?.*$" COMP ${COMP})
  44. if((NOT CMAKE_MATCH_1) AND (NOT CMAKE_MATCH_2))
  45. message(FATAL_ERROR "Unknown CMSIS component: ${COMP}")
  46. endif()
  47. if(CMAKE_MATCH_2)
  48. set(FAMILY ${CMAKE_MATCH_1})
  49. set(DEVICES "${CMAKE_MATCH_1}${CMAKE_MATCH_2}")
  50. else()
  51. set(FAMILY ${CMAKE_MATCH_1})
  52. stm32_get_devices_by_family(DEVICES FAMILY ${FAMILY} CORE ${CORE})
  53. endif()
  54. if(CMAKE_MATCH_3)
  55. set(CORE ${CMAKE_MATCH_3})
  56. set(CORE_C "::${CORE}")
  57. set(CORE_U "_${CORE}")
  58. else()
  59. unset(CORE)
  60. unset(CORE_C)
  61. unset(CORE_U)
  62. endif()
  63. string(TOLOWER ${FAMILY} FAMILY_L)
  64. if((NOT STM32_CMSIS_${FAMILY}_PATH) AND (NOT STM32_CUBE_${FAMILY}_PATH))
  65. set(STM32_CUBE_${FAMILY}_PATH /opt/STM32Cube${FAMILY} CACHE PATH "Path to STM32Cube${FAMILY}")
  66. message(STATUS "Neither STM32_CUBE_${FAMILY}_PATH nor STM32_CMSIS_${FAMILY}_PATH specified using default STM32_CUBE_${FAMILY}_PATH: ${STM32_CUBE_${FAMILY}_PATH}")
  67. endif()
  68. find_path(CMSIS_${FAMILY}${CORE_U}_CORE_PATH
  69. NAMES Include/cmsis_gcc.h
  70. PATHS "${STM32_CMSIS_PATH}" "${STM32_CUBE_${FAMILY}_PATH}/Drivers/CMSIS"
  71. NO_DEFAULT_PATH
  72. )
  73. if (NOT CMSIS_${FAMILY}${CORE_U}_CORE_PATH)
  74. continue()
  75. endif()
  76. find_path(CMSIS_${FAMILY}${CORE_U}_PATH
  77. NAMES Include/stm32${FAMILY_L}xx.h
  78. PATHS "${STM32_CMSIS_${FAMILY}_PATH}" "${STM32_CUBE_${FAMILY}_PATH}/Drivers/CMSIS/Device/ST/STM32${FAMILY}xx"
  79. NO_DEFAULT_PATH
  80. )
  81. if (NOT CMSIS_${FAMILY}${CORE_U}_PATH)
  82. continue()
  83. endif()
  84. list(APPEND CMSIS_INCLUDE_DIRS "${CMSIS_${FAMILY}${CORE_U}_CORE_PATH}/Include" "${CMSIS_${FAMILY}${CORE_U}_PATH}/Include")
  85. if(NOT CMSIS_${FAMILY}${CORE_U}_VERSION)
  86. find_file(CMSIS_${FAMILY}${CORE_U}_PDSC
  87. NAMES ARM.CMSIS.pdsc
  88. PATHS "${CMSIS_${FAMILY}${CORE_U}_CORE_PATH}"
  89. NO_DEFAULT_PATH
  90. )
  91. if (NOT CMSIS_${FAMILY}${CORE_U}_PDSC)
  92. set(CMSIS_${FAMILY}${CORE_U}_VERSION "0.0.0")
  93. else()
  94. file(STRINGS "${CMSIS_${FAMILY}${CORE_U}_PDSC}" VERSION_STRINGS REGEX "<release version=\"([0-9]*\\.[0-9]*\\.[0-9]*)\" date=\"[0-9]+\\-[0-9]+\\-[0-9]+\">")
  95. list(GET VERSION_STRINGS 0 STR)
  96. string(REGEX MATCH "<release version=\"([0-9]*)\\.([0-9]*)\\.([0-9]*)\" date=\"[0-9]+\\-[0-9]+\\-[0-9]+\">" MATCHED ${STR})
  97. set(CMSIS_${FAMILY}${CORE_U}_VERSION "${CMAKE_MATCH_1}.${CMAKE_MATCH_2}.${CMAKE_MATCH_3}" CACHE INTERNAL "CMSIS STM32${FAMILY}${CORE_U} version")
  98. endif()
  99. endif()
  100. set(CMSIS_${COMP}_VERSION ${CMSIS_${FAMILY}${CORE_U}_VERSION})
  101. set(CMSIS_VERSION ${CMSIS_${COMP}_VERSION})
  102. find_file(CMSIS_${FAMILY}${CORE_U}_SOURCE
  103. NAMES system_stm32${FAMILY_L}xx.c
  104. PATHS "${CMSIS_${FAMILY}${CORE_U}_PATH}/Source/Templates"
  105. NO_DEFAULT_PATH
  106. )
  107. list(APPEND CMSIS_SOURCES "${CMSIS_${FAMILY}${CORE_U}_SOURCE}")
  108. if (NOT CMSIS_${FAMILY}${CORE_U}_SOURCE)
  109. continue()
  110. endif()
  111. if(NOT (TARGET CMSIS::STM32::${FAMILY}${CORE_C}))
  112. add_library(CMSIS::STM32::${FAMILY}${CORE_C} INTERFACE IMPORTED)
  113. target_link_libraries(CMSIS::STM32::${FAMILY}${CORE_C} INTERFACE STM32::${FAMILY}${CORE_C})
  114. target_include_directories(CMSIS::STM32::${FAMILY}${CORE_C} INTERFACE "${CMSIS_${FAMILY}${CORE_U}_CORE_PATH}/Include")
  115. target_include_directories(CMSIS::STM32::${FAMILY}${CORE_C} INTERFACE "${CMSIS_${FAMILY}${CORE_U}_PATH}/Include")
  116. target_sources(CMSIS::STM32::${FAMILY}${CORE_C} INTERFACE "${CMSIS_${FAMILY}${CORE_U}_SOURCE}")
  117. endif()
  118. set(DEVICES_FOUND TRUE)
  119. foreach(DEVICE ${DEVICES})
  120. stm32_get_cores(DEV_CORES FAMILY ${FAMILY} DEVICE ${DEVICE})
  121. if(CORE AND (NOT ${CORE} IN_LIST DEV_CORES))
  122. continue()
  123. endif()
  124. stm32_get_chip_type(${FAMILY} ${DEVICE} TYPE)
  125. string(TOLOWER ${DEVICE} DEVICE_L)
  126. string(TOLOWER ${TYPE} TYPE_L)
  127. find_file(CMSIS_${FAMILY}${CORE_U}_${TYPE}_STARTUP
  128. NAMES startup_stm32${TYPE_L}.s
  129. PATHS "${CMSIS_${FAMILY}${CORE_U}_PATH}/Source/Templates/gcc"
  130. NO_DEFAULT_PATH
  131. )
  132. list(APPEND CMSIS_SOURCES "${CMSIS_${FAMILY}${CORE_U}_${TYPE}_STARTUP}")
  133. if(NOT CMSIS_${FAMILY}${CORE_U}_${TYPE}_STARTUP)
  134. set(DEVICES_FOUND FALSE)
  135. break()
  136. endif()
  137. if(NOT (TARGET CMSIS::STM32::${TYPE}${CORE_C}))
  138. add_library(CMSIS::STM32::${TYPE}${CORE_C} INTERFACE IMPORTED)
  139. target_link_libraries(CMSIS::STM32::${TYPE}${CORE_C} INTERFACE CMSIS::STM32::${FAMILY}${CORE_C} STM32::${TYPE}${CORE_C})
  140. target_sources(CMSIS::STM32::${TYPE}${CORE_C} INTERFACE "${CMSIS_${FAMILY}${CORE_U}_${TYPE}_STARTUP}")
  141. endif()
  142. add_library(CMSIS::STM32::${DEVICE}${CORE_C} INTERFACE IMPORTED)
  143. target_link_libraries(CMSIS::STM32::${DEVICE}${CORE_C} INTERFACE CMSIS::STM32::${TYPE}${CORE_C})
  144. cmsis_generate_default_linker_script(${FAMILY} ${DEVICE} "${CORE}")
  145. endforeach()
  146. if(DEVICES_FOUND)
  147. set(CMSIS_${COMP}_FOUND TRUE)
  148. else()
  149. set(CMSIS_${COMP}_FOUND FALSE)
  150. endif()
  151. list(REMOVE_DUPLICATES CMSIS_INCLUDE_DIRS)
  152. list(REMOVE_DUPLICATES CMSIS_SOURCES)
  153. endforeach()
  154. include(FindPackageHandleStandardArgs)
  155. find_package_handle_standard_args(CMSIS
  156. REQUIRED_VARS CMSIS_INCLUDE_DIRS CMSIS_SOURCES
  157. FOUND_VAR CMSIS_FOUND
  158. VERSION_VAR CMSIS_VERSION
  159. HANDLE_COMPONENTS
  160. )