/* * Copyright (c) 2016 Inria * 2017 Kaspar Schleiser * * This file is subject to the terms and conditions of the GNU General Public * License v2. See the file LICENSE for more details. * */ /** * @author Francisco Acosta * @author Kaspar Schleiser */ #include #include #include #include #include #include #include "riotboot/hdr.h" #include "common.h" /** * @brief Alignment required to set VTOR in Cortex-M0+/3/4/7 */ #define HDR_ALIGN (256) static void populate_hdr(riotboot_hdr_t *hdr, uint32_t ver, uint32_t addr) { /* ensure the buffer and header have 0's */ memset(hdr, '\0', sizeof(riotboot_hdr_t)); /* Generate image header */ hdr->magic_number = RIOTBOOT_MAGIC; hdr->version = ver; hdr->start_addr = addr; /* calculate header checksum */ hdr->chksum = riotboot_hdr_checksum(hdr); } int genhdr(int argc, char *argv[]) { const char generate_usage[] = " "; /* riotboot_hdr buffer */ uint8_t *hdr_buf; /* arguments storage variables */ unsigned long app_ver_arg = 0; unsigned long start_addr_arg = 0; unsigned long hdr_len_arg = 0; /* header variables */ size_t hdr_len = 0; uint32_t app_ver = 0; uint32_t start_addr = 0; /* helpers */ errno = 0; char *p; if (argc < 6) { fprintf(stderr, "usage: genhdr generate %s\n", generate_usage); return -1; } app_ver_arg = strtol(argv[2], &p, 0); if (errno != 0 || *p != '\0' || app_ver_arg > UINT32_MAX) { fprintf(stderr, "Error: APP_VER not valid!\n"); } else { app_ver = app_ver_arg; } start_addr_arg = strtol(argv[3], &p, 0); if (errno != 0 || *p != '\0' || start_addr_arg > UINT32_MAX) { fprintf(stderr, "Error: START_ADDR not valid!\n"); } else { start_addr = start_addr_arg; } hdr_len_arg = strtol(argv[4], &p, 0); if (errno != 0 || *p != '\0' || hdr_len_arg % HDR_ALIGN || hdr_len_arg > UINT32_MAX) { fprintf(stderr, "Error: HDR_LEN not valid!\n"); return -1; } else { hdr_len = hdr_len_arg; } /* prepare a 0 initialised buffer for riotboot_hdr_t */ hdr_buf = calloc(1, hdr_len); if (hdr_buf == NULL) { fprintf(stderr, "Error: not enough memory!\n"); return -1; } populate_hdr((riotboot_hdr_t*)hdr_buf, app_ver, start_addr); /* Write the header */ if (!to_file(argv[5], hdr_buf, hdr_len)) { fprintf(stderr, "Error: cannot write output\n"); free(hdr_buf); return 1; } free(hdr_buf); return 0; } int updatehdr(int argc, char *argv[]) { if (argc < 3) { fprintf(stderr, "usage: genhdr update \n"); return -1; } const char *file = argv[1]; riotboot_hdr_t hdr = { 0 }; int res = from_file(file, &hdr, sizeof(hdr)); if (res < (int)sizeof(hdr)) { fprintf(stderr, "Can't read header from %s\n", file); return -EIO; } if (hdr.magic_number != RIOTBOOT_MAGIC) { fprintf(stderr, "Invalid magic: %x\n", hdr.magic_number); return -EIO; } hdr.version = atoi(argv[2]); hdr.chksum = riotboot_hdr_checksum(&hdr); to_file(file, &hdr, sizeof(hdr)); return 0; } int readhdr(const char *file) { riotboot_hdr_t hdr = { 0 }; int res = from_file(file, &hdr, sizeof(hdr)); if (res < (int)sizeof(hdr)) { fprintf(stderr, "Can't read header from %s\n", file); return -EIO; } if (hdr.magic_number != RIOTBOOT_MAGIC) { fprintf(stderr, "Invalid magic: %x\n", hdr.magic_number); return -EIO; } printf("version: %u\n", hdr.version); printf("address: 0x%x\n", hdr.start_addr); printf("checksum: %svalid\n", riotboot_hdr_validate(&hdr) ? "in" : ""); return 0; }