mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-03 06:28:27 -04:00
AP_CheckFirmare: add support for signed firmwares
This commit is contained in:
parent
3f951c3e0b
commit
d434038b26
@ -8,15 +8,71 @@
|
||||
#if AP_CHECK_FIRMWARE_ENABLED
|
||||
|
||||
#if defined(HAL_BOOTLOADER_BUILD)
|
||||
|
||||
#if AP_SIGNED_FIRMWARE
|
||||
#include "../../Tools/AP_Bootloader/support.h"
|
||||
#include <string.h>
|
||||
#include "monocypher.h"
|
||||
|
||||
struct PACKED secure_data {
|
||||
uint8_t sig[8] = {0x4e, 0xcf, 0x4e, 0xa5, 0xa6, 0xb6, 0xf7, 0x29};
|
||||
struct PACKED {
|
||||
uint8_t key[32] = {};
|
||||
} public_key[10];
|
||||
};
|
||||
|
||||
const struct secure_data public_keys __attribute__((section(".ecc_raw")));
|
||||
|
||||
/*
|
||||
check a signature against bootloader keys
|
||||
*/
|
||||
static check_fw_result_t check_firmware_signature(const app_descriptor *ad,
|
||||
const uint8_t *flash1, uint32_t len1,
|
||||
const uint8_t *flash2, uint32_t len2)
|
||||
{
|
||||
// 8 byte signature version
|
||||
static const uint64_t sig_version = 30437LLU;
|
||||
|
||||
if (ad->signature_length != 72) {
|
||||
return check_fw_result_t::FAIL_REASON_BAD_FIRMWARE_SIGNATURE;
|
||||
}
|
||||
if (memcmp((const uint8_t*)&sig_version, ad->signature, sizeof(sig_version)) != 0) {
|
||||
return check_fw_result_t::FAIL_REASON_BAD_FIRMWARE_SIGNATURE;
|
||||
}
|
||||
|
||||
/*
|
||||
look over all public keys, if one matches then we are OK
|
||||
*/
|
||||
for (const auto &public_key : public_keys.public_key) {
|
||||
crypto_check_ctx ctx {};
|
||||
crypto_check_ctx_abstract *actx = (crypto_check_ctx_abstract*)&ctx;
|
||||
crypto_check_init(actx, &ad->signature[sizeof(sig_version)], public_key.key);
|
||||
|
||||
crypto_check_update(actx, flash1, len1);
|
||||
crypto_check_update(actx, flash2, len2);
|
||||
if (crypto_check_final(actx) == 0) {
|
||||
// good signature
|
||||
return check_fw_result_t::CHECK_FW_OK;
|
||||
}
|
||||
}
|
||||
// none of the public keys matched
|
||||
return check_fw_result_t::FAIL_REASON_VERIFICATION;
|
||||
}
|
||||
#endif // AP_SIGNED_FIRMWARE
|
||||
|
||||
/*
|
||||
check firmware CRC and board ID to see if it matches
|
||||
*/
|
||||
check_fw_result_t check_good_firmware(void)
|
||||
{
|
||||
#if AP_SIGNED_FIRMWARE
|
||||
const uint8_t sig[8] = { 0x41, 0xa3, 0xe5, 0xf2, 0x65, 0x69, 0x92, 0x07 };
|
||||
#else
|
||||
const uint8_t sig[8] = { 0x40, 0xa2, 0xe4, 0xf1, 0x64, 0x68, 0x91, 0x06 };
|
||||
const uint8_t *flash = (const uint8_t *)(FLASH_LOAD_ADDRESS + (FLASH_BOOTLOADER_LOAD_KB + APP_START_OFFSET_KB)*1024);
|
||||
#endif
|
||||
const uint8_t *flash1 = (const uint8_t *)(FLASH_LOAD_ADDRESS + (FLASH_BOOTLOADER_LOAD_KB + APP_START_OFFSET_KB)*1024);
|
||||
const uint32_t flash_size = (BOARD_FLASH_SIZE - (FLASH_BOOTLOADER_LOAD_KB + APP_START_OFFSET_KB))*1024;
|
||||
const app_descriptor *ad = (const app_descriptor *)memmem(flash, flash_size-sizeof(app_descriptor), sig, sizeof(sig));
|
||||
const app_descriptor *ad = (const app_descriptor *)memmem(flash1, flash_size-sizeof(app_descriptor), sig, sizeof(sig));
|
||||
if (ad == nullptr) {
|
||||
// no application signature
|
||||
return check_fw_result_t::FAIL_REASON_NO_APP_SIG;
|
||||
@ -35,19 +91,28 @@ check_fw_result_t check_good_firmware(void)
|
||||
return check_fw_result_t::FAIL_REASON_BAD_BOARD_ID;
|
||||
}
|
||||
|
||||
const uint8_t *flash2 = (const uint8_t *)&ad->version_major;
|
||||
const uint8_t desc_len = offsetof(app_descriptor, version_major) - offsetof(app_descriptor, image_crc1);
|
||||
uint32_t len1 = ((const uint8_t *)&ad->image_crc1) - flash;
|
||||
const uint32_t len1 = ((const uint8_t *)&ad->image_crc1) - flash1;
|
||||
|
||||
if ((len1 + desc_len) > ad->image_size) {
|
||||
return check_fw_result_t::FAIL_REASON_BAD_LENGTH_DESCRIPTOR;
|
||||
}
|
||||
|
||||
uint32_t len2 = ad->image_size - (len1 + desc_len);
|
||||
uint32_t crc1 = crc32_small(0, flash, len1);
|
||||
uint32_t crc2 = crc32_small(0, (const uint8_t *)&ad->version_major, len2);
|
||||
const uint32_t len2 = ad->image_size - (len1 + desc_len);
|
||||
uint32_t crc1 = crc32_small(0, flash1, len1);
|
||||
uint32_t crc2 = crc32_small(0, flash2, len2);
|
||||
if (crc1 != ad->image_crc1 || crc2 != ad->image_crc2) {
|
||||
return check_fw_result_t::FAIL_REASON_BAD_CRC;
|
||||
}
|
||||
return check_fw_result_t::CHECK_FW_OK;
|
||||
|
||||
check_fw_result_t ret = check_fw_result_t::CHECK_FW_OK;
|
||||
|
||||
#if AP_SIGNED_FIRMWARE
|
||||
ret = check_firmware_signature(ad, flash1, len1, flash2, len2);
|
||||
#endif
|
||||
|
||||
return ret;
|
||||
}
|
||||
#endif // HAL_BOOTLOADER_BUILD
|
||||
|
||||
|
@ -5,6 +5,7 @@
|
||||
|
||||
#include <AP_HAL/AP_HAL_Boards.h>
|
||||
#include <AP_OpenDroneID/AP_OpenDroneID_config.h>
|
||||
#include <AP_HAL/AP_HAL.h>
|
||||
|
||||
#ifndef AP_CHECK_FIRMWARE_ENABLED
|
||||
#define AP_CHECK_FIRMWARE_ENABLED AP_OPENDRONEID_ENABLED
|
||||
@ -21,6 +22,8 @@ enum class check_fw_result_t : uint8_t {
|
||||
FAIL_REASON_IN_UPDATE = 14,
|
||||
FAIL_REASON_WATCHDOG = 15,
|
||||
FAIL_REASON_BAD_LENGTH_DESCRIPTOR = 16,
|
||||
FAIL_REASON_BAD_FIRMWARE_SIGNATURE = 17,
|
||||
FAIL_REASON_VERIFICATION = 18,
|
||||
};
|
||||
|
||||
#ifndef FW_MAJOR
|
||||
@ -43,7 +46,11 @@ enum class check_fw_result_t : uint8_t {
|
||||
are filled in by set_app_descriptor() in the waf build
|
||||
*/
|
||||
struct app_descriptor {
|
||||
#if AP_SIGNED_FIRMWARE
|
||||
uint8_t sig[8] = { 0x41, 0xa3, 0xe5, 0xf2, 0x65, 0x69, 0x92, 0x07 };
|
||||
#else
|
||||
uint8_t sig[8] = { 0x40, 0xa2, 0xe4, 0xf1, 0x64, 0x68, 0x91, 0x06 };
|
||||
#endif
|
||||
// crc1 is the crc32 from firmware start to start of image_crc1
|
||||
uint32_t image_crc1 = 0;
|
||||
// crc2 is the crc32 from the start of version_major to the end of the firmware
|
||||
@ -51,6 +58,12 @@ struct app_descriptor {
|
||||
// total size of firmware image in bytes
|
||||
uint32_t image_size = 0;
|
||||
uint32_t git_hash = 0;
|
||||
|
||||
#if AP_SIGNED_FIRMWARE
|
||||
// firmware signature
|
||||
uint32_t signature_length = 0;
|
||||
uint8_t signature[72] = {};
|
||||
#endif
|
||||
// software version number
|
||||
uint8_t version_major = APP_FW_MAJOR;
|
||||
uint8_t version_minor = APP_FW_MINOR;
|
||||
@ -58,9 +71,15 @@ struct app_descriptor {
|
||||
// with high byte in HardwareVersion.major and low byte in HardwareVersion.minor
|
||||
uint16_t board_id = APJ_BOARD_ID;
|
||||
uint8_t reserved[8] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
|
||||
|
||||
};
|
||||
|
||||
#if AP_SIGNED_FIRMWARE
|
||||
#define APP_DESCRIPTOR_TOTAL_LENGTH (36+72+4)
|
||||
#else
|
||||
#define APP_DESCRIPTOR_TOTAL_LENGTH 36
|
||||
#endif
|
||||
|
||||
static_assert(sizeof(app_descriptor) == APP_DESCRIPTOR_TOTAL_LENGTH, "app_descriptor incorrect length");
|
||||
|
||||
#ifdef HAL_BOOTLOADER_BUILD
|
||||
|
Loading…
Reference in New Issue
Block a user