CMakeLists.txt 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. cmake_minimum_required(VERSION 3.15)
  2. include(cmake/Utilities.cmake)
  3. set (CMAKE_CXX_STANDARD 11)
  4. get_dependency_directory(prusa3dboards PRUSA_BOARDS_DIR)
  5. project(Prusa-Firmware)
  6. get_recommended_gcc_version(RECOMMENDED_TOOLCHAIN_VERSION)
  7. if(CMAKE_CROSSCOMPILING AND NOT CMAKE_CXX_COMPILER_VERSION VERSION_EQUAL
  8. ${RECOMMENDED_TOOLCHAIN_VERSION}
  9. )
  10. message(WARNING "Recommended AVR toolchain is ${RECOMMENDED_TOOLCHAIN_VERSION}"
  11. ", but you have ${CMAKE_CXX_COMPILER_VERSION}"
  12. )
  13. elseif(NOT CMAKE_CROSSCOMPILING AND NOT CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
  14. message(
  15. WARNING
  16. "Recommended compiler for host tools and unittests is GCC, you have ${CMAKE_CXX_COMPILER_ID}."
  17. )
  18. endif()
  19. # append custom C/C++ flags
  20. if(CUSTOM_COMPILE_OPTIONS)
  21. string(REPLACE " " ";" CUSTOM_COMPILE_OPTIONS "${CUSTOM_COMPILE_OPTIONS}")
  22. add_compile_options(${CUSTOM_COMPILE_OPTIONS})
  23. endif()
  24. #
  25. # Global Compiler & Linker Configuration
  26. #
  27. # include symbols
  28. add_compile_options(-g)
  29. # optimizations
  30. if(CMAKE_CROSSCOMPILING)
  31. if(CMAKE_BUILD_TYPE STREQUAL "Debug")
  32. add_compile_options(-Og)
  33. else()
  34. add_compile_options(-Os)
  35. endif()
  36. # mcu related settings
  37. set(MCU_FLAGS -mmcu=atmega2560 -DF_CPU=16000000L)
  38. add_compile_options(${MCU_FLAGS})
  39. add_link_options(${MCU_FLAGS})
  40. # split and gc sections
  41. add_compile_options(-ffunction-sections -fdata-sections)
  42. add_link_options(-Wl,--gc-sections)
  43. # disable exceptions and related metadata
  44. add_compile_options(-fno-exceptions -fno-unwind-tables)
  45. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:-fno-rtti>)
  46. add_link_options(-Wl,--defsym,__exidx_start=0,--defsym,__exidx_end=0)
  47. else()
  48. if(CMAKE_BUILD_TYPE STREQUAL "Debug")
  49. add_compile_options(-O0)
  50. else()
  51. add_compile_options(-O2)
  52. endif()
  53. endif()
  54. # enable all warnings (well, not all, but some)
  55. add_compile_options(-Wall -Wsign-compare)
  56. add_compile_options($<$<COMPILE_LANGUAGE:CXX>:-std=c++14>)
  57. # support _DEBUG macro (some code uses to recognize debug builds)
  58. if(CMAKE_BUILD_TYPE STREQUAL "Debug")
  59. add_compile_definitions(_DEBUG)
  60. endif()
  61. #
  62. # Firmware - get file lists.
  63. #
  64. file(GLOB FW_SOURCES RELATIVE ${PROJECT_SOURCE_DIR} ${PROJECT_SOURCE_DIR}/Firmware/*.c*)
  65. file(GLOB FW_HEADERS RELATIVE ${PROJECT_SOURCE_DIR} ${PROJECT_SOURCE_DIR}/Firmware/*.h*)
  66. file(GLOB AVR_SOURCES RELATIVE ${PROJECT_SOURCE_DIR} ${PRUSA_BOARDS_DIR}/cores/prusa_einsy_rambo/*.c*)
  67. # Setup language resources:
  68. file(GLOB LANG_VARIANTS RELATIVE ${PROJECT_SOURCE_DIR}/lang/po ${PROJECT_SOURCE_DIR}/lang/po/Firmware_??.po)
  69. string(REPLACE "Firmware_" "" LANG_VARIANTS "${LANG_VARIANTS}")
  70. string(REPLACE ".po" "" LANG_VARIANTS "${LANG_VARIANTS}")
  71. message("Languages found: ${LANG_VARIANTS}")
  72. add_library(avr_core STATIC ${AVR_SOURCES})
  73. target_include_directories(avr_core PRIVATE
  74. ${PRUSA_BOARDS_DIR}/cores/prusa_einsy_rambo/
  75. ${PRUSA_BOARDS_DIR}/variants/prusa_einsy_rambo/
  76. )
  77. target_compile_options(avr_core PUBLIC -mmcu=atmega2560)
  78. function(fw_add_variant variant_name)
  79. add_executable(${variant_name} ${FW_SOURCES} ${FW_HEADERS})
  80. set_target_properties(${variant_name} PROPERTIES CXX_STANDARD 14)
  81. # # configure linker script
  82. set(LINKER_SCRIPT ${PRUSA_BOARDS_DIR}/ldscripts/avr6.xn)
  83. target_link_options(${variant_name} PUBLIC -Wl,-T,${LINKER_SCRIPT})
  84. # limit the text section to 248K (256K - 8k reserved for the bootloader)
  85. target_link_options(${variant_name} PUBLIC -Wl,--defsym=__TEXT_REGION_LENGTH__=248K)
  86. # generate firmware.bin file
  87. objcopy(${variant_name} "ihex" ".hex")
  88. # produce ASM listing. Note we also specify the .map as a byproduct so it gets cleaned
  89. # because link_options doesn't have a "generated outputs" feature.
  90. add_custom_command(
  91. TARGET ${variant_name} POST_BUILD COMMAND ${CMAKE_OBJDUMP} -CSd ${variant_name} > ${variant_name}.asm
  92. BYPRODUCTS ${variant_name}.asm ${variant_name}.map
  93. )
  94. # inform about the firmware's size in terminal
  95. add_custom_command(
  96. TARGET ${variant_name} POST_BUILD COMMAND ${CMAKE_SIZE_UTIL} -C --mcu=atmega2560 ${variant_name}
  97. )
  98. report_size(${variant_name})
  99. # generate linker map file
  100. target_link_options(${variant_name} PUBLIC -Wl,-Map=${variant_name}.map)
  101. target_include_directories(${variant_name} PRIVATE Firmware
  102. ${PRUSA_BOARDS_DIR}/cores/prusa_einsy_rambo/
  103. ${PRUSA_BOARDS_DIR}/variants/prusa_einsy_rambo/
  104. ${PROJECT_SOURCE_DIR}/cmake/helpers/ # Add our magic config helper :)
  105. )
  106. target_compile_options(${variant_name} PRIVATE) # turn this on for lolz -Wdouble-promotion)
  107. string(REPLACE "-" "_" DEFINE_NAME "${variant_name}")
  108. target_compile_definitions(${variant_name} PRIVATE H${DEFINE_NAME} ARDUINO=10600 __AVR_ATmega2560__)
  109. target_link_libraries(${variant_name} avr_core)
  110. #Construct language map
  111. set(LANG_MAP ${CMAKE_CURRENT_BINARY_DIR}/lang/${variant_name}_lang.map)
  112. set(LANG_FWBIN ${CMAKE_BINARY_DIR}/${variant_name}.bin)
  113. set(LANG_FINAL_BIN ${CMAKE_CURRENT_BINARY_DIR}/lang/${variant_name}_lang.bin)
  114. set(LANG_FINAL_HEX ${CMAKE_CURRENT_BINARY_DIR}/lang/${variant_name}_lang.hex)
  115. add_custom_command(OUTPUT ${LANG_FWBIN}
  116. COMMAND "${CMAKE_OBJCOPY}" -I ihex -O binary ${CMAKE_BINARY_DIR}/${variant_name}.hex ${LANG_FWBIN}
  117. DEPENDS ${variant_name}
  118. )
  119. add_custom_command(OUTPUT ${LANG_MAP}
  120. COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/lang/lang-map.py "${CMAKE_BINARY_DIR}/${variant_name}" "${LANG_FWBIN}" > "${LANG_MAP}"
  121. DEPENDS ${LANG_FWBIN}
  122. )
  123. set(LANG_BINS "")
  124. foreach (LANG IN LISTS LANG_VARIANTS)
  125. set(LANG_BIN ${CMAKE_CURRENT_BINARY_DIR}/lang/${variant_name}_${LANG}.bin)
  126. set(PO_FILE "${CMAKE_CURRENT_SOURCE_DIR}/lang/po/Firmware_${LANG}.po")
  127. add_custom_command(OUTPUT ${LANG_BIN}
  128. # COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/lang/lang-check.py --no-warning --map "${LANG_MAP}" "${PO_FILE}"
  129. # COMMAND ${CMAKE_COMMAND} -E echo "Building lang_${LANG}.bin"
  130. COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/lang/lang-build.py ${LANG_MAP} ${PO_FILE} ${LANG_BIN}
  131. DEPENDS ${LANG_MAP}
  132. COMMENT "Generating ${variant_name}_${LANG}.bin from .po"
  133. )
  134. LIST(APPEND LANG_BINS ${LANG_BIN})
  135. endforeach()
  136. add_custom_command( OUTPUT ${LANG_FINAL_BIN}
  137. # TODO - needs differentiation for platforms, e.g. copy /b on Win
  138. COMMAND cat ${LANG_BINS} > ${LANG_FINAL_BIN}
  139. DEPENDS ${LANG_BINS}
  140. COMMENT "Merging language binaries"
  141. )
  142. add_custom_command( OUTPUT ${LANG_FINAL_HEX}
  143. # TODO - needs differentiation for platforms, e.g. copy /b on Win
  144. COMMAND ${CMAKE_OBJCOPY} -I binary -O ihex ${LANG_FINAL_BIN} ${LANG_FINAL_HEX}
  145. DEPENDS ${LANG_FINAL_BIN}
  146. COMMENT "Generating Hex for language data"
  147. )
  148. set(LANG_HEX ${CMAKE_BINARY_DIR}/${variant_name}-lang.hex)
  149. add_custom_target(${variant_name}-languages
  150. COMMAND ${CMAKE_COMMAND} -E copy ${CMAKE_BINARY_DIR}/${variant_name}.hex ${LANG_HEX}
  151. COMMAND cat ${LANG_FINAL_HEX} >> ${LANG_HEX}
  152. COMMENT "Generating final ${variant_name}-lang.hex"
  153. BYPRODUCTS ${LANG_HEX}
  154. DEPENDS ${LANG_FINAL_HEX}
  155. )
  156. endfunction()
  157. if(CMAKE_CROSSCOMPILING)
  158. add_custom_target(All_Firmware)
  159. file(GLOB FW_VARIANTS RELATIVE ${PROJECT_SOURCE_DIR}/Firmware/variants ${PROJECT_SOURCE_DIR}/Firmware/variants/*.h)
  160. foreach(THIS_VAR IN LISTS FW_VARIANTS)
  161. string(REPLACE ".h" "" TRIMMED_NAME "${THIS_VAR}")
  162. message("Variant added: ${TRIMMED_NAME}")
  163. fw_add_variant(${TRIMMED_NAME})
  164. add_dependencies(All_Firmware ${TRIMMED_NAME})
  165. endforeach(THIS_VAR IN LISTS FW_VARIANTS)
  166. endif()
  167. if(NOT CMAKE_CROSSCOMPILING)
  168. # do not build the firmware by default (tests are the focus if not crosscompiling)
  169. project(cmake_test)
  170. # Prepare "Catch" library for other executables
  171. set(CATCH_INCLUDE_DIR Catch2)
  172. add_library(Catch INTERFACE)
  173. target_include_directories(Catch INTERFACE ${CATCH_INCLUDE_DIR})
  174. # Make test executable
  175. set(TEST_SOURCES
  176. Tests/tests.cpp
  177. Tests/Example_test.cpp
  178. Tests/Timer_test.cpp
  179. Tests/AutoDeplete_test.cpp
  180. Tests/PrusaStatistics_test.cpp
  181. Firmware/Timer.cpp
  182. Firmware/AutoDeplete.cpp
  183. )
  184. add_executable(tests ${TEST_SOURCES})
  185. target_include_directories(tests PRIVATE Tests)
  186. target_link_libraries(tests Catch)
  187. endif()