123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361 |
- /*
- * Copyright (c) 2017-2022, STMicroelectronics - All Rights Reserved
- *
- * SPDX-License-Identifier: BSD-3-Clause
- */
- #include <asm/byteorder.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <stdint.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <sys/mman.h>
- #include <sys/stat.h>
- #include <sys/types.h>
- #include <unistd.h>
- /* 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;
- }
|