/* * Copyright (c) 2017-2022, STMicroelectronics - All Rights Reserved * * SPDX-License-Identifier: BSD-3-Clause */ #include #include #include #include #include #include #include #include #include #include #include /* Magic = 'S' 'T' 'M' 0x32 */ #define HEADER_MAGIC __be32_to_cpu(0x53544D32) #define VER_MAJOR 2 #define VER_MINOR 1 #define VER_VARIANT 0 #define HEADER_VERSION_V1 0x1 #define HEADER_VERSION_V2 0x2 #define PADDING_HEADER_MAGIC __be32_to_cpu(0x5354FFFF) #define PADDING_HEADER_FLAG (1 << 31) #define PADDING_HEADER_LENGTH 0x180 struct stm32_header_v1 { uint32_t magic_number; uint8_t image_signature[64]; uint32_t image_checksum; uint8_t header_version[4]; uint32_t image_length; uint32_t image_entry_point; uint32_t reserved1; uint32_t load_address; uint32_t reserved2; uint32_t version_number; uint32_t option_flags; uint32_t ecdsa_algorithm; uint8_t ecdsa_public_key[64]; uint8_t padding[83]; uint8_t binary_type; }; struct stm32_header_v2 { uint32_t magic_number; uint8_t image_signature[64]; uint32_t image_checksum; uint8_t header_version[4]; uint32_t image_length; uint32_t image_entry_point; uint32_t reserved1; uint32_t load_address; uint32_t reserved2; uint32_t version_number; uint32_t extension_flags; uint32_t extension_headers_length; uint32_t binary_type; uint8_t padding[16]; uint32_t extension_header_type; uint32_t extension_header_length; uint8_t extension_padding[376]; }; static void stm32image_default_header(void *ptr) { struct stm32_header_v1 *header = (struct stm32_header_v1 *)ptr; if (!header) { return; } header->magic_number = HEADER_MAGIC; header->version_number = __cpu_to_le32(0); } static uint32_t stm32image_checksum(void *start, uint32_t len, uint32_t header_size) { uint32_t csum = 0; uint8_t *p; if (len < header_size) { return 0; } p = (unsigned char *)start + header_size; len -= header_size; while (len > 0) { csum += *p; p++; len--; } return csum; } static void stm32image_print_header(const void *ptr) { struct stm32_header_v1 *stm32hdr = (struct stm32_header_v1 *)ptr; struct stm32_header_v2 *stm32hdr_v2 = (struct stm32_header_v2 *)ptr; printf("Image Type : ST Microelectronics STM32 V%d.%d\n", stm32hdr->header_version[VER_MAJOR], stm32hdr->header_version[VER_MINOR]); printf("Image Size : %lu bytes\n", (unsigned long)__le32_to_cpu(stm32hdr->image_length)); printf("Image Load : 0x%08x\n", __le32_to_cpu(stm32hdr->load_address)); printf("Entry Point : 0x%08x\n", __le32_to_cpu(stm32hdr->image_entry_point)); printf("Checksum : 0x%08x\n", __le32_to_cpu(stm32hdr->image_checksum)); switch (stm32hdr->header_version[VER_MAJOR]) { case HEADER_VERSION_V1: printf("Option : 0x%08x\n", __le32_to_cpu(stm32hdr->option_flags)); break; case HEADER_VERSION_V2: printf("Extension : 0x%08x\n", __le32_to_cpu(stm32hdr_v2->extension_flags)); break; default: printf("Incorrect header version\n"); } printf("Version : 0x%08x\n", __le32_to_cpu(stm32hdr->version_number)); } static int stm32image_set_header(void *ptr, struct stat *sbuf, int ifd, uint32_t loadaddr, uint32_t ep, uint32_t ver, uint32_t major, uint32_t minor, uint32_t binary_type, uint32_t header_size) { struct stm32_header_v1 *stm32hdr = (struct stm32_header_v1 *)ptr; struct stm32_header_v2 *stm32hdr_v2 = (struct stm32_header_v2 *)ptr; uint32_t ext_size = 0U; uint32_t ext_flags = 0U; stm32image_default_header(ptr); stm32hdr->header_version[VER_MAJOR] = major; stm32hdr->header_version[VER_MINOR] = minor; stm32hdr->load_address = __cpu_to_le32(loadaddr); stm32hdr->image_entry_point = __cpu_to_le32(ep); stm32hdr->image_length = __cpu_to_le32((uint32_t)sbuf->st_size - header_size); stm32hdr->image_checksum = __cpu_to_le32(stm32image_checksum(ptr, sbuf->st_size, header_size)); switch (stm32hdr->header_version[VER_MAJOR]) { case HEADER_VERSION_V1: /* Default option for header v1 : bit0 => no signature */ stm32hdr->option_flags = __cpu_to_le32(0x00000001); stm32hdr->ecdsa_algorithm = __cpu_to_le32(1); stm32hdr->binary_type = (uint8_t)binary_type; break; case HEADER_VERSION_V2: stm32hdr_v2->binary_type = binary_type; ext_size += PADDING_HEADER_LENGTH; ext_flags |= PADDING_HEADER_FLAG; stm32hdr_v2->extension_flags = __cpu_to_le32(ext_flags); stm32hdr_v2->extension_headers_length = __cpu_to_le32(ext_size); stm32hdr_v2->extension_header_type = PADDING_HEADER_MAGIC; stm32hdr_v2->extension_header_length = __cpu_to_le32(PADDING_HEADER_LENGTH); break; default: return -1; } stm32hdr->version_number = __cpu_to_le32(ver); return 0; } static int stm32image_create_header_file(char *srcname, char *destname, uint32_t loadaddr, uint32_t entry, uint32_t version, uint32_t major, uint32_t minor, uint32_t binary_type) { int src_fd, dest_fd, header_size; struct stat sbuf; unsigned char *ptr; void *stm32image_header; dest_fd = open(destname, O_RDWR | O_CREAT | O_TRUNC | O_APPEND, 0666); if (dest_fd == -1) { fprintf(stderr, "Can't open %s: %s\n", destname, strerror(errno)); return -1; } src_fd = open(srcname, O_RDONLY); if (src_fd == -1) { fprintf(stderr, "Can't open %s: %s\n", srcname, strerror(errno)); return -1; } if (fstat(src_fd, &sbuf) < 0) { return -1; } ptr = mmap(NULL, sbuf.st_size, PROT_READ, MAP_SHARED, src_fd, 0); if (ptr == MAP_FAILED) { fprintf(stderr, "Can't read %s\n", srcname); return -1; } switch (major) { case HEADER_VERSION_V1: stm32image_header = malloc(sizeof(struct stm32_header_v1)); header_size = sizeof(struct stm32_header_v1); break; case HEADER_VERSION_V2: stm32image_header = malloc(sizeof(struct stm32_header_v2)); header_size = sizeof(struct stm32_header_v2); break; default: return -1; } memset(stm32image_header, 0, header_size); if (write(dest_fd, stm32image_header, header_size) != header_size) { fprintf(stderr, "Write error %s: %s\n", destname, strerror(errno)); free(stm32image_header); return -1; } free(stm32image_header); if (write(dest_fd, ptr, sbuf.st_size) != sbuf.st_size) { fprintf(stderr, "Write error on %s: %s\n", destname, strerror(errno)); return -1; } munmap((void *)ptr, sbuf.st_size); close(src_fd); if (fstat(dest_fd, &sbuf) < 0) { return -1; } ptr = mmap(0, sbuf.st_size, PROT_READ | PROT_WRITE, MAP_SHARED, dest_fd, 0); if (ptr == MAP_FAILED) { fprintf(stderr, "Can't write %s\n", destname); return -1; } if (stm32image_set_header(ptr, &sbuf, dest_fd, loadaddr, entry, version, major, minor, binary_type, header_size) != 0) { return -1; } stm32image_print_header(ptr); munmap((void *)ptr, sbuf.st_size); close(dest_fd); return 0; } int main(int argc, char *argv[]) { int opt; int loadaddr = -1; int entry = -1; int err = 0; int version = 0; int binary_type = -1; int major = HEADER_VERSION_V2; int minor = 0; char *dest = NULL; char *src = NULL; while ((opt = getopt(argc, argv, ":b:s:d:l:e:v:m:n:")) != -1) { switch (opt) { case 'b': binary_type = strtol(optarg, NULL, 0); break; case 's': src = optarg; break; case 'd': dest = optarg; break; case 'l': loadaddr = strtol(optarg, NULL, 0); break; case 'e': entry = strtol(optarg, NULL, 0); break; case 'v': version = strtol(optarg, NULL, 0); break; case 'm': major = strtol(optarg, NULL, 0); break; case 'n': minor = strtol(optarg, NULL, 0); break; default: fprintf(stderr, "Usage : %s [-s srcfile] [-d destfile] [-l loadaddr] [-e entry_point] [-m major] [-n minor] [-b binary_type]\n", argv[0]); return -1; } } if (!src) { fprintf(stderr, "Missing -s option\n"); return -1; } if (!dest) { fprintf(stderr, "Missing -d option\n"); return -1; } if (loadaddr == -1) { fprintf(stderr, "Missing -l option\n"); return -1; } if (entry == -1) { fprintf(stderr, "Missing -e option\n"); return -1; } if (binary_type == -1) { fprintf(stderr, "Missing -b option\n"); return -1; } err = stm32image_create_header_file(src, dest, loadaddr, entry, version, major, minor, binary_type); return err; }