CMakeLists.txt 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. cmake_minimum_required(VERSION 3.5)
  2. if(NOT SDKCONFIG)
  3. message(FATAL_ERROR "Bootloader subproject expects the SDKCONFIG variable to be passed "
  4. "in by the parent build process.")
  5. endif()
  6. if(NOT IDF_PATH)
  7. message(FATAL_ERROR "Bootloader subproject expects the IDF_PATH variable to be passed "
  8. "in by the parent build process.")
  9. endif()
  10. if(NOT IDF_TARGET)
  11. message(FATAL_ERROR "Bootloader subproject expects the IDF_TARGET variable to be passed "
  12. "in by the parent build process.")
  13. endif()
  14. set(COMPONENTS bootloader esptool_py partition_table soc bootloader_support log spi_flash micro-ecc main efuse)
  15. set(BOOTLOADER_BUILD 1)
  16. include("${IDF_PATH}/tools/cmake/project.cmake")
  17. set(common_req log esp_rom esp_common xtensa)
  18. if(LEGACY_INCLUDE_COMMON_HEADERS)
  19. list(APPEND common_req soc)
  20. endif()
  21. idf_build_set_property(__COMPONENT_REQUIRES_COMMON "${common_req}")
  22. idf_build_set_property(__OUTPUT_SDKCONFIG 0)
  23. project(bootloader)
  24. idf_build_set_property(COMPILE_DEFINITIONS "-DBOOTLOADER_BUILD=1" APPEND)
  25. idf_build_set_property(COMPILE_OPTIONS "-fno-stack-protector" APPEND)
  26. idf_component_get_property(main_args esptool_py FLASH_ARGS)
  27. idf_component_get_property(sub_args esptool_py FLASH_SUB_ARGS)
  28. # String for printing flash command
  29. string(REPLACE ";" " " esptoolpy_write_flash
  30. "${ESPTOOLPY} --port=(PORT) --baud=(BAUD) ${main_args} "
  31. "write_flash ${sub_args}")
  32. string(REPLACE ";" " " espsecurepy "${ESPSECUREPY}")
  33. string(REPLACE ";" " " espefusepy "${ESPEFUSEPY}")
  34. if(CONFIG_SECURE_BOOTLOADER_REFLASHABLE)
  35. if(CONFIG_SECURE_BOOTLOADER_KEY_ENCODING_192BIT)
  36. set(key_digest_len 192)
  37. else()
  38. set(key_digest_len 256)
  39. endif()
  40. get_filename_component(bootloader_digest_bin
  41. "bootloader-reflash-digest.bin"
  42. ABSOLUTE BASE_DIR "${CMAKE_BINARY_DIR}")
  43. get_filename_component(secure_bootloader_key
  44. "secure-bootloader-key-${key_digest_len}.bin"
  45. ABSOLUTE BASE_DIR "${CMAKE_BINARY_DIR}")
  46. add_custom_command(OUTPUT "${secure_bootloader_key}"
  47. COMMAND ${ESPSECUREPY} digest_private_key
  48. --keylen "${key_digest_len}"
  49. --keyfile "${SECURE_BOOT_SIGNING_KEY}"
  50. "${secure_bootloader_key}"
  51. VERBATIM)
  52. if(CONFIG_SECURE_BOOT_BUILD_SIGNED_BINARIES)
  53. add_custom_target(gen_secure_bootloader_key ALL DEPENDS "${secure_bootloader_key}")
  54. else()
  55. if(NOT EXISTS "${secure_bootloader_key}")
  56. message(FATAL_ERROR
  57. "No pre-generated key for a reflashable secure bootloader is available, "
  58. "due to signing configuration."
  59. "\nTo generate one, you can use this command:"
  60. "\n\t${espsecurepy} generate_flash_encryption_key ${secure_bootloader_key}"
  61. "\nIf a signing key is present, then instead use:"
  62. "\n\t${espsecurepy} digest_private_key "
  63. "--keylen (192/256) --keyfile KEYFILE "
  64. "${secure_bootloader_key}")
  65. endif()
  66. add_custom_target(gen_secure_bootloader_key)
  67. endif()
  68. add_custom_command(OUTPUT "${bootloader_digest_bin}"
  69. COMMAND ${CMAKE_COMMAND} -E echo "DIGEST ${bootloader_digest_bin}"
  70. COMMAND ${ESPSECUREPY} digest_secure_bootloader --keyfile "${secure_bootloader_key}"
  71. -o "${bootloader_digest_bin}" "${CMAKE_BINARY_DIR}/bootloader.bin"
  72. MAIN_DEPENDENCY "${CMAKE_BINARY_DIR}/.bin_timestamp"
  73. DEPENDS gen_secure_bootloader_key gen_project_binary
  74. VERBATIM)
  75. add_custom_target (gen_bootloader_digest_bin ALL DEPENDS "${bootloader_digest_bin}")
  76. endif()
  77. if(CONFIG_SECURE_BOOT_V2_ENABLED)
  78. if(CONFIG_SECURE_BOOT_BUILD_SIGNED_BINARIES)
  79. get_filename_component(secure_boot_signing_key
  80. "${SECURE_BOOT_SIGNING_KEY}" ABSOLUTE BASE_DIR "${project_dir}")
  81. if(NOT EXISTS "${secure_boot_signing_key}")
  82. message(FATAL_ERROR
  83. "Secure Boot Signing Key Not found."
  84. "\nGenerate the Secure Boot V2 RSA-PSS 3072 Key."
  85. "\nTo generate one, you can use this command:"
  86. "\n\t${espsecurepy} generate_signing_key --version 2 ${SECURE_BOOT_SIGNING_KEY}")
  87. endif()
  88. set(bootloader_unsigned_bin "bootloader-unsigned.bin")
  89. add_custom_command(OUTPUT ".signed_bin_timestamp"
  90. COMMAND ${CMAKE_COMMAND} -E copy "${CMAKE_BINARY_DIR}/${PROJECT_BIN}" "${CMAKE_BINARY_DIR}/${bootloader_unsigned_bin}"
  91. COMMAND ${ESPSECUREPY} sign_data --version 2 --keyfile "${secure_boot_signing_key}"
  92. -o "${CMAKE_BINARY_DIR}/${PROJECT_BIN}" "${CMAKE_BINARY_DIR}/${bootloader_unsigned_bin}"
  93. COMMAND ${CMAKE_COMMAND} -E echo "Generated signed binary image ${build_dir}/${PROJECT_BIN}"
  94. "from ${CMAKE_BINARY_DIR}/${bootloader_unsigned_bin}"
  95. COMMAND ${CMAKE_COMMAND} -E md5sum "${CMAKE_BINARY_DIR}/${PROJECT_BIN}" > "${CMAKE_BINARY_DIR}/.signed_bin_timestamp"
  96. DEPENDS "${build_dir}/.bin_timestamp"
  97. VERBATIM
  98. COMMENT "Generated the signed Bootloader")
  99. else()
  100. add_custom_command(OUTPUT ".signed_bin_timestamp"
  101. VERBATIM
  102. COMMENT "Bootloader generated but not signed")
  103. endif()
  104. add_custom_target (gen_signed_bootloader ALL DEPENDS "${build_dir}/.signed_bin_timestamp")
  105. endif()
  106. if(CONFIG_SECURE_BOOTLOADER_ONE_TIME_FLASH)
  107. add_custom_command(TARGET bootloader.elf POST_BUILD
  108. COMMAND ${CMAKE_COMMAND} -E echo
  109. "=============================================================================="
  110. COMMAND ${CMAKE_COMMAND} -E echo
  111. "Bootloader built. Secure boot enabled, so bootloader not flashed automatically."
  112. COMMAND ${CMAKE_COMMAND} -E echo
  113. "One-time flash command is:"
  114. COMMAND ${CMAKE_COMMAND} -E echo
  115. "\t${esptoolpy_write_flash} ${BOOTLOADER_OFFSET} ${CMAKE_BINARY_DIR}/bootloader.bin"
  116. COMMAND ${CMAKE_COMMAND} -E echo
  117. "* IMPORTANT: After first boot, BOOTLOADER CANNOT BE RE-FLASHED on same device"
  118. VERBATIM)
  119. elseif(CONFIG_SECURE_BOOTLOADER_REFLASHABLE)
  120. add_custom_command(TARGET bootloader.elf POST_BUILD
  121. COMMAND ${CMAKE_COMMAND} -E echo
  122. "=============================================================================="
  123. COMMAND ${CMAKE_COMMAND} -E echo
  124. "Bootloader built and secure digest generated."
  125. COMMAND ${CMAKE_COMMAND} -E echo
  126. "Secure boot enabled, so bootloader not flashed automatically."
  127. COMMAND ${CMAKE_COMMAND} -E echo
  128. "Burn secure boot key to efuse using:"
  129. COMMAND ${CMAKE_COMMAND} -E echo
  130. "\t${espefusepy} burn_key secure_boot_v1 ${secure_bootloader_key}"
  131. COMMAND ${CMAKE_COMMAND} -E echo
  132. "First time flash command is:"
  133. COMMAND ${CMAKE_COMMAND} -E echo
  134. "\t${esptoolpy_write_flash} ${BOOTLOADER_OFFSET} ${CMAKE_BINARY_DIR}/bootloader.bin"
  135. COMMAND ${CMAKE_COMMAND} -E echo
  136. "=============================================================================="
  137. COMMAND ${CMAKE_COMMAND} -E echo
  138. "To reflash the bootloader after initial flash:"
  139. COMMAND ${CMAKE_COMMAND} -E echo
  140. "\t${esptoolpy_write_flash} 0x0 ${bootloader_digest_bin}"
  141. COMMAND ${CMAKE_COMMAND} -E echo
  142. "=============================================================================="
  143. COMMAND ${CMAKE_COMMAND} -E echo
  144. "* After first boot, only re-flashes of this kind (with same key) will be accepted."
  145. COMMAND ${CMAKE_COMMAND} -E echo
  146. "* Not recommended to re-use the same secure boot keyfile on multiple production devices."
  147. DEPENDS gen_secure_bootloader_key gen_bootloader_digest_bin
  148. VERBATIM)
  149. elseif(CONFIG_SECURE_BOOT_V2_ENABLED AND CONFIG_IDF_TARGET_ESP32S2)
  150. add_custom_command(TARGET bootloader.elf POST_BUILD
  151. COMMAND ${CMAKE_COMMAND} -E echo
  152. "=============================================================================="
  153. COMMAND ${CMAKE_COMMAND} -E echo
  154. "Bootloader built. Secure boot enabled, so bootloader not flashed automatically."
  155. COMMAND ${CMAKE_COMMAND} -E echo
  156. "To sign the bootloader with additional private keys."
  157. COMMAND ${CMAKE_COMMAND} -E echo
  158. "\t${espsecurepy} sign_data -k secure_boot_signing_key2.pem -v 2 --append_signatures -o signed_bootloader.bin build/bootloader/bootloader.bin"
  159. COMMAND ${CMAKE_COMMAND} -E echo
  160. "Secure boot enabled, so bootloader not flashed automatically."
  161. COMMAND ${CMAKE_COMMAND} -E echo
  162. "\t${esptoolpy_write_flash} ${BOOTLOADER_OFFSET} ${CMAKE_BINARY_DIR}/bootloader.bin"
  163. COMMAND ${CMAKE_COMMAND} -E echo
  164. "=============================================================================="
  165. DEPENDS gen_signed_bootloader
  166. VERBATIM)
  167. elseif(CONFIG_SECURE_BOOT_V2_ENABLED)
  168. add_custom_command(TARGET bootloader.elf POST_BUILD
  169. COMMAND ${CMAKE_COMMAND} -E echo
  170. "=============================================================================="
  171. COMMAND ${CMAKE_COMMAND} -E echo
  172. "Bootloader built. Secure boot enabled, so bootloader not flashed automatically."
  173. COMMAND ${CMAKE_COMMAND} -E echo
  174. "Secure boot enabled, so bootloader not flashed automatically."
  175. COMMAND ${CMAKE_COMMAND} -E echo
  176. "\t${esptoolpy_write_flash} ${BOOTLOADER_OFFSET} ${CMAKE_BINARY_DIR}/bootloader.bin"
  177. COMMAND ${CMAKE_COMMAND} -E echo
  178. "=============================================================================="
  179. DEPENDS gen_signed_bootloader
  180. VERBATIM)
  181. endif()