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 54e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid); 55e0f083a0Sdp-arm static image_t *lookup_image_from_uuid(const uuid_t *uuid); 56819281eeSdp-arm static int info_cmd(int argc, char *argv[]); 57819281eeSdp-arm static void info_usage(void); 58819281eeSdp-arm static int create_cmd(int argc, char *argv[]); 59819281eeSdp-arm static void create_usage(void); 60819281eeSdp-arm static int update_cmd(int argc, char *argv[]); 61819281eeSdp-arm static void update_usage(void); 62819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]); 63819281eeSdp-arm static void unpack_usage(void); 64819281eeSdp-arm static int remove_cmd(int argc, char *argv[]); 65819281eeSdp-arm static void remove_usage(void); 66819281eeSdp-arm static int version_cmd(int argc, char *argv[]); 67819281eeSdp-arm static void version_usage(void); 68819281eeSdp-arm static int help_cmd(int argc, char *argv[]); 69819281eeSdp-arm static void usage(void); 70819281eeSdp-arm 71819281eeSdp-arm /* Available subcommands. */ 72819281eeSdp-arm static cmd_t cmds[] = { 73819281eeSdp-arm { .name = "info", .handler = info_cmd, .usage = info_usage }, 74819281eeSdp-arm { .name = "create", .handler = create_cmd, .usage = create_usage }, 75819281eeSdp-arm { .name = "update", .handler = update_cmd, .usage = update_usage }, 76819281eeSdp-arm { .name = "unpack", .handler = unpack_cmd, .usage = unpack_usage }, 77819281eeSdp-arm { .name = "remove", .handler = remove_cmd, .usage = remove_usage }, 78819281eeSdp-arm { .name = "version", .handler = version_cmd, .usage = version_usage }, 79819281eeSdp-arm { .name = "help", .handler = help_cmd, .usage = NULL }, 80819281eeSdp-arm }; 81819281eeSdp-arm 82e0f083a0Sdp-arm static image_desc_t *image_desc_head; 83e0f083a0Sdp-arm static size_t nr_image_descs; 84e0f083a0Sdp-arm static image_t *image_head; 85819281eeSdp-arm static size_t nr_images; 86819281eeSdp-arm static uuid_t uuid_null = { 0 }; 87819281eeSdp-arm static int verbose; 88819281eeSdp-arm 8960b499feSdp-arm static void vlog(int prio, const char *msg, va_list ap) 90819281eeSdp-arm { 91819281eeSdp-arm char *prefix[] = { "DEBUG", "WARN", "ERROR" }; 92819281eeSdp-arm 93819281eeSdp-arm fprintf(stderr, "%s: ", prefix[prio]); 94819281eeSdp-arm vfprintf(stderr, msg, ap); 95819281eeSdp-arm fputc('\n', stderr); 96819281eeSdp-arm } 97819281eeSdp-arm 9860b499feSdp-arm static void log_dbgx(const char *msg, ...) 99819281eeSdp-arm { 100819281eeSdp-arm va_list ap; 101819281eeSdp-arm 102819281eeSdp-arm va_start(ap, msg); 103819281eeSdp-arm vlog(LOG_DBG, msg, ap); 104819281eeSdp-arm va_end(ap); 105819281eeSdp-arm } 106819281eeSdp-arm 10760b499feSdp-arm static void log_warnx(const char *msg, ...) 108819281eeSdp-arm { 109819281eeSdp-arm va_list ap; 110819281eeSdp-arm 111819281eeSdp-arm va_start(ap, msg); 112819281eeSdp-arm vlog(LOG_WARN, msg, ap); 113819281eeSdp-arm va_end(ap); 114819281eeSdp-arm } 115819281eeSdp-arm 11660b499feSdp-arm static void log_err(const char *msg, ...) 117819281eeSdp-arm { 118819281eeSdp-arm char buf[512]; 119819281eeSdp-arm va_list ap; 120819281eeSdp-arm 121819281eeSdp-arm va_start(ap, msg); 122819281eeSdp-arm snprintf(buf, sizeof(buf), "%s: %s", msg, strerror(errno)); 123819281eeSdp-arm vlog(LOG_ERR, buf, ap); 124819281eeSdp-arm va_end(ap); 125819281eeSdp-arm exit(1); 126819281eeSdp-arm } 127819281eeSdp-arm 12860b499feSdp-arm static void log_errx(const char *msg, ...) 129819281eeSdp-arm { 130819281eeSdp-arm va_list ap; 131819281eeSdp-arm 132819281eeSdp-arm va_start(ap, msg); 133819281eeSdp-arm vlog(LOG_ERR, msg, ap); 134819281eeSdp-arm va_end(ap); 135819281eeSdp-arm exit(1); 136819281eeSdp-arm } 137819281eeSdp-arm 138a22f6285Sdp-arm static char *xstrdup(const char *s, const char *msg) 139a22f6285Sdp-arm { 140a22f6285Sdp-arm char *d; 141a22f6285Sdp-arm 142a22f6285Sdp-arm d = strdup(s); 143a22f6285Sdp-arm if (d == NULL) 1449fc9ff1fSdp-arm log_errx("strdup: %s", msg); 145a22f6285Sdp-arm return d; 146a22f6285Sdp-arm } 147a22f6285Sdp-arm 148a22f6285Sdp-arm static void *xmalloc(size_t size, const char *msg) 149a22f6285Sdp-arm { 150a22f6285Sdp-arm void *d; 151a22f6285Sdp-arm 152a22f6285Sdp-arm d = malloc(size); 153a22f6285Sdp-arm if (d == NULL) 1549fc9ff1fSdp-arm log_errx("malloc: %s", msg); 155a22f6285Sdp-arm return d; 156a22f6285Sdp-arm } 157a22f6285Sdp-arm 158e0f083a0Sdp-arm static image_desc_t *new_image_desc(const uuid_t *uuid, 159e0f083a0Sdp-arm const char *name, const char *cmdline_name) 160e0f083a0Sdp-arm { 161e0f083a0Sdp-arm image_desc_t *desc; 162e0f083a0Sdp-arm 163e0f083a0Sdp-arm desc = xmalloc(sizeof(*desc), 164e0f083a0Sdp-arm "failed to allocate memory for image descriptor"); 165e0f083a0Sdp-arm memcpy(&desc->uuid, uuid, sizeof(uuid_t)); 166e0f083a0Sdp-arm desc->name = xstrdup(name, 167e0f083a0Sdp-arm "failed to allocate memory for image name"); 168e0f083a0Sdp-arm desc->cmdline_name = xstrdup(cmdline_name, 169e0f083a0Sdp-arm "failed to allocate memory for image command line name"); 170e0f083a0Sdp-arm desc->action = DO_UNSPEC; 171e0f083a0Sdp-arm desc->action_arg = NULL; 172e0f083a0Sdp-arm return desc; 173e0f083a0Sdp-arm } 174e0f083a0Sdp-arm 175*d02fcebeSdp-arm static void set_image_desc_action(image_desc_t *desc, int action, 176*d02fcebeSdp-arm const char *arg) 177*d02fcebeSdp-arm { 178*d02fcebeSdp-arm assert(desc != NULL); 179*d02fcebeSdp-arm 180*d02fcebeSdp-arm if (desc->action_arg != DO_UNSPEC) 181*d02fcebeSdp-arm free(desc->action_arg); 182*d02fcebeSdp-arm desc->action = action; 183*d02fcebeSdp-arm desc->action_arg = NULL; 184*d02fcebeSdp-arm if (arg != NULL) 185*d02fcebeSdp-arm desc->action_arg = xstrdup(arg, 186*d02fcebeSdp-arm "failed to allocate memory for argument"); 187*d02fcebeSdp-arm } 188*d02fcebeSdp-arm 189e0f083a0Sdp-arm static void free_image_desc(image_desc_t *desc) 190e0f083a0Sdp-arm { 191e0f083a0Sdp-arm free(desc->name); 192e0f083a0Sdp-arm free(desc->cmdline_name); 193e0f083a0Sdp-arm free(desc->action_arg); 194e0f083a0Sdp-arm free(desc); 195e0f083a0Sdp-arm } 196e0f083a0Sdp-arm 197e0f083a0Sdp-arm static void add_image_desc(image_desc_t *desc) 198e0f083a0Sdp-arm { 199e0f083a0Sdp-arm assert(lookup_image_desc_from_uuid(&desc->uuid) == NULL); 200e0f083a0Sdp-arm desc->next = image_desc_head; 201e0f083a0Sdp-arm image_desc_head = desc; 202e0f083a0Sdp-arm nr_image_descs++; 203e0f083a0Sdp-arm } 204e0f083a0Sdp-arm 205e0f083a0Sdp-arm static void free_image_descs(void) 206e0f083a0Sdp-arm { 207e0f083a0Sdp-arm image_desc_t *desc = image_desc_head, *tmp; 208e0f083a0Sdp-arm 209e0f083a0Sdp-arm while (desc != NULL) { 210e0f083a0Sdp-arm tmp = desc->next; 211e0f083a0Sdp-arm free_image_desc(desc); 212e0f083a0Sdp-arm desc = tmp; 213e0f083a0Sdp-arm nr_image_descs--; 214e0f083a0Sdp-arm } 215e0f083a0Sdp-arm assert(nr_image_descs == 0); 216e0f083a0Sdp-arm } 217e0f083a0Sdp-arm 218e0f083a0Sdp-arm static void fill_image_descs(void) 219e0f083a0Sdp-arm { 220e0f083a0Sdp-arm toc_entry_t *toc_entry; 221e0f083a0Sdp-arm 222e0f083a0Sdp-arm for (toc_entry = toc_entries; 223e0f083a0Sdp-arm toc_entry->cmdline_name != NULL; 224e0f083a0Sdp-arm toc_entry++) { 225e0f083a0Sdp-arm image_desc_t *desc; 226e0f083a0Sdp-arm 227e0f083a0Sdp-arm desc = new_image_desc(&toc_entry->uuid, 228e0f083a0Sdp-arm toc_entry->name, 229e0f083a0Sdp-arm toc_entry->cmdline_name); 230e0f083a0Sdp-arm add_image_desc(desc); 231e0f083a0Sdp-arm } 232e0f083a0Sdp-arm } 233e0f083a0Sdp-arm 234819281eeSdp-arm static void add_image(image_t *image) 235819281eeSdp-arm { 236e0f083a0Sdp-arm assert(lookup_image_from_uuid(&image->uuid) == NULL); 237e0f083a0Sdp-arm image->next = image_head; 238e0f083a0Sdp-arm image_head = image; 239e0f083a0Sdp-arm nr_images++; 240819281eeSdp-arm } 241819281eeSdp-arm 242819281eeSdp-arm static void free_image(image_t *image) 243819281eeSdp-arm { 244819281eeSdp-arm free(image->buffer); 245819281eeSdp-arm free(image); 246819281eeSdp-arm } 247819281eeSdp-arm 248819281eeSdp-arm static void remove_image(image_t *image) 249819281eeSdp-arm { 250e0f083a0Sdp-arm image_t *tmp = image_head, *prev; 251819281eeSdp-arm 252e0f083a0Sdp-arm if (tmp == image) { 253e0f083a0Sdp-arm image_head = tmp->next; 254e0f083a0Sdp-arm free_image(tmp); 255e0f083a0Sdp-arm } else { 256e0f083a0Sdp-arm while (tmp != NULL && tmp != image) { 257e0f083a0Sdp-arm prev = tmp; 258e0f083a0Sdp-arm tmp = tmp->next; 259819281eeSdp-arm } 260e0f083a0Sdp-arm assert(tmp != NULL); 261e0f083a0Sdp-arm prev->next = tmp->next; 262e0f083a0Sdp-arm free_image(tmp); 263819281eeSdp-arm } 264819281eeSdp-arm nr_images--; 265819281eeSdp-arm } 266819281eeSdp-arm 267819281eeSdp-arm static void free_images(void) 268819281eeSdp-arm { 269e0f083a0Sdp-arm image_t *image = image_head, *tmp; 270819281eeSdp-arm 271e0f083a0Sdp-arm while (image != NULL) { 272e0f083a0Sdp-arm tmp = image->next; 273e0f083a0Sdp-arm free_image(image); 274e0f083a0Sdp-arm image = tmp; 275e0f083a0Sdp-arm nr_images--; 276819281eeSdp-arm } 277e0f083a0Sdp-arm assert(nr_images == 0); 278819281eeSdp-arm } 279819281eeSdp-arm 280e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid) 281819281eeSdp-arm { 282e0f083a0Sdp-arm image_desc_t *desc; 283819281eeSdp-arm 284e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 285e0f083a0Sdp-arm if (memcmp(&desc->uuid, uuid, sizeof(uuid_t)) == 0) 286e0f083a0Sdp-arm return desc; 287819281eeSdp-arm return NULL; 288819281eeSdp-arm } 289819281eeSdp-arm 290e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_opt(const char *opt) 291e0f083a0Sdp-arm { 292e0f083a0Sdp-arm image_desc_t *desc; 293e0f083a0Sdp-arm 294e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 295e0f083a0Sdp-arm if (strcmp(desc->cmdline_name, opt) == 0) 296e0f083a0Sdp-arm return desc; 297e0f083a0Sdp-arm return NULL; 298e0f083a0Sdp-arm } 299e0f083a0Sdp-arm 300e0f083a0Sdp-arm static image_t *lookup_image_from_uuid(const uuid_t *uuid) 301b04efcceSdp-arm { 302b04efcceSdp-arm image_t *image; 303b04efcceSdp-arm 304e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 305b04efcceSdp-arm if (memcmp(&image->uuid, uuid, sizeof(uuid_t)) == 0) 306b04efcceSdp-arm return image; 307b04efcceSdp-arm return NULL; 308b04efcceSdp-arm } 309b04efcceSdp-arm 310fcab6bbeSdp-arm static void uuid_to_str(char *s, size_t len, const uuid_t *u) 311fcab6bbeSdp-arm { 312fcab6bbeSdp-arm assert(len >= (_UUID_STR_LEN + 1)); 313fcab6bbeSdp-arm 314fcab6bbeSdp-arm snprintf(s, len, "%08X-%04X-%04X-%04X-%04X%04X%04X", 315fcab6bbeSdp-arm u->time_low, 316fcab6bbeSdp-arm u->time_mid, 317fcab6bbeSdp-arm u->time_hi_and_version, 318fcab6bbeSdp-arm ((uint16_t)u->clock_seq_hi_and_reserved << 8) | u->clock_seq_low, 319fcab6bbeSdp-arm ((uint16_t)u->node[0] << 8) | u->node[1], 320fcab6bbeSdp-arm ((uint16_t)u->node[2] << 8) | u->node[3], 321fcab6bbeSdp-arm ((uint16_t)u->node[4] << 8) | u->node[5]); 322fcab6bbeSdp-arm } 323fcab6bbeSdp-arm 324fcab6bbeSdp-arm static void uuid_from_str(uuid_t *u, const char *s) 325fcab6bbeSdp-arm { 326fcab6bbeSdp-arm int n; 327fcab6bbeSdp-arm 328fcab6bbeSdp-arm if (s == NULL) 329fcab6bbeSdp-arm log_errx("UUID cannot be NULL"); 330fcab6bbeSdp-arm if (strlen(s) != _UUID_STR_LEN) 331fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 332fcab6bbeSdp-arm 333fcab6bbeSdp-arm n = sscanf(s, 334fcab6bbeSdp-arm "%8x-%4hx-%4hx-%2hhx%2hhx-%2hhx%2hhx%2hhx%2hhx%2hhx%2hhx", 335fcab6bbeSdp-arm &u->time_low, &u->time_mid, &u->time_hi_and_version, 336fcab6bbeSdp-arm &u->clock_seq_hi_and_reserved, &u->clock_seq_low, &u->node[0], 337fcab6bbeSdp-arm &u->node[1], &u->node[2], &u->node[3], &u->node[4], &u->node[5]); 338fcab6bbeSdp-arm /* 339fcab6bbeSdp-arm * Given the format specifier above, we expect 11 items to be scanned 340fcab6bbeSdp-arm * for a properly formatted UUID. 341fcab6bbeSdp-arm */ 342fcab6bbeSdp-arm if (n != 11) 343fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 344fcab6bbeSdp-arm } 345fcab6bbeSdp-arm 34660b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out) 347819281eeSdp-arm { 348819281eeSdp-arm struct stat st; 349819281eeSdp-arm FILE *fp; 350819281eeSdp-arm char *buf, *bufend; 351819281eeSdp-arm fip_toc_header_t *toc_header; 352819281eeSdp-arm fip_toc_entry_t *toc_entry; 353819281eeSdp-arm int terminated = 0; 354819281eeSdp-arm 355819281eeSdp-arm fp = fopen(filename, "r"); 356819281eeSdp-arm if (fp == NULL) 357819281eeSdp-arm log_err("fopen %s", filename); 358819281eeSdp-arm 359819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 360819281eeSdp-arm log_err("fstat %s", filename); 361819281eeSdp-arm 362a22f6285Sdp-arm buf = xmalloc(st.st_size, "failed to load file into memory"); 363819281eeSdp-arm if (fread(buf, 1, st.st_size, fp) != st.st_size) 364819281eeSdp-arm log_errx("Failed to read %s", filename); 365819281eeSdp-arm bufend = buf + st.st_size; 366819281eeSdp-arm fclose(fp); 367819281eeSdp-arm 368819281eeSdp-arm if (st.st_size < sizeof(fip_toc_header_t)) 369819281eeSdp-arm log_errx("FIP %s is truncated", filename); 370819281eeSdp-arm 371819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 372819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 373819281eeSdp-arm 374819281eeSdp-arm if (toc_header->name != TOC_HEADER_NAME) 375819281eeSdp-arm log_errx("%s is not a FIP file", filename); 376819281eeSdp-arm 377819281eeSdp-arm /* Return the ToC header if the caller wants it. */ 378819281eeSdp-arm if (toc_header_out != NULL) 379819281eeSdp-arm *toc_header_out = *toc_header; 380819281eeSdp-arm 381819281eeSdp-arm /* Walk through each ToC entry in the file. */ 382819281eeSdp-arm while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) { 383fcab6bbeSdp-arm image_t *image; 384fcab6bbeSdp-arm image_desc_t *desc; 385fcab6bbeSdp-arm 386819281eeSdp-arm /* Found the ToC terminator, we are done. */ 387819281eeSdp-arm if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) { 388819281eeSdp-arm terminated = 1; 389819281eeSdp-arm break; 390819281eeSdp-arm } 391819281eeSdp-arm 392819281eeSdp-arm /* 393819281eeSdp-arm * Build a new image out of the ToC entry and add it to the 394819281eeSdp-arm * table of images. 395819281eeSdp-arm */ 396a22f6285Sdp-arm image = xmalloc(sizeof(*image), 397a22f6285Sdp-arm "failed to allocate memory for image"); 398819281eeSdp-arm memcpy(&image->uuid, &toc_entry->uuid, sizeof(uuid_t)); 399a22f6285Sdp-arm image->buffer = xmalloc(toc_entry->size, 400a22f6285Sdp-arm "failed to allocate image buffer, is FIP file corrupted?"); 401819281eeSdp-arm /* Overflow checks before memory copy. */ 402819281eeSdp-arm if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address) 403819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 404819281eeSdp-arm if (toc_entry->size + toc_entry->offset_address > st.st_size) 405819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 406819281eeSdp-arm 407819281eeSdp-arm memcpy(image->buffer, buf + toc_entry->offset_address, 408819281eeSdp-arm toc_entry->size); 409819281eeSdp-arm image->size = toc_entry->size; 410819281eeSdp-arm 411fcab6bbeSdp-arm /* If this is an unknown image, create a descriptor for it. */ 412fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&image->uuid); 413fcab6bbeSdp-arm if (desc == NULL) { 414fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 415fcab6bbeSdp-arm 416fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &image->uuid); 417fcab6bbeSdp-arm snprintf(filename, sizeof(filename), "%s%s", 418fcab6bbeSdp-arm name, ".bin"); 419fcab6bbeSdp-arm desc = new_image_desc(&image->uuid, name, "blob"); 420fcab6bbeSdp-arm desc->action = DO_UNPACK; 421fcab6bbeSdp-arm desc->action_arg = xstrdup(filename, 422fcab6bbeSdp-arm "failed to allocate memory for blob filename"); 423fcab6bbeSdp-arm add_image_desc(desc); 424fcab6bbeSdp-arm } 425fcab6bbeSdp-arm 426819281eeSdp-arm add_image(image); 427819281eeSdp-arm 428819281eeSdp-arm toc_entry++; 429819281eeSdp-arm } 430819281eeSdp-arm 431819281eeSdp-arm if (terminated == 0) 432819281eeSdp-arm log_errx("FIP %s does not have a ToC terminator entry", 433819281eeSdp-arm filename); 434819281eeSdp-arm free(buf); 435819281eeSdp-arm return 0; 436819281eeSdp-arm } 437819281eeSdp-arm 43860b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename) 439819281eeSdp-arm { 440819281eeSdp-arm struct stat st; 441819281eeSdp-arm image_t *image; 442819281eeSdp-arm FILE *fp; 443819281eeSdp-arm 444b04efcceSdp-arm assert(uuid != NULL); 445b04efcceSdp-arm 446819281eeSdp-arm fp = fopen(filename, "r"); 447819281eeSdp-arm if (fp == NULL) 448819281eeSdp-arm log_err("fopen %s", filename); 449819281eeSdp-arm 450819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 451819281eeSdp-arm log_errx("fstat %s", filename); 452819281eeSdp-arm 453a22f6285Sdp-arm image = xmalloc(sizeof(*image), "failed to allocate memory for image"); 454b04efcceSdp-arm memcpy(&image->uuid, uuid, sizeof(uuid_t)); 455a22f6285Sdp-arm image->buffer = xmalloc(st.st_size, "failed to allocate image buffer"); 456819281eeSdp-arm if (fread(image->buffer, 1, st.st_size, fp) != st.st_size) 457819281eeSdp-arm log_errx("Failed to read %s", filename); 458819281eeSdp-arm image->size = st.st_size; 459819281eeSdp-arm 460819281eeSdp-arm fclose(fp); 461819281eeSdp-arm return image; 462819281eeSdp-arm } 463819281eeSdp-arm 46460b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename) 465819281eeSdp-arm { 466819281eeSdp-arm FILE *fp; 467819281eeSdp-arm 468819281eeSdp-arm fp = fopen(filename, "w"); 469819281eeSdp-arm if (fp == NULL) 470819281eeSdp-arm log_err("fopen"); 471819281eeSdp-arm if (fwrite(image->buffer, 1, image->size, fp) != image->size) 472819281eeSdp-arm log_errx("Failed to write %s", filename); 473819281eeSdp-arm fclose(fp); 474819281eeSdp-arm return 0; 475819281eeSdp-arm } 476819281eeSdp-arm 477e0f083a0Sdp-arm static struct option *add_opt(struct option *opts, size_t *nr_opts, 478e0f083a0Sdp-arm const char *name, int has_arg, int val) 479819281eeSdp-arm { 480e0f083a0Sdp-arm opts = realloc(opts, (*nr_opts + 1) * sizeof(*opts)); 481e0f083a0Sdp-arm if (opts == NULL) 482e0f083a0Sdp-arm log_err("realloc"); 483e0f083a0Sdp-arm opts[*nr_opts].name = name; 484e0f083a0Sdp-arm opts[*nr_opts].has_arg = has_arg; 485e0f083a0Sdp-arm opts[*nr_opts].flag = NULL; 486e0f083a0Sdp-arm opts[*nr_opts].val = val; 487e0f083a0Sdp-arm ++*nr_opts; 488e0f083a0Sdp-arm return opts; 489819281eeSdp-arm } 490819281eeSdp-arm 491e0f083a0Sdp-arm static struct option *fill_common_opts(struct option *opts, size_t *nr_opts, 492e0f083a0Sdp-arm int has_arg) 493819281eeSdp-arm { 494e0f083a0Sdp-arm image_desc_t *desc; 495e0f083a0Sdp-arm 496e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 497e0f083a0Sdp-arm opts = add_opt(opts, nr_opts, desc->cmdline_name, has_arg, 498e0f083a0Sdp-arm OPT_TOC_ENTRY); 499e0f083a0Sdp-arm return opts; 500819281eeSdp-arm } 501819281eeSdp-arm 502e0f083a0Sdp-arm static void md_print(const unsigned char *md, size_t len) 5039df69ba3Sdp-arm { 5049df69ba3Sdp-arm size_t i; 5059df69ba3Sdp-arm 5069df69ba3Sdp-arm for (i = 0; i < len; i++) 5079df69ba3Sdp-arm printf("%02x", md[i]); 5089df69ba3Sdp-arm } 5099df69ba3Sdp-arm 510819281eeSdp-arm static int info_cmd(int argc, char *argv[]) 511819281eeSdp-arm { 512819281eeSdp-arm image_t *image; 513819281eeSdp-arm uint64_t image_offset; 514fcab6bbeSdp-arm uint64_t image_size; 515819281eeSdp-arm fip_toc_header_t toc_header; 516819281eeSdp-arm 517819281eeSdp-arm if (argc != 2) 51885ee2778Sdp-arm info_usage(); 519819281eeSdp-arm argc--, argv++; 520819281eeSdp-arm 521819281eeSdp-arm parse_fip(argv[0], &toc_header); 522819281eeSdp-arm 523819281eeSdp-arm if (verbose) { 524819281eeSdp-arm log_dbgx("toc_header[name]: 0x%llX", 525819281eeSdp-arm (unsigned long long)toc_header.name); 526819281eeSdp-arm log_dbgx("toc_header[serial_number]: 0x%llX", 527819281eeSdp-arm (unsigned long long)toc_header.serial_number); 528819281eeSdp-arm log_dbgx("toc_header[flags]: 0x%llX", 529819281eeSdp-arm (unsigned long long)toc_header.flags); 530819281eeSdp-arm } 531819281eeSdp-arm 532819281eeSdp-arm image_offset = sizeof(fip_toc_header_t) + 533819281eeSdp-arm (sizeof(fip_toc_entry_t) * (nr_images + 1)); 534819281eeSdp-arm 535e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) { 536e0f083a0Sdp-arm image_desc_t *desc; 537b04efcceSdp-arm 538e0f083a0Sdp-arm desc = lookup_image_desc_from_uuid(&image->uuid); 539fcab6bbeSdp-arm assert(desc != NULL); 540e0f083a0Sdp-arm printf("%s: ", desc->name); 541819281eeSdp-arm image_size = image->size; 542819281eeSdp-arm printf("offset=0x%llX, size=0x%llX", 543819281eeSdp-arm (unsigned long long)image_offset, 544819281eeSdp-arm (unsigned long long)image_size); 545e0f083a0Sdp-arm if (desc != NULL) 5469df69ba3Sdp-arm printf(", cmdline=\"--%s\"", 547e0f083a0Sdp-arm desc->cmdline_name); 5489df69ba3Sdp-arm if (verbose) { 5499df69ba3Sdp-arm unsigned char md[SHA256_DIGEST_LENGTH]; 5509df69ba3Sdp-arm 5519df69ba3Sdp-arm SHA256(image->buffer, image_size, md); 5529df69ba3Sdp-arm printf(", sha256="); 5539df69ba3Sdp-arm md_print(md, sizeof(md)); 5549df69ba3Sdp-arm } 555819281eeSdp-arm putchar('\n'); 556819281eeSdp-arm image_offset += image_size; 557819281eeSdp-arm } 558819281eeSdp-arm 559819281eeSdp-arm free_images(); 560819281eeSdp-arm return 0; 561819281eeSdp-arm } 562819281eeSdp-arm 563819281eeSdp-arm static void info_usage(void) 564819281eeSdp-arm { 565819281eeSdp-arm printf("fiptool info FIP_FILENAME\n"); 56685ee2778Sdp-arm exit(1); 567819281eeSdp-arm } 568819281eeSdp-arm 569e0f083a0Sdp-arm static int pack_images(const char *filename, uint64_t toc_flags) 570819281eeSdp-arm { 571819281eeSdp-arm FILE *fp; 572819281eeSdp-arm image_t *image; 573819281eeSdp-arm fip_toc_header_t *toc_header; 574819281eeSdp-arm fip_toc_entry_t *toc_entry; 575819281eeSdp-arm char *buf; 576819281eeSdp-arm uint64_t entry_offset, buf_size, payload_size; 577819281eeSdp-arm 578819281eeSdp-arm /* Calculate total payload size and allocate scratch buffer. */ 579819281eeSdp-arm payload_size = 0; 580e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 581e0f083a0Sdp-arm payload_size += image->size; 582819281eeSdp-arm 583819281eeSdp-arm buf_size = sizeof(fip_toc_header_t) + 584819281eeSdp-arm sizeof(fip_toc_entry_t) * (nr_images + 1); 585819281eeSdp-arm buf = calloc(1, buf_size); 586819281eeSdp-arm if (buf == NULL) 587819281eeSdp-arm log_err("calloc"); 588819281eeSdp-arm 589819281eeSdp-arm /* Build up header and ToC entries from the image table. */ 590819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 591819281eeSdp-arm toc_header->name = TOC_HEADER_NAME; 592819281eeSdp-arm toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER; 593819281eeSdp-arm toc_header->flags = toc_flags; 594819281eeSdp-arm 595819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 596819281eeSdp-arm 597819281eeSdp-arm entry_offset = buf_size; 598e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) { 599819281eeSdp-arm memcpy(&toc_entry->uuid, &image->uuid, sizeof(uuid_t)); 600819281eeSdp-arm toc_entry->offset_address = entry_offset; 601819281eeSdp-arm toc_entry->size = image->size; 602819281eeSdp-arm toc_entry->flags = 0; 603819281eeSdp-arm entry_offset += toc_entry->size; 604819281eeSdp-arm toc_entry++; 605819281eeSdp-arm } 606819281eeSdp-arm 607819281eeSdp-arm /* Append a null uuid entry to mark the end of ToC entries. */ 608819281eeSdp-arm memcpy(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)); 609819281eeSdp-arm toc_entry->offset_address = entry_offset; 610819281eeSdp-arm toc_entry->size = 0; 611819281eeSdp-arm toc_entry->flags = 0; 612819281eeSdp-arm 613819281eeSdp-arm /* Generate the FIP file. */ 614819281eeSdp-arm fp = fopen(filename, "w"); 615819281eeSdp-arm if (fp == NULL) 616819281eeSdp-arm log_err("fopen %s", filename); 617819281eeSdp-arm 618819281eeSdp-arm if (verbose) 619819281eeSdp-arm log_dbgx("Metadata size: %zu bytes", buf_size); 620819281eeSdp-arm 621819281eeSdp-arm if (fwrite(buf, 1, buf_size, fp) != buf_size) 622819281eeSdp-arm log_errx("Failed to write image to %s", filename); 623819281eeSdp-arm free(buf); 624819281eeSdp-arm 625819281eeSdp-arm if (verbose) 626819281eeSdp-arm log_dbgx("Payload size: %zu bytes", payload_size); 627819281eeSdp-arm 628e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 629819281eeSdp-arm if (fwrite(image->buffer, 1, image->size, fp) != image->size) 630819281eeSdp-arm log_errx("Failed to write image to %s", filename); 631819281eeSdp-arm 632819281eeSdp-arm fclose(fp); 633819281eeSdp-arm return 0; 634819281eeSdp-arm } 635819281eeSdp-arm 636819281eeSdp-arm /* 637819281eeSdp-arm * This function is shared between the create and update subcommands. 638819281eeSdp-arm * The difference between the two subcommands is that when the FIP file 639819281eeSdp-arm * is created, the parsing of an existing FIP is skipped. This results 640819281eeSdp-arm * in update_fip() creating the new FIP file from scratch because the 641819281eeSdp-arm * internal image table is not populated. 642819281eeSdp-arm */ 643819281eeSdp-arm static void update_fip(void) 644819281eeSdp-arm { 645e0f083a0Sdp-arm image_desc_t *desc; 646819281eeSdp-arm 647819281eeSdp-arm /* Add or replace images in the FIP file. */ 648e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 649fcab6bbeSdp-arm image_t *new_image, *old_image; 650fcab6bbeSdp-arm 651e0f083a0Sdp-arm if (desc->action != DO_PACK) 652819281eeSdp-arm continue; 653819281eeSdp-arm 654e0f083a0Sdp-arm new_image = read_image_from_file(&desc->uuid, 655e0f083a0Sdp-arm desc->action_arg); 656e0f083a0Sdp-arm old_image = lookup_image_from_uuid(&desc->uuid); 657b04efcceSdp-arm if (old_image != NULL) { 658e0f083a0Sdp-arm if (verbose) { 659fcab6bbeSdp-arm log_dbgx("Replacing %s with %s", 660e0f083a0Sdp-arm desc->cmdline_name, 661e0f083a0Sdp-arm desc->action_arg); 662e0f083a0Sdp-arm } 663e0f083a0Sdp-arm remove_image(old_image); 664e0f083a0Sdp-arm add_image(new_image); 665819281eeSdp-arm } else { 666819281eeSdp-arm if (verbose) 667819281eeSdp-arm log_dbgx("Adding image %s", 668e0f083a0Sdp-arm desc->action_arg); 669b04efcceSdp-arm add_image(new_image); 670819281eeSdp-arm } 671819281eeSdp-arm } 672819281eeSdp-arm } 673819281eeSdp-arm 674e0f083a0Sdp-arm static void parse_plat_toc_flags(const char *arg, unsigned long long *toc_flags) 675819281eeSdp-arm { 676819281eeSdp-arm unsigned long long flags; 677819281eeSdp-arm char *endptr; 678819281eeSdp-arm 679819281eeSdp-arm errno = 0; 680819281eeSdp-arm flags = strtoull(arg, &endptr, 16); 681819281eeSdp-arm if (*endptr != '\0' || flags > UINT16_MAX || errno != 0) 682819281eeSdp-arm log_errx("Invalid platform ToC flags: %s", arg); 683819281eeSdp-arm /* Platform ToC flags is a 16-bit field occupying bits [32-47]. */ 684819281eeSdp-arm *toc_flags |= flags << 32; 685819281eeSdp-arm } 686819281eeSdp-arm 687fcab6bbeSdp-arm static void parse_blob_opt(char *arg, uuid_t *uuid, char *filename, size_t len) 688fcab6bbeSdp-arm { 689fcab6bbeSdp-arm char *p; 690fcab6bbeSdp-arm 691fcab6bbeSdp-arm for (p = strtok(arg, ","); p != NULL; p = strtok(NULL, ",")) { 692fcab6bbeSdp-arm if (strncmp(p, "uuid=", strlen("uuid=")) == 0) { 693fcab6bbeSdp-arm p += strlen("uuid="); 694fcab6bbeSdp-arm uuid_from_str(uuid, p); 695fcab6bbeSdp-arm } else if (strncmp(p, "file=", strlen("file=")) == 0) { 696fcab6bbeSdp-arm p += strlen("file="); 697fcab6bbeSdp-arm snprintf(filename, len, "%s", p); 698fcab6bbeSdp-arm } 699fcab6bbeSdp-arm } 700fcab6bbeSdp-arm } 701fcab6bbeSdp-arm 702819281eeSdp-arm static int create_cmd(int argc, char *argv[]) 703819281eeSdp-arm { 704e0f083a0Sdp-arm struct option *opts = NULL; 705e0f083a0Sdp-arm size_t nr_opts = 0; 706819281eeSdp-arm unsigned long long toc_flags = 0; 707819281eeSdp-arm 708819281eeSdp-arm if (argc < 2) 70985ee2778Sdp-arm create_usage(); 710819281eeSdp-arm 711e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 712e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 713819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 714fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 715e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 716819281eeSdp-arm 717819281eeSdp-arm while (1) { 71820f87e78Sdp-arm int c, opt_index = 0; 719819281eeSdp-arm 720fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:", opts, &opt_index); 721819281eeSdp-arm if (c == -1) 722819281eeSdp-arm break; 723819281eeSdp-arm 724819281eeSdp-arm switch (c) { 725819281eeSdp-arm case OPT_TOC_ENTRY: { 726e0f083a0Sdp-arm image_desc_t *desc; 727819281eeSdp-arm 728e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 729*d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 730819281eeSdp-arm break; 731819281eeSdp-arm } 732819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: 733819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 734819281eeSdp-arm break; 735fcab6bbeSdp-arm case 'b': { 736fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 737fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 738fcab6bbeSdp-arm uuid_t uuid = { 0 }; 739fcab6bbeSdp-arm image_desc_t *desc; 740fcab6bbeSdp-arm 741fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 742fcab6bbeSdp-arm filename, sizeof(filename)); 743fcab6bbeSdp-arm 744fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 745fcab6bbeSdp-arm filename[0] == '\0') 746fcab6bbeSdp-arm create_usage(); 747fcab6bbeSdp-arm 748fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 749*d02fcebeSdp-arm if (desc == NULL) { 750fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 751fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 752fcab6bbeSdp-arm add_image_desc(desc); 753fcab6bbeSdp-arm } 754*d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 755fcab6bbeSdp-arm break; 756fcab6bbeSdp-arm } 757819281eeSdp-arm default: 75885ee2778Sdp-arm create_usage(); 759819281eeSdp-arm } 760819281eeSdp-arm } 761819281eeSdp-arm argc -= optind; 762819281eeSdp-arm argv += optind; 763e0f083a0Sdp-arm free(opts); 764819281eeSdp-arm 765819281eeSdp-arm if (argc == 0) 76685ee2778Sdp-arm create_usage(); 767819281eeSdp-arm 768819281eeSdp-arm update_fip(); 769819281eeSdp-arm 770819281eeSdp-arm pack_images(argv[0], toc_flags); 771819281eeSdp-arm free_images(); 772819281eeSdp-arm return 0; 773819281eeSdp-arm } 774819281eeSdp-arm 775819281eeSdp-arm static void create_usage(void) 776819281eeSdp-arm { 777819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 778819281eeSdp-arm 779fcab6bbeSdp-arm printf("fiptool create [--blob uuid=...,file=...] " 780fcab6bbeSdp-arm "[--plat-toc-flags <value>] [opts] FIP_FILENAME\n"); 781fcab6bbeSdp-arm printf(" --blob uuid=...,file=...\tAdd an image with the given UUID " 782fcab6bbeSdp-arm "pointed to by file.\n"); 783819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 784819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 785819281eeSdp-arm fputc('\n', stderr); 786819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 787819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 788819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 789819281eeSdp-arm toc_entry->name); 79085ee2778Sdp-arm exit(1); 791819281eeSdp-arm } 792819281eeSdp-arm 793819281eeSdp-arm static int update_cmd(int argc, char *argv[]) 794819281eeSdp-arm { 795e0f083a0Sdp-arm struct option *opts = NULL; 796e0f083a0Sdp-arm size_t nr_opts = 0; 797fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 798819281eeSdp-arm fip_toc_header_t toc_header = { 0 }; 799819281eeSdp-arm unsigned long long toc_flags = 0; 800819281eeSdp-arm int pflag = 0; 801819281eeSdp-arm 802819281eeSdp-arm if (argc < 2) 80385ee2778Sdp-arm update_usage(); 804819281eeSdp-arm 805e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 806fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 807e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 808e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 809819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 810e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 811819281eeSdp-arm 812819281eeSdp-arm while (1) { 81320f87e78Sdp-arm int c, opt_index = 0; 814819281eeSdp-arm 815fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:o:", opts, &opt_index); 816819281eeSdp-arm if (c == -1) 817819281eeSdp-arm break; 818819281eeSdp-arm 819819281eeSdp-arm switch (c) { 820819281eeSdp-arm case OPT_TOC_ENTRY: { 821e0f083a0Sdp-arm image_desc_t *desc; 822819281eeSdp-arm 823e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 824*d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 825819281eeSdp-arm break; 826819281eeSdp-arm } 827e0f083a0Sdp-arm case OPT_PLAT_TOC_FLAGS: 828819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 829819281eeSdp-arm pflag = 1; 830819281eeSdp-arm break; 831fcab6bbeSdp-arm case 'b': { 832fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 833fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 834fcab6bbeSdp-arm uuid_t uuid = { 0 }; 835fcab6bbeSdp-arm image_desc_t *desc; 836fcab6bbeSdp-arm 837fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 838fcab6bbeSdp-arm filename, sizeof(filename)); 839fcab6bbeSdp-arm 840fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 841fcab6bbeSdp-arm filename[0] == '\0') 842fcab6bbeSdp-arm update_usage(); 843fcab6bbeSdp-arm 844fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 845*d02fcebeSdp-arm if (desc == NULL) { 846fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 847fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 848fcab6bbeSdp-arm add_image_desc(desc); 849fcab6bbeSdp-arm } 850*d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 851fcab6bbeSdp-arm break; 852fcab6bbeSdp-arm } 853819281eeSdp-arm case 'o': 854819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 855819281eeSdp-arm break; 856819281eeSdp-arm default: 85785ee2778Sdp-arm update_usage(); 858819281eeSdp-arm } 859819281eeSdp-arm } 860819281eeSdp-arm argc -= optind; 861819281eeSdp-arm argv += optind; 862e0f083a0Sdp-arm free(opts); 863819281eeSdp-arm 864819281eeSdp-arm if (argc == 0) 86585ee2778Sdp-arm update_usage(); 866819281eeSdp-arm 867819281eeSdp-arm if (outfile[0] == '\0') 868819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 869819281eeSdp-arm 870819281eeSdp-arm if (access(outfile, F_OK) == 0) 871819281eeSdp-arm parse_fip(argv[0], &toc_header); 872819281eeSdp-arm 873819281eeSdp-arm if (pflag) 874819281eeSdp-arm toc_header.flags &= ~(0xffffULL << 32); 875819281eeSdp-arm toc_flags = (toc_header.flags |= toc_flags); 876819281eeSdp-arm 877819281eeSdp-arm update_fip(); 878819281eeSdp-arm 879819281eeSdp-arm pack_images(outfile, toc_flags); 880819281eeSdp-arm free_images(); 881819281eeSdp-arm return 0; 882819281eeSdp-arm } 883819281eeSdp-arm 884819281eeSdp-arm static void update_usage(void) 885819281eeSdp-arm { 886819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 887819281eeSdp-arm 888fcab6bbeSdp-arm printf("fiptool update [--blob uuid=...,file=...] [--out FIP_FILENAME] " 889819281eeSdp-arm "[--plat-toc-flags <value>] [opts] FIP_FILENAME\n"); 890fcab6bbeSdp-arm printf(" --blob uuid=...,file=...\tAdd or update an image " 891fcab6bbeSdp-arm "with the given UUID pointed to by file.\n"); 892819281eeSdp-arm printf(" --out FIP_FILENAME\t\tSet an alternative output FIP file.\n"); 893819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 894819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 895819281eeSdp-arm fputc('\n', stderr); 896819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 897819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 898819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 899819281eeSdp-arm toc_entry->name); 90085ee2778Sdp-arm exit(1); 901819281eeSdp-arm } 902819281eeSdp-arm 903819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]) 904819281eeSdp-arm { 905e0f083a0Sdp-arm struct option *opts = NULL; 906e0f083a0Sdp-arm size_t nr_opts = 0; 907fcab6bbeSdp-arm char outdir[PATH_MAX] = { 0 }; 908e0f083a0Sdp-arm image_desc_t *desc; 909819281eeSdp-arm int fflag = 0; 910819281eeSdp-arm int unpack_all = 1; 911819281eeSdp-arm 912819281eeSdp-arm if (argc < 2) 91385ee2778Sdp-arm unpack_usage(); 914819281eeSdp-arm 915e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 916fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 917e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 918e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 919e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 920819281eeSdp-arm 921819281eeSdp-arm while (1) { 92220f87e78Sdp-arm int c, opt_index = 0; 923819281eeSdp-arm 924fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 925819281eeSdp-arm if (c == -1) 926819281eeSdp-arm break; 927819281eeSdp-arm 928819281eeSdp-arm switch (c) { 929e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 930e0f083a0Sdp-arm image_desc_t *desc; 931e0f083a0Sdp-arm 932e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 933*d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, optarg); 934e0f083a0Sdp-arm unpack_all = 0; 935819281eeSdp-arm break; 936e0f083a0Sdp-arm } 937fcab6bbeSdp-arm case 'b': { 938fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 939fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 940fcab6bbeSdp-arm uuid_t uuid = { 0 }; 941fcab6bbeSdp-arm image_desc_t *desc; 942fcab6bbeSdp-arm 943fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 944fcab6bbeSdp-arm filename, sizeof(filename)); 945fcab6bbeSdp-arm 946fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 947fcab6bbeSdp-arm filename[0] == '\0') 948fcab6bbeSdp-arm unpack_usage(); 949fcab6bbeSdp-arm 950fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 951*d02fcebeSdp-arm if (desc == NULL) { 952fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 953fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 954fcab6bbeSdp-arm add_image_desc(desc); 955fcab6bbeSdp-arm } 956*d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, filename); 957fcab6bbeSdp-arm unpack_all = 0; 958fcab6bbeSdp-arm break; 959fcab6bbeSdp-arm } 960819281eeSdp-arm case 'f': 961819281eeSdp-arm fflag = 1; 962819281eeSdp-arm break; 963819281eeSdp-arm case 'o': 964819281eeSdp-arm snprintf(outdir, sizeof(outdir), "%s", optarg); 965819281eeSdp-arm break; 966819281eeSdp-arm default: 96785ee2778Sdp-arm unpack_usage(); 968819281eeSdp-arm } 969819281eeSdp-arm } 970819281eeSdp-arm argc -= optind; 971819281eeSdp-arm argv += optind; 972e0f083a0Sdp-arm free(opts); 973819281eeSdp-arm 974819281eeSdp-arm if (argc == 0) 97585ee2778Sdp-arm unpack_usage(); 976819281eeSdp-arm 977819281eeSdp-arm parse_fip(argv[0], NULL); 978819281eeSdp-arm 979819281eeSdp-arm if (outdir[0] != '\0') 980819281eeSdp-arm if (chdir(outdir) == -1) 981819281eeSdp-arm log_err("chdir %s", outdir); 982819281eeSdp-arm 983819281eeSdp-arm /* Unpack all specified images. */ 984e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 985fcab6bbeSdp-arm char file[PATH_MAX]; 986b04efcceSdp-arm image_t *image; 987b04efcceSdp-arm 988e0f083a0Sdp-arm if (!unpack_all && desc->action != DO_UNPACK) 989819281eeSdp-arm continue; 990819281eeSdp-arm 991819281eeSdp-arm /* Build filename. */ 992e0f083a0Sdp-arm if (desc->action_arg == NULL) 993819281eeSdp-arm snprintf(file, sizeof(file), "%s.bin", 994e0f083a0Sdp-arm desc->cmdline_name); 995819281eeSdp-arm else 996819281eeSdp-arm snprintf(file, sizeof(file), "%s", 997e0f083a0Sdp-arm desc->action_arg); 998819281eeSdp-arm 999e0f083a0Sdp-arm image = lookup_image_from_uuid(&desc->uuid); 1000b04efcceSdp-arm if (image == NULL) { 1001b04efcceSdp-arm if (!unpack_all) 1002fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 1003819281eeSdp-arm file, argv[0]); 1004819281eeSdp-arm continue; 1005819281eeSdp-arm } 1006819281eeSdp-arm 1007819281eeSdp-arm if (access(file, F_OK) != 0 || fflag) { 1008819281eeSdp-arm if (verbose) 1009819281eeSdp-arm log_dbgx("Unpacking %s", file); 1010b04efcceSdp-arm write_image_to_file(image, file); 1011819281eeSdp-arm } else { 1012819281eeSdp-arm log_warnx("File %s already exists, use --force to overwrite it", 1013819281eeSdp-arm file); 1014819281eeSdp-arm } 1015819281eeSdp-arm } 1016819281eeSdp-arm 1017819281eeSdp-arm free_images(); 1018819281eeSdp-arm return 0; 1019819281eeSdp-arm } 1020819281eeSdp-arm 1021819281eeSdp-arm static void unpack_usage(void) 1022819281eeSdp-arm { 1023819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 1024819281eeSdp-arm 1025fcab6bbeSdp-arm printf("fiptool unpack [--blob uuid=...,file=...] [--force] " 1026fcab6bbeSdp-arm "[--out <path>] [opts] FIP_FILENAME\n"); 1027fcab6bbeSdp-arm printf(" --blob uuid=...,file=...\tUnpack an image with the given UUID " 1028fcab6bbeSdp-arm "to file.\n"); 1029fcab6bbeSdp-arm printf(" --force\t\t\tIf the output file already exists, use --force to " 1030819281eeSdp-arm "overwrite it.\n"); 1031fcab6bbeSdp-arm printf(" --out path\t\t\tSet the output directory path.\n"); 1032819281eeSdp-arm fputc('\n', stderr); 1033819281eeSdp-arm printf("Specific images are unpacked with the following options:\n"); 1034819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1035819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 1036819281eeSdp-arm toc_entry->name); 1037819281eeSdp-arm fputc('\n', stderr); 1038819281eeSdp-arm printf("If no options are provided, all images will be unpacked.\n"); 103985ee2778Sdp-arm exit(1); 1040819281eeSdp-arm } 1041819281eeSdp-arm 1042819281eeSdp-arm static int remove_cmd(int argc, char *argv[]) 1043819281eeSdp-arm { 1044e0f083a0Sdp-arm struct option *opts = NULL; 1045e0f083a0Sdp-arm size_t nr_opts = 0; 1046fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 1047819281eeSdp-arm fip_toc_header_t toc_header; 1048e0f083a0Sdp-arm image_desc_t *desc; 1049819281eeSdp-arm int fflag = 0; 1050819281eeSdp-arm 1051819281eeSdp-arm if (argc < 2) 105285ee2778Sdp-arm remove_usage(); 1053819281eeSdp-arm 1054e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, no_argument); 1055fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 1056e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 1057e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 1058e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 1059819281eeSdp-arm 1060819281eeSdp-arm while (1) { 106120f87e78Sdp-arm int c, opt_index = 0; 1062819281eeSdp-arm 1063fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 1064819281eeSdp-arm if (c == -1) 1065819281eeSdp-arm break; 1066819281eeSdp-arm 1067819281eeSdp-arm switch (c) { 1068e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 1069e0f083a0Sdp-arm image_desc_t *desc; 1070e0f083a0Sdp-arm 1071e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 1072*d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1073819281eeSdp-arm break; 1074e0f083a0Sdp-arm } 1075fcab6bbeSdp-arm case 'b': { 1076fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 1077fcab6bbeSdp-arm uuid_t uuid = { 0 }; 1078fcab6bbeSdp-arm image_desc_t *desc; 1079fcab6bbeSdp-arm 1080fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 1081fcab6bbeSdp-arm filename, sizeof(filename)); 1082fcab6bbeSdp-arm 1083fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0) 1084fcab6bbeSdp-arm remove_usage(); 1085fcab6bbeSdp-arm 1086fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 1087*d02fcebeSdp-arm if (desc == NULL) { 1088fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 1089fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 1090fcab6bbeSdp-arm add_image_desc(desc); 1091fcab6bbeSdp-arm } 1092*d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1093fcab6bbeSdp-arm break; 1094fcab6bbeSdp-arm } 1095819281eeSdp-arm case 'f': 1096819281eeSdp-arm fflag = 1; 1097819281eeSdp-arm break; 1098819281eeSdp-arm case 'o': 1099819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 1100819281eeSdp-arm break; 1101819281eeSdp-arm default: 110285ee2778Sdp-arm remove_usage(); 1103819281eeSdp-arm } 1104819281eeSdp-arm } 1105819281eeSdp-arm argc -= optind; 1106819281eeSdp-arm argv += optind; 1107e0f083a0Sdp-arm free(opts); 1108819281eeSdp-arm 1109819281eeSdp-arm if (argc == 0) 111085ee2778Sdp-arm remove_usage(); 1111819281eeSdp-arm 1112819281eeSdp-arm if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag) 1113819281eeSdp-arm log_errx("File %s already exists, use --force to overwrite it", 1114819281eeSdp-arm outfile); 1115819281eeSdp-arm 1116819281eeSdp-arm if (outfile[0] == '\0') 1117819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 1118819281eeSdp-arm 1119819281eeSdp-arm parse_fip(argv[0], &toc_header); 1120819281eeSdp-arm 1121e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 1122b04efcceSdp-arm image_t *image; 1123b04efcceSdp-arm 1124e0f083a0Sdp-arm if (desc->action != DO_REMOVE) 1125819281eeSdp-arm continue; 1126fcab6bbeSdp-arm 1127e0f083a0Sdp-arm image = lookup_image_from_uuid(&desc->uuid); 1128b04efcceSdp-arm if (image != NULL) { 1129819281eeSdp-arm if (verbose) 1130fcab6bbeSdp-arm log_dbgx("Removing %s", 1131e0f083a0Sdp-arm desc->cmdline_name); 1132b04efcceSdp-arm remove_image(image); 1133819281eeSdp-arm } else { 1134fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 1135e0f083a0Sdp-arm desc->cmdline_name, argv[0]); 1136819281eeSdp-arm } 1137819281eeSdp-arm } 1138819281eeSdp-arm 1139819281eeSdp-arm pack_images(outfile, toc_header.flags); 1140819281eeSdp-arm free_images(); 1141819281eeSdp-arm return 0; 1142819281eeSdp-arm } 1143819281eeSdp-arm 1144819281eeSdp-arm static void remove_usage(void) 1145819281eeSdp-arm { 1146819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 1147819281eeSdp-arm 1148fcab6bbeSdp-arm printf("fiptool remove [--blob uuid=...] [--force] " 1149fcab6bbeSdp-arm "[--out FIP_FILENAME] [opts] FIP_FILENAME\n"); 1150fcab6bbeSdp-arm printf(" --blob uuid=...\tRemove an image with the given UUID.\n"); 1151819281eeSdp-arm printf(" --force\t\tIf the output FIP file already exists, use --force to " 1152819281eeSdp-arm "overwrite it.\n"); 1153819281eeSdp-arm printf(" --out FIP_FILENAME\tSet an alternative output FIP file.\n"); 1154819281eeSdp-arm fputc('\n', stderr); 1155819281eeSdp-arm printf("Specific images are removed with the following options:\n"); 1156819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1157819281eeSdp-arm printf(" --%-16s\t%s\n", toc_entry->cmdline_name, 1158819281eeSdp-arm toc_entry->name); 115985ee2778Sdp-arm exit(1); 1160819281eeSdp-arm } 1161819281eeSdp-arm 1162819281eeSdp-arm static int version_cmd(int argc, char *argv[]) 1163819281eeSdp-arm { 1164819281eeSdp-arm #ifdef VERSION 1165819281eeSdp-arm puts(VERSION); 1166819281eeSdp-arm #else 1167819281eeSdp-arm /* If built from fiptool directory, VERSION is not set. */ 1168819281eeSdp-arm puts("Unknown version"); 1169819281eeSdp-arm #endif 1170819281eeSdp-arm return 0; 1171819281eeSdp-arm } 1172819281eeSdp-arm 1173819281eeSdp-arm static void version_usage(void) 1174819281eeSdp-arm { 1175819281eeSdp-arm printf("fiptool version\n"); 117685ee2778Sdp-arm exit(1); 1177819281eeSdp-arm } 1178819281eeSdp-arm 1179819281eeSdp-arm static int help_cmd(int argc, char *argv[]) 1180819281eeSdp-arm { 1181819281eeSdp-arm int i; 1182819281eeSdp-arm 1183819281eeSdp-arm if (argc < 2) 1184819281eeSdp-arm usage(); 1185819281eeSdp-arm argc--, argv++; 1186819281eeSdp-arm 1187819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1188819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0 && 118985ee2778Sdp-arm cmds[i].usage != NULL) 1190819281eeSdp-arm cmds[i].usage(); 1191819281eeSdp-arm } 1192819281eeSdp-arm if (i == NELEM(cmds)) 1193819281eeSdp-arm printf("No help for subcommand '%s'\n", argv[0]); 1194819281eeSdp-arm return 0; 1195819281eeSdp-arm } 1196819281eeSdp-arm 1197819281eeSdp-arm static void usage(void) 1198819281eeSdp-arm { 1199819281eeSdp-arm printf("usage: [--verbose] fiptool <command> [<args>]\n"); 1200819281eeSdp-arm printf("Global options supported:\n"); 1201819281eeSdp-arm printf(" --verbose\tEnable verbose output for all commands.\n"); 1202819281eeSdp-arm fputc('\n', stderr); 1203819281eeSdp-arm printf("Commands supported:\n"); 1204819281eeSdp-arm printf(" info\t\tList images contained in FIP.\n"); 1205819281eeSdp-arm printf(" create\tCreate a new FIP with the given images.\n"); 1206819281eeSdp-arm printf(" update\tUpdate an existing FIP with the given images.\n"); 1207819281eeSdp-arm printf(" unpack\tUnpack images from FIP.\n"); 1208819281eeSdp-arm printf(" remove\tRemove images from FIP.\n"); 1209819281eeSdp-arm printf(" version\tShow fiptool version.\n"); 1210819281eeSdp-arm printf(" help\t\tShow help for given command.\n"); 1211819281eeSdp-arm exit(1); 1212819281eeSdp-arm } 1213819281eeSdp-arm 1214819281eeSdp-arm int main(int argc, char *argv[]) 1215819281eeSdp-arm { 1216819281eeSdp-arm int i, ret = 0; 1217819281eeSdp-arm 1218cc672bb2Sdp-arm while (1) { 1219cc672bb2Sdp-arm int c, opt_index = 0; 1220cc672bb2Sdp-arm static struct option opts[] = { 1221cc672bb2Sdp-arm { "verbose", no_argument, NULL, 'v' }, 1222cc672bb2Sdp-arm { NULL, no_argument, NULL, 0 } 1223cc672bb2Sdp-arm }; 1224819281eeSdp-arm 1225cc672bb2Sdp-arm /* 1226cc672bb2Sdp-arm * Set POSIX mode so getopt stops at the first non-option 1227cc672bb2Sdp-arm * which is the subcommand. 1228cc672bb2Sdp-arm */ 1229cc672bb2Sdp-arm c = getopt_long(argc, argv, "+v", opts, &opt_index); 1230cc672bb2Sdp-arm if (c == -1) 1231cc672bb2Sdp-arm break; 1232cc672bb2Sdp-arm 1233cc672bb2Sdp-arm switch (c) { 1234cc672bb2Sdp-arm case 'v': 1235819281eeSdp-arm verbose = 1; 1236cc672bb2Sdp-arm break; 1237cc672bb2Sdp-arm default: 1238c9cb4089SMasahiro Yamada usage(); 1239819281eeSdp-arm } 1240cc672bb2Sdp-arm } 1241cc672bb2Sdp-arm argc -= optind; 1242cc672bb2Sdp-arm argv += optind; 1243cc672bb2Sdp-arm /* Reset optind for subsequent getopt processing. */ 1244cc672bb2Sdp-arm optind = 0; 1245cc672bb2Sdp-arm 1246cc672bb2Sdp-arm if (argc == 0) 1247cc672bb2Sdp-arm usage(); 1248819281eeSdp-arm 1249e0f083a0Sdp-arm fill_image_descs(); 1250819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1251819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0) { 1252819281eeSdp-arm ret = cmds[i].handler(argc, argv); 1253819281eeSdp-arm break; 1254819281eeSdp-arm } 1255819281eeSdp-arm } 1256819281eeSdp-arm if (i == NELEM(cmds)) 1257819281eeSdp-arm usage(); 1258e0f083a0Sdp-arm free_image_descs(); 1259819281eeSdp-arm return ret; 1260819281eeSdp-arm } 1261