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 53819281eeSdp-arm 54819281eeSdp-arm static int info_cmd(int argc, char *argv[]); 55819281eeSdp-arm static void info_usage(void); 56819281eeSdp-arm static int create_cmd(int argc, char *argv[]); 57819281eeSdp-arm static void create_usage(void); 58819281eeSdp-arm static int update_cmd(int argc, char *argv[]); 59819281eeSdp-arm static void update_usage(void); 60819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]); 61819281eeSdp-arm static void unpack_usage(void); 62819281eeSdp-arm static int remove_cmd(int argc, char *argv[]); 63819281eeSdp-arm static void remove_usage(void); 64819281eeSdp-arm static int version_cmd(int argc, char *argv[]); 65819281eeSdp-arm static void version_usage(void); 66819281eeSdp-arm static int help_cmd(int argc, char *argv[]); 67819281eeSdp-arm static void usage(void); 68819281eeSdp-arm 69819281eeSdp-arm /* Available subcommands. */ 70819281eeSdp-arm static cmd_t cmds[] = { 71819281eeSdp-arm { .name = "info", .handler = info_cmd, .usage = info_usage }, 72819281eeSdp-arm { .name = "create", .handler = create_cmd, .usage = create_usage }, 73819281eeSdp-arm { .name = "update", .handler = update_cmd, .usage = update_usage }, 74819281eeSdp-arm { .name = "unpack", .handler = unpack_cmd, .usage = unpack_usage }, 75819281eeSdp-arm { .name = "remove", .handler = remove_cmd, .usage = remove_usage }, 76819281eeSdp-arm { .name = "version", .handler = version_cmd, .usage = version_usage }, 77819281eeSdp-arm { .name = "help", .handler = help_cmd, .usage = NULL }, 78819281eeSdp-arm }; 79819281eeSdp-arm 80819281eeSdp-arm static image_t *images[MAX_IMAGES]; 81819281eeSdp-arm static size_t nr_images; 82819281eeSdp-arm static uuid_t uuid_null = { 0 }; 83819281eeSdp-arm static int verbose; 84819281eeSdp-arm 8560b499feSdp-arm static void vlog(int prio, const char *msg, va_list ap) 86819281eeSdp-arm { 87819281eeSdp-arm char *prefix[] = { "DEBUG", "WARN", "ERROR" }; 88819281eeSdp-arm 89819281eeSdp-arm fprintf(stderr, "%s: ", prefix[prio]); 90819281eeSdp-arm vfprintf(stderr, msg, ap); 91819281eeSdp-arm fputc('\n', stderr); 92819281eeSdp-arm } 93819281eeSdp-arm 9460b499feSdp-arm static void log_dbgx(const char *msg, ...) 95819281eeSdp-arm { 96819281eeSdp-arm va_list ap; 97819281eeSdp-arm 98819281eeSdp-arm va_start(ap, msg); 99819281eeSdp-arm vlog(LOG_DBG, msg, ap); 100819281eeSdp-arm va_end(ap); 101819281eeSdp-arm } 102819281eeSdp-arm 10360b499feSdp-arm static void log_warnx(const char *msg, ...) 104819281eeSdp-arm { 105819281eeSdp-arm va_list ap; 106819281eeSdp-arm 107819281eeSdp-arm va_start(ap, msg); 108819281eeSdp-arm vlog(LOG_WARN, msg, ap); 109819281eeSdp-arm va_end(ap); 110819281eeSdp-arm } 111819281eeSdp-arm 11260b499feSdp-arm static void log_err(const char *msg, ...) 113819281eeSdp-arm { 114819281eeSdp-arm char buf[512]; 115819281eeSdp-arm va_list ap; 116819281eeSdp-arm 117819281eeSdp-arm va_start(ap, msg); 118819281eeSdp-arm snprintf(buf, sizeof(buf), "%s: %s", msg, strerror(errno)); 119819281eeSdp-arm vlog(LOG_ERR, buf, ap); 120819281eeSdp-arm va_end(ap); 121819281eeSdp-arm exit(1); 122819281eeSdp-arm } 123819281eeSdp-arm 12460b499feSdp-arm static void log_errx(const char *msg, ...) 125819281eeSdp-arm { 126819281eeSdp-arm va_list ap; 127819281eeSdp-arm 128819281eeSdp-arm va_start(ap, msg); 129819281eeSdp-arm vlog(LOG_ERR, msg, ap); 130819281eeSdp-arm va_end(ap); 131819281eeSdp-arm exit(1); 132819281eeSdp-arm } 133819281eeSdp-arm 134*a22f6285Sdp-arm static char *xstrdup(const char *s, const char *msg) 135*a22f6285Sdp-arm { 136*a22f6285Sdp-arm char *d; 137*a22f6285Sdp-arm 138*a22f6285Sdp-arm d = strdup(s); 139*a22f6285Sdp-arm if (d == NULL) 140*a22f6285Sdp-arm log_errx("strdup: ", msg); 141*a22f6285Sdp-arm return d; 142*a22f6285Sdp-arm } 143*a22f6285Sdp-arm 144*a22f6285Sdp-arm static void *xmalloc(size_t size, const char *msg) 145*a22f6285Sdp-arm { 146*a22f6285Sdp-arm void *d; 147*a22f6285Sdp-arm 148*a22f6285Sdp-arm d = malloc(size); 149*a22f6285Sdp-arm if (d == NULL) 150*a22f6285Sdp-arm log_errx("malloc: ", msg); 151*a22f6285Sdp-arm return d; 152*a22f6285Sdp-arm } 153*a22f6285Sdp-arm 154819281eeSdp-arm static void add_image(image_t *image) 155819281eeSdp-arm { 156819281eeSdp-arm if (nr_images + 1 > MAX_IMAGES) 157819281eeSdp-arm log_errx("Too many images"); 158819281eeSdp-arm images[nr_images++] = image; 159819281eeSdp-arm } 160819281eeSdp-arm 161819281eeSdp-arm static void free_image(image_t *image) 162819281eeSdp-arm { 163819281eeSdp-arm free(image->buffer); 164819281eeSdp-arm free(image); 165819281eeSdp-arm } 166819281eeSdp-arm 167819281eeSdp-arm static void replace_image(image_t *image_dst, image_t *image_src) 168819281eeSdp-arm { 169819281eeSdp-arm int i; 170819281eeSdp-arm 171819281eeSdp-arm for (i = 0; i < nr_images; i++) { 172819281eeSdp-arm if (images[i] == image_dst) { 173819281eeSdp-arm free_image(images[i]); 174819281eeSdp-arm images[i] = image_src; 175819281eeSdp-arm break; 176819281eeSdp-arm } 177819281eeSdp-arm } 178819281eeSdp-arm assert(i != nr_images); 179819281eeSdp-arm } 180819281eeSdp-arm 181819281eeSdp-arm static void remove_image(image_t *image) 182819281eeSdp-arm { 183819281eeSdp-arm int i; 184819281eeSdp-arm 185819281eeSdp-arm for (i = 0; i < nr_images; i++) { 186819281eeSdp-arm if (images[i] == image) { 187819281eeSdp-arm free_image(images[i]); 188819281eeSdp-arm images[i] = NULL; 189819281eeSdp-arm break; 190819281eeSdp-arm } 191819281eeSdp-arm } 192819281eeSdp-arm assert(i != nr_images); 193819281eeSdp-arm 194819281eeSdp-arm /* Compact array. */ 195819281eeSdp-arm memmove(&images[i], &images[i + 1], 196819281eeSdp-arm (nr_images - i - 1) * sizeof(*images)); 197819281eeSdp-arm nr_images--; 198819281eeSdp-arm } 199819281eeSdp-arm 200819281eeSdp-arm static void free_images(void) 201819281eeSdp-arm { 202819281eeSdp-arm int i; 203819281eeSdp-arm 204819281eeSdp-arm for (i = 0; i < nr_images; i++) { 205819281eeSdp-arm free_image(images[i]); 206819281eeSdp-arm images[i] = NULL; 207819281eeSdp-arm } 208819281eeSdp-arm } 209819281eeSdp-arm 210b04efcceSdp-arm static toc_entry_t *lookup_entry_from_uuid(uuid_t *uuid) 211819281eeSdp-arm { 212819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 213819281eeSdp-arm 214819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 215819281eeSdp-arm if (memcmp(&toc_entry->uuid, uuid, sizeof(uuid_t)) == 0) 216819281eeSdp-arm return toc_entry; 217819281eeSdp-arm return NULL; 218819281eeSdp-arm } 219819281eeSdp-arm 220b04efcceSdp-arm static image_t *lookup_image_from_uuid(uuid_t *uuid) 221b04efcceSdp-arm { 222b04efcceSdp-arm image_t *image; 223b04efcceSdp-arm int i; 224b04efcceSdp-arm 225b04efcceSdp-arm for (i = 0; i < nr_images; i++) { 226b04efcceSdp-arm image = images[i]; 227b04efcceSdp-arm if (memcmp(&image->uuid, uuid, sizeof(uuid_t)) == 0) 228b04efcceSdp-arm return image; 229b04efcceSdp-arm } 230b04efcceSdp-arm return NULL; 231b04efcceSdp-arm } 232b04efcceSdp-arm 23360b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out) 234819281eeSdp-arm { 235819281eeSdp-arm struct stat st; 236819281eeSdp-arm FILE *fp; 237819281eeSdp-arm char *buf, *bufend; 238819281eeSdp-arm fip_toc_header_t *toc_header; 239819281eeSdp-arm fip_toc_entry_t *toc_entry; 240819281eeSdp-arm image_t *image; 241819281eeSdp-arm int terminated = 0; 242819281eeSdp-arm 243819281eeSdp-arm fp = fopen(filename, "r"); 244819281eeSdp-arm if (fp == NULL) 245819281eeSdp-arm log_err("fopen %s", filename); 246819281eeSdp-arm 247819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 248819281eeSdp-arm log_err("fstat %s", filename); 249819281eeSdp-arm 250*a22f6285Sdp-arm buf = xmalloc(st.st_size, "failed to load file into memory"); 251819281eeSdp-arm if (fread(buf, 1, st.st_size, fp) != st.st_size) 252819281eeSdp-arm log_errx("Failed to read %s", filename); 253819281eeSdp-arm bufend = buf + st.st_size; 254819281eeSdp-arm fclose(fp); 255819281eeSdp-arm 256819281eeSdp-arm if (st.st_size < sizeof(fip_toc_header_t)) 257819281eeSdp-arm log_errx("FIP %s is truncated", filename); 258819281eeSdp-arm 259819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 260819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 261819281eeSdp-arm 262819281eeSdp-arm if (toc_header->name != TOC_HEADER_NAME) 263819281eeSdp-arm log_errx("%s is not a FIP file", filename); 264819281eeSdp-arm 265819281eeSdp-arm /* Return the ToC header if the caller wants it. */ 266819281eeSdp-arm if (toc_header_out != NULL) 267819281eeSdp-arm *toc_header_out = *toc_header; 268819281eeSdp-arm 269819281eeSdp-arm /* Walk through each ToC entry in the file. */ 270819281eeSdp-arm while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) { 271819281eeSdp-arm /* Found the ToC terminator, we are done. */ 272819281eeSdp-arm if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) { 273819281eeSdp-arm terminated = 1; 274819281eeSdp-arm break; 275819281eeSdp-arm } 276819281eeSdp-arm 277819281eeSdp-arm /* 278819281eeSdp-arm * Build a new image out of the ToC entry and add it to the 279819281eeSdp-arm * table of images. 280819281eeSdp-arm */ 281*a22f6285Sdp-arm image = xmalloc(sizeof(*image), 282*a22f6285Sdp-arm "failed to allocate memory for image"); 283819281eeSdp-arm memcpy(&image->uuid, &toc_entry->uuid, sizeof(uuid_t)); 284*a22f6285Sdp-arm image->buffer = xmalloc(toc_entry->size, 285*a22f6285Sdp-arm "failed to allocate image buffer, is FIP file corrupted?"); 286819281eeSdp-arm /* Overflow checks before memory copy. */ 287819281eeSdp-arm if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address) 288819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 289819281eeSdp-arm if (toc_entry->size + toc_entry->offset_address > st.st_size) 290819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 291819281eeSdp-arm 292819281eeSdp-arm memcpy(image->buffer, buf + toc_entry->offset_address, 293819281eeSdp-arm toc_entry->size); 294819281eeSdp-arm image->size = toc_entry->size; 295819281eeSdp-arm 296819281eeSdp-arm add_image(image); 297819281eeSdp-arm 298819281eeSdp-arm toc_entry++; 299819281eeSdp-arm } 300819281eeSdp-arm 301819281eeSdp-arm if (terminated == 0) 302819281eeSdp-arm log_errx("FIP %s does not have a ToC terminator entry", 303819281eeSdp-arm filename); 304819281eeSdp-arm free(buf); 305819281eeSdp-arm return 0; 306819281eeSdp-arm } 307819281eeSdp-arm 30860b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename) 309819281eeSdp-arm { 310819281eeSdp-arm struct stat st; 311819281eeSdp-arm image_t *image; 312819281eeSdp-arm FILE *fp; 313819281eeSdp-arm 314b04efcceSdp-arm assert(uuid != NULL); 315b04efcceSdp-arm 316819281eeSdp-arm fp = fopen(filename, "r"); 317819281eeSdp-arm if (fp == NULL) 318819281eeSdp-arm log_err("fopen %s", filename); 319819281eeSdp-arm 320819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 321819281eeSdp-arm log_errx("fstat %s", filename); 322819281eeSdp-arm 323*a22f6285Sdp-arm image = xmalloc(sizeof(*image), "failed to allocate memory for image"); 324b04efcceSdp-arm memcpy(&image->uuid, uuid, sizeof(uuid_t)); 325*a22f6285Sdp-arm image->buffer = xmalloc(st.st_size, "failed to allocate image buffer"); 326819281eeSdp-arm if (fread(image->buffer, 1, st.st_size, fp) != st.st_size) 327819281eeSdp-arm log_errx("Failed to read %s", filename); 328819281eeSdp-arm image->size = st.st_size; 329819281eeSdp-arm 330819281eeSdp-arm fclose(fp); 331819281eeSdp-arm return image; 332819281eeSdp-arm } 333819281eeSdp-arm 33460b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename) 335819281eeSdp-arm { 336819281eeSdp-arm FILE *fp; 337819281eeSdp-arm 338819281eeSdp-arm fp = fopen(filename, "w"); 339819281eeSdp-arm if (fp == NULL) 340819281eeSdp-arm log_err("fopen"); 341819281eeSdp-arm if (fwrite(image->buffer, 1, image->size, fp) != image->size) 342819281eeSdp-arm log_errx("Failed to write %s", filename); 343819281eeSdp-arm fclose(fp); 344819281eeSdp-arm return 0; 345819281eeSdp-arm } 346819281eeSdp-arm 347819281eeSdp-arm static int fill_common_opts(struct option *opts, int has_arg) 348819281eeSdp-arm { 349819281eeSdp-arm int i; 350819281eeSdp-arm 351819281eeSdp-arm for (i = 0; toc_entries[i].cmdline_name != NULL; i++) { 352819281eeSdp-arm opts[i].name = toc_entries[i].cmdline_name; 353819281eeSdp-arm opts[i].has_arg = has_arg; 354819281eeSdp-arm opts[i].flag = NULL; 355819281eeSdp-arm opts[i].val = 0; 356819281eeSdp-arm } 357819281eeSdp-arm return i; 358819281eeSdp-arm } 359819281eeSdp-arm 360819281eeSdp-arm static void add_opt(struct option *opts, int idx, char *name, 361819281eeSdp-arm int has_arg, int val) 362819281eeSdp-arm { 363819281eeSdp-arm opts[idx].name = name; 364819281eeSdp-arm opts[idx].has_arg = has_arg; 365819281eeSdp-arm opts[idx].flag = NULL; 366819281eeSdp-arm opts[idx].val = val; 367819281eeSdp-arm } 368819281eeSdp-arm 3699df69ba3Sdp-arm static void md_print(unsigned char *md, size_t len) 3709df69ba3Sdp-arm { 3719df69ba3Sdp-arm size_t i; 3729df69ba3Sdp-arm 3739df69ba3Sdp-arm for (i = 0; i < len; i++) 3749df69ba3Sdp-arm printf("%02x", md[i]); 3759df69ba3Sdp-arm } 3769df69ba3Sdp-arm 377819281eeSdp-arm static int info_cmd(int argc, char *argv[]) 378819281eeSdp-arm { 379819281eeSdp-arm image_t *image; 380819281eeSdp-arm uint64_t image_offset; 381819281eeSdp-arm uint64_t image_size = 0; 382819281eeSdp-arm fip_toc_header_t toc_header; 383819281eeSdp-arm int i; 384819281eeSdp-arm 385819281eeSdp-arm if (argc != 2) 38685ee2778Sdp-arm info_usage(); 387819281eeSdp-arm argc--, argv++; 388819281eeSdp-arm 389819281eeSdp-arm parse_fip(argv[0], &toc_header); 390819281eeSdp-arm 391819281eeSdp-arm if (verbose) { 392819281eeSdp-arm log_dbgx("toc_header[name]: 0x%llX", 393819281eeSdp-arm (unsigned long long)toc_header.name); 394819281eeSdp-arm log_dbgx("toc_header[serial_number]: 0x%llX", 395819281eeSdp-arm (unsigned long long)toc_header.serial_number); 396819281eeSdp-arm log_dbgx("toc_header[flags]: 0x%llX", 397819281eeSdp-arm (unsigned long long)toc_header.flags); 398819281eeSdp-arm } 399819281eeSdp-arm 400819281eeSdp-arm image_offset = sizeof(fip_toc_header_t) + 401819281eeSdp-arm (sizeof(fip_toc_entry_t) * (nr_images + 1)); 402819281eeSdp-arm 403819281eeSdp-arm for (i = 0; i < nr_images; i++) { 404b04efcceSdp-arm toc_entry_t *toc_entry; 405b04efcceSdp-arm 406819281eeSdp-arm image = images[i]; 407b04efcceSdp-arm toc_entry = lookup_entry_from_uuid(&image->uuid); 408b04efcceSdp-arm if (toc_entry != NULL) 409b04efcceSdp-arm printf("%s: ", toc_entry->name); 410819281eeSdp-arm else 411819281eeSdp-arm printf("Unknown entry: "); 412819281eeSdp-arm image_size = image->size; 413819281eeSdp-arm printf("offset=0x%llX, size=0x%llX", 414819281eeSdp-arm (unsigned long long)image_offset, 415819281eeSdp-arm (unsigned long long)image_size); 416b04efcceSdp-arm if (toc_entry != NULL) 4179df69ba3Sdp-arm printf(", cmdline=\"--%s\"", 418b04efcceSdp-arm toc_entry->cmdline_name); 4199df69ba3Sdp-arm if (verbose) { 4209df69ba3Sdp-arm unsigned char md[SHA256_DIGEST_LENGTH]; 4219df69ba3Sdp-arm 4229df69ba3Sdp-arm SHA256(image->buffer, image_size, md); 4239df69ba3Sdp-arm printf(", sha256="); 4249df69ba3Sdp-arm md_print(md, sizeof(md)); 4259df69ba3Sdp-arm } 426819281eeSdp-arm putchar('\n'); 427819281eeSdp-arm image_offset += image_size; 428819281eeSdp-arm } 429819281eeSdp-arm 430819281eeSdp-arm free_images(); 431819281eeSdp-arm return 0; 432819281eeSdp-arm } 433819281eeSdp-arm 434819281eeSdp-arm static void info_usage(void) 435819281eeSdp-arm { 436819281eeSdp-arm printf("fiptool info FIP_FILENAME\n"); 43785ee2778Sdp-arm exit(1); 438819281eeSdp-arm } 439819281eeSdp-arm 440819281eeSdp-arm static int pack_images(char *filename, uint64_t toc_flags) 441819281eeSdp-arm { 442819281eeSdp-arm FILE *fp; 443819281eeSdp-arm image_t *image; 444819281eeSdp-arm fip_toc_header_t *toc_header; 445819281eeSdp-arm fip_toc_entry_t *toc_entry; 446819281eeSdp-arm char *buf; 447819281eeSdp-arm uint64_t entry_offset, buf_size, payload_size; 448819281eeSdp-arm int i; 449819281eeSdp-arm 450819281eeSdp-arm /* Calculate total payload size and allocate scratch buffer. */ 451819281eeSdp-arm payload_size = 0; 452819281eeSdp-arm for (i = 0; i < nr_images; i++) 453819281eeSdp-arm payload_size += images[i]->size; 454819281eeSdp-arm 455819281eeSdp-arm buf_size = sizeof(fip_toc_header_t) + 456819281eeSdp-arm sizeof(fip_toc_entry_t) * (nr_images + 1); 457819281eeSdp-arm buf = calloc(1, buf_size); 458819281eeSdp-arm if (buf == NULL) 459819281eeSdp-arm log_err("calloc"); 460819281eeSdp-arm 461819281eeSdp-arm /* Build up header and ToC entries from the image table. */ 462819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 463819281eeSdp-arm toc_header->name = TOC_HEADER_NAME; 464819281eeSdp-arm toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER; 465819281eeSdp-arm toc_header->flags = toc_flags; 466819281eeSdp-arm 467819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 468819281eeSdp-arm 469819281eeSdp-arm entry_offset = buf_size; 470819281eeSdp-arm for (i = 0; i < nr_images; i++) { 471819281eeSdp-arm image = images[i]; 472819281eeSdp-arm memcpy(&toc_entry->uuid, &image->uuid, sizeof(uuid_t)); 473819281eeSdp-arm toc_entry->offset_address = entry_offset; 474819281eeSdp-arm toc_entry->size = image->size; 475819281eeSdp-arm toc_entry->flags = 0; 476819281eeSdp-arm entry_offset += toc_entry->size; 477819281eeSdp-arm toc_entry++; 478819281eeSdp-arm } 479819281eeSdp-arm 480819281eeSdp-arm /* Append a null uuid entry to mark the end of ToC entries. */ 481819281eeSdp-arm memcpy(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)); 482819281eeSdp-arm toc_entry->offset_address = entry_offset; 483819281eeSdp-arm toc_entry->size = 0; 484819281eeSdp-arm toc_entry->flags = 0; 485819281eeSdp-arm 486819281eeSdp-arm /* Generate the FIP file. */ 487819281eeSdp-arm fp = fopen(filename, "w"); 488819281eeSdp-arm if (fp == NULL) 489819281eeSdp-arm log_err("fopen %s", filename); 490819281eeSdp-arm 491819281eeSdp-arm if (verbose) 492819281eeSdp-arm log_dbgx("Metadata size: %zu bytes", buf_size); 493819281eeSdp-arm 494819281eeSdp-arm if (fwrite(buf, 1, buf_size, fp) != buf_size) 495819281eeSdp-arm log_errx("Failed to write image to %s", filename); 496819281eeSdp-arm free(buf); 497819281eeSdp-arm 498819281eeSdp-arm if (verbose) 499819281eeSdp-arm log_dbgx("Payload size: %zu bytes", payload_size); 500819281eeSdp-arm 501819281eeSdp-arm for (i = 0; i < nr_images; i++) { 502819281eeSdp-arm image = images[i]; 503819281eeSdp-arm if (fwrite(image->buffer, 1, image->size, fp) != image->size) 504819281eeSdp-arm log_errx("Failed to write image to %s", filename); 505819281eeSdp-arm } 506819281eeSdp-arm 507819281eeSdp-arm fclose(fp); 508819281eeSdp-arm return 0; 509819281eeSdp-arm } 510819281eeSdp-arm 511819281eeSdp-arm /* 512819281eeSdp-arm * This function is shared between the create and update subcommands. 513819281eeSdp-arm * The difference between the two subcommands is that when the FIP file 514819281eeSdp-arm * is created, the parsing of an existing FIP is skipped. This results 515819281eeSdp-arm * in update_fip() creating the new FIP file from scratch because the 516819281eeSdp-arm * internal image table is not populated. 517819281eeSdp-arm */ 518819281eeSdp-arm static void update_fip(void) 519819281eeSdp-arm { 520819281eeSdp-arm toc_entry_t *toc_entry; 521b04efcceSdp-arm image_t *new_image, *old_image; 522819281eeSdp-arm 523819281eeSdp-arm /* Add or replace images in the FIP file. */ 524819281eeSdp-arm for (toc_entry = toc_entries; 525819281eeSdp-arm toc_entry->cmdline_name != NULL; 526819281eeSdp-arm toc_entry++) { 527819281eeSdp-arm if (toc_entry->action != DO_PACK) 528819281eeSdp-arm continue; 529819281eeSdp-arm 530b04efcceSdp-arm new_image = read_image_from_file(&toc_entry->uuid, 531b04efcceSdp-arm toc_entry->action_arg); 532b04efcceSdp-arm old_image = lookup_image_from_uuid(&toc_entry->uuid); 533b04efcceSdp-arm if (old_image != NULL) { 534819281eeSdp-arm if (verbose) 535819281eeSdp-arm log_dbgx("Replacing image %s.bin with %s", 536819281eeSdp-arm toc_entry->cmdline_name, 537819281eeSdp-arm toc_entry->action_arg); 538b04efcceSdp-arm replace_image(old_image, new_image); 539819281eeSdp-arm } else { 540819281eeSdp-arm if (verbose) 541819281eeSdp-arm log_dbgx("Adding image %s", 542819281eeSdp-arm toc_entry->action_arg); 543b04efcceSdp-arm add_image(new_image); 544819281eeSdp-arm } 545819281eeSdp-arm 546819281eeSdp-arm free(toc_entry->action_arg); 547819281eeSdp-arm toc_entry->action_arg = NULL; 548819281eeSdp-arm } 549819281eeSdp-arm } 550819281eeSdp-arm 551819281eeSdp-arm static void parse_plat_toc_flags(char *arg, unsigned long long *toc_flags) 552819281eeSdp-arm { 553819281eeSdp-arm unsigned long long flags; 554819281eeSdp-arm char *endptr; 555819281eeSdp-arm 556819281eeSdp-arm errno = 0; 557819281eeSdp-arm flags = strtoull(arg, &endptr, 16); 558819281eeSdp-arm if (*endptr != '\0' || flags > UINT16_MAX || errno != 0) 559819281eeSdp-arm log_errx("Invalid platform ToC flags: %s", arg); 560819281eeSdp-arm /* Platform ToC flags is a 16-bit field occupying bits [32-47]. */ 561819281eeSdp-arm *toc_flags |= flags << 32; 562819281eeSdp-arm } 563819281eeSdp-arm 564819281eeSdp-arm static int create_cmd(int argc, char *argv[]) 565819281eeSdp-arm { 566819281eeSdp-arm struct option opts[toc_entries_len + 1]; 567819281eeSdp-arm unsigned long long toc_flags = 0; 568819281eeSdp-arm int i; 569819281eeSdp-arm 570819281eeSdp-arm if (argc < 2) 57185ee2778Sdp-arm create_usage(); 572819281eeSdp-arm 573819281eeSdp-arm i = fill_common_opts(opts, required_argument); 574819281eeSdp-arm add_opt(opts, i, "plat-toc-flags", required_argument, 575819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 576819281eeSdp-arm add_opt(opts, ++i, NULL, 0, 0); 577819281eeSdp-arm 578819281eeSdp-arm while (1) { 57920f87e78Sdp-arm int c, opt_index = 0; 580819281eeSdp-arm 58160b499feSdp-arm c = getopt_long(argc, argv, "", opts, &opt_index); 582819281eeSdp-arm if (c == -1) 583819281eeSdp-arm break; 584819281eeSdp-arm 585819281eeSdp-arm switch (c) { 586819281eeSdp-arm case OPT_TOC_ENTRY: { 587819281eeSdp-arm toc_entry_t *toc_entry; 588819281eeSdp-arm 589819281eeSdp-arm toc_entry = &toc_entries[opt_index]; 590819281eeSdp-arm toc_entry->action = DO_PACK; 591*a22f6285Sdp-arm toc_entry->action_arg = xstrdup(optarg, 592*a22f6285Sdp-arm "failed to allocate memory for argument"); 593819281eeSdp-arm break; 594819281eeSdp-arm } 595819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: 596819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 597819281eeSdp-arm break; 598819281eeSdp-arm default: 59985ee2778Sdp-arm create_usage(); 600819281eeSdp-arm } 601819281eeSdp-arm } 602819281eeSdp-arm argc -= optind; 603819281eeSdp-arm argv += optind; 604819281eeSdp-arm 605819281eeSdp-arm if (argc == 0) 60685ee2778Sdp-arm create_usage(); 607819281eeSdp-arm 608819281eeSdp-arm update_fip(); 609819281eeSdp-arm 610819281eeSdp-arm pack_images(argv[0], toc_flags); 611819281eeSdp-arm free_images(); 612819281eeSdp-arm return 0; 613819281eeSdp-arm } 614819281eeSdp-arm 615819281eeSdp-arm static void create_usage(void) 616819281eeSdp-arm { 617819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 618819281eeSdp-arm 61923fcb90dSdp-arm printf("fiptool create [--plat-toc-flags <value>] [opts] FIP_FILENAME\n"); 620819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 621819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 622819281eeSdp-arm fputc('\n', stderr); 623819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 624819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 625819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 626819281eeSdp-arm toc_entry->name); 62785ee2778Sdp-arm exit(1); 628819281eeSdp-arm } 629819281eeSdp-arm 630819281eeSdp-arm static int update_cmd(int argc, char *argv[]) 631819281eeSdp-arm { 632819281eeSdp-arm struct option opts[toc_entries_len + 2]; 633819281eeSdp-arm char outfile[FILENAME_MAX] = { 0 }; 634819281eeSdp-arm fip_toc_header_t toc_header = { 0 }; 635819281eeSdp-arm unsigned long long toc_flags = 0; 636819281eeSdp-arm int pflag = 0; 637819281eeSdp-arm int i; 638819281eeSdp-arm 639819281eeSdp-arm if (argc < 2) 64085ee2778Sdp-arm update_usage(); 641819281eeSdp-arm 642819281eeSdp-arm i = fill_common_opts(opts, required_argument); 643819281eeSdp-arm add_opt(opts, i, "out", required_argument, 'o'); 644819281eeSdp-arm add_opt(opts, ++i, "plat-toc-flags", required_argument, 645819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 646819281eeSdp-arm add_opt(opts, ++i, NULL, 0, 0); 647819281eeSdp-arm 648819281eeSdp-arm while (1) { 64920f87e78Sdp-arm int c, opt_index = 0; 650819281eeSdp-arm 651819281eeSdp-arm c = getopt_long(argc, argv, "o:", opts, &opt_index); 652819281eeSdp-arm if (c == -1) 653819281eeSdp-arm break; 654819281eeSdp-arm 655819281eeSdp-arm switch (c) { 656819281eeSdp-arm case OPT_TOC_ENTRY: { 657819281eeSdp-arm toc_entry_t *toc_entry; 658819281eeSdp-arm 659819281eeSdp-arm toc_entry = &toc_entries[opt_index]; 660819281eeSdp-arm toc_entry->action = DO_PACK; 661*a22f6285Sdp-arm toc_entry->action_arg = xstrdup(optarg, 662*a22f6285Sdp-arm "failed to allocate memory for argument"); 663819281eeSdp-arm break; 664819281eeSdp-arm } 665819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: { 666819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 667819281eeSdp-arm pflag = 1; 668819281eeSdp-arm break; 669819281eeSdp-arm } 670819281eeSdp-arm case 'o': 671819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 672819281eeSdp-arm break; 673819281eeSdp-arm default: 67485ee2778Sdp-arm update_usage(); 675819281eeSdp-arm } 676819281eeSdp-arm } 677819281eeSdp-arm argc -= optind; 678819281eeSdp-arm argv += optind; 679819281eeSdp-arm 680819281eeSdp-arm if (argc == 0) 68185ee2778Sdp-arm update_usage(); 682819281eeSdp-arm 683819281eeSdp-arm if (outfile[0] == '\0') 684819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 685819281eeSdp-arm 686819281eeSdp-arm if (access(outfile, F_OK) == 0) 687819281eeSdp-arm parse_fip(argv[0], &toc_header); 688819281eeSdp-arm 689819281eeSdp-arm if (pflag) 690819281eeSdp-arm toc_header.flags &= ~(0xffffULL << 32); 691819281eeSdp-arm toc_flags = (toc_header.flags |= toc_flags); 692819281eeSdp-arm 693819281eeSdp-arm update_fip(); 694819281eeSdp-arm 695819281eeSdp-arm pack_images(outfile, toc_flags); 696819281eeSdp-arm free_images(); 697819281eeSdp-arm return 0; 698819281eeSdp-arm } 699819281eeSdp-arm 700819281eeSdp-arm static void update_usage(void) 701819281eeSdp-arm { 702819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 703819281eeSdp-arm 70423fcb90dSdp-arm printf("fiptool update [--out FIP_FILENAME] " 705819281eeSdp-arm "[--plat-toc-flags <value>] [opts] FIP_FILENAME\n"); 706819281eeSdp-arm printf(" --out FIP_FILENAME\t\tSet an alternative output FIP file.\n"); 707819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 708819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 709819281eeSdp-arm fputc('\n', stderr); 710819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 711819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 712819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 713819281eeSdp-arm toc_entry->name); 71485ee2778Sdp-arm exit(1); 715819281eeSdp-arm } 716819281eeSdp-arm 717819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]) 718819281eeSdp-arm { 719819281eeSdp-arm struct option opts[toc_entries_len + 3]; 720819281eeSdp-arm char file[FILENAME_MAX], outdir[PATH_MAX] = { 0 }; 721819281eeSdp-arm toc_entry_t *toc_entry; 722819281eeSdp-arm int fflag = 0; 723819281eeSdp-arm int unpack_all = 1; 724819281eeSdp-arm int i; 725819281eeSdp-arm 726819281eeSdp-arm if (argc < 2) 72785ee2778Sdp-arm unpack_usage(); 728819281eeSdp-arm 729819281eeSdp-arm i = fill_common_opts(opts, required_argument); 730819281eeSdp-arm add_opt(opts, i, "force", no_argument, 'f'); 731819281eeSdp-arm add_opt(opts, ++i, "out", required_argument, 'o'); 732819281eeSdp-arm add_opt(opts, ++i, NULL, 0, 0); 733819281eeSdp-arm 734819281eeSdp-arm while (1) { 73520f87e78Sdp-arm int c, opt_index = 0; 736819281eeSdp-arm 737819281eeSdp-arm c = getopt_long(argc, argv, "fo:", opts, &opt_index); 738819281eeSdp-arm if (c == -1) 739819281eeSdp-arm break; 740819281eeSdp-arm 741819281eeSdp-arm switch (c) { 742819281eeSdp-arm case OPT_TOC_ENTRY: 743819281eeSdp-arm unpack_all = 0; 744819281eeSdp-arm toc_entry = &toc_entries[opt_index]; 745819281eeSdp-arm toc_entry->action = DO_UNPACK; 746*a22f6285Sdp-arm toc_entry->action_arg = xstrdup(optarg, 747*a22f6285Sdp-arm "failed to allocate memory for argument"); 748819281eeSdp-arm break; 749819281eeSdp-arm case 'f': 750819281eeSdp-arm fflag = 1; 751819281eeSdp-arm break; 752819281eeSdp-arm case 'o': 753819281eeSdp-arm snprintf(outdir, sizeof(outdir), "%s", optarg); 754819281eeSdp-arm break; 755819281eeSdp-arm default: 75685ee2778Sdp-arm unpack_usage(); 757819281eeSdp-arm } 758819281eeSdp-arm } 759819281eeSdp-arm argc -= optind; 760819281eeSdp-arm argv += optind; 761819281eeSdp-arm 762819281eeSdp-arm if (argc == 0) 76385ee2778Sdp-arm unpack_usage(); 764819281eeSdp-arm 765819281eeSdp-arm parse_fip(argv[0], NULL); 766819281eeSdp-arm 767819281eeSdp-arm if (outdir[0] != '\0') 768819281eeSdp-arm if (chdir(outdir) == -1) 769819281eeSdp-arm log_err("chdir %s", outdir); 770819281eeSdp-arm 771819281eeSdp-arm /* Unpack all specified images. */ 772819281eeSdp-arm for (toc_entry = toc_entries; 773819281eeSdp-arm toc_entry->cmdline_name != NULL; 774819281eeSdp-arm toc_entry++) { 775b04efcceSdp-arm image_t *image; 776b04efcceSdp-arm 777b04efcceSdp-arm if (!unpack_all && toc_entry->action != DO_UNPACK) 778819281eeSdp-arm continue; 779819281eeSdp-arm 780819281eeSdp-arm /* Build filename. */ 781819281eeSdp-arm if (toc_entry->action_arg == NULL) 782819281eeSdp-arm snprintf(file, sizeof(file), "%s.bin", 783819281eeSdp-arm toc_entry->cmdline_name); 784819281eeSdp-arm else 785819281eeSdp-arm snprintf(file, sizeof(file), "%s", 786819281eeSdp-arm toc_entry->action_arg); 787819281eeSdp-arm 788b04efcceSdp-arm image = lookup_image_from_uuid(&toc_entry->uuid); 789b04efcceSdp-arm if (image == NULL) { 790b04efcceSdp-arm if (!unpack_all) 791819281eeSdp-arm log_warnx("Requested image %s is not in %s", 792819281eeSdp-arm file, argv[0]); 793819281eeSdp-arm free(toc_entry->action_arg); 794819281eeSdp-arm toc_entry->action_arg = NULL; 795819281eeSdp-arm continue; 796819281eeSdp-arm } 797819281eeSdp-arm 798819281eeSdp-arm if (access(file, F_OK) != 0 || fflag) { 799819281eeSdp-arm if (verbose) 800819281eeSdp-arm log_dbgx("Unpacking %s", file); 801b04efcceSdp-arm write_image_to_file(image, file); 802819281eeSdp-arm } else { 803819281eeSdp-arm log_warnx("File %s already exists, use --force to overwrite it", 804819281eeSdp-arm file); 805819281eeSdp-arm } 806819281eeSdp-arm 807819281eeSdp-arm free(toc_entry->action_arg); 808819281eeSdp-arm toc_entry->action_arg = NULL; 809819281eeSdp-arm } 810819281eeSdp-arm 811819281eeSdp-arm free_images(); 812819281eeSdp-arm return 0; 813819281eeSdp-arm } 814819281eeSdp-arm 815819281eeSdp-arm static void unpack_usage(void) 816819281eeSdp-arm { 817819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 818819281eeSdp-arm 819819281eeSdp-arm printf("fiptool unpack [--force] [--out <path>] [opts] FIP_FILENAME\n"); 820819281eeSdp-arm printf(" --force\tIf the output file already exists, use --force to " 821819281eeSdp-arm "overwrite it.\n"); 822819281eeSdp-arm printf(" --out path\tSet the output directory path.\n"); 823819281eeSdp-arm fputc('\n', stderr); 824819281eeSdp-arm printf("Specific images are unpacked with the following options:\n"); 825819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 826819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 827819281eeSdp-arm toc_entry->name); 828819281eeSdp-arm fputc('\n', stderr); 829819281eeSdp-arm printf("If no options are provided, all images will be unpacked.\n"); 83085ee2778Sdp-arm exit(1); 831819281eeSdp-arm } 832819281eeSdp-arm 833819281eeSdp-arm static int remove_cmd(int argc, char *argv[]) 834819281eeSdp-arm { 835819281eeSdp-arm struct option opts[toc_entries_len + 2]; 836819281eeSdp-arm char outfile[FILENAME_MAX] = { 0 }; 837819281eeSdp-arm fip_toc_header_t toc_header; 838819281eeSdp-arm toc_entry_t *toc_entry; 839819281eeSdp-arm int fflag = 0; 840819281eeSdp-arm int i; 841819281eeSdp-arm 842819281eeSdp-arm if (argc < 2) 84385ee2778Sdp-arm remove_usage(); 844819281eeSdp-arm 845819281eeSdp-arm i = fill_common_opts(opts, no_argument); 846819281eeSdp-arm add_opt(opts, i, "force", no_argument, 'f'); 847819281eeSdp-arm add_opt(opts, ++i, "out", required_argument, 'o'); 848819281eeSdp-arm add_opt(opts, ++i, NULL, 0, 0); 849819281eeSdp-arm 850819281eeSdp-arm while (1) { 85120f87e78Sdp-arm int c, opt_index = 0; 852819281eeSdp-arm 853819281eeSdp-arm c = getopt_long(argc, argv, "fo:", opts, &opt_index); 854819281eeSdp-arm if (c == -1) 855819281eeSdp-arm break; 856819281eeSdp-arm 857819281eeSdp-arm switch (c) { 858819281eeSdp-arm case OPT_TOC_ENTRY: 859819281eeSdp-arm toc_entry = &toc_entries[opt_index]; 860819281eeSdp-arm toc_entry->action = DO_REMOVE; 861819281eeSdp-arm break; 862819281eeSdp-arm case 'f': 863819281eeSdp-arm fflag = 1; 864819281eeSdp-arm break; 865819281eeSdp-arm case 'o': 866819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 867819281eeSdp-arm break; 868819281eeSdp-arm default: 86985ee2778Sdp-arm remove_usage(); 870819281eeSdp-arm } 871819281eeSdp-arm } 872819281eeSdp-arm argc -= optind; 873819281eeSdp-arm argv += optind; 874819281eeSdp-arm 875819281eeSdp-arm if (argc == 0) 87685ee2778Sdp-arm remove_usage(); 877819281eeSdp-arm 878819281eeSdp-arm if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag) 879819281eeSdp-arm log_errx("File %s already exists, use --force to overwrite it", 880819281eeSdp-arm outfile); 881819281eeSdp-arm 882819281eeSdp-arm if (outfile[0] == '\0') 883819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 884819281eeSdp-arm 885819281eeSdp-arm parse_fip(argv[0], &toc_header); 886819281eeSdp-arm 887819281eeSdp-arm for (toc_entry = toc_entries; 888819281eeSdp-arm toc_entry->cmdline_name != NULL; 889819281eeSdp-arm toc_entry++) { 890b04efcceSdp-arm image_t *image; 891b04efcceSdp-arm 892819281eeSdp-arm if (toc_entry->action != DO_REMOVE) 893819281eeSdp-arm continue; 894b04efcceSdp-arm image = lookup_image_from_uuid(&toc_entry->uuid); 895b04efcceSdp-arm if (image != NULL) { 896819281eeSdp-arm if (verbose) 897819281eeSdp-arm log_dbgx("Removing %s.bin", 898819281eeSdp-arm toc_entry->cmdline_name); 899b04efcceSdp-arm remove_image(image); 900819281eeSdp-arm } else { 901819281eeSdp-arm log_warnx("Requested image %s.bin is not in %s", 902819281eeSdp-arm toc_entry->cmdline_name, argv[0]); 903819281eeSdp-arm } 904819281eeSdp-arm } 905819281eeSdp-arm 906819281eeSdp-arm pack_images(outfile, toc_header.flags); 907819281eeSdp-arm free_images(); 908819281eeSdp-arm return 0; 909819281eeSdp-arm } 910819281eeSdp-arm 911819281eeSdp-arm static void remove_usage(void) 912819281eeSdp-arm { 913819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 914819281eeSdp-arm 915819281eeSdp-arm printf("fiptool remove [--force] [--out FIP_FILENAME] [opts] FIP_FILENAME\n"); 916819281eeSdp-arm printf(" --force\t\tIf the output FIP file already exists, use --force to " 917819281eeSdp-arm "overwrite it.\n"); 918819281eeSdp-arm printf(" --out FIP_FILENAME\tSet an alternative output FIP file.\n"); 919819281eeSdp-arm fputc('\n', stderr); 920819281eeSdp-arm printf("Specific images are removed with the following options:\n"); 921819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 922819281eeSdp-arm printf(" --%-16s\t%s\n", toc_entry->cmdline_name, 923819281eeSdp-arm toc_entry->name); 92485ee2778Sdp-arm exit(1); 925819281eeSdp-arm } 926819281eeSdp-arm 927819281eeSdp-arm static int version_cmd(int argc, char *argv[]) 928819281eeSdp-arm { 929819281eeSdp-arm #ifdef VERSION 930819281eeSdp-arm puts(VERSION); 931819281eeSdp-arm #else 932819281eeSdp-arm /* If built from fiptool directory, VERSION is not set. */ 933819281eeSdp-arm puts("Unknown version"); 934819281eeSdp-arm #endif 935819281eeSdp-arm return 0; 936819281eeSdp-arm } 937819281eeSdp-arm 938819281eeSdp-arm static void version_usage(void) 939819281eeSdp-arm { 940819281eeSdp-arm printf("fiptool version\n"); 94185ee2778Sdp-arm exit(1); 942819281eeSdp-arm } 943819281eeSdp-arm 944819281eeSdp-arm static int help_cmd(int argc, char *argv[]) 945819281eeSdp-arm { 946819281eeSdp-arm int i; 947819281eeSdp-arm 948819281eeSdp-arm if (argc < 2) 949819281eeSdp-arm usage(); 950819281eeSdp-arm argc--, argv++; 951819281eeSdp-arm 952819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 953819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0 && 95485ee2778Sdp-arm cmds[i].usage != NULL) 955819281eeSdp-arm cmds[i].usage(); 956819281eeSdp-arm } 957819281eeSdp-arm if (i == NELEM(cmds)) 958819281eeSdp-arm printf("No help for subcommand '%s'\n", argv[0]); 959819281eeSdp-arm return 0; 960819281eeSdp-arm } 961819281eeSdp-arm 962819281eeSdp-arm static void usage(void) 963819281eeSdp-arm { 964819281eeSdp-arm printf("usage: [--verbose] fiptool <command> [<args>]\n"); 965819281eeSdp-arm printf("Global options supported:\n"); 966819281eeSdp-arm printf(" --verbose\tEnable verbose output for all commands.\n"); 967819281eeSdp-arm fputc('\n', stderr); 968819281eeSdp-arm printf("Commands supported:\n"); 969819281eeSdp-arm printf(" info\t\tList images contained in FIP.\n"); 970819281eeSdp-arm printf(" create\tCreate a new FIP with the given images.\n"); 971819281eeSdp-arm printf(" update\tUpdate an existing FIP with the given images.\n"); 972819281eeSdp-arm printf(" unpack\tUnpack images from FIP.\n"); 973819281eeSdp-arm printf(" remove\tRemove images from FIP.\n"); 974819281eeSdp-arm printf(" version\tShow fiptool version.\n"); 975819281eeSdp-arm printf(" help\t\tShow help for given command.\n"); 976819281eeSdp-arm exit(1); 977819281eeSdp-arm } 978819281eeSdp-arm 979819281eeSdp-arm int main(int argc, char *argv[]) 980819281eeSdp-arm { 981819281eeSdp-arm int i, ret = 0; 982819281eeSdp-arm 983cc672bb2Sdp-arm while (1) { 984cc672bb2Sdp-arm int c, opt_index = 0; 985cc672bb2Sdp-arm static struct option opts[] = { 986cc672bb2Sdp-arm { "verbose", no_argument, NULL, 'v' }, 987cc672bb2Sdp-arm { NULL, no_argument, NULL, 0 } 988cc672bb2Sdp-arm }; 989819281eeSdp-arm 990cc672bb2Sdp-arm /* 991cc672bb2Sdp-arm * Set POSIX mode so getopt stops at the first non-option 992cc672bb2Sdp-arm * which is the subcommand. 993cc672bb2Sdp-arm */ 994cc672bb2Sdp-arm c = getopt_long(argc, argv, "+v", opts, &opt_index); 995cc672bb2Sdp-arm if (c == -1) 996cc672bb2Sdp-arm break; 997cc672bb2Sdp-arm 998cc672bb2Sdp-arm switch (c) { 999cc672bb2Sdp-arm case 'v': 1000819281eeSdp-arm verbose = 1; 1001cc672bb2Sdp-arm break; 1002cc672bb2Sdp-arm default: 1003c9cb4089SMasahiro Yamada usage(); 1004819281eeSdp-arm } 1005cc672bb2Sdp-arm } 1006cc672bb2Sdp-arm argc -= optind; 1007cc672bb2Sdp-arm argv += optind; 1008cc672bb2Sdp-arm /* Reset optind for subsequent getopt processing. */ 1009cc672bb2Sdp-arm optind = 0; 1010cc672bb2Sdp-arm 1011cc672bb2Sdp-arm if (argc == 0) 1012cc672bb2Sdp-arm usage(); 1013819281eeSdp-arm 1014819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1015819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0) { 1016819281eeSdp-arm ret = cmds[i].handler(argc, argv); 1017819281eeSdp-arm break; 1018819281eeSdp-arm } 1019819281eeSdp-arm } 1020819281eeSdp-arm if (i == NELEM(cmds)) 1021819281eeSdp-arm usage(); 1022819281eeSdp-arm return ret; 1023819281eeSdp-arm } 1024