2021-05-10 04:35:07 +02:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: 2015-2021 Espressif Systems (Shanghai) CO LTD
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
2018-01-31 14:45:12 +01:00
|
|
|
#include <string.h>
|
2016-11-11 17:00:34 +11:00
|
|
|
#include "esp_flash_partitions.h"
|
|
|
|
#include "esp_log.h"
|
2020-07-21 17:01:28 +08:00
|
|
|
#include "esp_rom_md5.h"
|
2020-11-26 19:56:13 +11:00
|
|
|
#if CONFIG_IDF_TARGET_ESP32C3
|
|
|
|
#include "esp32c3/rom/spi_flash.h"
|
|
|
|
#elif CONFIG_IDF_TARGET_ESP32S2
|
|
|
|
#include "esp32s2/rom/spi_flash.h"
|
2021-03-17 18:48:05 +08:00
|
|
|
#elif CONFIG_IDF_TARGET_ESP32S3
|
|
|
|
#include "esp32s3/rom/spi_flash.h"
|
2021-06-10 19:28:18 +08:00
|
|
|
#elif CONFIG_IDF_TARGET_ESP32H2
|
|
|
|
#include "esp32h2/rom/spi_flash.h"
|
2020-11-26 19:56:13 +11:00
|
|
|
#else
|
|
|
|
#include "esp32/rom/spi_flash.h"
|
|
|
|
#endif
|
2016-11-11 17:00:34 +11:00
|
|
|
|
|
|
|
static const char *TAG = "flash_parts";
|
|
|
|
|
2018-07-13 15:23:04 +10:00
|
|
|
esp_err_t esp_partition_table_verify(const esp_partition_info_t *partition_table, bool log_errors, int *num_partitions)
|
2016-11-11 17:00:34 +11:00
|
|
|
{
|
2018-01-31 14:45:12 +01:00
|
|
|
int md5_found = 0;
|
2020-11-17 12:48:35 +08:00
|
|
|
size_t num_parts;
|
2018-01-31 14:45:12 +01:00
|
|
|
uint32_t chip_size = g_rom_flashchip.chip_size;
|
|
|
|
*num_partitions = 0;
|
2016-11-11 17:00:34 +11:00
|
|
|
|
2018-01-31 14:45:12 +01:00
|
|
|
for (num_parts = 0; num_parts < ESP_PARTITION_TABLE_MAX_ENTRIES; num_parts++) {
|
|
|
|
const esp_partition_info_t *part = &partition_table[num_parts];
|
|
|
|
|
|
|
|
if (part->magic == ESP_PARTITION_MAGIC) {
|
|
|
|
const esp_partition_pos_t *pos = &part->pos;
|
|
|
|
if (pos->offset > chip_size || pos->offset + pos->size > chip_size) {
|
|
|
|
if (log_errors) {
|
|
|
|
ESP_LOGE(TAG, "partition %d invalid - offset 0x%x size 0x%x exceeds flash chip size 0x%x",
|
|
|
|
num_parts, pos->offset, pos->size, chip_size);
|
|
|
|
}
|
|
|
|
return ESP_ERR_INVALID_SIZE;
|
|
|
|
}
|
|
|
|
} else if (part->magic == ESP_PARTITION_MAGIC_MD5) {
|
|
|
|
if (md5_found) {
|
|
|
|
if (log_errors) {
|
|
|
|
ESP_LOGE(TAG, "Only one MD5 checksum is allowed");
|
|
|
|
}
|
|
|
|
return ESP_ERR_INVALID_STATE;
|
|
|
|
}
|
2016-12-30 13:16:22 +11:00
|
|
|
|
2018-01-31 14:45:12 +01:00
|
|
|
struct MD5Context context;
|
|
|
|
unsigned char digest[16];
|
2020-07-21 17:01:28 +08:00
|
|
|
esp_rom_md5_init(&context);
|
|
|
|
esp_rom_md5_update(&context, (unsigned char *) partition_table, num_parts * sizeof(esp_partition_info_t));
|
|
|
|
esp_rom_md5_final(digest, &context);
|
2018-01-31 14:45:12 +01:00
|
|
|
|
2021-02-04 11:12:04 +11:00
|
|
|
unsigned char *md5sum = ((unsigned char *) part) + ESP_PARTITION_MD5_OFFSET;
|
2018-01-31 14:45:12 +01:00
|
|
|
|
|
|
|
if (memcmp(md5sum, digest, sizeof(digest)) != 0) {
|
|
|
|
if (log_errors) {
|
|
|
|
ESP_LOGE(TAG, "Incorrect MD5 checksum");
|
|
|
|
}
|
|
|
|
return ESP_ERR_INVALID_STATE;
|
|
|
|
}
|
|
|
|
//MD5 checksum matches and we continue with the next interation in
|
|
|
|
//order to detect the end of the partition table
|
|
|
|
md5_found = 1;
|
|
|
|
} else if (part->magic == 0xFFFF
|
|
|
|
&& part->type == PART_TYPE_END
|
|
|
|
&& part->subtype == PART_SUBTYPE_END) {
|
|
|
|
ESP_LOGD(TAG, "partition table verified, %d entries", num_parts);
|
|
|
|
*num_partitions = num_parts - md5_found; //do not count the partition where the MD5 checksum is held
|
|
|
|
return ESP_OK;
|
|
|
|
} else {
|
|
|
|
if (log_errors) {
|
|
|
|
ESP_LOGE(TAG, "partition %d invalid magic number 0x%x", num_parts, part->magic);
|
|
|
|
}
|
|
|
|
return ESP_ERR_INVALID_STATE;
|
2016-12-30 13:16:22 +11:00
|
|
|
}
|
|
|
|
}
|
2016-11-11 17:00:34 +11:00
|
|
|
|
2018-01-31 14:45:12 +01:00
|
|
|
if (log_errors) {
|
|
|
|
ESP_LOGE(TAG, "partition table has no terminating entry, not valid");
|
|
|
|
}
|
|
|
|
return ESP_ERR_INVALID_STATE;
|
2016-11-11 17:00:34 +11:00
|
|
|
}
|