1819281eeSdp-arm /* 2819281eeSdp-arm * Copyright (c) 2016, ARM Limited and Contributors. All rights reserved. 3819281eeSdp-arm * 4819281eeSdp-arm * Redistribution and use in source and binary forms, with or without 5819281eeSdp-arm * modification, are permitted provided that the following conditions are met: 6819281eeSdp-arm * 7819281eeSdp-arm * Redistributions of source code must retain the above copyright notice, this 8819281eeSdp-arm * list of conditions and the following disclaimer. 9819281eeSdp-arm * 10819281eeSdp-arm * Redistributions in binary form must reproduce the above copyright notice, 11819281eeSdp-arm * this list of conditions and the following disclaimer in the documentation 12819281eeSdp-arm * and/or other materials provided with the distribution. 13819281eeSdp-arm * 14819281eeSdp-arm * Neither the name of ARM nor the names of its contributors may be used 15819281eeSdp-arm * to endorse or promote products derived from this software without specific 16819281eeSdp-arm * prior written permission. 17819281eeSdp-arm * 18819281eeSdp-arm * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 19819281eeSdp-arm * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20819281eeSdp-arm * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 21819281eeSdp-arm * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 22819281eeSdp-arm * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 23819281eeSdp-arm * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 24819281eeSdp-arm * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 25819281eeSdp-arm * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 26819281eeSdp-arm * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 27819281eeSdp-arm * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 28819281eeSdp-arm * POSSIBILITY OF SUCH DAMAGE. 29819281eeSdp-arm */ 30819281eeSdp-arm 31819281eeSdp-arm #include <sys/types.h> 32819281eeSdp-arm #include <sys/stat.h> 33819281eeSdp-arm 34819281eeSdp-arm #include <assert.h> 35819281eeSdp-arm #include <errno.h> 36819281eeSdp-arm #include <getopt.h> 37819281eeSdp-arm #include <limits.h> 38819281eeSdp-arm #include <stdarg.h> 39819281eeSdp-arm #include <stdint.h> 40819281eeSdp-arm #include <stdio.h> 41819281eeSdp-arm #include <stdlib.h> 42819281eeSdp-arm #include <string.h> 43819281eeSdp-arm #include <unistd.h> 44819281eeSdp-arm 459df69ba3Sdp-arm #include <openssl/sha.h> 469df69ba3Sdp-arm 47819281eeSdp-arm #include "fiptool.h" 48819281eeSdp-arm #include "firmware_image_package.h" 49819281eeSdp-arm #include "tbbr_config.h" 50819281eeSdp-arm 51819281eeSdp-arm #define OPT_TOC_ENTRY 0 52819281eeSdp-arm #define OPT_PLAT_TOC_FLAGS 1 53*1c75d5dfSMasahiro Yamada #define OPT_ALIGN 2 54819281eeSdp-arm 55e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid); 56e0f083a0Sdp-arm static image_t *lookup_image_from_uuid(const uuid_t *uuid); 57819281eeSdp-arm static int info_cmd(int argc, char *argv[]); 58819281eeSdp-arm static void info_usage(void); 59819281eeSdp-arm static int create_cmd(int argc, char *argv[]); 60819281eeSdp-arm static void create_usage(void); 61819281eeSdp-arm static int update_cmd(int argc, char *argv[]); 62819281eeSdp-arm static void update_usage(void); 63819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]); 64819281eeSdp-arm static void unpack_usage(void); 65819281eeSdp-arm static int remove_cmd(int argc, char *argv[]); 66819281eeSdp-arm static void remove_usage(void); 67819281eeSdp-arm static int version_cmd(int argc, char *argv[]); 68819281eeSdp-arm static void version_usage(void); 69819281eeSdp-arm static int help_cmd(int argc, char *argv[]); 70819281eeSdp-arm static void usage(void); 71819281eeSdp-arm 72819281eeSdp-arm /* Available subcommands. */ 73819281eeSdp-arm static cmd_t cmds[] = { 74819281eeSdp-arm { .name = "info", .handler = info_cmd, .usage = info_usage }, 75819281eeSdp-arm { .name = "create", .handler = create_cmd, .usage = create_usage }, 76819281eeSdp-arm { .name = "update", .handler = update_cmd, .usage = update_usage }, 77819281eeSdp-arm { .name = "unpack", .handler = unpack_cmd, .usage = unpack_usage }, 78819281eeSdp-arm { .name = "remove", .handler = remove_cmd, .usage = remove_usage }, 79819281eeSdp-arm { .name = "version", .handler = version_cmd, .usage = version_usage }, 80819281eeSdp-arm { .name = "help", .handler = help_cmd, .usage = NULL }, 81819281eeSdp-arm }; 82819281eeSdp-arm 83e0f083a0Sdp-arm static image_desc_t *image_desc_head; 84e0f083a0Sdp-arm static size_t nr_image_descs; 85e0f083a0Sdp-arm static image_t *image_head; 86819281eeSdp-arm static size_t nr_images; 87819281eeSdp-arm static uuid_t uuid_null = { 0 }; 88819281eeSdp-arm static int verbose; 89819281eeSdp-arm 9060b499feSdp-arm static void vlog(int prio, const char *msg, va_list ap) 91819281eeSdp-arm { 92819281eeSdp-arm char *prefix[] = { "DEBUG", "WARN", "ERROR" }; 93819281eeSdp-arm 94819281eeSdp-arm fprintf(stderr, "%s: ", prefix[prio]); 95819281eeSdp-arm vfprintf(stderr, msg, ap); 96819281eeSdp-arm fputc('\n', stderr); 97819281eeSdp-arm } 98819281eeSdp-arm 9960b499feSdp-arm static void log_dbgx(const char *msg, ...) 100819281eeSdp-arm { 101819281eeSdp-arm va_list ap; 102819281eeSdp-arm 103819281eeSdp-arm va_start(ap, msg); 104819281eeSdp-arm vlog(LOG_DBG, msg, ap); 105819281eeSdp-arm va_end(ap); 106819281eeSdp-arm } 107819281eeSdp-arm 10860b499feSdp-arm static void log_warnx(const char *msg, ...) 109819281eeSdp-arm { 110819281eeSdp-arm va_list ap; 111819281eeSdp-arm 112819281eeSdp-arm va_start(ap, msg); 113819281eeSdp-arm vlog(LOG_WARN, msg, ap); 114819281eeSdp-arm va_end(ap); 115819281eeSdp-arm } 116819281eeSdp-arm 11760b499feSdp-arm static void log_err(const char *msg, ...) 118819281eeSdp-arm { 119819281eeSdp-arm char buf[512]; 120819281eeSdp-arm va_list ap; 121819281eeSdp-arm 122819281eeSdp-arm va_start(ap, msg); 123819281eeSdp-arm snprintf(buf, sizeof(buf), "%s: %s", msg, strerror(errno)); 124819281eeSdp-arm vlog(LOG_ERR, buf, ap); 125819281eeSdp-arm va_end(ap); 126819281eeSdp-arm exit(1); 127819281eeSdp-arm } 128819281eeSdp-arm 12960b499feSdp-arm static void log_errx(const char *msg, ...) 130819281eeSdp-arm { 131819281eeSdp-arm va_list ap; 132819281eeSdp-arm 133819281eeSdp-arm va_start(ap, msg); 134819281eeSdp-arm vlog(LOG_ERR, msg, ap); 135819281eeSdp-arm va_end(ap); 136819281eeSdp-arm exit(1); 137819281eeSdp-arm } 138819281eeSdp-arm 139a22f6285Sdp-arm static char *xstrdup(const char *s, const char *msg) 140a22f6285Sdp-arm { 141a22f6285Sdp-arm char *d; 142a22f6285Sdp-arm 143a22f6285Sdp-arm d = strdup(s); 144a22f6285Sdp-arm if (d == NULL) 1459fc9ff1fSdp-arm log_errx("strdup: %s", msg); 146a22f6285Sdp-arm return d; 147a22f6285Sdp-arm } 148a22f6285Sdp-arm 149a22f6285Sdp-arm static void *xmalloc(size_t size, const char *msg) 150a22f6285Sdp-arm { 151a22f6285Sdp-arm void *d; 152a22f6285Sdp-arm 153a22f6285Sdp-arm d = malloc(size); 154a22f6285Sdp-arm if (d == NULL) 1559fc9ff1fSdp-arm log_errx("malloc: %s", msg); 156a22f6285Sdp-arm return d; 157a22f6285Sdp-arm } 158a22f6285Sdp-arm 159696ccba6SMasahiro Yamada static void *xzalloc(size_t size, const char *msg) 160696ccba6SMasahiro Yamada { 161696ccba6SMasahiro Yamada return memset(xmalloc(size, msg), 0, size); 162696ccba6SMasahiro Yamada } 163696ccba6SMasahiro Yamada 164a1da83f5SMasahiro Yamada static void xfwrite(void *buf, size_t size, FILE *fp, const char *filename) 165a1da83f5SMasahiro Yamada { 166a1da83f5SMasahiro Yamada if (fwrite(buf, 1, size, fp) != size) 167a1da83f5SMasahiro Yamada log_errx("Failed to write %s", filename); 168a1da83f5SMasahiro Yamada } 169a1da83f5SMasahiro Yamada 170e0f083a0Sdp-arm static image_desc_t *new_image_desc(const uuid_t *uuid, 171e0f083a0Sdp-arm const char *name, const char *cmdline_name) 172e0f083a0Sdp-arm { 173e0f083a0Sdp-arm image_desc_t *desc; 174e0f083a0Sdp-arm 175696ccba6SMasahiro Yamada desc = xzalloc(sizeof(*desc), 176e0f083a0Sdp-arm "failed to allocate memory for image descriptor"); 177e0f083a0Sdp-arm memcpy(&desc->uuid, uuid, sizeof(uuid_t)); 178e0f083a0Sdp-arm desc->name = xstrdup(name, 179e0f083a0Sdp-arm "failed to allocate memory for image name"); 180e0f083a0Sdp-arm desc->cmdline_name = xstrdup(cmdline_name, 181e0f083a0Sdp-arm "failed to allocate memory for image command line name"); 182e0f083a0Sdp-arm desc->action = DO_UNSPEC; 183e0f083a0Sdp-arm return desc; 184e0f083a0Sdp-arm } 185e0f083a0Sdp-arm 186d02fcebeSdp-arm static void set_image_desc_action(image_desc_t *desc, int action, 187d02fcebeSdp-arm const char *arg) 188d02fcebeSdp-arm { 189d02fcebeSdp-arm assert(desc != NULL); 190d02fcebeSdp-arm 191d02fcebeSdp-arm if (desc->action_arg != DO_UNSPEC) 192d02fcebeSdp-arm free(desc->action_arg); 193d02fcebeSdp-arm desc->action = action; 194d02fcebeSdp-arm desc->action_arg = NULL; 195d02fcebeSdp-arm if (arg != NULL) 196d02fcebeSdp-arm desc->action_arg = xstrdup(arg, 197d02fcebeSdp-arm "failed to allocate memory for argument"); 198d02fcebeSdp-arm } 199d02fcebeSdp-arm 200e0f083a0Sdp-arm static void free_image_desc(image_desc_t *desc) 201e0f083a0Sdp-arm { 202e0f083a0Sdp-arm free(desc->name); 203e0f083a0Sdp-arm free(desc->cmdline_name); 204e0f083a0Sdp-arm free(desc->action_arg); 205e0f083a0Sdp-arm free(desc); 206e0f083a0Sdp-arm } 207e0f083a0Sdp-arm 208e0f083a0Sdp-arm static void add_image_desc(image_desc_t *desc) 209e0f083a0Sdp-arm { 21011c0a4ffSMasahiro Yamada image_desc_t **p = &image_desc_head; 21111c0a4ffSMasahiro Yamada 21211c0a4ffSMasahiro Yamada while (*p) 21311c0a4ffSMasahiro Yamada p = &(*p)->next; 21411c0a4ffSMasahiro Yamada 215e9e0d287SMasahiro Yamada assert(*p == NULL); 21611c0a4ffSMasahiro Yamada *p = desc; 217e0f083a0Sdp-arm nr_image_descs++; 218e0f083a0Sdp-arm } 219e0f083a0Sdp-arm 220e0f083a0Sdp-arm static void free_image_descs(void) 221e0f083a0Sdp-arm { 222e0f083a0Sdp-arm image_desc_t *desc = image_desc_head, *tmp; 223e0f083a0Sdp-arm 224e0f083a0Sdp-arm while (desc != NULL) { 225e0f083a0Sdp-arm tmp = desc->next; 226e0f083a0Sdp-arm free_image_desc(desc); 227e0f083a0Sdp-arm desc = tmp; 228e0f083a0Sdp-arm nr_image_descs--; 229e0f083a0Sdp-arm } 230e0f083a0Sdp-arm assert(nr_image_descs == 0); 231e0f083a0Sdp-arm } 232e0f083a0Sdp-arm 233e0f083a0Sdp-arm static void fill_image_descs(void) 234e0f083a0Sdp-arm { 235e0f083a0Sdp-arm toc_entry_t *toc_entry; 236e0f083a0Sdp-arm 237e0f083a0Sdp-arm for (toc_entry = toc_entries; 238e0f083a0Sdp-arm toc_entry->cmdline_name != NULL; 239e0f083a0Sdp-arm toc_entry++) { 240e0f083a0Sdp-arm image_desc_t *desc; 241e0f083a0Sdp-arm 242e0f083a0Sdp-arm desc = new_image_desc(&toc_entry->uuid, 243e0f083a0Sdp-arm toc_entry->name, 244e0f083a0Sdp-arm toc_entry->cmdline_name); 245e0f083a0Sdp-arm add_image_desc(desc); 246e0f083a0Sdp-arm } 247e0f083a0Sdp-arm } 248e0f083a0Sdp-arm 249819281eeSdp-arm static void add_image(image_t *image) 250819281eeSdp-arm { 25111c0a4ffSMasahiro Yamada image_t **p = &image_head; 25211c0a4ffSMasahiro Yamada 25311c0a4ffSMasahiro Yamada while (*p) 25411c0a4ffSMasahiro Yamada p = &(*p)->next; 25511c0a4ffSMasahiro Yamada 256e9e0d287SMasahiro Yamada assert(*p == NULL); 25711c0a4ffSMasahiro Yamada *p = image; 25811c0a4ffSMasahiro Yamada 259e0f083a0Sdp-arm nr_images++; 260819281eeSdp-arm } 261819281eeSdp-arm 262ea39d557SMasahiro Yamada static void replace_image(image_t *image) 263ea39d557SMasahiro Yamada { 264ea39d557SMasahiro Yamada image_t **p = &image_head; 265ea39d557SMasahiro Yamada 266ea39d557SMasahiro Yamada while (*p) { 26765caa3d0SMasahiro Yamada if (!memcmp(&(*p)->toc_e.uuid, &image->toc_e.uuid, 26865caa3d0SMasahiro Yamada sizeof(image->toc_e.uuid))) 269ea39d557SMasahiro Yamada break; 270ea39d557SMasahiro Yamada p = &(*p)->next; 271ea39d557SMasahiro Yamada } 272ea39d557SMasahiro Yamada 273ea39d557SMasahiro Yamada assert(*p != NULL); 274ea39d557SMasahiro Yamada 275ea39d557SMasahiro Yamada image->next = (*p)->next; 276ea39d557SMasahiro Yamada *p = image; 277ea39d557SMasahiro Yamada } 278ea39d557SMasahiro Yamada 279819281eeSdp-arm static void free_image(image_t *image) 280819281eeSdp-arm { 281819281eeSdp-arm free(image->buffer); 282819281eeSdp-arm free(image); 283819281eeSdp-arm } 284819281eeSdp-arm 285819281eeSdp-arm static void remove_image(image_t *image) 286819281eeSdp-arm { 2879e866d34SMasahiro Yamada image_t *tmp, **p = &image_head; 288819281eeSdp-arm 2899e866d34SMasahiro Yamada while (*p) { 2909e866d34SMasahiro Yamada if (*p == image) 2919e866d34SMasahiro Yamada break; 2929e866d34SMasahiro Yamada p = &(*p)->next; 293819281eeSdp-arm } 2949e866d34SMasahiro Yamada 2959e866d34SMasahiro Yamada assert(*p != NULL); 2969e866d34SMasahiro Yamada 2979e866d34SMasahiro Yamada tmp = *p; 2989e866d34SMasahiro Yamada *p = tmp->next; 299e0f083a0Sdp-arm free_image(tmp); 3009e866d34SMasahiro Yamada 301819281eeSdp-arm nr_images--; 302819281eeSdp-arm } 303819281eeSdp-arm 304819281eeSdp-arm static void free_images(void) 305819281eeSdp-arm { 306e0f083a0Sdp-arm image_t *image = image_head, *tmp; 307819281eeSdp-arm 308e0f083a0Sdp-arm while (image != NULL) { 309e0f083a0Sdp-arm tmp = image->next; 310e0f083a0Sdp-arm free_image(image); 311e0f083a0Sdp-arm image = tmp; 312e0f083a0Sdp-arm nr_images--; 313819281eeSdp-arm } 314e0f083a0Sdp-arm assert(nr_images == 0); 315819281eeSdp-arm } 316819281eeSdp-arm 317e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid) 318819281eeSdp-arm { 319e0f083a0Sdp-arm image_desc_t *desc; 320819281eeSdp-arm 321e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 322e0f083a0Sdp-arm if (memcmp(&desc->uuid, uuid, sizeof(uuid_t)) == 0) 323e0f083a0Sdp-arm return desc; 324819281eeSdp-arm return NULL; 325819281eeSdp-arm } 326819281eeSdp-arm 327e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_opt(const char *opt) 328e0f083a0Sdp-arm { 329e0f083a0Sdp-arm image_desc_t *desc; 330e0f083a0Sdp-arm 331e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 332e0f083a0Sdp-arm if (strcmp(desc->cmdline_name, opt) == 0) 333e0f083a0Sdp-arm return desc; 334e0f083a0Sdp-arm return NULL; 335e0f083a0Sdp-arm } 336e0f083a0Sdp-arm 337e0f083a0Sdp-arm static image_t *lookup_image_from_uuid(const uuid_t *uuid) 338b04efcceSdp-arm { 339b04efcceSdp-arm image_t *image; 340b04efcceSdp-arm 341e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 34265caa3d0SMasahiro Yamada if (!memcmp(&image->toc_e.uuid, uuid, sizeof(*uuid))) 343b04efcceSdp-arm return image; 344b04efcceSdp-arm return NULL; 345b04efcceSdp-arm } 346b04efcceSdp-arm 347fcab6bbeSdp-arm static void uuid_to_str(char *s, size_t len, const uuid_t *u) 348fcab6bbeSdp-arm { 349fcab6bbeSdp-arm assert(len >= (_UUID_STR_LEN + 1)); 350fcab6bbeSdp-arm 351fcab6bbeSdp-arm snprintf(s, len, "%08X-%04X-%04X-%04X-%04X%04X%04X", 352fcab6bbeSdp-arm u->time_low, 353fcab6bbeSdp-arm u->time_mid, 354fcab6bbeSdp-arm u->time_hi_and_version, 355fcab6bbeSdp-arm ((uint16_t)u->clock_seq_hi_and_reserved << 8) | u->clock_seq_low, 356fcab6bbeSdp-arm ((uint16_t)u->node[0] << 8) | u->node[1], 357fcab6bbeSdp-arm ((uint16_t)u->node[2] << 8) | u->node[3], 358fcab6bbeSdp-arm ((uint16_t)u->node[4] << 8) | u->node[5]); 359fcab6bbeSdp-arm } 360fcab6bbeSdp-arm 361fcab6bbeSdp-arm static void uuid_from_str(uuid_t *u, const char *s) 362fcab6bbeSdp-arm { 363fcab6bbeSdp-arm int n; 364fcab6bbeSdp-arm 365fcab6bbeSdp-arm if (s == NULL) 366fcab6bbeSdp-arm log_errx("UUID cannot be NULL"); 367fcab6bbeSdp-arm if (strlen(s) != _UUID_STR_LEN) 368fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 369fcab6bbeSdp-arm 370fcab6bbeSdp-arm n = sscanf(s, 371fcab6bbeSdp-arm "%8x-%4hx-%4hx-%2hhx%2hhx-%2hhx%2hhx%2hhx%2hhx%2hhx%2hhx", 372fcab6bbeSdp-arm &u->time_low, &u->time_mid, &u->time_hi_and_version, 373fcab6bbeSdp-arm &u->clock_seq_hi_and_reserved, &u->clock_seq_low, &u->node[0], 374fcab6bbeSdp-arm &u->node[1], &u->node[2], &u->node[3], &u->node[4], &u->node[5]); 375fcab6bbeSdp-arm /* 376fcab6bbeSdp-arm * Given the format specifier above, we expect 11 items to be scanned 377fcab6bbeSdp-arm * for a properly formatted UUID. 378fcab6bbeSdp-arm */ 379fcab6bbeSdp-arm if (n != 11) 380fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 381fcab6bbeSdp-arm } 382fcab6bbeSdp-arm 38360b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out) 384819281eeSdp-arm { 385819281eeSdp-arm struct stat st; 386819281eeSdp-arm FILE *fp; 387819281eeSdp-arm char *buf, *bufend; 388819281eeSdp-arm fip_toc_header_t *toc_header; 389819281eeSdp-arm fip_toc_entry_t *toc_entry; 390819281eeSdp-arm int terminated = 0; 391819281eeSdp-arm 392819281eeSdp-arm fp = fopen(filename, "r"); 393819281eeSdp-arm if (fp == NULL) 394819281eeSdp-arm log_err("fopen %s", filename); 395819281eeSdp-arm 396819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 397819281eeSdp-arm log_err("fstat %s", filename); 398819281eeSdp-arm 399a22f6285Sdp-arm buf = xmalloc(st.st_size, "failed to load file into memory"); 400819281eeSdp-arm if (fread(buf, 1, st.st_size, fp) != st.st_size) 401819281eeSdp-arm log_errx("Failed to read %s", filename); 402819281eeSdp-arm bufend = buf + st.st_size; 403819281eeSdp-arm fclose(fp); 404819281eeSdp-arm 405819281eeSdp-arm if (st.st_size < sizeof(fip_toc_header_t)) 406819281eeSdp-arm log_errx("FIP %s is truncated", filename); 407819281eeSdp-arm 408819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 409819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 410819281eeSdp-arm 411819281eeSdp-arm if (toc_header->name != TOC_HEADER_NAME) 412819281eeSdp-arm log_errx("%s is not a FIP file", filename); 413819281eeSdp-arm 414819281eeSdp-arm /* Return the ToC header if the caller wants it. */ 415819281eeSdp-arm if (toc_header_out != NULL) 416819281eeSdp-arm *toc_header_out = *toc_header; 417819281eeSdp-arm 418819281eeSdp-arm /* Walk through each ToC entry in the file. */ 419819281eeSdp-arm while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) { 420fcab6bbeSdp-arm image_t *image; 421fcab6bbeSdp-arm image_desc_t *desc; 422fcab6bbeSdp-arm 423819281eeSdp-arm /* Found the ToC terminator, we are done. */ 424819281eeSdp-arm if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) { 425819281eeSdp-arm terminated = 1; 426819281eeSdp-arm break; 427819281eeSdp-arm } 428819281eeSdp-arm 429819281eeSdp-arm /* 430819281eeSdp-arm * Build a new image out of the ToC entry and add it to the 431819281eeSdp-arm * table of images. 432819281eeSdp-arm */ 43311c0a4ffSMasahiro Yamada image = xzalloc(sizeof(*image), 434a22f6285Sdp-arm "failed to allocate memory for image"); 43565caa3d0SMasahiro Yamada image->toc_e = *toc_entry; 436a22f6285Sdp-arm image->buffer = xmalloc(toc_entry->size, 437a22f6285Sdp-arm "failed to allocate image buffer, is FIP file corrupted?"); 438819281eeSdp-arm /* Overflow checks before memory copy. */ 439819281eeSdp-arm if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address) 440819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 441819281eeSdp-arm if (toc_entry->size + toc_entry->offset_address > st.st_size) 442819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 443819281eeSdp-arm 444819281eeSdp-arm memcpy(image->buffer, buf + toc_entry->offset_address, 445819281eeSdp-arm toc_entry->size); 446819281eeSdp-arm 447fcab6bbeSdp-arm /* If this is an unknown image, create a descriptor for it. */ 44865caa3d0SMasahiro Yamada desc = lookup_image_desc_from_uuid(&toc_entry->uuid); 449fcab6bbeSdp-arm if (desc == NULL) { 450fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 451fcab6bbeSdp-arm 45265caa3d0SMasahiro Yamada uuid_to_str(name, sizeof(name), &toc_entry->uuid); 453fcab6bbeSdp-arm snprintf(filename, sizeof(filename), "%s%s", 454fcab6bbeSdp-arm name, ".bin"); 45565caa3d0SMasahiro Yamada desc = new_image_desc(&toc_entry->uuid, name, "blob"); 456fcab6bbeSdp-arm desc->action = DO_UNPACK; 457fcab6bbeSdp-arm desc->action_arg = xstrdup(filename, 458fcab6bbeSdp-arm "failed to allocate memory for blob filename"); 459fcab6bbeSdp-arm add_image_desc(desc); 460fcab6bbeSdp-arm } 461fcab6bbeSdp-arm 462819281eeSdp-arm add_image(image); 463819281eeSdp-arm 464819281eeSdp-arm toc_entry++; 465819281eeSdp-arm } 466819281eeSdp-arm 467819281eeSdp-arm if (terminated == 0) 468819281eeSdp-arm log_errx("FIP %s does not have a ToC terminator entry", 469819281eeSdp-arm filename); 470819281eeSdp-arm free(buf); 471819281eeSdp-arm return 0; 472819281eeSdp-arm } 473819281eeSdp-arm 47460b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename) 475819281eeSdp-arm { 476819281eeSdp-arm struct stat st; 477819281eeSdp-arm image_t *image; 478819281eeSdp-arm FILE *fp; 479819281eeSdp-arm 480b04efcceSdp-arm assert(uuid != NULL); 481b04efcceSdp-arm 482819281eeSdp-arm fp = fopen(filename, "r"); 483819281eeSdp-arm if (fp == NULL) 484819281eeSdp-arm log_err("fopen %s", filename); 485819281eeSdp-arm 486819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 487819281eeSdp-arm log_errx("fstat %s", filename); 488819281eeSdp-arm 48911c0a4ffSMasahiro Yamada image = xzalloc(sizeof(*image), "failed to allocate memory for image"); 49065caa3d0SMasahiro Yamada image->toc_e.uuid = *uuid; 491a22f6285Sdp-arm image->buffer = xmalloc(st.st_size, "failed to allocate image buffer"); 492819281eeSdp-arm if (fread(image->buffer, 1, st.st_size, fp) != st.st_size) 493819281eeSdp-arm log_errx("Failed to read %s", filename); 49465caa3d0SMasahiro Yamada image->toc_e.size = st.st_size; 495819281eeSdp-arm 496819281eeSdp-arm fclose(fp); 497819281eeSdp-arm return image; 498819281eeSdp-arm } 499819281eeSdp-arm 50060b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename) 501819281eeSdp-arm { 502819281eeSdp-arm FILE *fp; 503819281eeSdp-arm 504819281eeSdp-arm fp = fopen(filename, "w"); 505819281eeSdp-arm if (fp == NULL) 506819281eeSdp-arm log_err("fopen"); 50765caa3d0SMasahiro Yamada xfwrite(image->buffer, image->toc_e.size, fp, filename); 508819281eeSdp-arm fclose(fp); 509819281eeSdp-arm return 0; 510819281eeSdp-arm } 511819281eeSdp-arm 512e0f083a0Sdp-arm static struct option *add_opt(struct option *opts, size_t *nr_opts, 513e0f083a0Sdp-arm const char *name, int has_arg, int val) 514819281eeSdp-arm { 515e0f083a0Sdp-arm opts = realloc(opts, (*nr_opts + 1) * sizeof(*opts)); 516e0f083a0Sdp-arm if (opts == NULL) 517e0f083a0Sdp-arm log_err("realloc"); 518e0f083a0Sdp-arm opts[*nr_opts].name = name; 519e0f083a0Sdp-arm opts[*nr_opts].has_arg = has_arg; 520e0f083a0Sdp-arm opts[*nr_opts].flag = NULL; 521e0f083a0Sdp-arm opts[*nr_opts].val = val; 522e0f083a0Sdp-arm ++*nr_opts; 523e0f083a0Sdp-arm return opts; 524819281eeSdp-arm } 525819281eeSdp-arm 526e0f083a0Sdp-arm static struct option *fill_common_opts(struct option *opts, size_t *nr_opts, 527e0f083a0Sdp-arm int has_arg) 528819281eeSdp-arm { 529e0f083a0Sdp-arm image_desc_t *desc; 530e0f083a0Sdp-arm 531e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 532e0f083a0Sdp-arm opts = add_opt(opts, nr_opts, desc->cmdline_name, has_arg, 533e0f083a0Sdp-arm OPT_TOC_ENTRY); 534e0f083a0Sdp-arm return opts; 535819281eeSdp-arm } 536819281eeSdp-arm 537e0f083a0Sdp-arm static void md_print(const unsigned char *md, size_t len) 5389df69ba3Sdp-arm { 5399df69ba3Sdp-arm size_t i; 5409df69ba3Sdp-arm 5419df69ba3Sdp-arm for (i = 0; i < len; i++) 5429df69ba3Sdp-arm printf("%02x", md[i]); 5439df69ba3Sdp-arm } 5449df69ba3Sdp-arm 545819281eeSdp-arm static int info_cmd(int argc, char *argv[]) 546819281eeSdp-arm { 547819281eeSdp-arm image_t *image; 548819281eeSdp-arm fip_toc_header_t toc_header; 549819281eeSdp-arm 550819281eeSdp-arm if (argc != 2) 55185ee2778Sdp-arm info_usage(); 552819281eeSdp-arm argc--, argv++; 553819281eeSdp-arm 554819281eeSdp-arm parse_fip(argv[0], &toc_header); 555819281eeSdp-arm 556819281eeSdp-arm if (verbose) { 557819281eeSdp-arm log_dbgx("toc_header[name]: 0x%llX", 558819281eeSdp-arm (unsigned long long)toc_header.name); 559819281eeSdp-arm log_dbgx("toc_header[serial_number]: 0x%llX", 560819281eeSdp-arm (unsigned long long)toc_header.serial_number); 561819281eeSdp-arm log_dbgx("toc_header[flags]: 0x%llX", 562819281eeSdp-arm (unsigned long long)toc_header.flags); 563819281eeSdp-arm } 564819281eeSdp-arm 565e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) { 566e0f083a0Sdp-arm image_desc_t *desc; 567b04efcceSdp-arm 56865caa3d0SMasahiro Yamada desc = lookup_image_desc_from_uuid(&image->toc_e.uuid); 569fcab6bbeSdp-arm assert(desc != NULL); 57065caa3d0SMasahiro Yamada printf("%s: offset=0x%llX, size=0x%llX, cmdline=\"--%s\"", 57165caa3d0SMasahiro Yamada desc->name, 57265caa3d0SMasahiro Yamada (unsigned long long)image->toc_e.offset_address, 57365caa3d0SMasahiro Yamada (unsigned long long)image->toc_e.size, 574e0f083a0Sdp-arm desc->cmdline_name); 5759df69ba3Sdp-arm if (verbose) { 5769df69ba3Sdp-arm unsigned char md[SHA256_DIGEST_LENGTH]; 5779df69ba3Sdp-arm 57865caa3d0SMasahiro Yamada SHA256(image->buffer, image->toc_e.size, md); 5799df69ba3Sdp-arm printf(", sha256="); 5809df69ba3Sdp-arm md_print(md, sizeof(md)); 5819df69ba3Sdp-arm } 582819281eeSdp-arm putchar('\n'); 583819281eeSdp-arm } 584819281eeSdp-arm 585819281eeSdp-arm free_images(); 586819281eeSdp-arm return 0; 587819281eeSdp-arm } 588819281eeSdp-arm 589819281eeSdp-arm static void info_usage(void) 590819281eeSdp-arm { 591819281eeSdp-arm printf("fiptool info FIP_FILENAME\n"); 59285ee2778Sdp-arm exit(1); 593819281eeSdp-arm } 594819281eeSdp-arm 595*1c75d5dfSMasahiro Yamada static int pack_images(const char *filename, uint64_t toc_flags, unsigned long align) 596819281eeSdp-arm { 597819281eeSdp-arm FILE *fp; 598819281eeSdp-arm image_t *image; 599819281eeSdp-arm fip_toc_header_t *toc_header; 600819281eeSdp-arm fip_toc_entry_t *toc_entry; 601819281eeSdp-arm char *buf; 60265caa3d0SMasahiro Yamada uint64_t entry_offset, buf_size, payload_size = 0; 603819281eeSdp-arm 604819281eeSdp-arm buf_size = sizeof(fip_toc_header_t) + 605819281eeSdp-arm sizeof(fip_toc_entry_t) * (nr_images + 1); 606819281eeSdp-arm buf = calloc(1, buf_size); 607819281eeSdp-arm if (buf == NULL) 608819281eeSdp-arm log_err("calloc"); 609819281eeSdp-arm 610819281eeSdp-arm /* Build up header and ToC entries from the image table. */ 611819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 612819281eeSdp-arm toc_header->name = TOC_HEADER_NAME; 613819281eeSdp-arm toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER; 614819281eeSdp-arm toc_header->flags = toc_flags; 615819281eeSdp-arm 616819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 617819281eeSdp-arm 618819281eeSdp-arm entry_offset = buf_size; 619e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) { 62065caa3d0SMasahiro Yamada payload_size += image->toc_e.size; 621*1c75d5dfSMasahiro Yamada entry_offset = (entry_offset + align - 1) & ~(align - 1); 62265caa3d0SMasahiro Yamada image->toc_e.offset_address = entry_offset; 62365caa3d0SMasahiro Yamada *toc_entry++ = image->toc_e; 62465caa3d0SMasahiro Yamada entry_offset += image->toc_e.size; 625819281eeSdp-arm } 626819281eeSdp-arm 627819281eeSdp-arm /* Append a null uuid entry to mark the end of ToC entries. */ 62865caa3d0SMasahiro Yamada memset(toc_entry, 0, sizeof(*toc_entry)); 629819281eeSdp-arm toc_entry->offset_address = entry_offset; 630819281eeSdp-arm 631819281eeSdp-arm /* Generate the FIP file. */ 632819281eeSdp-arm fp = fopen(filename, "w"); 633819281eeSdp-arm if (fp == NULL) 634819281eeSdp-arm log_err("fopen %s", filename); 635819281eeSdp-arm 636819281eeSdp-arm if (verbose) 637819281eeSdp-arm log_dbgx("Metadata size: %zu bytes", buf_size); 638819281eeSdp-arm 639a1da83f5SMasahiro Yamada xfwrite(buf, buf_size, fp, filename); 640819281eeSdp-arm free(buf); 641819281eeSdp-arm 642819281eeSdp-arm if (verbose) 643819281eeSdp-arm log_dbgx("Payload size: %zu bytes", payload_size); 644819281eeSdp-arm 645*1c75d5dfSMasahiro Yamada for (image = image_head; image != NULL; image = image->next) { 646*1c75d5dfSMasahiro Yamada if (fseek(fp, image->toc_e.offset_address, SEEK_SET)) 647*1c75d5dfSMasahiro Yamada log_errx("Failed to set file position"); 648*1c75d5dfSMasahiro Yamada 64965caa3d0SMasahiro Yamada xfwrite(image->buffer, image->toc_e.size, fp, filename); 650*1c75d5dfSMasahiro Yamada } 651819281eeSdp-arm 652819281eeSdp-arm fclose(fp); 653819281eeSdp-arm return 0; 654819281eeSdp-arm } 655819281eeSdp-arm 656819281eeSdp-arm /* 657819281eeSdp-arm * This function is shared between the create and update subcommands. 658819281eeSdp-arm * The difference between the two subcommands is that when the FIP file 659819281eeSdp-arm * is created, the parsing of an existing FIP is skipped. This results 660819281eeSdp-arm * in update_fip() creating the new FIP file from scratch because the 661819281eeSdp-arm * internal image table is not populated. 662819281eeSdp-arm */ 663819281eeSdp-arm static void update_fip(void) 664819281eeSdp-arm { 665e0f083a0Sdp-arm image_desc_t *desc; 666819281eeSdp-arm 667819281eeSdp-arm /* Add or replace images in the FIP file. */ 668e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 669fcab6bbeSdp-arm image_t *new_image, *old_image; 670fcab6bbeSdp-arm 671e0f083a0Sdp-arm if (desc->action != DO_PACK) 672819281eeSdp-arm continue; 673819281eeSdp-arm 674e0f083a0Sdp-arm new_image = read_image_from_file(&desc->uuid, 675e0f083a0Sdp-arm desc->action_arg); 676e0f083a0Sdp-arm old_image = lookup_image_from_uuid(&desc->uuid); 677b04efcceSdp-arm if (old_image != NULL) { 678e0f083a0Sdp-arm if (verbose) { 679fcab6bbeSdp-arm log_dbgx("Replacing %s with %s", 680e0f083a0Sdp-arm desc->cmdline_name, 681e0f083a0Sdp-arm desc->action_arg); 682e0f083a0Sdp-arm } 683ea39d557SMasahiro Yamada replace_image(new_image); 684819281eeSdp-arm } else { 685819281eeSdp-arm if (verbose) 686819281eeSdp-arm log_dbgx("Adding image %s", 687e0f083a0Sdp-arm desc->action_arg); 688b04efcceSdp-arm add_image(new_image); 689819281eeSdp-arm } 690819281eeSdp-arm } 691819281eeSdp-arm } 692819281eeSdp-arm 693e0f083a0Sdp-arm static void parse_plat_toc_flags(const char *arg, unsigned long long *toc_flags) 694819281eeSdp-arm { 695819281eeSdp-arm unsigned long long flags; 696819281eeSdp-arm char *endptr; 697819281eeSdp-arm 698819281eeSdp-arm errno = 0; 699819281eeSdp-arm flags = strtoull(arg, &endptr, 16); 700819281eeSdp-arm if (*endptr != '\0' || flags > UINT16_MAX || errno != 0) 701819281eeSdp-arm log_errx("Invalid platform ToC flags: %s", arg); 702819281eeSdp-arm /* Platform ToC flags is a 16-bit field occupying bits [32-47]. */ 703819281eeSdp-arm *toc_flags |= flags << 32; 704819281eeSdp-arm } 705819281eeSdp-arm 706*1c75d5dfSMasahiro Yamada static int is_power_of_2(unsigned long x) 707*1c75d5dfSMasahiro Yamada { 708*1c75d5dfSMasahiro Yamada return x && !(x & (x - 1)); 709*1c75d5dfSMasahiro Yamada } 710*1c75d5dfSMasahiro Yamada 711*1c75d5dfSMasahiro Yamada static unsigned long get_image_align(char *arg) 712*1c75d5dfSMasahiro Yamada { 713*1c75d5dfSMasahiro Yamada char *endptr; 714*1c75d5dfSMasahiro Yamada unsigned long align; 715*1c75d5dfSMasahiro Yamada 716*1c75d5dfSMasahiro Yamada errno = 0; 717*1c75d5dfSMasahiro Yamada align = strtoul(arg, &endptr, 10); 718*1c75d5dfSMasahiro Yamada if (*endptr != '\0' || !is_power_of_2(align) || errno != 0) 719*1c75d5dfSMasahiro Yamada log_errx("Invalid alignment: %s", arg); 720*1c75d5dfSMasahiro Yamada 721*1c75d5dfSMasahiro Yamada return align; 722*1c75d5dfSMasahiro Yamada } 723*1c75d5dfSMasahiro Yamada 724fcab6bbeSdp-arm static void parse_blob_opt(char *arg, uuid_t *uuid, char *filename, size_t len) 725fcab6bbeSdp-arm { 726fcab6bbeSdp-arm char *p; 727fcab6bbeSdp-arm 728fcab6bbeSdp-arm for (p = strtok(arg, ","); p != NULL; p = strtok(NULL, ",")) { 729fcab6bbeSdp-arm if (strncmp(p, "uuid=", strlen("uuid=")) == 0) { 730fcab6bbeSdp-arm p += strlen("uuid="); 731fcab6bbeSdp-arm uuid_from_str(uuid, p); 732fcab6bbeSdp-arm } else if (strncmp(p, "file=", strlen("file=")) == 0) { 733fcab6bbeSdp-arm p += strlen("file="); 734fcab6bbeSdp-arm snprintf(filename, len, "%s", p); 735fcab6bbeSdp-arm } 736fcab6bbeSdp-arm } 737fcab6bbeSdp-arm } 738fcab6bbeSdp-arm 739819281eeSdp-arm static int create_cmd(int argc, char *argv[]) 740819281eeSdp-arm { 741e0f083a0Sdp-arm struct option *opts = NULL; 742e0f083a0Sdp-arm size_t nr_opts = 0; 743819281eeSdp-arm unsigned long long toc_flags = 0; 744*1c75d5dfSMasahiro Yamada unsigned long align = 1; 745819281eeSdp-arm 746819281eeSdp-arm if (argc < 2) 74785ee2778Sdp-arm create_usage(); 748819281eeSdp-arm 749e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 750e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 751819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 752*1c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 753fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 754e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 755819281eeSdp-arm 756819281eeSdp-arm while (1) { 75720f87e78Sdp-arm int c, opt_index = 0; 758819281eeSdp-arm 759fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:", opts, &opt_index); 760819281eeSdp-arm if (c == -1) 761819281eeSdp-arm break; 762819281eeSdp-arm 763819281eeSdp-arm switch (c) { 764819281eeSdp-arm case OPT_TOC_ENTRY: { 765e0f083a0Sdp-arm image_desc_t *desc; 766819281eeSdp-arm 767e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 768d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 769819281eeSdp-arm break; 770819281eeSdp-arm } 771819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: 772819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 773819281eeSdp-arm break; 774*1c75d5dfSMasahiro Yamada case OPT_ALIGN: 775*1c75d5dfSMasahiro Yamada align = get_image_align(optarg); 776*1c75d5dfSMasahiro Yamada break; 777fcab6bbeSdp-arm case 'b': { 778fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 779fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 780fcab6bbeSdp-arm uuid_t uuid = { 0 }; 781fcab6bbeSdp-arm image_desc_t *desc; 782fcab6bbeSdp-arm 783fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 784fcab6bbeSdp-arm filename, sizeof(filename)); 785fcab6bbeSdp-arm 786fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 787fcab6bbeSdp-arm filename[0] == '\0') 788fcab6bbeSdp-arm create_usage(); 789fcab6bbeSdp-arm 790fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 791d02fcebeSdp-arm if (desc == NULL) { 792fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 793fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 794fcab6bbeSdp-arm add_image_desc(desc); 795fcab6bbeSdp-arm } 796d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 797fcab6bbeSdp-arm break; 798fcab6bbeSdp-arm } 799819281eeSdp-arm default: 80085ee2778Sdp-arm create_usage(); 801819281eeSdp-arm } 802819281eeSdp-arm } 803819281eeSdp-arm argc -= optind; 804819281eeSdp-arm argv += optind; 805e0f083a0Sdp-arm free(opts); 806819281eeSdp-arm 807819281eeSdp-arm if (argc == 0) 80885ee2778Sdp-arm create_usage(); 809819281eeSdp-arm 810819281eeSdp-arm update_fip(); 811819281eeSdp-arm 812*1c75d5dfSMasahiro Yamada pack_images(argv[0], toc_flags, align); 813819281eeSdp-arm free_images(); 814819281eeSdp-arm return 0; 815819281eeSdp-arm } 816819281eeSdp-arm 817819281eeSdp-arm static void create_usage(void) 818819281eeSdp-arm { 819819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 820819281eeSdp-arm 821ee079320SMasahiro Yamada printf("fiptool create [opts] FIP_FILENAME\n"); 822ee079320SMasahiro Yamada printf("\n"); 823ee079320SMasahiro Yamada printf("Options:\n"); 824*1c75d5dfSMasahiro Yamada printf(" --align <value>\t\tEach image is aligned to <value> (default: 1).\n"); 825fcab6bbeSdp-arm printf(" --blob uuid=...,file=...\tAdd an image with the given UUID " 826fcab6bbeSdp-arm "pointed to by file.\n"); 827819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 828819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 829819281eeSdp-arm fputc('\n', stderr); 830819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 831819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 832819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 833819281eeSdp-arm toc_entry->name); 83485ee2778Sdp-arm exit(1); 835819281eeSdp-arm } 836819281eeSdp-arm 837819281eeSdp-arm static int update_cmd(int argc, char *argv[]) 838819281eeSdp-arm { 839e0f083a0Sdp-arm struct option *opts = NULL; 840e0f083a0Sdp-arm size_t nr_opts = 0; 841fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 842819281eeSdp-arm fip_toc_header_t toc_header = { 0 }; 843819281eeSdp-arm unsigned long long toc_flags = 0; 844*1c75d5dfSMasahiro Yamada unsigned long align = 1; 845819281eeSdp-arm int pflag = 0; 846819281eeSdp-arm 847819281eeSdp-arm if (argc < 2) 84885ee2778Sdp-arm update_usage(); 849819281eeSdp-arm 850e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 851*1c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 852fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 853e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 854e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 855819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 856e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 857819281eeSdp-arm 858819281eeSdp-arm while (1) { 85920f87e78Sdp-arm int c, opt_index = 0; 860819281eeSdp-arm 861fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:o:", opts, &opt_index); 862819281eeSdp-arm if (c == -1) 863819281eeSdp-arm break; 864819281eeSdp-arm 865819281eeSdp-arm switch (c) { 866819281eeSdp-arm case OPT_TOC_ENTRY: { 867e0f083a0Sdp-arm image_desc_t *desc; 868819281eeSdp-arm 869e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 870d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 871819281eeSdp-arm break; 872819281eeSdp-arm } 873e0f083a0Sdp-arm case OPT_PLAT_TOC_FLAGS: 874819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 875819281eeSdp-arm pflag = 1; 876819281eeSdp-arm break; 877fcab6bbeSdp-arm case 'b': { 878fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 879fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 880fcab6bbeSdp-arm uuid_t uuid = { 0 }; 881fcab6bbeSdp-arm image_desc_t *desc; 882fcab6bbeSdp-arm 883fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 884fcab6bbeSdp-arm filename, sizeof(filename)); 885fcab6bbeSdp-arm 886fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 887fcab6bbeSdp-arm filename[0] == '\0') 888fcab6bbeSdp-arm update_usage(); 889fcab6bbeSdp-arm 890fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 891d02fcebeSdp-arm if (desc == NULL) { 892fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 893fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 894fcab6bbeSdp-arm add_image_desc(desc); 895fcab6bbeSdp-arm } 896d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 897fcab6bbeSdp-arm break; 898fcab6bbeSdp-arm } 899*1c75d5dfSMasahiro Yamada case OPT_ALIGN: 900*1c75d5dfSMasahiro Yamada align = get_image_align(optarg); 901*1c75d5dfSMasahiro Yamada break; 902819281eeSdp-arm case 'o': 903819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 904819281eeSdp-arm break; 905819281eeSdp-arm default: 90685ee2778Sdp-arm update_usage(); 907819281eeSdp-arm } 908819281eeSdp-arm } 909819281eeSdp-arm argc -= optind; 910819281eeSdp-arm argv += optind; 911e0f083a0Sdp-arm free(opts); 912819281eeSdp-arm 913819281eeSdp-arm if (argc == 0) 91485ee2778Sdp-arm update_usage(); 915819281eeSdp-arm 916819281eeSdp-arm if (outfile[0] == '\0') 917819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 918819281eeSdp-arm 91940866aafSMasahiro Yamada if (access(argv[0], F_OK) == 0) 920819281eeSdp-arm parse_fip(argv[0], &toc_header); 921819281eeSdp-arm 922819281eeSdp-arm if (pflag) 923819281eeSdp-arm toc_header.flags &= ~(0xffffULL << 32); 924819281eeSdp-arm toc_flags = (toc_header.flags |= toc_flags); 925819281eeSdp-arm 926819281eeSdp-arm update_fip(); 927819281eeSdp-arm 928*1c75d5dfSMasahiro Yamada pack_images(outfile, toc_flags, align); 929819281eeSdp-arm free_images(); 930819281eeSdp-arm return 0; 931819281eeSdp-arm } 932819281eeSdp-arm 933819281eeSdp-arm static void update_usage(void) 934819281eeSdp-arm { 935819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 936819281eeSdp-arm 937ee079320SMasahiro Yamada printf("fiptool update [opts] FIP_FILENAME\n"); 938ee079320SMasahiro Yamada printf("\n"); 939ee079320SMasahiro Yamada printf("Options:\n"); 940*1c75d5dfSMasahiro Yamada printf(" --align <value>\t\tEach image is aligned to <value> (default: 1).\n"); 941fcab6bbeSdp-arm printf(" --blob uuid=...,file=...\tAdd or update an image " 942fcab6bbeSdp-arm "with the given UUID pointed to by file.\n"); 943819281eeSdp-arm printf(" --out FIP_FILENAME\t\tSet an alternative output FIP file.\n"); 944819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 945819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 946819281eeSdp-arm fputc('\n', stderr); 947819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 948819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 949819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 950819281eeSdp-arm toc_entry->name); 95185ee2778Sdp-arm exit(1); 952819281eeSdp-arm } 953819281eeSdp-arm 954819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]) 955819281eeSdp-arm { 956e0f083a0Sdp-arm struct option *opts = NULL; 957e0f083a0Sdp-arm size_t nr_opts = 0; 958fcab6bbeSdp-arm char outdir[PATH_MAX] = { 0 }; 959e0f083a0Sdp-arm image_desc_t *desc; 960819281eeSdp-arm int fflag = 0; 961819281eeSdp-arm int unpack_all = 1; 962819281eeSdp-arm 963819281eeSdp-arm if (argc < 2) 96485ee2778Sdp-arm unpack_usage(); 965819281eeSdp-arm 966e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 967fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 968e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 969e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 970e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 971819281eeSdp-arm 972819281eeSdp-arm while (1) { 97320f87e78Sdp-arm int c, opt_index = 0; 974819281eeSdp-arm 975fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 976819281eeSdp-arm if (c == -1) 977819281eeSdp-arm break; 978819281eeSdp-arm 979819281eeSdp-arm switch (c) { 980e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 981e0f083a0Sdp-arm image_desc_t *desc; 982e0f083a0Sdp-arm 983e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 984d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, optarg); 985e0f083a0Sdp-arm unpack_all = 0; 986819281eeSdp-arm break; 987e0f083a0Sdp-arm } 988fcab6bbeSdp-arm case 'b': { 989fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 990fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 991fcab6bbeSdp-arm uuid_t uuid = { 0 }; 992fcab6bbeSdp-arm image_desc_t *desc; 993fcab6bbeSdp-arm 994fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 995fcab6bbeSdp-arm filename, sizeof(filename)); 996fcab6bbeSdp-arm 997fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 998fcab6bbeSdp-arm filename[0] == '\0') 999fcab6bbeSdp-arm unpack_usage(); 1000fcab6bbeSdp-arm 1001fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 1002d02fcebeSdp-arm if (desc == NULL) { 1003fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 1004fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 1005fcab6bbeSdp-arm add_image_desc(desc); 1006fcab6bbeSdp-arm } 1007d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, filename); 1008fcab6bbeSdp-arm unpack_all = 0; 1009fcab6bbeSdp-arm break; 1010fcab6bbeSdp-arm } 1011819281eeSdp-arm case 'f': 1012819281eeSdp-arm fflag = 1; 1013819281eeSdp-arm break; 1014819281eeSdp-arm case 'o': 1015819281eeSdp-arm snprintf(outdir, sizeof(outdir), "%s", optarg); 1016819281eeSdp-arm break; 1017819281eeSdp-arm default: 101885ee2778Sdp-arm unpack_usage(); 1019819281eeSdp-arm } 1020819281eeSdp-arm } 1021819281eeSdp-arm argc -= optind; 1022819281eeSdp-arm argv += optind; 1023e0f083a0Sdp-arm free(opts); 1024819281eeSdp-arm 1025819281eeSdp-arm if (argc == 0) 102685ee2778Sdp-arm unpack_usage(); 1027819281eeSdp-arm 1028819281eeSdp-arm parse_fip(argv[0], NULL); 1029819281eeSdp-arm 1030819281eeSdp-arm if (outdir[0] != '\0') 1031819281eeSdp-arm if (chdir(outdir) == -1) 1032819281eeSdp-arm log_err("chdir %s", outdir); 1033819281eeSdp-arm 1034819281eeSdp-arm /* Unpack all specified images. */ 1035e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 1036fcab6bbeSdp-arm char file[PATH_MAX]; 1037b04efcceSdp-arm image_t *image; 1038b04efcceSdp-arm 1039e0f083a0Sdp-arm if (!unpack_all && desc->action != DO_UNPACK) 1040819281eeSdp-arm continue; 1041819281eeSdp-arm 1042819281eeSdp-arm /* Build filename. */ 1043e0f083a0Sdp-arm if (desc->action_arg == NULL) 1044819281eeSdp-arm snprintf(file, sizeof(file), "%s.bin", 1045e0f083a0Sdp-arm desc->cmdline_name); 1046819281eeSdp-arm else 1047819281eeSdp-arm snprintf(file, sizeof(file), "%s", 1048e0f083a0Sdp-arm desc->action_arg); 1049819281eeSdp-arm 1050e0f083a0Sdp-arm image = lookup_image_from_uuid(&desc->uuid); 1051b04efcceSdp-arm if (image == NULL) { 1052b04efcceSdp-arm if (!unpack_all) 1053fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 1054819281eeSdp-arm file, argv[0]); 1055819281eeSdp-arm continue; 1056819281eeSdp-arm } 1057819281eeSdp-arm 1058819281eeSdp-arm if (access(file, F_OK) != 0 || fflag) { 1059819281eeSdp-arm if (verbose) 1060819281eeSdp-arm log_dbgx("Unpacking %s", file); 1061b04efcceSdp-arm write_image_to_file(image, file); 1062819281eeSdp-arm } else { 1063819281eeSdp-arm log_warnx("File %s already exists, use --force to overwrite it", 1064819281eeSdp-arm file); 1065819281eeSdp-arm } 1066819281eeSdp-arm } 1067819281eeSdp-arm 1068819281eeSdp-arm free_images(); 1069819281eeSdp-arm return 0; 1070819281eeSdp-arm } 1071819281eeSdp-arm 1072819281eeSdp-arm static void unpack_usage(void) 1073819281eeSdp-arm { 1074819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 1075819281eeSdp-arm 1076ee079320SMasahiro Yamada printf("fiptool unpack [opts] FIP_FILENAME\n"); 1077ee079320SMasahiro Yamada printf("\n"); 1078ee079320SMasahiro Yamada printf("Options:\n"); 1079fcab6bbeSdp-arm printf(" --blob uuid=...,file=...\tUnpack an image with the given UUID " 1080fcab6bbeSdp-arm "to file.\n"); 1081fcab6bbeSdp-arm printf(" --force\t\t\tIf the output file already exists, use --force to " 1082819281eeSdp-arm "overwrite it.\n"); 1083fcab6bbeSdp-arm printf(" --out path\t\t\tSet the output directory path.\n"); 1084819281eeSdp-arm fputc('\n', stderr); 1085819281eeSdp-arm printf("Specific images are unpacked with the following options:\n"); 1086819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1087819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 1088819281eeSdp-arm toc_entry->name); 1089819281eeSdp-arm fputc('\n', stderr); 1090819281eeSdp-arm printf("If no options are provided, all images will be unpacked.\n"); 109185ee2778Sdp-arm exit(1); 1092819281eeSdp-arm } 1093819281eeSdp-arm 1094819281eeSdp-arm static int remove_cmd(int argc, char *argv[]) 1095819281eeSdp-arm { 1096e0f083a0Sdp-arm struct option *opts = NULL; 1097e0f083a0Sdp-arm size_t nr_opts = 0; 1098fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 1099819281eeSdp-arm fip_toc_header_t toc_header; 1100e0f083a0Sdp-arm image_desc_t *desc; 1101*1c75d5dfSMasahiro Yamada unsigned long align = 1; 1102819281eeSdp-arm int fflag = 0; 1103819281eeSdp-arm 1104819281eeSdp-arm if (argc < 2) 110585ee2778Sdp-arm remove_usage(); 1106819281eeSdp-arm 1107e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, no_argument); 1108*1c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 1109fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 1110e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 1111e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 1112e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 1113819281eeSdp-arm 1114819281eeSdp-arm while (1) { 111520f87e78Sdp-arm int c, opt_index = 0; 1116819281eeSdp-arm 1117fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 1118819281eeSdp-arm if (c == -1) 1119819281eeSdp-arm break; 1120819281eeSdp-arm 1121819281eeSdp-arm switch (c) { 1122e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 1123e0f083a0Sdp-arm image_desc_t *desc; 1124e0f083a0Sdp-arm 1125e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 1126d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1127819281eeSdp-arm break; 1128e0f083a0Sdp-arm } 1129*1c75d5dfSMasahiro Yamada case OPT_ALIGN: 1130*1c75d5dfSMasahiro Yamada align = get_image_align(optarg); 1131*1c75d5dfSMasahiro Yamada break; 1132fcab6bbeSdp-arm case 'b': { 1133fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 1134fcab6bbeSdp-arm uuid_t uuid = { 0 }; 1135fcab6bbeSdp-arm image_desc_t *desc; 1136fcab6bbeSdp-arm 1137fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 1138fcab6bbeSdp-arm filename, sizeof(filename)); 1139fcab6bbeSdp-arm 1140fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0) 1141fcab6bbeSdp-arm remove_usage(); 1142fcab6bbeSdp-arm 1143fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 1144d02fcebeSdp-arm if (desc == NULL) { 1145fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 1146fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 1147fcab6bbeSdp-arm add_image_desc(desc); 1148fcab6bbeSdp-arm } 1149d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1150fcab6bbeSdp-arm break; 1151fcab6bbeSdp-arm } 1152819281eeSdp-arm case 'f': 1153819281eeSdp-arm fflag = 1; 1154819281eeSdp-arm break; 1155819281eeSdp-arm case 'o': 1156819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 1157819281eeSdp-arm break; 1158819281eeSdp-arm default: 115985ee2778Sdp-arm remove_usage(); 1160819281eeSdp-arm } 1161819281eeSdp-arm } 1162819281eeSdp-arm argc -= optind; 1163819281eeSdp-arm argv += optind; 1164e0f083a0Sdp-arm free(opts); 1165819281eeSdp-arm 1166819281eeSdp-arm if (argc == 0) 116785ee2778Sdp-arm remove_usage(); 1168819281eeSdp-arm 1169819281eeSdp-arm if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag) 1170819281eeSdp-arm log_errx("File %s already exists, use --force to overwrite it", 1171819281eeSdp-arm outfile); 1172819281eeSdp-arm 1173819281eeSdp-arm if (outfile[0] == '\0') 1174819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 1175819281eeSdp-arm 1176819281eeSdp-arm parse_fip(argv[0], &toc_header); 1177819281eeSdp-arm 1178e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 1179b04efcceSdp-arm image_t *image; 1180b04efcceSdp-arm 1181e0f083a0Sdp-arm if (desc->action != DO_REMOVE) 1182819281eeSdp-arm continue; 1183fcab6bbeSdp-arm 1184e0f083a0Sdp-arm image = lookup_image_from_uuid(&desc->uuid); 1185b04efcceSdp-arm if (image != NULL) { 1186819281eeSdp-arm if (verbose) 1187fcab6bbeSdp-arm log_dbgx("Removing %s", 1188e0f083a0Sdp-arm desc->cmdline_name); 1189b04efcceSdp-arm remove_image(image); 1190819281eeSdp-arm } else { 1191fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 1192e0f083a0Sdp-arm desc->cmdline_name, argv[0]); 1193819281eeSdp-arm } 1194819281eeSdp-arm } 1195819281eeSdp-arm 1196*1c75d5dfSMasahiro Yamada pack_images(outfile, toc_header.flags, align); 1197819281eeSdp-arm free_images(); 1198819281eeSdp-arm return 0; 1199819281eeSdp-arm } 1200819281eeSdp-arm 1201819281eeSdp-arm static void remove_usage(void) 1202819281eeSdp-arm { 1203819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 1204819281eeSdp-arm 1205ee079320SMasahiro Yamada printf("fiptool remove [opts] FIP_FILENAME\n"); 1206ee079320SMasahiro Yamada printf("\n"); 1207ee079320SMasahiro Yamada printf("Options:\n"); 1208*1c75d5dfSMasahiro Yamada printf(" --align <value>\tEach image is aligned to <value> (default: 1).\n"); 1209fcab6bbeSdp-arm printf(" --blob uuid=...\tRemove an image with the given UUID.\n"); 1210819281eeSdp-arm printf(" --force\t\tIf the output FIP file already exists, use --force to " 1211819281eeSdp-arm "overwrite it.\n"); 1212819281eeSdp-arm printf(" --out FIP_FILENAME\tSet an alternative output FIP file.\n"); 1213819281eeSdp-arm fputc('\n', stderr); 1214819281eeSdp-arm printf("Specific images are removed with the following options:\n"); 1215819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1216819281eeSdp-arm printf(" --%-16s\t%s\n", toc_entry->cmdline_name, 1217819281eeSdp-arm toc_entry->name); 121885ee2778Sdp-arm exit(1); 1219819281eeSdp-arm } 1220819281eeSdp-arm 1221819281eeSdp-arm static int version_cmd(int argc, char *argv[]) 1222819281eeSdp-arm { 1223819281eeSdp-arm #ifdef VERSION 1224819281eeSdp-arm puts(VERSION); 1225819281eeSdp-arm #else 1226819281eeSdp-arm /* If built from fiptool directory, VERSION is not set. */ 1227819281eeSdp-arm puts("Unknown version"); 1228819281eeSdp-arm #endif 1229819281eeSdp-arm return 0; 1230819281eeSdp-arm } 1231819281eeSdp-arm 1232819281eeSdp-arm static void version_usage(void) 1233819281eeSdp-arm { 1234819281eeSdp-arm printf("fiptool version\n"); 123585ee2778Sdp-arm exit(1); 1236819281eeSdp-arm } 1237819281eeSdp-arm 1238819281eeSdp-arm static int help_cmd(int argc, char *argv[]) 1239819281eeSdp-arm { 1240819281eeSdp-arm int i; 1241819281eeSdp-arm 1242819281eeSdp-arm if (argc < 2) 1243819281eeSdp-arm usage(); 1244819281eeSdp-arm argc--, argv++; 1245819281eeSdp-arm 1246819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1247819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0 && 124885ee2778Sdp-arm cmds[i].usage != NULL) 1249819281eeSdp-arm cmds[i].usage(); 1250819281eeSdp-arm } 1251819281eeSdp-arm if (i == NELEM(cmds)) 1252819281eeSdp-arm printf("No help for subcommand '%s'\n", argv[0]); 1253819281eeSdp-arm return 0; 1254819281eeSdp-arm } 1255819281eeSdp-arm 1256819281eeSdp-arm static void usage(void) 1257819281eeSdp-arm { 12584f96a498SMasahiro Yamada printf("usage: fiptool [--verbose] <command> [<args>]\n"); 1259819281eeSdp-arm printf("Global options supported:\n"); 1260819281eeSdp-arm printf(" --verbose\tEnable verbose output for all commands.\n"); 1261819281eeSdp-arm fputc('\n', stderr); 1262819281eeSdp-arm printf("Commands supported:\n"); 1263819281eeSdp-arm printf(" info\t\tList images contained in FIP.\n"); 1264819281eeSdp-arm printf(" create\tCreate a new FIP with the given images.\n"); 1265819281eeSdp-arm printf(" update\tUpdate an existing FIP with the given images.\n"); 1266819281eeSdp-arm printf(" unpack\tUnpack images from FIP.\n"); 1267819281eeSdp-arm printf(" remove\tRemove images from FIP.\n"); 1268819281eeSdp-arm printf(" version\tShow fiptool version.\n"); 1269819281eeSdp-arm printf(" help\t\tShow help for given command.\n"); 1270819281eeSdp-arm exit(1); 1271819281eeSdp-arm } 1272819281eeSdp-arm 1273819281eeSdp-arm int main(int argc, char *argv[]) 1274819281eeSdp-arm { 1275819281eeSdp-arm int i, ret = 0; 1276819281eeSdp-arm 1277cc672bb2Sdp-arm while (1) { 1278cc672bb2Sdp-arm int c, opt_index = 0; 1279cc672bb2Sdp-arm static struct option opts[] = { 1280cc672bb2Sdp-arm { "verbose", no_argument, NULL, 'v' }, 1281cc672bb2Sdp-arm { NULL, no_argument, NULL, 0 } 1282cc672bb2Sdp-arm }; 1283819281eeSdp-arm 1284cc672bb2Sdp-arm /* 1285cc672bb2Sdp-arm * Set POSIX mode so getopt stops at the first non-option 1286cc672bb2Sdp-arm * which is the subcommand. 1287cc672bb2Sdp-arm */ 1288cc672bb2Sdp-arm c = getopt_long(argc, argv, "+v", opts, &opt_index); 1289cc672bb2Sdp-arm if (c == -1) 1290cc672bb2Sdp-arm break; 1291cc672bb2Sdp-arm 1292cc672bb2Sdp-arm switch (c) { 1293cc672bb2Sdp-arm case 'v': 1294819281eeSdp-arm verbose = 1; 1295cc672bb2Sdp-arm break; 1296cc672bb2Sdp-arm default: 1297c9cb4089SMasahiro Yamada usage(); 1298819281eeSdp-arm } 1299cc672bb2Sdp-arm } 1300cc672bb2Sdp-arm argc -= optind; 1301cc672bb2Sdp-arm argv += optind; 1302cc672bb2Sdp-arm /* Reset optind for subsequent getopt processing. */ 1303cc672bb2Sdp-arm optind = 0; 1304cc672bb2Sdp-arm 1305cc672bb2Sdp-arm if (argc == 0) 1306cc672bb2Sdp-arm usage(); 1307819281eeSdp-arm 1308e0f083a0Sdp-arm fill_image_descs(); 1309819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1310819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0) { 1311819281eeSdp-arm ret = cmds[i].handler(argc, argv); 1312819281eeSdp-arm break; 1313819281eeSdp-arm } 1314819281eeSdp-arm } 1315819281eeSdp-arm if (i == NELEM(cmds)) 1316819281eeSdp-arm usage(); 1317e0f083a0Sdp-arm free_image_descs(); 1318819281eeSdp-arm return ret; 1319819281eeSdp-arm } 1320