1819281eeSdp-arm /* 22a6c1a8fSMasahiro Yamada * Copyright (c) 2016-2017, ARM Limited and Contributors. All rights reserved. 3819281eeSdp-arm * 482cb2c1aSdp-arm * SPDX-License-Identifier: BSD-3-Clause 5819281eeSdp-arm */ 6819281eeSdp-arm 7819281eeSdp-arm #include <sys/types.h> 8819281eeSdp-arm #include <sys/stat.h> 9819281eeSdp-arm 10819281eeSdp-arm #include <assert.h> 11819281eeSdp-arm #include <errno.h> 12819281eeSdp-arm #include <getopt.h> 13819281eeSdp-arm #include <limits.h> 14819281eeSdp-arm #include <stdarg.h> 15819281eeSdp-arm #include <stdint.h> 16819281eeSdp-arm #include <stdio.h> 17819281eeSdp-arm #include <stdlib.h> 18819281eeSdp-arm #include <string.h> 19819281eeSdp-arm #include <unistd.h> 20819281eeSdp-arm 219df69ba3Sdp-arm #include <openssl/sha.h> 22769fb6b6SIsla Mitchell 232a6c1a8fSMasahiro Yamada #include <firmware_image_package.h> 242a6c1a8fSMasahiro Yamada 25819281eeSdp-arm #include "fiptool.h" 26819281eeSdp-arm #include "tbbr_config.h" 27819281eeSdp-arm 28819281eeSdp-arm #define OPT_TOC_ENTRY 0 29819281eeSdp-arm #define OPT_PLAT_TOC_FLAGS 1 301c75d5dfSMasahiro Yamada #define OPT_ALIGN 2 31819281eeSdp-arm 32819281eeSdp-arm static int info_cmd(int argc, char *argv[]); 33819281eeSdp-arm static void info_usage(void); 34819281eeSdp-arm static int create_cmd(int argc, char *argv[]); 35819281eeSdp-arm static void create_usage(void); 36819281eeSdp-arm static int update_cmd(int argc, char *argv[]); 37819281eeSdp-arm static void update_usage(void); 38819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]); 39819281eeSdp-arm static void unpack_usage(void); 40819281eeSdp-arm static int remove_cmd(int argc, char *argv[]); 41819281eeSdp-arm static void remove_usage(void); 42819281eeSdp-arm static int version_cmd(int argc, char *argv[]); 43819281eeSdp-arm static void version_usage(void); 44819281eeSdp-arm static int help_cmd(int argc, char *argv[]); 45819281eeSdp-arm static void usage(void); 46819281eeSdp-arm 47819281eeSdp-arm /* Available subcommands. */ 48819281eeSdp-arm static cmd_t cmds[] = { 49819281eeSdp-arm { .name = "info", .handler = info_cmd, .usage = info_usage }, 50819281eeSdp-arm { .name = "create", .handler = create_cmd, .usage = create_usage }, 51819281eeSdp-arm { .name = "update", .handler = update_cmd, .usage = update_usage }, 52819281eeSdp-arm { .name = "unpack", .handler = unpack_cmd, .usage = unpack_usage }, 53819281eeSdp-arm { .name = "remove", .handler = remove_cmd, .usage = remove_usage }, 54819281eeSdp-arm { .name = "version", .handler = version_cmd, .usage = version_usage }, 55819281eeSdp-arm { .name = "help", .handler = help_cmd, .usage = NULL }, 56819281eeSdp-arm }; 57819281eeSdp-arm 58e0f083a0Sdp-arm static image_desc_t *image_desc_head; 59e0f083a0Sdp-arm static size_t nr_image_descs; 60819281eeSdp-arm static uuid_t uuid_null = { 0 }; 61819281eeSdp-arm static int verbose; 62819281eeSdp-arm 6360b499feSdp-arm static void vlog(int prio, const char *msg, va_list ap) 64819281eeSdp-arm { 65819281eeSdp-arm char *prefix[] = { "DEBUG", "WARN", "ERROR" }; 66819281eeSdp-arm 67819281eeSdp-arm fprintf(stderr, "%s: ", prefix[prio]); 68819281eeSdp-arm vfprintf(stderr, msg, ap); 69819281eeSdp-arm fputc('\n', stderr); 70819281eeSdp-arm } 71819281eeSdp-arm 7260b499feSdp-arm static void log_dbgx(const char *msg, ...) 73819281eeSdp-arm { 74819281eeSdp-arm va_list ap; 75819281eeSdp-arm 76819281eeSdp-arm va_start(ap, msg); 77819281eeSdp-arm vlog(LOG_DBG, msg, ap); 78819281eeSdp-arm va_end(ap); 79819281eeSdp-arm } 80819281eeSdp-arm 8160b499feSdp-arm static void log_warnx(const char *msg, ...) 82819281eeSdp-arm { 83819281eeSdp-arm va_list ap; 84819281eeSdp-arm 85819281eeSdp-arm va_start(ap, msg); 86819281eeSdp-arm vlog(LOG_WARN, msg, ap); 87819281eeSdp-arm va_end(ap); 88819281eeSdp-arm } 89819281eeSdp-arm 9060b499feSdp-arm static void log_err(const char *msg, ...) 91819281eeSdp-arm { 92819281eeSdp-arm char buf[512]; 93819281eeSdp-arm va_list ap; 94819281eeSdp-arm 95819281eeSdp-arm va_start(ap, msg); 96819281eeSdp-arm snprintf(buf, sizeof(buf), "%s: %s", msg, strerror(errno)); 97819281eeSdp-arm vlog(LOG_ERR, buf, ap); 98819281eeSdp-arm va_end(ap); 99819281eeSdp-arm exit(1); 100819281eeSdp-arm } 101819281eeSdp-arm 10260b499feSdp-arm static void log_errx(const char *msg, ...) 103819281eeSdp-arm { 104819281eeSdp-arm va_list ap; 105819281eeSdp-arm 106819281eeSdp-arm va_start(ap, msg); 107819281eeSdp-arm vlog(LOG_ERR, msg, ap); 108819281eeSdp-arm va_end(ap); 109819281eeSdp-arm exit(1); 110819281eeSdp-arm } 111819281eeSdp-arm 112a22f6285Sdp-arm static char *xstrdup(const char *s, const char *msg) 113a22f6285Sdp-arm { 114a22f6285Sdp-arm char *d; 115a22f6285Sdp-arm 116a22f6285Sdp-arm d = strdup(s); 117a22f6285Sdp-arm if (d == NULL) 1189fc9ff1fSdp-arm log_errx("strdup: %s", msg); 119a22f6285Sdp-arm return d; 120a22f6285Sdp-arm } 121a22f6285Sdp-arm 122a22f6285Sdp-arm static void *xmalloc(size_t size, const char *msg) 123a22f6285Sdp-arm { 124a22f6285Sdp-arm void *d; 125a22f6285Sdp-arm 126a22f6285Sdp-arm d = malloc(size); 127a22f6285Sdp-arm if (d == NULL) 1289fc9ff1fSdp-arm log_errx("malloc: %s", msg); 129a22f6285Sdp-arm return d; 130a22f6285Sdp-arm } 131a22f6285Sdp-arm 132696ccba6SMasahiro Yamada static void *xzalloc(size_t size, const char *msg) 133696ccba6SMasahiro Yamada { 134696ccba6SMasahiro Yamada return memset(xmalloc(size, msg), 0, size); 135696ccba6SMasahiro Yamada } 136696ccba6SMasahiro Yamada 137a1da83f5SMasahiro Yamada static void xfwrite(void *buf, size_t size, FILE *fp, const char *filename) 138a1da83f5SMasahiro Yamada { 139a1da83f5SMasahiro Yamada if (fwrite(buf, 1, size, fp) != size) 140a1da83f5SMasahiro Yamada log_errx("Failed to write %s", filename); 141a1da83f5SMasahiro Yamada } 142a1da83f5SMasahiro Yamada 143e0f083a0Sdp-arm static image_desc_t *new_image_desc(const uuid_t *uuid, 144e0f083a0Sdp-arm const char *name, const char *cmdline_name) 145e0f083a0Sdp-arm { 146e0f083a0Sdp-arm image_desc_t *desc; 147e0f083a0Sdp-arm 148696ccba6SMasahiro Yamada desc = xzalloc(sizeof(*desc), 149e0f083a0Sdp-arm "failed to allocate memory for image descriptor"); 150e0f083a0Sdp-arm memcpy(&desc->uuid, uuid, sizeof(uuid_t)); 151e0f083a0Sdp-arm desc->name = xstrdup(name, 152e0f083a0Sdp-arm "failed to allocate memory for image name"); 153e0f083a0Sdp-arm desc->cmdline_name = xstrdup(cmdline_name, 154e0f083a0Sdp-arm "failed to allocate memory for image command line name"); 155e0f083a0Sdp-arm desc->action = DO_UNSPEC; 156e0f083a0Sdp-arm return desc; 157e0f083a0Sdp-arm } 158e0f083a0Sdp-arm 159d02fcebeSdp-arm static void set_image_desc_action(image_desc_t *desc, int action, 160d02fcebeSdp-arm const char *arg) 161d02fcebeSdp-arm { 162d02fcebeSdp-arm assert(desc != NULL); 163d02fcebeSdp-arm 164d02fcebeSdp-arm if (desc->action_arg != DO_UNSPEC) 165d02fcebeSdp-arm free(desc->action_arg); 166d02fcebeSdp-arm desc->action = action; 167d02fcebeSdp-arm desc->action_arg = NULL; 168d02fcebeSdp-arm if (arg != NULL) 169d02fcebeSdp-arm desc->action_arg = xstrdup(arg, 170d02fcebeSdp-arm "failed to allocate memory for argument"); 171d02fcebeSdp-arm } 172d02fcebeSdp-arm 173e0f083a0Sdp-arm static void free_image_desc(image_desc_t *desc) 174e0f083a0Sdp-arm { 175e0f083a0Sdp-arm free(desc->name); 176e0f083a0Sdp-arm free(desc->cmdline_name); 177e0f083a0Sdp-arm free(desc->action_arg); 178b9589fe5Sdp-arm free(desc->image); 179e0f083a0Sdp-arm free(desc); 180e0f083a0Sdp-arm } 181e0f083a0Sdp-arm 182e0f083a0Sdp-arm static void add_image_desc(image_desc_t *desc) 183e0f083a0Sdp-arm { 18411c0a4ffSMasahiro Yamada image_desc_t **p = &image_desc_head; 18511c0a4ffSMasahiro Yamada 18611c0a4ffSMasahiro Yamada while (*p) 18711c0a4ffSMasahiro Yamada p = &(*p)->next; 18811c0a4ffSMasahiro Yamada 189e9e0d287SMasahiro Yamada assert(*p == NULL); 19011c0a4ffSMasahiro Yamada *p = desc; 191e0f083a0Sdp-arm nr_image_descs++; 192e0f083a0Sdp-arm } 193e0f083a0Sdp-arm 194e0f083a0Sdp-arm static void free_image_descs(void) 195e0f083a0Sdp-arm { 196e0f083a0Sdp-arm image_desc_t *desc = image_desc_head, *tmp; 197e0f083a0Sdp-arm 198e0f083a0Sdp-arm while (desc != NULL) { 199e0f083a0Sdp-arm tmp = desc->next; 200e0f083a0Sdp-arm free_image_desc(desc); 201e0f083a0Sdp-arm desc = tmp; 202e0f083a0Sdp-arm nr_image_descs--; 203e0f083a0Sdp-arm } 204e0f083a0Sdp-arm assert(nr_image_descs == 0); 205e0f083a0Sdp-arm } 206e0f083a0Sdp-arm 207e0f083a0Sdp-arm static void fill_image_descs(void) 208e0f083a0Sdp-arm { 209e0f083a0Sdp-arm toc_entry_t *toc_entry; 210e0f083a0Sdp-arm 211e0f083a0Sdp-arm for (toc_entry = toc_entries; 212e0f083a0Sdp-arm toc_entry->cmdline_name != NULL; 213e0f083a0Sdp-arm toc_entry++) { 214e0f083a0Sdp-arm image_desc_t *desc; 215e0f083a0Sdp-arm 216e0f083a0Sdp-arm desc = new_image_desc(&toc_entry->uuid, 217e0f083a0Sdp-arm toc_entry->name, 218e0f083a0Sdp-arm toc_entry->cmdline_name); 219e0f083a0Sdp-arm add_image_desc(desc); 220e0f083a0Sdp-arm } 221e0f083a0Sdp-arm } 222e0f083a0Sdp-arm 223e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid) 224819281eeSdp-arm { 225e0f083a0Sdp-arm image_desc_t *desc; 226819281eeSdp-arm 227e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 228e0f083a0Sdp-arm if (memcmp(&desc->uuid, uuid, sizeof(uuid_t)) == 0) 229e0f083a0Sdp-arm return desc; 230819281eeSdp-arm return NULL; 231819281eeSdp-arm } 232819281eeSdp-arm 233e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_opt(const char *opt) 234e0f083a0Sdp-arm { 235e0f083a0Sdp-arm image_desc_t *desc; 236e0f083a0Sdp-arm 237e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 238e0f083a0Sdp-arm if (strcmp(desc->cmdline_name, opt) == 0) 239e0f083a0Sdp-arm return desc; 240e0f083a0Sdp-arm return NULL; 241e0f083a0Sdp-arm } 242e0f083a0Sdp-arm 243fcab6bbeSdp-arm static void uuid_to_str(char *s, size_t len, const uuid_t *u) 244fcab6bbeSdp-arm { 245fcab6bbeSdp-arm assert(len >= (_UUID_STR_LEN + 1)); 246fcab6bbeSdp-arm 247fcab6bbeSdp-arm snprintf(s, len, "%08X-%04X-%04X-%04X-%04X%04X%04X", 248fcab6bbeSdp-arm u->time_low, 249fcab6bbeSdp-arm u->time_mid, 250fcab6bbeSdp-arm u->time_hi_and_version, 251fcab6bbeSdp-arm ((uint16_t)u->clock_seq_hi_and_reserved << 8) | u->clock_seq_low, 252fcab6bbeSdp-arm ((uint16_t)u->node[0] << 8) | u->node[1], 253fcab6bbeSdp-arm ((uint16_t)u->node[2] << 8) | u->node[3], 254fcab6bbeSdp-arm ((uint16_t)u->node[4] << 8) | u->node[5]); 255fcab6bbeSdp-arm } 256fcab6bbeSdp-arm 257fcab6bbeSdp-arm static void uuid_from_str(uuid_t *u, const char *s) 258fcab6bbeSdp-arm { 259fcab6bbeSdp-arm int n; 260fcab6bbeSdp-arm 261fcab6bbeSdp-arm if (s == NULL) 262fcab6bbeSdp-arm log_errx("UUID cannot be NULL"); 263fcab6bbeSdp-arm if (strlen(s) != _UUID_STR_LEN) 264fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 265fcab6bbeSdp-arm 266fcab6bbeSdp-arm n = sscanf(s, 267fcab6bbeSdp-arm "%8x-%4hx-%4hx-%2hhx%2hhx-%2hhx%2hhx%2hhx%2hhx%2hhx%2hhx", 268fcab6bbeSdp-arm &u->time_low, &u->time_mid, &u->time_hi_and_version, 269fcab6bbeSdp-arm &u->clock_seq_hi_and_reserved, &u->clock_seq_low, &u->node[0], 270fcab6bbeSdp-arm &u->node[1], &u->node[2], &u->node[3], &u->node[4], &u->node[5]); 271fcab6bbeSdp-arm /* 272fcab6bbeSdp-arm * Given the format specifier above, we expect 11 items to be scanned 273fcab6bbeSdp-arm * for a properly formatted UUID. 274fcab6bbeSdp-arm */ 275fcab6bbeSdp-arm if (n != 11) 276fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 277fcab6bbeSdp-arm } 278fcab6bbeSdp-arm 27960b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out) 280819281eeSdp-arm { 281819281eeSdp-arm struct stat st; 282819281eeSdp-arm FILE *fp; 283819281eeSdp-arm char *buf, *bufend; 284819281eeSdp-arm fip_toc_header_t *toc_header; 285819281eeSdp-arm fip_toc_entry_t *toc_entry; 286819281eeSdp-arm int terminated = 0; 287819281eeSdp-arm 288*55745deaSEvan Lloyd fp = fopen(filename, "rb"); 289819281eeSdp-arm if (fp == NULL) 290819281eeSdp-arm log_err("fopen %s", filename); 291819281eeSdp-arm 292819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 293819281eeSdp-arm log_err("fstat %s", filename); 294819281eeSdp-arm 295a22f6285Sdp-arm buf = xmalloc(st.st_size, "failed to load file into memory"); 296819281eeSdp-arm if (fread(buf, 1, st.st_size, fp) != st.st_size) 297819281eeSdp-arm log_errx("Failed to read %s", filename); 298819281eeSdp-arm bufend = buf + st.st_size; 299819281eeSdp-arm fclose(fp); 300819281eeSdp-arm 301819281eeSdp-arm if (st.st_size < sizeof(fip_toc_header_t)) 302819281eeSdp-arm log_errx("FIP %s is truncated", filename); 303819281eeSdp-arm 304819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 305819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 306819281eeSdp-arm 307819281eeSdp-arm if (toc_header->name != TOC_HEADER_NAME) 308819281eeSdp-arm log_errx("%s is not a FIP file", filename); 309819281eeSdp-arm 310819281eeSdp-arm /* Return the ToC header if the caller wants it. */ 311819281eeSdp-arm if (toc_header_out != NULL) 312819281eeSdp-arm *toc_header_out = *toc_header; 313819281eeSdp-arm 314819281eeSdp-arm /* Walk through each ToC entry in the file. */ 315819281eeSdp-arm while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) { 316fcab6bbeSdp-arm image_t *image; 317fcab6bbeSdp-arm image_desc_t *desc; 318fcab6bbeSdp-arm 319819281eeSdp-arm /* Found the ToC terminator, we are done. */ 320819281eeSdp-arm if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) { 321819281eeSdp-arm terminated = 1; 322819281eeSdp-arm break; 323819281eeSdp-arm } 324819281eeSdp-arm 325819281eeSdp-arm /* 326819281eeSdp-arm * Build a new image out of the ToC entry and add it to the 327819281eeSdp-arm * table of images. 328819281eeSdp-arm */ 32911c0a4ffSMasahiro Yamada image = xzalloc(sizeof(*image), 330a22f6285Sdp-arm "failed to allocate memory for image"); 33165caa3d0SMasahiro Yamada image->toc_e = *toc_entry; 332a22f6285Sdp-arm image->buffer = xmalloc(toc_entry->size, 333a22f6285Sdp-arm "failed to allocate image buffer, is FIP file corrupted?"); 334819281eeSdp-arm /* Overflow checks before memory copy. */ 335819281eeSdp-arm if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address) 336819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 337819281eeSdp-arm if (toc_entry->size + toc_entry->offset_address > st.st_size) 338819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 339819281eeSdp-arm 340819281eeSdp-arm memcpy(image->buffer, buf + toc_entry->offset_address, 341819281eeSdp-arm toc_entry->size); 342819281eeSdp-arm 343fcab6bbeSdp-arm /* If this is an unknown image, create a descriptor for it. */ 34465caa3d0SMasahiro Yamada desc = lookup_image_desc_from_uuid(&toc_entry->uuid); 345fcab6bbeSdp-arm if (desc == NULL) { 346fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 347fcab6bbeSdp-arm 34865caa3d0SMasahiro Yamada uuid_to_str(name, sizeof(name), &toc_entry->uuid); 349fcab6bbeSdp-arm snprintf(filename, sizeof(filename), "%s%s", 350fcab6bbeSdp-arm name, ".bin"); 35165caa3d0SMasahiro Yamada desc = new_image_desc(&toc_entry->uuid, name, "blob"); 352fcab6bbeSdp-arm desc->action = DO_UNPACK; 353fcab6bbeSdp-arm desc->action_arg = xstrdup(filename, 354fcab6bbeSdp-arm "failed to allocate memory for blob filename"); 355fcab6bbeSdp-arm add_image_desc(desc); 356fcab6bbeSdp-arm } 357fcab6bbeSdp-arm 358b9589fe5Sdp-arm assert(desc->image == NULL); 359b9589fe5Sdp-arm desc->image = image; 360819281eeSdp-arm 361819281eeSdp-arm toc_entry++; 362819281eeSdp-arm } 363819281eeSdp-arm 364819281eeSdp-arm if (terminated == 0) 365819281eeSdp-arm log_errx("FIP %s does not have a ToC terminator entry", 366819281eeSdp-arm filename); 367819281eeSdp-arm free(buf); 368819281eeSdp-arm return 0; 369819281eeSdp-arm } 370819281eeSdp-arm 37160b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename) 372819281eeSdp-arm { 373819281eeSdp-arm struct stat st; 374819281eeSdp-arm image_t *image; 375819281eeSdp-arm FILE *fp; 376819281eeSdp-arm 377b04efcceSdp-arm assert(uuid != NULL); 378b04efcceSdp-arm 379*55745deaSEvan Lloyd fp = fopen(filename, "rb"); 380819281eeSdp-arm if (fp == NULL) 381819281eeSdp-arm log_err("fopen %s", filename); 382819281eeSdp-arm 383819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 384819281eeSdp-arm log_errx("fstat %s", filename); 385819281eeSdp-arm 38611c0a4ffSMasahiro Yamada image = xzalloc(sizeof(*image), "failed to allocate memory for image"); 38765caa3d0SMasahiro Yamada image->toc_e.uuid = *uuid; 388a22f6285Sdp-arm image->buffer = xmalloc(st.st_size, "failed to allocate image buffer"); 389819281eeSdp-arm if (fread(image->buffer, 1, st.st_size, fp) != st.st_size) 390819281eeSdp-arm log_errx("Failed to read %s", filename); 39165caa3d0SMasahiro Yamada image->toc_e.size = st.st_size; 392819281eeSdp-arm 393819281eeSdp-arm fclose(fp); 394819281eeSdp-arm return image; 395819281eeSdp-arm } 396819281eeSdp-arm 39760b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename) 398819281eeSdp-arm { 399819281eeSdp-arm FILE *fp; 400819281eeSdp-arm 401*55745deaSEvan Lloyd fp = fopen(filename, "wb"); 402819281eeSdp-arm if (fp == NULL) 403819281eeSdp-arm log_err("fopen"); 40465caa3d0SMasahiro Yamada xfwrite(image->buffer, image->toc_e.size, fp, filename); 405819281eeSdp-arm fclose(fp); 406819281eeSdp-arm return 0; 407819281eeSdp-arm } 408819281eeSdp-arm 409e0f083a0Sdp-arm static struct option *add_opt(struct option *opts, size_t *nr_opts, 410e0f083a0Sdp-arm const char *name, int has_arg, int val) 411819281eeSdp-arm { 412e0f083a0Sdp-arm opts = realloc(opts, (*nr_opts + 1) * sizeof(*opts)); 413e0f083a0Sdp-arm if (opts == NULL) 414e0f083a0Sdp-arm log_err("realloc"); 415e0f083a0Sdp-arm opts[*nr_opts].name = name; 416e0f083a0Sdp-arm opts[*nr_opts].has_arg = has_arg; 417e0f083a0Sdp-arm opts[*nr_opts].flag = NULL; 418e0f083a0Sdp-arm opts[*nr_opts].val = val; 419e0f083a0Sdp-arm ++*nr_opts; 420e0f083a0Sdp-arm return opts; 421819281eeSdp-arm } 422819281eeSdp-arm 423e0f083a0Sdp-arm static struct option *fill_common_opts(struct option *opts, size_t *nr_opts, 424e0f083a0Sdp-arm int has_arg) 425819281eeSdp-arm { 426e0f083a0Sdp-arm image_desc_t *desc; 427e0f083a0Sdp-arm 428e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 429e0f083a0Sdp-arm opts = add_opt(opts, nr_opts, desc->cmdline_name, has_arg, 430e0f083a0Sdp-arm OPT_TOC_ENTRY); 431e0f083a0Sdp-arm return opts; 432819281eeSdp-arm } 433819281eeSdp-arm 434e0f083a0Sdp-arm static void md_print(const unsigned char *md, size_t len) 4359df69ba3Sdp-arm { 4369df69ba3Sdp-arm size_t i; 4379df69ba3Sdp-arm 4389df69ba3Sdp-arm for (i = 0; i < len; i++) 4399df69ba3Sdp-arm printf("%02x", md[i]); 4409df69ba3Sdp-arm } 4419df69ba3Sdp-arm 442819281eeSdp-arm static int info_cmd(int argc, char *argv[]) 443819281eeSdp-arm { 444b9589fe5Sdp-arm image_desc_t *desc; 445819281eeSdp-arm fip_toc_header_t toc_header; 446819281eeSdp-arm 447819281eeSdp-arm if (argc != 2) 44885ee2778Sdp-arm info_usage(); 449819281eeSdp-arm argc--, argv++; 450819281eeSdp-arm 451819281eeSdp-arm parse_fip(argv[0], &toc_header); 452819281eeSdp-arm 453819281eeSdp-arm if (verbose) { 454819281eeSdp-arm log_dbgx("toc_header[name]: 0x%llX", 455819281eeSdp-arm (unsigned long long)toc_header.name); 456819281eeSdp-arm log_dbgx("toc_header[serial_number]: 0x%llX", 457819281eeSdp-arm (unsigned long long)toc_header.serial_number); 458819281eeSdp-arm log_dbgx("toc_header[flags]: 0x%llX", 459819281eeSdp-arm (unsigned long long)toc_header.flags); 460819281eeSdp-arm } 461819281eeSdp-arm 462b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 463b9589fe5Sdp-arm image_t *image = desc->image; 464b04efcceSdp-arm 465b9589fe5Sdp-arm if (image == NULL) 466b9589fe5Sdp-arm continue; 46765caa3d0SMasahiro Yamada printf("%s: offset=0x%llX, size=0x%llX, cmdline=\"--%s\"", 46865caa3d0SMasahiro Yamada desc->name, 46965caa3d0SMasahiro Yamada (unsigned long long)image->toc_e.offset_address, 47065caa3d0SMasahiro Yamada (unsigned long long)image->toc_e.size, 471e0f083a0Sdp-arm desc->cmdline_name); 4729df69ba3Sdp-arm if (verbose) { 4739df69ba3Sdp-arm unsigned char md[SHA256_DIGEST_LENGTH]; 4749df69ba3Sdp-arm 47565caa3d0SMasahiro Yamada SHA256(image->buffer, image->toc_e.size, md); 4769df69ba3Sdp-arm printf(", sha256="); 4779df69ba3Sdp-arm md_print(md, sizeof(md)); 4789df69ba3Sdp-arm } 479819281eeSdp-arm putchar('\n'); 480819281eeSdp-arm } 481819281eeSdp-arm 482819281eeSdp-arm return 0; 483819281eeSdp-arm } 484819281eeSdp-arm 485819281eeSdp-arm static void info_usage(void) 486819281eeSdp-arm { 487819281eeSdp-arm printf("fiptool info FIP_FILENAME\n"); 48885ee2778Sdp-arm exit(1); 489819281eeSdp-arm } 490819281eeSdp-arm 4911c75d5dfSMasahiro Yamada static int pack_images(const char *filename, uint64_t toc_flags, unsigned long align) 492819281eeSdp-arm { 493819281eeSdp-arm FILE *fp; 494b9589fe5Sdp-arm image_desc_t *desc; 495819281eeSdp-arm fip_toc_header_t *toc_header; 496819281eeSdp-arm fip_toc_entry_t *toc_entry; 497819281eeSdp-arm char *buf; 49865caa3d0SMasahiro Yamada uint64_t entry_offset, buf_size, payload_size = 0; 499b9589fe5Sdp-arm size_t nr_images = 0; 500b9589fe5Sdp-arm 501b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 502b9589fe5Sdp-arm if (desc->image != NULL) 503b9589fe5Sdp-arm nr_images++; 504819281eeSdp-arm 505819281eeSdp-arm buf_size = sizeof(fip_toc_header_t) + 506819281eeSdp-arm sizeof(fip_toc_entry_t) * (nr_images + 1); 507819281eeSdp-arm buf = calloc(1, buf_size); 508819281eeSdp-arm if (buf == NULL) 509819281eeSdp-arm log_err("calloc"); 510819281eeSdp-arm 511819281eeSdp-arm /* Build up header and ToC entries from the image table. */ 512819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 513819281eeSdp-arm toc_header->name = TOC_HEADER_NAME; 514819281eeSdp-arm toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER; 515819281eeSdp-arm toc_header->flags = toc_flags; 516819281eeSdp-arm 517819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 518819281eeSdp-arm 519819281eeSdp-arm entry_offset = buf_size; 520b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 521b9589fe5Sdp-arm image_t *image = desc->image; 522b9589fe5Sdp-arm 523b9589fe5Sdp-arm if (image == NULL) 524b9589fe5Sdp-arm continue; 52565caa3d0SMasahiro Yamada payload_size += image->toc_e.size; 5261c75d5dfSMasahiro Yamada entry_offset = (entry_offset + align - 1) & ~(align - 1); 52765caa3d0SMasahiro Yamada image->toc_e.offset_address = entry_offset; 52865caa3d0SMasahiro Yamada *toc_entry++ = image->toc_e; 52965caa3d0SMasahiro Yamada entry_offset += image->toc_e.size; 530819281eeSdp-arm } 531819281eeSdp-arm 532819281eeSdp-arm /* Append a null uuid entry to mark the end of ToC entries. */ 53365caa3d0SMasahiro Yamada memset(toc_entry, 0, sizeof(*toc_entry)); 534819281eeSdp-arm toc_entry->offset_address = entry_offset; 535819281eeSdp-arm 536819281eeSdp-arm /* Generate the FIP file. */ 537*55745deaSEvan Lloyd fp = fopen(filename, "wb"); 538819281eeSdp-arm if (fp == NULL) 539819281eeSdp-arm log_err("fopen %s", filename); 540819281eeSdp-arm 541819281eeSdp-arm if (verbose) 542819281eeSdp-arm log_dbgx("Metadata size: %zu bytes", buf_size); 543819281eeSdp-arm 544a1da83f5SMasahiro Yamada xfwrite(buf, buf_size, fp, filename); 545819281eeSdp-arm free(buf); 546819281eeSdp-arm 547819281eeSdp-arm if (verbose) 548819281eeSdp-arm log_dbgx("Payload size: %zu bytes", payload_size); 549819281eeSdp-arm 550b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 551b9589fe5Sdp-arm image_t *image = desc->image; 552b9589fe5Sdp-arm 553b9589fe5Sdp-arm if (image == NULL) 554b9589fe5Sdp-arm continue; 5551c75d5dfSMasahiro Yamada if (fseek(fp, image->toc_e.offset_address, SEEK_SET)) 5561c75d5dfSMasahiro Yamada log_errx("Failed to set file position"); 5571c75d5dfSMasahiro Yamada 55865caa3d0SMasahiro Yamada xfwrite(image->buffer, image->toc_e.size, fp, filename); 5591c75d5dfSMasahiro Yamada } 560819281eeSdp-arm 561819281eeSdp-arm fclose(fp); 562819281eeSdp-arm return 0; 563819281eeSdp-arm } 564819281eeSdp-arm 565819281eeSdp-arm /* 566819281eeSdp-arm * This function is shared between the create and update subcommands. 567819281eeSdp-arm * The difference between the two subcommands is that when the FIP file 568819281eeSdp-arm * is created, the parsing of an existing FIP is skipped. This results 569819281eeSdp-arm * in update_fip() creating the new FIP file from scratch because the 570819281eeSdp-arm * internal image table is not populated. 571819281eeSdp-arm */ 572819281eeSdp-arm static void update_fip(void) 573819281eeSdp-arm { 574e0f083a0Sdp-arm image_desc_t *desc; 575819281eeSdp-arm 576819281eeSdp-arm /* Add or replace images in the FIP file. */ 577e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 578b9589fe5Sdp-arm image_t *image; 579fcab6bbeSdp-arm 580e0f083a0Sdp-arm if (desc->action != DO_PACK) 581819281eeSdp-arm continue; 582819281eeSdp-arm 583b9589fe5Sdp-arm image = read_image_from_file(&desc->uuid, 584e0f083a0Sdp-arm desc->action_arg); 585b9589fe5Sdp-arm if (desc->image != NULL) { 586e0f083a0Sdp-arm if (verbose) { 587fcab6bbeSdp-arm log_dbgx("Replacing %s with %s", 588e0f083a0Sdp-arm desc->cmdline_name, 589e0f083a0Sdp-arm desc->action_arg); 590e0f083a0Sdp-arm } 591b9589fe5Sdp-arm free(desc->image); 592b9589fe5Sdp-arm desc->image = image; 593819281eeSdp-arm } else { 594819281eeSdp-arm if (verbose) 595819281eeSdp-arm log_dbgx("Adding image %s", 596e0f083a0Sdp-arm desc->action_arg); 597b9589fe5Sdp-arm desc->image = image; 598819281eeSdp-arm } 599819281eeSdp-arm } 600819281eeSdp-arm } 601819281eeSdp-arm 602e0f083a0Sdp-arm static void parse_plat_toc_flags(const char *arg, unsigned long long *toc_flags) 603819281eeSdp-arm { 604819281eeSdp-arm unsigned long long flags; 605819281eeSdp-arm char *endptr; 606819281eeSdp-arm 607819281eeSdp-arm errno = 0; 608819281eeSdp-arm flags = strtoull(arg, &endptr, 16); 609819281eeSdp-arm if (*endptr != '\0' || flags > UINT16_MAX || errno != 0) 610819281eeSdp-arm log_errx("Invalid platform ToC flags: %s", arg); 611819281eeSdp-arm /* Platform ToC flags is a 16-bit field occupying bits [32-47]. */ 612819281eeSdp-arm *toc_flags |= flags << 32; 613819281eeSdp-arm } 614819281eeSdp-arm 6151c75d5dfSMasahiro Yamada static int is_power_of_2(unsigned long x) 6161c75d5dfSMasahiro Yamada { 6171c75d5dfSMasahiro Yamada return x && !(x & (x - 1)); 6181c75d5dfSMasahiro Yamada } 6191c75d5dfSMasahiro Yamada 6201c75d5dfSMasahiro Yamada static unsigned long get_image_align(char *arg) 6211c75d5dfSMasahiro Yamada { 6221c75d5dfSMasahiro Yamada char *endptr; 6231c75d5dfSMasahiro Yamada unsigned long align; 6241c75d5dfSMasahiro Yamada 6251c75d5dfSMasahiro Yamada errno = 0; 626fb5f7949SAndreas Färber align = strtoul(arg, &endptr, 0); 6271c75d5dfSMasahiro Yamada if (*endptr != '\0' || !is_power_of_2(align) || errno != 0) 6281c75d5dfSMasahiro Yamada log_errx("Invalid alignment: %s", arg); 6291c75d5dfSMasahiro Yamada 6301c75d5dfSMasahiro Yamada return align; 6311c75d5dfSMasahiro Yamada } 6321c75d5dfSMasahiro Yamada 633fcab6bbeSdp-arm static void parse_blob_opt(char *arg, uuid_t *uuid, char *filename, size_t len) 634fcab6bbeSdp-arm { 635fcab6bbeSdp-arm char *p; 636fcab6bbeSdp-arm 637fcab6bbeSdp-arm for (p = strtok(arg, ","); p != NULL; p = strtok(NULL, ",")) { 638fcab6bbeSdp-arm if (strncmp(p, "uuid=", strlen("uuid=")) == 0) { 639fcab6bbeSdp-arm p += strlen("uuid="); 640fcab6bbeSdp-arm uuid_from_str(uuid, p); 641fcab6bbeSdp-arm } else if (strncmp(p, "file=", strlen("file=")) == 0) { 642fcab6bbeSdp-arm p += strlen("file="); 643fcab6bbeSdp-arm snprintf(filename, len, "%s", p); 644fcab6bbeSdp-arm } 645fcab6bbeSdp-arm } 646fcab6bbeSdp-arm } 647fcab6bbeSdp-arm 648819281eeSdp-arm static int create_cmd(int argc, char *argv[]) 649819281eeSdp-arm { 650e0f083a0Sdp-arm struct option *opts = NULL; 651e0f083a0Sdp-arm size_t nr_opts = 0; 652819281eeSdp-arm unsigned long long toc_flags = 0; 6531c75d5dfSMasahiro Yamada unsigned long align = 1; 654819281eeSdp-arm 655819281eeSdp-arm if (argc < 2) 65685ee2778Sdp-arm create_usage(); 657819281eeSdp-arm 658e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 659e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 660819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 6611c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 662fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 663e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 664819281eeSdp-arm 665819281eeSdp-arm while (1) { 66620f87e78Sdp-arm int c, opt_index = 0; 667819281eeSdp-arm 668fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:", opts, &opt_index); 669819281eeSdp-arm if (c == -1) 670819281eeSdp-arm break; 671819281eeSdp-arm 672819281eeSdp-arm switch (c) { 673819281eeSdp-arm case OPT_TOC_ENTRY: { 674e0f083a0Sdp-arm image_desc_t *desc; 675819281eeSdp-arm 676e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 677d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 678819281eeSdp-arm break; 679819281eeSdp-arm } 680819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: 681819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 682819281eeSdp-arm break; 6831c75d5dfSMasahiro Yamada case OPT_ALIGN: 6841c75d5dfSMasahiro Yamada align = get_image_align(optarg); 6851c75d5dfSMasahiro Yamada break; 686fcab6bbeSdp-arm case 'b': { 687fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 688fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 689fcab6bbeSdp-arm uuid_t uuid = { 0 }; 690fcab6bbeSdp-arm image_desc_t *desc; 691fcab6bbeSdp-arm 692fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 693fcab6bbeSdp-arm filename, sizeof(filename)); 694fcab6bbeSdp-arm 695fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 696fcab6bbeSdp-arm filename[0] == '\0') 697fcab6bbeSdp-arm create_usage(); 698fcab6bbeSdp-arm 699fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 700d02fcebeSdp-arm if (desc == NULL) { 701fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 702fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 703fcab6bbeSdp-arm add_image_desc(desc); 704fcab6bbeSdp-arm } 705d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 706fcab6bbeSdp-arm break; 707fcab6bbeSdp-arm } 708819281eeSdp-arm default: 70985ee2778Sdp-arm create_usage(); 710819281eeSdp-arm } 711819281eeSdp-arm } 712819281eeSdp-arm argc -= optind; 713819281eeSdp-arm argv += optind; 714e0f083a0Sdp-arm free(opts); 715819281eeSdp-arm 716819281eeSdp-arm if (argc == 0) 71785ee2778Sdp-arm create_usage(); 718819281eeSdp-arm 719819281eeSdp-arm update_fip(); 720819281eeSdp-arm 7211c75d5dfSMasahiro Yamada pack_images(argv[0], toc_flags, align); 722819281eeSdp-arm return 0; 723819281eeSdp-arm } 724819281eeSdp-arm 725819281eeSdp-arm static void create_usage(void) 726819281eeSdp-arm { 727819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 728819281eeSdp-arm 729ee079320SMasahiro Yamada printf("fiptool create [opts] FIP_FILENAME\n"); 730ee079320SMasahiro Yamada printf("\n"); 731ee079320SMasahiro Yamada printf("Options:\n"); 7321c75d5dfSMasahiro Yamada printf(" --align <value>\t\tEach image is aligned to <value> (default: 1).\n"); 733802b42a0SMasahiro Yamada printf(" --blob uuid=...,file=...\tAdd an image with the given UUID pointed to by file.\n"); 734802b42a0SMasahiro Yamada printf(" --plat-toc-flags <value>\t16-bit platform specific flag field occupying bits 32-47 in 64-bit ToC header.\n"); 73595d2b268SMasahiro Yamada printf("\n"); 736819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 737819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 738819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 739819281eeSdp-arm toc_entry->name); 74085ee2778Sdp-arm exit(1); 741819281eeSdp-arm } 742819281eeSdp-arm 743819281eeSdp-arm static int update_cmd(int argc, char *argv[]) 744819281eeSdp-arm { 745e0f083a0Sdp-arm struct option *opts = NULL; 746e0f083a0Sdp-arm size_t nr_opts = 0; 747fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 748819281eeSdp-arm fip_toc_header_t toc_header = { 0 }; 749819281eeSdp-arm unsigned long long toc_flags = 0; 7501c75d5dfSMasahiro Yamada unsigned long align = 1; 751819281eeSdp-arm int pflag = 0; 752819281eeSdp-arm 753819281eeSdp-arm if (argc < 2) 75485ee2778Sdp-arm update_usage(); 755819281eeSdp-arm 756e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 7571c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 758fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 759e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 760e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 761819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 762e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 763819281eeSdp-arm 764819281eeSdp-arm while (1) { 76520f87e78Sdp-arm int c, opt_index = 0; 766819281eeSdp-arm 767fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:o:", opts, &opt_index); 768819281eeSdp-arm if (c == -1) 769819281eeSdp-arm break; 770819281eeSdp-arm 771819281eeSdp-arm switch (c) { 772819281eeSdp-arm case OPT_TOC_ENTRY: { 773e0f083a0Sdp-arm image_desc_t *desc; 774819281eeSdp-arm 775e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 776d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 777819281eeSdp-arm break; 778819281eeSdp-arm } 779e0f083a0Sdp-arm case OPT_PLAT_TOC_FLAGS: 780819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 781819281eeSdp-arm pflag = 1; 782819281eeSdp-arm break; 783fcab6bbeSdp-arm case 'b': { 784fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 785fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 786fcab6bbeSdp-arm uuid_t uuid = { 0 }; 787fcab6bbeSdp-arm image_desc_t *desc; 788fcab6bbeSdp-arm 789fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 790fcab6bbeSdp-arm filename, sizeof(filename)); 791fcab6bbeSdp-arm 792fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 793fcab6bbeSdp-arm filename[0] == '\0') 794fcab6bbeSdp-arm update_usage(); 795fcab6bbeSdp-arm 796fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 797d02fcebeSdp-arm if (desc == NULL) { 798fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 799fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 800fcab6bbeSdp-arm add_image_desc(desc); 801fcab6bbeSdp-arm } 802d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 803fcab6bbeSdp-arm break; 804fcab6bbeSdp-arm } 8051c75d5dfSMasahiro Yamada case OPT_ALIGN: 8061c75d5dfSMasahiro Yamada align = get_image_align(optarg); 8071c75d5dfSMasahiro Yamada break; 808819281eeSdp-arm case 'o': 809819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 810819281eeSdp-arm break; 811819281eeSdp-arm default: 81285ee2778Sdp-arm update_usage(); 813819281eeSdp-arm } 814819281eeSdp-arm } 815819281eeSdp-arm argc -= optind; 816819281eeSdp-arm argv += optind; 817e0f083a0Sdp-arm free(opts); 818819281eeSdp-arm 819819281eeSdp-arm if (argc == 0) 82085ee2778Sdp-arm update_usage(); 821819281eeSdp-arm 822819281eeSdp-arm if (outfile[0] == '\0') 823819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 824819281eeSdp-arm 82540866aafSMasahiro Yamada if (access(argv[0], F_OK) == 0) 826819281eeSdp-arm parse_fip(argv[0], &toc_header); 827819281eeSdp-arm 828819281eeSdp-arm if (pflag) 829819281eeSdp-arm toc_header.flags &= ~(0xffffULL << 32); 830819281eeSdp-arm toc_flags = (toc_header.flags |= toc_flags); 831819281eeSdp-arm 832819281eeSdp-arm update_fip(); 833819281eeSdp-arm 8341c75d5dfSMasahiro Yamada pack_images(outfile, toc_flags, align); 835819281eeSdp-arm return 0; 836819281eeSdp-arm } 837819281eeSdp-arm 838819281eeSdp-arm static void update_usage(void) 839819281eeSdp-arm { 840819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 841819281eeSdp-arm 842ee079320SMasahiro Yamada printf("fiptool update [opts] FIP_FILENAME\n"); 843ee079320SMasahiro Yamada printf("\n"); 844ee079320SMasahiro Yamada printf("Options:\n"); 8451c75d5dfSMasahiro Yamada printf(" --align <value>\t\tEach image is aligned to <value> (default: 1).\n"); 846802b42a0SMasahiro Yamada printf(" --blob uuid=...,file=...\tAdd or update an image with the given UUID pointed to by file.\n"); 847819281eeSdp-arm printf(" --out FIP_FILENAME\t\tSet an alternative output FIP file.\n"); 848802b42a0SMasahiro Yamada printf(" --plat-toc-flags <value>\t16-bit platform specific flag field occupying bits 32-47 in 64-bit ToC header.\n"); 84995d2b268SMasahiro Yamada printf("\n"); 850819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 851819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 852819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 853819281eeSdp-arm toc_entry->name); 85485ee2778Sdp-arm exit(1); 855819281eeSdp-arm } 856819281eeSdp-arm 857819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]) 858819281eeSdp-arm { 859e0f083a0Sdp-arm struct option *opts = NULL; 860e0f083a0Sdp-arm size_t nr_opts = 0; 861fcab6bbeSdp-arm char outdir[PATH_MAX] = { 0 }; 862e0f083a0Sdp-arm image_desc_t *desc; 863819281eeSdp-arm int fflag = 0; 864819281eeSdp-arm int unpack_all = 1; 865819281eeSdp-arm 866819281eeSdp-arm if (argc < 2) 86785ee2778Sdp-arm unpack_usage(); 868819281eeSdp-arm 869e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 870fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 871e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 872e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 873e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 874819281eeSdp-arm 875819281eeSdp-arm while (1) { 87620f87e78Sdp-arm int c, opt_index = 0; 877819281eeSdp-arm 878fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 879819281eeSdp-arm if (c == -1) 880819281eeSdp-arm break; 881819281eeSdp-arm 882819281eeSdp-arm switch (c) { 883e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 884e0f083a0Sdp-arm image_desc_t *desc; 885e0f083a0Sdp-arm 886e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 887d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, optarg); 888e0f083a0Sdp-arm unpack_all = 0; 889819281eeSdp-arm break; 890e0f083a0Sdp-arm } 891fcab6bbeSdp-arm case 'b': { 892fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 893fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 894fcab6bbeSdp-arm uuid_t uuid = { 0 }; 895fcab6bbeSdp-arm image_desc_t *desc; 896fcab6bbeSdp-arm 897fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 898fcab6bbeSdp-arm filename, sizeof(filename)); 899fcab6bbeSdp-arm 900fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 901fcab6bbeSdp-arm filename[0] == '\0') 902fcab6bbeSdp-arm unpack_usage(); 903fcab6bbeSdp-arm 904fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 905d02fcebeSdp-arm if (desc == NULL) { 906fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 907fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 908fcab6bbeSdp-arm add_image_desc(desc); 909fcab6bbeSdp-arm } 910d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, filename); 911fcab6bbeSdp-arm unpack_all = 0; 912fcab6bbeSdp-arm break; 913fcab6bbeSdp-arm } 914819281eeSdp-arm case 'f': 915819281eeSdp-arm fflag = 1; 916819281eeSdp-arm break; 917819281eeSdp-arm case 'o': 918819281eeSdp-arm snprintf(outdir, sizeof(outdir), "%s", optarg); 919819281eeSdp-arm break; 920819281eeSdp-arm default: 92185ee2778Sdp-arm unpack_usage(); 922819281eeSdp-arm } 923819281eeSdp-arm } 924819281eeSdp-arm argc -= optind; 925819281eeSdp-arm argv += optind; 926e0f083a0Sdp-arm free(opts); 927819281eeSdp-arm 928819281eeSdp-arm if (argc == 0) 92985ee2778Sdp-arm unpack_usage(); 930819281eeSdp-arm 931819281eeSdp-arm parse_fip(argv[0], NULL); 932819281eeSdp-arm 933819281eeSdp-arm if (outdir[0] != '\0') 934819281eeSdp-arm if (chdir(outdir) == -1) 935819281eeSdp-arm log_err("chdir %s", outdir); 936819281eeSdp-arm 937819281eeSdp-arm /* Unpack all specified images. */ 938e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 939fcab6bbeSdp-arm char file[PATH_MAX]; 940b9589fe5Sdp-arm image_t *image = desc->image; 941b04efcceSdp-arm 942e0f083a0Sdp-arm if (!unpack_all && desc->action != DO_UNPACK) 943819281eeSdp-arm continue; 944819281eeSdp-arm 945819281eeSdp-arm /* Build filename. */ 946e0f083a0Sdp-arm if (desc->action_arg == NULL) 947819281eeSdp-arm snprintf(file, sizeof(file), "%s.bin", 948e0f083a0Sdp-arm desc->cmdline_name); 949819281eeSdp-arm else 950819281eeSdp-arm snprintf(file, sizeof(file), "%s", 951e0f083a0Sdp-arm desc->action_arg); 952819281eeSdp-arm 953b04efcceSdp-arm if (image == NULL) { 954b04efcceSdp-arm if (!unpack_all) 955fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 956819281eeSdp-arm file, argv[0]); 957819281eeSdp-arm continue; 958819281eeSdp-arm } 959819281eeSdp-arm 960819281eeSdp-arm if (access(file, F_OK) != 0 || fflag) { 961819281eeSdp-arm if (verbose) 962819281eeSdp-arm log_dbgx("Unpacking %s", file); 963b04efcceSdp-arm write_image_to_file(image, file); 964819281eeSdp-arm } else { 965819281eeSdp-arm log_warnx("File %s already exists, use --force to overwrite it", 966819281eeSdp-arm file); 967819281eeSdp-arm } 968819281eeSdp-arm } 969819281eeSdp-arm 970819281eeSdp-arm return 0; 971819281eeSdp-arm } 972819281eeSdp-arm 973819281eeSdp-arm static void unpack_usage(void) 974819281eeSdp-arm { 975819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 976819281eeSdp-arm 977ee079320SMasahiro Yamada printf("fiptool unpack [opts] FIP_FILENAME\n"); 978ee079320SMasahiro Yamada printf("\n"); 979ee079320SMasahiro Yamada printf("Options:\n"); 980802b42a0SMasahiro Yamada printf(" --blob uuid=...,file=...\tUnpack an image with the given UUID to file.\n"); 981802b42a0SMasahiro Yamada printf(" --force\t\t\tIf the output file already exists, use --force to overwrite it.\n"); 982fcab6bbeSdp-arm printf(" --out path\t\t\tSet the output directory path.\n"); 98395d2b268SMasahiro Yamada printf("\n"); 984819281eeSdp-arm printf("Specific images are unpacked with the following options:\n"); 985819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 986819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 987819281eeSdp-arm toc_entry->name); 98895d2b268SMasahiro Yamada printf("\n"); 989819281eeSdp-arm printf("If no options are provided, all images will be unpacked.\n"); 99085ee2778Sdp-arm exit(1); 991819281eeSdp-arm } 992819281eeSdp-arm 993819281eeSdp-arm static int remove_cmd(int argc, char *argv[]) 994819281eeSdp-arm { 995e0f083a0Sdp-arm struct option *opts = NULL; 996e0f083a0Sdp-arm size_t nr_opts = 0; 997fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 998819281eeSdp-arm fip_toc_header_t toc_header; 999e0f083a0Sdp-arm image_desc_t *desc; 10001c75d5dfSMasahiro Yamada unsigned long align = 1; 1001819281eeSdp-arm int fflag = 0; 1002819281eeSdp-arm 1003819281eeSdp-arm if (argc < 2) 100485ee2778Sdp-arm remove_usage(); 1005819281eeSdp-arm 1006e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, no_argument); 10071c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 1008fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 1009e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 1010e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 1011e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 1012819281eeSdp-arm 1013819281eeSdp-arm while (1) { 101420f87e78Sdp-arm int c, opt_index = 0; 1015819281eeSdp-arm 1016fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 1017819281eeSdp-arm if (c == -1) 1018819281eeSdp-arm break; 1019819281eeSdp-arm 1020819281eeSdp-arm switch (c) { 1021e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 1022e0f083a0Sdp-arm image_desc_t *desc; 1023e0f083a0Sdp-arm 1024e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 1025d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1026819281eeSdp-arm break; 1027e0f083a0Sdp-arm } 10281c75d5dfSMasahiro Yamada case OPT_ALIGN: 10291c75d5dfSMasahiro Yamada align = get_image_align(optarg); 10301c75d5dfSMasahiro Yamada break; 1031fcab6bbeSdp-arm case 'b': { 1032fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 1033fcab6bbeSdp-arm uuid_t uuid = { 0 }; 1034fcab6bbeSdp-arm image_desc_t *desc; 1035fcab6bbeSdp-arm 1036fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 1037fcab6bbeSdp-arm filename, sizeof(filename)); 1038fcab6bbeSdp-arm 1039fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0) 1040fcab6bbeSdp-arm remove_usage(); 1041fcab6bbeSdp-arm 1042fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 1043d02fcebeSdp-arm if (desc == NULL) { 1044fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 1045fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 1046fcab6bbeSdp-arm add_image_desc(desc); 1047fcab6bbeSdp-arm } 1048d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1049fcab6bbeSdp-arm break; 1050fcab6bbeSdp-arm } 1051819281eeSdp-arm case 'f': 1052819281eeSdp-arm fflag = 1; 1053819281eeSdp-arm break; 1054819281eeSdp-arm case 'o': 1055819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 1056819281eeSdp-arm break; 1057819281eeSdp-arm default: 105885ee2778Sdp-arm remove_usage(); 1059819281eeSdp-arm } 1060819281eeSdp-arm } 1061819281eeSdp-arm argc -= optind; 1062819281eeSdp-arm argv += optind; 1063e0f083a0Sdp-arm free(opts); 1064819281eeSdp-arm 1065819281eeSdp-arm if (argc == 0) 106685ee2778Sdp-arm remove_usage(); 1067819281eeSdp-arm 1068819281eeSdp-arm if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag) 1069819281eeSdp-arm log_errx("File %s already exists, use --force to overwrite it", 1070819281eeSdp-arm outfile); 1071819281eeSdp-arm 1072819281eeSdp-arm if (outfile[0] == '\0') 1073819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 1074819281eeSdp-arm 1075819281eeSdp-arm parse_fip(argv[0], &toc_header); 1076819281eeSdp-arm 1077e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 1078e0f083a0Sdp-arm if (desc->action != DO_REMOVE) 1079819281eeSdp-arm continue; 1080fcab6bbeSdp-arm 1081b9589fe5Sdp-arm if (desc->image != NULL) { 1082819281eeSdp-arm if (verbose) 1083fcab6bbeSdp-arm log_dbgx("Removing %s", 1084e0f083a0Sdp-arm desc->cmdline_name); 1085b9589fe5Sdp-arm free(desc->image); 1086b9589fe5Sdp-arm desc->image = NULL; 1087819281eeSdp-arm } else { 1088fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 1089e0f083a0Sdp-arm desc->cmdline_name, argv[0]); 1090819281eeSdp-arm } 1091819281eeSdp-arm } 1092819281eeSdp-arm 10931c75d5dfSMasahiro Yamada pack_images(outfile, toc_header.flags, align); 1094819281eeSdp-arm return 0; 1095819281eeSdp-arm } 1096819281eeSdp-arm 1097819281eeSdp-arm static void remove_usage(void) 1098819281eeSdp-arm { 1099819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 1100819281eeSdp-arm 1101ee079320SMasahiro Yamada printf("fiptool remove [opts] FIP_FILENAME\n"); 1102ee079320SMasahiro Yamada printf("\n"); 1103ee079320SMasahiro Yamada printf("Options:\n"); 11041c75d5dfSMasahiro Yamada printf(" --align <value>\tEach image is aligned to <value> (default: 1).\n"); 1105fcab6bbeSdp-arm printf(" --blob uuid=...\tRemove an image with the given UUID.\n"); 1106802b42a0SMasahiro Yamada printf(" --force\t\tIf the output FIP file already exists, use --force to overwrite it.\n"); 1107819281eeSdp-arm printf(" --out FIP_FILENAME\tSet an alternative output FIP file.\n"); 110895d2b268SMasahiro Yamada printf("\n"); 1109819281eeSdp-arm printf("Specific images are removed with the following options:\n"); 1110819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1111819281eeSdp-arm printf(" --%-16s\t%s\n", toc_entry->cmdline_name, 1112819281eeSdp-arm toc_entry->name); 111385ee2778Sdp-arm exit(1); 1114819281eeSdp-arm } 1115819281eeSdp-arm 1116819281eeSdp-arm static int version_cmd(int argc, char *argv[]) 1117819281eeSdp-arm { 1118819281eeSdp-arm #ifdef VERSION 1119819281eeSdp-arm puts(VERSION); 1120819281eeSdp-arm #else 1121819281eeSdp-arm /* If built from fiptool directory, VERSION is not set. */ 1122819281eeSdp-arm puts("Unknown version"); 1123819281eeSdp-arm #endif 1124819281eeSdp-arm return 0; 1125819281eeSdp-arm } 1126819281eeSdp-arm 1127819281eeSdp-arm static void version_usage(void) 1128819281eeSdp-arm { 1129819281eeSdp-arm printf("fiptool version\n"); 113085ee2778Sdp-arm exit(1); 1131819281eeSdp-arm } 1132819281eeSdp-arm 1133819281eeSdp-arm static int help_cmd(int argc, char *argv[]) 1134819281eeSdp-arm { 1135819281eeSdp-arm int i; 1136819281eeSdp-arm 1137819281eeSdp-arm if (argc < 2) 1138819281eeSdp-arm usage(); 1139819281eeSdp-arm argc--, argv++; 1140819281eeSdp-arm 1141819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1142819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0 && 114385ee2778Sdp-arm cmds[i].usage != NULL) 1144819281eeSdp-arm cmds[i].usage(); 1145819281eeSdp-arm } 1146819281eeSdp-arm if (i == NELEM(cmds)) 1147819281eeSdp-arm printf("No help for subcommand '%s'\n", argv[0]); 1148819281eeSdp-arm return 0; 1149819281eeSdp-arm } 1150819281eeSdp-arm 1151819281eeSdp-arm static void usage(void) 1152819281eeSdp-arm { 11534f96a498SMasahiro Yamada printf("usage: fiptool [--verbose] <command> [<args>]\n"); 1154819281eeSdp-arm printf("Global options supported:\n"); 1155819281eeSdp-arm printf(" --verbose\tEnable verbose output for all commands.\n"); 115695d2b268SMasahiro Yamada printf("\n"); 1157819281eeSdp-arm printf("Commands supported:\n"); 1158819281eeSdp-arm printf(" info\t\tList images contained in FIP.\n"); 1159819281eeSdp-arm printf(" create\tCreate a new FIP with the given images.\n"); 1160819281eeSdp-arm printf(" update\tUpdate an existing FIP with the given images.\n"); 1161819281eeSdp-arm printf(" unpack\tUnpack images from FIP.\n"); 1162819281eeSdp-arm printf(" remove\tRemove images from FIP.\n"); 1163819281eeSdp-arm printf(" version\tShow fiptool version.\n"); 1164819281eeSdp-arm printf(" help\t\tShow help for given command.\n"); 1165819281eeSdp-arm exit(1); 1166819281eeSdp-arm } 1167819281eeSdp-arm 1168819281eeSdp-arm int main(int argc, char *argv[]) 1169819281eeSdp-arm { 1170819281eeSdp-arm int i, ret = 0; 1171819281eeSdp-arm 1172cc672bb2Sdp-arm while (1) { 1173cc672bb2Sdp-arm int c, opt_index = 0; 1174cc672bb2Sdp-arm static struct option opts[] = { 1175cc672bb2Sdp-arm { "verbose", no_argument, NULL, 'v' }, 1176cc672bb2Sdp-arm { NULL, no_argument, NULL, 0 } 1177cc672bb2Sdp-arm }; 1178819281eeSdp-arm 1179cc672bb2Sdp-arm /* 1180cc672bb2Sdp-arm * Set POSIX mode so getopt stops at the first non-option 1181cc672bb2Sdp-arm * which is the subcommand. 1182cc672bb2Sdp-arm */ 1183cc672bb2Sdp-arm c = getopt_long(argc, argv, "+v", opts, &opt_index); 1184cc672bb2Sdp-arm if (c == -1) 1185cc672bb2Sdp-arm break; 1186cc672bb2Sdp-arm 1187cc672bb2Sdp-arm switch (c) { 1188cc672bb2Sdp-arm case 'v': 1189819281eeSdp-arm verbose = 1; 1190cc672bb2Sdp-arm break; 1191cc672bb2Sdp-arm default: 1192c9cb4089SMasahiro Yamada usage(); 1193819281eeSdp-arm } 1194cc672bb2Sdp-arm } 1195cc672bb2Sdp-arm argc -= optind; 1196cc672bb2Sdp-arm argv += optind; 1197cc672bb2Sdp-arm /* Reset optind for subsequent getopt processing. */ 1198cc672bb2Sdp-arm optind = 0; 1199cc672bb2Sdp-arm 1200cc672bb2Sdp-arm if (argc == 0) 1201cc672bb2Sdp-arm usage(); 1202819281eeSdp-arm 1203e0f083a0Sdp-arm fill_image_descs(); 1204819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1205819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0) { 1206819281eeSdp-arm ret = cmds[i].handler(argc, argv); 1207819281eeSdp-arm break; 1208819281eeSdp-arm } 1209819281eeSdp-arm } 1210819281eeSdp-arm if (i == NELEM(cmds)) 1211819281eeSdp-arm usage(); 1212e0f083a0Sdp-arm free_image_descs(); 1213819281eeSdp-arm return ret; 1214819281eeSdp-arm } 1215