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 531c75d5dfSMasahiro Yamada #define OPT_ALIGN 2 54819281eeSdp-arm 55819281eeSdp-arm static int info_cmd(int argc, char *argv[]); 56819281eeSdp-arm static void info_usage(void); 57819281eeSdp-arm static int create_cmd(int argc, char *argv[]); 58819281eeSdp-arm static void create_usage(void); 59819281eeSdp-arm static int update_cmd(int argc, char *argv[]); 60819281eeSdp-arm static void update_usage(void); 61819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]); 62819281eeSdp-arm static void unpack_usage(void); 63819281eeSdp-arm static int remove_cmd(int argc, char *argv[]); 64819281eeSdp-arm static void remove_usage(void); 65819281eeSdp-arm static int version_cmd(int argc, char *argv[]); 66819281eeSdp-arm static void version_usage(void); 67819281eeSdp-arm static int help_cmd(int argc, char *argv[]); 68819281eeSdp-arm static void usage(void); 69819281eeSdp-arm 70819281eeSdp-arm /* Available subcommands. */ 71819281eeSdp-arm static cmd_t cmds[] = { 72819281eeSdp-arm { .name = "info", .handler = info_cmd, .usage = info_usage }, 73819281eeSdp-arm { .name = "create", .handler = create_cmd, .usage = create_usage }, 74819281eeSdp-arm { .name = "update", .handler = update_cmd, .usage = update_usage }, 75819281eeSdp-arm { .name = "unpack", .handler = unpack_cmd, .usage = unpack_usage }, 76819281eeSdp-arm { .name = "remove", .handler = remove_cmd, .usage = remove_usage }, 77819281eeSdp-arm { .name = "version", .handler = version_cmd, .usage = version_usage }, 78819281eeSdp-arm { .name = "help", .handler = help_cmd, .usage = NULL }, 79819281eeSdp-arm }; 80819281eeSdp-arm 81e0f083a0Sdp-arm static image_desc_t *image_desc_head; 82e0f083a0Sdp-arm static size_t nr_image_descs; 83819281eeSdp-arm static uuid_t uuid_null = { 0 }; 84819281eeSdp-arm static int verbose; 85819281eeSdp-arm 8660b499feSdp-arm static void vlog(int prio, const char *msg, va_list ap) 87819281eeSdp-arm { 88819281eeSdp-arm char *prefix[] = { "DEBUG", "WARN", "ERROR" }; 89819281eeSdp-arm 90819281eeSdp-arm fprintf(stderr, "%s: ", prefix[prio]); 91819281eeSdp-arm vfprintf(stderr, msg, ap); 92819281eeSdp-arm fputc('\n', stderr); 93819281eeSdp-arm } 94819281eeSdp-arm 9560b499feSdp-arm static void log_dbgx(const char *msg, ...) 96819281eeSdp-arm { 97819281eeSdp-arm va_list ap; 98819281eeSdp-arm 99819281eeSdp-arm va_start(ap, msg); 100819281eeSdp-arm vlog(LOG_DBG, msg, ap); 101819281eeSdp-arm va_end(ap); 102819281eeSdp-arm } 103819281eeSdp-arm 10460b499feSdp-arm static void log_warnx(const char *msg, ...) 105819281eeSdp-arm { 106819281eeSdp-arm va_list ap; 107819281eeSdp-arm 108819281eeSdp-arm va_start(ap, msg); 109819281eeSdp-arm vlog(LOG_WARN, msg, ap); 110819281eeSdp-arm va_end(ap); 111819281eeSdp-arm } 112819281eeSdp-arm 11360b499feSdp-arm static void log_err(const char *msg, ...) 114819281eeSdp-arm { 115819281eeSdp-arm char buf[512]; 116819281eeSdp-arm va_list ap; 117819281eeSdp-arm 118819281eeSdp-arm va_start(ap, msg); 119819281eeSdp-arm snprintf(buf, sizeof(buf), "%s: %s", msg, strerror(errno)); 120819281eeSdp-arm vlog(LOG_ERR, buf, ap); 121819281eeSdp-arm va_end(ap); 122819281eeSdp-arm exit(1); 123819281eeSdp-arm } 124819281eeSdp-arm 12560b499feSdp-arm static void log_errx(const char *msg, ...) 126819281eeSdp-arm { 127819281eeSdp-arm va_list ap; 128819281eeSdp-arm 129819281eeSdp-arm va_start(ap, msg); 130819281eeSdp-arm vlog(LOG_ERR, msg, ap); 131819281eeSdp-arm va_end(ap); 132819281eeSdp-arm exit(1); 133819281eeSdp-arm } 134819281eeSdp-arm 135a22f6285Sdp-arm static char *xstrdup(const char *s, const char *msg) 136a22f6285Sdp-arm { 137a22f6285Sdp-arm char *d; 138a22f6285Sdp-arm 139a22f6285Sdp-arm d = strdup(s); 140a22f6285Sdp-arm if (d == NULL) 1419fc9ff1fSdp-arm log_errx("strdup: %s", msg); 142a22f6285Sdp-arm return d; 143a22f6285Sdp-arm } 144a22f6285Sdp-arm 145a22f6285Sdp-arm static void *xmalloc(size_t size, const char *msg) 146a22f6285Sdp-arm { 147a22f6285Sdp-arm void *d; 148a22f6285Sdp-arm 149a22f6285Sdp-arm d = malloc(size); 150a22f6285Sdp-arm if (d == NULL) 1519fc9ff1fSdp-arm log_errx("malloc: %s", msg); 152a22f6285Sdp-arm return d; 153a22f6285Sdp-arm } 154a22f6285Sdp-arm 155696ccba6SMasahiro Yamada static void *xzalloc(size_t size, const char *msg) 156696ccba6SMasahiro Yamada { 157696ccba6SMasahiro Yamada return memset(xmalloc(size, msg), 0, size); 158696ccba6SMasahiro Yamada } 159696ccba6SMasahiro Yamada 160a1da83f5SMasahiro Yamada static void xfwrite(void *buf, size_t size, FILE *fp, const char *filename) 161a1da83f5SMasahiro Yamada { 162a1da83f5SMasahiro Yamada if (fwrite(buf, 1, size, fp) != size) 163a1da83f5SMasahiro Yamada log_errx("Failed to write %s", filename); 164a1da83f5SMasahiro Yamada } 165a1da83f5SMasahiro Yamada 166e0f083a0Sdp-arm static image_desc_t *new_image_desc(const uuid_t *uuid, 167e0f083a0Sdp-arm const char *name, const char *cmdline_name) 168e0f083a0Sdp-arm { 169e0f083a0Sdp-arm image_desc_t *desc; 170e0f083a0Sdp-arm 171696ccba6SMasahiro Yamada desc = xzalloc(sizeof(*desc), 172e0f083a0Sdp-arm "failed to allocate memory for image descriptor"); 173e0f083a0Sdp-arm memcpy(&desc->uuid, uuid, sizeof(uuid_t)); 174e0f083a0Sdp-arm desc->name = xstrdup(name, 175e0f083a0Sdp-arm "failed to allocate memory for image name"); 176e0f083a0Sdp-arm desc->cmdline_name = xstrdup(cmdline_name, 177e0f083a0Sdp-arm "failed to allocate memory for image command line name"); 178e0f083a0Sdp-arm desc->action = DO_UNSPEC; 179e0f083a0Sdp-arm return desc; 180e0f083a0Sdp-arm } 181e0f083a0Sdp-arm 182d02fcebeSdp-arm static void set_image_desc_action(image_desc_t *desc, int action, 183d02fcebeSdp-arm const char *arg) 184d02fcebeSdp-arm { 185d02fcebeSdp-arm assert(desc != NULL); 186d02fcebeSdp-arm 187d02fcebeSdp-arm if (desc->action_arg != DO_UNSPEC) 188d02fcebeSdp-arm free(desc->action_arg); 189d02fcebeSdp-arm desc->action = action; 190d02fcebeSdp-arm desc->action_arg = NULL; 191d02fcebeSdp-arm if (arg != NULL) 192d02fcebeSdp-arm desc->action_arg = xstrdup(arg, 193d02fcebeSdp-arm "failed to allocate memory for argument"); 194d02fcebeSdp-arm } 195d02fcebeSdp-arm 196e0f083a0Sdp-arm static void free_image_desc(image_desc_t *desc) 197e0f083a0Sdp-arm { 198e0f083a0Sdp-arm free(desc->name); 199e0f083a0Sdp-arm free(desc->cmdline_name); 200e0f083a0Sdp-arm free(desc->action_arg); 201b9589fe5Sdp-arm free(desc->image); 202e0f083a0Sdp-arm free(desc); 203e0f083a0Sdp-arm } 204e0f083a0Sdp-arm 205e0f083a0Sdp-arm static void add_image_desc(image_desc_t *desc) 206e0f083a0Sdp-arm { 20711c0a4ffSMasahiro Yamada image_desc_t **p = &image_desc_head; 20811c0a4ffSMasahiro Yamada 20911c0a4ffSMasahiro Yamada while (*p) 21011c0a4ffSMasahiro Yamada p = &(*p)->next; 21111c0a4ffSMasahiro Yamada 212e9e0d287SMasahiro Yamada assert(*p == NULL); 21311c0a4ffSMasahiro Yamada *p = desc; 214e0f083a0Sdp-arm nr_image_descs++; 215e0f083a0Sdp-arm } 216e0f083a0Sdp-arm 217e0f083a0Sdp-arm static void free_image_descs(void) 218e0f083a0Sdp-arm { 219e0f083a0Sdp-arm image_desc_t *desc = image_desc_head, *tmp; 220e0f083a0Sdp-arm 221e0f083a0Sdp-arm while (desc != NULL) { 222e0f083a0Sdp-arm tmp = desc->next; 223e0f083a0Sdp-arm free_image_desc(desc); 224e0f083a0Sdp-arm desc = tmp; 225e0f083a0Sdp-arm nr_image_descs--; 226e0f083a0Sdp-arm } 227e0f083a0Sdp-arm assert(nr_image_descs == 0); 228e0f083a0Sdp-arm } 229e0f083a0Sdp-arm 230e0f083a0Sdp-arm static void fill_image_descs(void) 231e0f083a0Sdp-arm { 232e0f083a0Sdp-arm toc_entry_t *toc_entry; 233e0f083a0Sdp-arm 234e0f083a0Sdp-arm for (toc_entry = toc_entries; 235e0f083a0Sdp-arm toc_entry->cmdline_name != NULL; 236e0f083a0Sdp-arm toc_entry++) { 237e0f083a0Sdp-arm image_desc_t *desc; 238e0f083a0Sdp-arm 239e0f083a0Sdp-arm desc = new_image_desc(&toc_entry->uuid, 240e0f083a0Sdp-arm toc_entry->name, 241e0f083a0Sdp-arm toc_entry->cmdline_name); 242e0f083a0Sdp-arm add_image_desc(desc); 243e0f083a0Sdp-arm } 244e0f083a0Sdp-arm } 245e0f083a0Sdp-arm 246e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid) 247819281eeSdp-arm { 248e0f083a0Sdp-arm image_desc_t *desc; 249819281eeSdp-arm 250e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 251e0f083a0Sdp-arm if (memcmp(&desc->uuid, uuid, sizeof(uuid_t)) == 0) 252e0f083a0Sdp-arm return desc; 253819281eeSdp-arm return NULL; 254819281eeSdp-arm } 255819281eeSdp-arm 256e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_opt(const char *opt) 257e0f083a0Sdp-arm { 258e0f083a0Sdp-arm image_desc_t *desc; 259e0f083a0Sdp-arm 260e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 261e0f083a0Sdp-arm if (strcmp(desc->cmdline_name, opt) == 0) 262e0f083a0Sdp-arm return desc; 263e0f083a0Sdp-arm return NULL; 264e0f083a0Sdp-arm } 265e0f083a0Sdp-arm 266fcab6bbeSdp-arm static void uuid_to_str(char *s, size_t len, const uuid_t *u) 267fcab6bbeSdp-arm { 268fcab6bbeSdp-arm assert(len >= (_UUID_STR_LEN + 1)); 269fcab6bbeSdp-arm 270fcab6bbeSdp-arm snprintf(s, len, "%08X-%04X-%04X-%04X-%04X%04X%04X", 271fcab6bbeSdp-arm u->time_low, 272fcab6bbeSdp-arm u->time_mid, 273fcab6bbeSdp-arm u->time_hi_and_version, 274fcab6bbeSdp-arm ((uint16_t)u->clock_seq_hi_and_reserved << 8) | u->clock_seq_low, 275fcab6bbeSdp-arm ((uint16_t)u->node[0] << 8) | u->node[1], 276fcab6bbeSdp-arm ((uint16_t)u->node[2] << 8) | u->node[3], 277fcab6bbeSdp-arm ((uint16_t)u->node[4] << 8) | u->node[5]); 278fcab6bbeSdp-arm } 279fcab6bbeSdp-arm 280fcab6bbeSdp-arm static void uuid_from_str(uuid_t *u, const char *s) 281fcab6bbeSdp-arm { 282fcab6bbeSdp-arm int n; 283fcab6bbeSdp-arm 284fcab6bbeSdp-arm if (s == NULL) 285fcab6bbeSdp-arm log_errx("UUID cannot be NULL"); 286fcab6bbeSdp-arm if (strlen(s) != _UUID_STR_LEN) 287fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 288fcab6bbeSdp-arm 289fcab6bbeSdp-arm n = sscanf(s, 290fcab6bbeSdp-arm "%8x-%4hx-%4hx-%2hhx%2hhx-%2hhx%2hhx%2hhx%2hhx%2hhx%2hhx", 291fcab6bbeSdp-arm &u->time_low, &u->time_mid, &u->time_hi_and_version, 292fcab6bbeSdp-arm &u->clock_seq_hi_and_reserved, &u->clock_seq_low, &u->node[0], 293fcab6bbeSdp-arm &u->node[1], &u->node[2], &u->node[3], &u->node[4], &u->node[5]); 294fcab6bbeSdp-arm /* 295fcab6bbeSdp-arm * Given the format specifier above, we expect 11 items to be scanned 296fcab6bbeSdp-arm * for a properly formatted UUID. 297fcab6bbeSdp-arm */ 298fcab6bbeSdp-arm if (n != 11) 299fcab6bbeSdp-arm log_errx("Invalid UUID: %s", s); 300fcab6bbeSdp-arm } 301fcab6bbeSdp-arm 30260b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out) 303819281eeSdp-arm { 304819281eeSdp-arm struct stat st; 305819281eeSdp-arm FILE *fp; 306819281eeSdp-arm char *buf, *bufend; 307819281eeSdp-arm fip_toc_header_t *toc_header; 308819281eeSdp-arm fip_toc_entry_t *toc_entry; 309819281eeSdp-arm int terminated = 0; 310819281eeSdp-arm 311819281eeSdp-arm fp = fopen(filename, "r"); 312819281eeSdp-arm if (fp == NULL) 313819281eeSdp-arm log_err("fopen %s", filename); 314819281eeSdp-arm 315819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 316819281eeSdp-arm log_err("fstat %s", filename); 317819281eeSdp-arm 318a22f6285Sdp-arm buf = xmalloc(st.st_size, "failed to load file into memory"); 319819281eeSdp-arm if (fread(buf, 1, st.st_size, fp) != st.st_size) 320819281eeSdp-arm log_errx("Failed to read %s", filename); 321819281eeSdp-arm bufend = buf + st.st_size; 322819281eeSdp-arm fclose(fp); 323819281eeSdp-arm 324819281eeSdp-arm if (st.st_size < sizeof(fip_toc_header_t)) 325819281eeSdp-arm log_errx("FIP %s is truncated", filename); 326819281eeSdp-arm 327819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 328819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 329819281eeSdp-arm 330819281eeSdp-arm if (toc_header->name != TOC_HEADER_NAME) 331819281eeSdp-arm log_errx("%s is not a FIP file", filename); 332819281eeSdp-arm 333819281eeSdp-arm /* Return the ToC header if the caller wants it. */ 334819281eeSdp-arm if (toc_header_out != NULL) 335819281eeSdp-arm *toc_header_out = *toc_header; 336819281eeSdp-arm 337819281eeSdp-arm /* Walk through each ToC entry in the file. */ 338819281eeSdp-arm while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) { 339fcab6bbeSdp-arm image_t *image; 340fcab6bbeSdp-arm image_desc_t *desc; 341fcab6bbeSdp-arm 342819281eeSdp-arm /* Found the ToC terminator, we are done. */ 343819281eeSdp-arm if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) { 344819281eeSdp-arm terminated = 1; 345819281eeSdp-arm break; 346819281eeSdp-arm } 347819281eeSdp-arm 348819281eeSdp-arm /* 349819281eeSdp-arm * Build a new image out of the ToC entry and add it to the 350819281eeSdp-arm * table of images. 351819281eeSdp-arm */ 35211c0a4ffSMasahiro Yamada image = xzalloc(sizeof(*image), 353a22f6285Sdp-arm "failed to allocate memory for image"); 35465caa3d0SMasahiro Yamada image->toc_e = *toc_entry; 355a22f6285Sdp-arm image->buffer = xmalloc(toc_entry->size, 356a22f6285Sdp-arm "failed to allocate image buffer, is FIP file corrupted?"); 357819281eeSdp-arm /* Overflow checks before memory copy. */ 358819281eeSdp-arm if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address) 359819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 360819281eeSdp-arm if (toc_entry->size + toc_entry->offset_address > st.st_size) 361819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 362819281eeSdp-arm 363819281eeSdp-arm memcpy(image->buffer, buf + toc_entry->offset_address, 364819281eeSdp-arm toc_entry->size); 365819281eeSdp-arm 366fcab6bbeSdp-arm /* If this is an unknown image, create a descriptor for it. */ 36765caa3d0SMasahiro Yamada desc = lookup_image_desc_from_uuid(&toc_entry->uuid); 368fcab6bbeSdp-arm if (desc == NULL) { 369fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 370fcab6bbeSdp-arm 37165caa3d0SMasahiro Yamada uuid_to_str(name, sizeof(name), &toc_entry->uuid); 372fcab6bbeSdp-arm snprintf(filename, sizeof(filename), "%s%s", 373fcab6bbeSdp-arm name, ".bin"); 37465caa3d0SMasahiro Yamada desc = new_image_desc(&toc_entry->uuid, name, "blob"); 375fcab6bbeSdp-arm desc->action = DO_UNPACK; 376fcab6bbeSdp-arm desc->action_arg = xstrdup(filename, 377fcab6bbeSdp-arm "failed to allocate memory for blob filename"); 378fcab6bbeSdp-arm add_image_desc(desc); 379fcab6bbeSdp-arm } 380fcab6bbeSdp-arm 381b9589fe5Sdp-arm assert(desc->image == NULL); 382b9589fe5Sdp-arm desc->image = image; 383819281eeSdp-arm 384819281eeSdp-arm toc_entry++; 385819281eeSdp-arm } 386819281eeSdp-arm 387819281eeSdp-arm if (terminated == 0) 388819281eeSdp-arm log_errx("FIP %s does not have a ToC terminator entry", 389819281eeSdp-arm filename); 390819281eeSdp-arm free(buf); 391819281eeSdp-arm return 0; 392819281eeSdp-arm } 393819281eeSdp-arm 39460b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename) 395819281eeSdp-arm { 396819281eeSdp-arm struct stat st; 397819281eeSdp-arm image_t *image; 398819281eeSdp-arm FILE *fp; 399819281eeSdp-arm 400b04efcceSdp-arm assert(uuid != NULL); 401b04efcceSdp-arm 402819281eeSdp-arm fp = fopen(filename, "r"); 403819281eeSdp-arm if (fp == NULL) 404819281eeSdp-arm log_err("fopen %s", filename); 405819281eeSdp-arm 406819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 407819281eeSdp-arm log_errx("fstat %s", filename); 408819281eeSdp-arm 40911c0a4ffSMasahiro Yamada image = xzalloc(sizeof(*image), "failed to allocate memory for image"); 41065caa3d0SMasahiro Yamada image->toc_e.uuid = *uuid; 411a22f6285Sdp-arm image->buffer = xmalloc(st.st_size, "failed to allocate image buffer"); 412819281eeSdp-arm if (fread(image->buffer, 1, st.st_size, fp) != st.st_size) 413819281eeSdp-arm log_errx("Failed to read %s", filename); 41465caa3d0SMasahiro Yamada image->toc_e.size = st.st_size; 415819281eeSdp-arm 416819281eeSdp-arm fclose(fp); 417819281eeSdp-arm return image; 418819281eeSdp-arm } 419819281eeSdp-arm 42060b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename) 421819281eeSdp-arm { 422819281eeSdp-arm FILE *fp; 423819281eeSdp-arm 424819281eeSdp-arm fp = fopen(filename, "w"); 425819281eeSdp-arm if (fp == NULL) 426819281eeSdp-arm log_err("fopen"); 42765caa3d0SMasahiro Yamada xfwrite(image->buffer, image->toc_e.size, fp, filename); 428819281eeSdp-arm fclose(fp); 429819281eeSdp-arm return 0; 430819281eeSdp-arm } 431819281eeSdp-arm 432e0f083a0Sdp-arm static struct option *add_opt(struct option *opts, size_t *nr_opts, 433e0f083a0Sdp-arm const char *name, int has_arg, int val) 434819281eeSdp-arm { 435e0f083a0Sdp-arm opts = realloc(opts, (*nr_opts + 1) * sizeof(*opts)); 436e0f083a0Sdp-arm if (opts == NULL) 437e0f083a0Sdp-arm log_err("realloc"); 438e0f083a0Sdp-arm opts[*nr_opts].name = name; 439e0f083a0Sdp-arm opts[*nr_opts].has_arg = has_arg; 440e0f083a0Sdp-arm opts[*nr_opts].flag = NULL; 441e0f083a0Sdp-arm opts[*nr_opts].val = val; 442e0f083a0Sdp-arm ++*nr_opts; 443e0f083a0Sdp-arm return opts; 444819281eeSdp-arm } 445819281eeSdp-arm 446e0f083a0Sdp-arm static struct option *fill_common_opts(struct option *opts, size_t *nr_opts, 447e0f083a0Sdp-arm int has_arg) 448819281eeSdp-arm { 449e0f083a0Sdp-arm image_desc_t *desc; 450e0f083a0Sdp-arm 451e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 452e0f083a0Sdp-arm opts = add_opt(opts, nr_opts, desc->cmdline_name, has_arg, 453e0f083a0Sdp-arm OPT_TOC_ENTRY); 454e0f083a0Sdp-arm return opts; 455819281eeSdp-arm } 456819281eeSdp-arm 457e0f083a0Sdp-arm static void md_print(const unsigned char *md, size_t len) 4589df69ba3Sdp-arm { 4599df69ba3Sdp-arm size_t i; 4609df69ba3Sdp-arm 4619df69ba3Sdp-arm for (i = 0; i < len; i++) 4629df69ba3Sdp-arm printf("%02x", md[i]); 4639df69ba3Sdp-arm } 4649df69ba3Sdp-arm 465819281eeSdp-arm static int info_cmd(int argc, char *argv[]) 466819281eeSdp-arm { 467b9589fe5Sdp-arm image_desc_t *desc; 468819281eeSdp-arm fip_toc_header_t toc_header; 469819281eeSdp-arm 470819281eeSdp-arm if (argc != 2) 47185ee2778Sdp-arm info_usage(); 472819281eeSdp-arm argc--, argv++; 473819281eeSdp-arm 474819281eeSdp-arm parse_fip(argv[0], &toc_header); 475819281eeSdp-arm 476819281eeSdp-arm if (verbose) { 477819281eeSdp-arm log_dbgx("toc_header[name]: 0x%llX", 478819281eeSdp-arm (unsigned long long)toc_header.name); 479819281eeSdp-arm log_dbgx("toc_header[serial_number]: 0x%llX", 480819281eeSdp-arm (unsigned long long)toc_header.serial_number); 481819281eeSdp-arm log_dbgx("toc_header[flags]: 0x%llX", 482819281eeSdp-arm (unsigned long long)toc_header.flags); 483819281eeSdp-arm } 484819281eeSdp-arm 485b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 486b9589fe5Sdp-arm image_t *image = desc->image; 487b04efcceSdp-arm 488b9589fe5Sdp-arm if (image == NULL) 489b9589fe5Sdp-arm continue; 49065caa3d0SMasahiro Yamada printf("%s: offset=0x%llX, size=0x%llX, cmdline=\"--%s\"", 49165caa3d0SMasahiro Yamada desc->name, 49265caa3d0SMasahiro Yamada (unsigned long long)image->toc_e.offset_address, 49365caa3d0SMasahiro Yamada (unsigned long long)image->toc_e.size, 494e0f083a0Sdp-arm desc->cmdline_name); 4959df69ba3Sdp-arm if (verbose) { 4969df69ba3Sdp-arm unsigned char md[SHA256_DIGEST_LENGTH]; 4979df69ba3Sdp-arm 49865caa3d0SMasahiro Yamada SHA256(image->buffer, image->toc_e.size, md); 4999df69ba3Sdp-arm printf(", sha256="); 5009df69ba3Sdp-arm md_print(md, sizeof(md)); 5019df69ba3Sdp-arm } 502819281eeSdp-arm putchar('\n'); 503819281eeSdp-arm } 504819281eeSdp-arm 505819281eeSdp-arm return 0; 506819281eeSdp-arm } 507819281eeSdp-arm 508819281eeSdp-arm static void info_usage(void) 509819281eeSdp-arm { 510819281eeSdp-arm printf("fiptool info FIP_FILENAME\n"); 51185ee2778Sdp-arm exit(1); 512819281eeSdp-arm } 513819281eeSdp-arm 5141c75d5dfSMasahiro Yamada static int pack_images(const char *filename, uint64_t toc_flags, unsigned long align) 515819281eeSdp-arm { 516819281eeSdp-arm FILE *fp; 517b9589fe5Sdp-arm image_desc_t *desc; 518819281eeSdp-arm fip_toc_header_t *toc_header; 519819281eeSdp-arm fip_toc_entry_t *toc_entry; 520819281eeSdp-arm char *buf; 52165caa3d0SMasahiro Yamada uint64_t entry_offset, buf_size, payload_size = 0; 522b9589fe5Sdp-arm size_t nr_images = 0; 523b9589fe5Sdp-arm 524b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 525b9589fe5Sdp-arm if (desc->image != NULL) 526b9589fe5Sdp-arm nr_images++; 527819281eeSdp-arm 528819281eeSdp-arm buf_size = sizeof(fip_toc_header_t) + 529819281eeSdp-arm sizeof(fip_toc_entry_t) * (nr_images + 1); 530819281eeSdp-arm buf = calloc(1, buf_size); 531819281eeSdp-arm if (buf == NULL) 532819281eeSdp-arm log_err("calloc"); 533819281eeSdp-arm 534819281eeSdp-arm /* Build up header and ToC entries from the image table. */ 535819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 536819281eeSdp-arm toc_header->name = TOC_HEADER_NAME; 537819281eeSdp-arm toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER; 538819281eeSdp-arm toc_header->flags = toc_flags; 539819281eeSdp-arm 540819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 541819281eeSdp-arm 542819281eeSdp-arm entry_offset = buf_size; 543b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 544b9589fe5Sdp-arm image_t *image = desc->image; 545b9589fe5Sdp-arm 546b9589fe5Sdp-arm if (image == NULL) 547b9589fe5Sdp-arm continue; 54865caa3d0SMasahiro Yamada payload_size += image->toc_e.size; 5491c75d5dfSMasahiro Yamada entry_offset = (entry_offset + align - 1) & ~(align - 1); 55065caa3d0SMasahiro Yamada image->toc_e.offset_address = entry_offset; 55165caa3d0SMasahiro Yamada *toc_entry++ = image->toc_e; 55265caa3d0SMasahiro Yamada entry_offset += image->toc_e.size; 553819281eeSdp-arm } 554819281eeSdp-arm 555819281eeSdp-arm /* Append a null uuid entry to mark the end of ToC entries. */ 55665caa3d0SMasahiro Yamada memset(toc_entry, 0, sizeof(*toc_entry)); 557819281eeSdp-arm toc_entry->offset_address = entry_offset; 558819281eeSdp-arm 559819281eeSdp-arm /* Generate the FIP file. */ 560819281eeSdp-arm fp = fopen(filename, "w"); 561819281eeSdp-arm if (fp == NULL) 562819281eeSdp-arm log_err("fopen %s", filename); 563819281eeSdp-arm 564819281eeSdp-arm if (verbose) 565819281eeSdp-arm log_dbgx("Metadata size: %zu bytes", buf_size); 566819281eeSdp-arm 567a1da83f5SMasahiro Yamada xfwrite(buf, buf_size, fp, filename); 568819281eeSdp-arm free(buf); 569819281eeSdp-arm 570819281eeSdp-arm if (verbose) 571819281eeSdp-arm log_dbgx("Payload size: %zu bytes", payload_size); 572819281eeSdp-arm 573b9589fe5Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 574b9589fe5Sdp-arm image_t *image = desc->image; 575b9589fe5Sdp-arm 576b9589fe5Sdp-arm if (image == NULL) 577b9589fe5Sdp-arm continue; 5781c75d5dfSMasahiro Yamada if (fseek(fp, image->toc_e.offset_address, SEEK_SET)) 5791c75d5dfSMasahiro Yamada log_errx("Failed to set file position"); 5801c75d5dfSMasahiro Yamada 58165caa3d0SMasahiro Yamada xfwrite(image->buffer, image->toc_e.size, fp, filename); 5821c75d5dfSMasahiro Yamada } 583819281eeSdp-arm 584819281eeSdp-arm fclose(fp); 585819281eeSdp-arm return 0; 586819281eeSdp-arm } 587819281eeSdp-arm 588819281eeSdp-arm /* 589819281eeSdp-arm * This function is shared between the create and update subcommands. 590819281eeSdp-arm * The difference between the two subcommands is that when the FIP file 591819281eeSdp-arm * is created, the parsing of an existing FIP is skipped. This results 592819281eeSdp-arm * in update_fip() creating the new FIP file from scratch because the 593819281eeSdp-arm * internal image table is not populated. 594819281eeSdp-arm */ 595819281eeSdp-arm static void update_fip(void) 596819281eeSdp-arm { 597e0f083a0Sdp-arm image_desc_t *desc; 598819281eeSdp-arm 599819281eeSdp-arm /* Add or replace images in the FIP file. */ 600e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 601b9589fe5Sdp-arm image_t *image; 602fcab6bbeSdp-arm 603e0f083a0Sdp-arm if (desc->action != DO_PACK) 604819281eeSdp-arm continue; 605819281eeSdp-arm 606b9589fe5Sdp-arm image = read_image_from_file(&desc->uuid, 607e0f083a0Sdp-arm desc->action_arg); 608b9589fe5Sdp-arm if (desc->image != NULL) { 609e0f083a0Sdp-arm if (verbose) { 610fcab6bbeSdp-arm log_dbgx("Replacing %s with %s", 611e0f083a0Sdp-arm desc->cmdline_name, 612e0f083a0Sdp-arm desc->action_arg); 613e0f083a0Sdp-arm } 614b9589fe5Sdp-arm free(desc->image); 615b9589fe5Sdp-arm desc->image = image; 616819281eeSdp-arm } else { 617819281eeSdp-arm if (verbose) 618819281eeSdp-arm log_dbgx("Adding image %s", 619e0f083a0Sdp-arm desc->action_arg); 620b9589fe5Sdp-arm desc->image = image; 621819281eeSdp-arm } 622819281eeSdp-arm } 623819281eeSdp-arm } 624819281eeSdp-arm 625e0f083a0Sdp-arm static void parse_plat_toc_flags(const char *arg, unsigned long long *toc_flags) 626819281eeSdp-arm { 627819281eeSdp-arm unsigned long long flags; 628819281eeSdp-arm char *endptr; 629819281eeSdp-arm 630819281eeSdp-arm errno = 0; 631819281eeSdp-arm flags = strtoull(arg, &endptr, 16); 632819281eeSdp-arm if (*endptr != '\0' || flags > UINT16_MAX || errno != 0) 633819281eeSdp-arm log_errx("Invalid platform ToC flags: %s", arg); 634819281eeSdp-arm /* Platform ToC flags is a 16-bit field occupying bits [32-47]. */ 635819281eeSdp-arm *toc_flags |= flags << 32; 636819281eeSdp-arm } 637819281eeSdp-arm 6381c75d5dfSMasahiro Yamada static int is_power_of_2(unsigned long x) 6391c75d5dfSMasahiro Yamada { 6401c75d5dfSMasahiro Yamada return x && !(x & (x - 1)); 6411c75d5dfSMasahiro Yamada } 6421c75d5dfSMasahiro Yamada 6431c75d5dfSMasahiro Yamada static unsigned long get_image_align(char *arg) 6441c75d5dfSMasahiro Yamada { 6451c75d5dfSMasahiro Yamada char *endptr; 6461c75d5dfSMasahiro Yamada unsigned long align; 6471c75d5dfSMasahiro Yamada 6481c75d5dfSMasahiro Yamada errno = 0; 649*fb5f7949SAndreas Färber align = strtoul(arg, &endptr, 0); 6501c75d5dfSMasahiro Yamada if (*endptr != '\0' || !is_power_of_2(align) || errno != 0) 6511c75d5dfSMasahiro Yamada log_errx("Invalid alignment: %s", arg); 6521c75d5dfSMasahiro Yamada 6531c75d5dfSMasahiro Yamada return align; 6541c75d5dfSMasahiro Yamada } 6551c75d5dfSMasahiro Yamada 656fcab6bbeSdp-arm static void parse_blob_opt(char *arg, uuid_t *uuid, char *filename, size_t len) 657fcab6bbeSdp-arm { 658fcab6bbeSdp-arm char *p; 659fcab6bbeSdp-arm 660fcab6bbeSdp-arm for (p = strtok(arg, ","); p != NULL; p = strtok(NULL, ",")) { 661fcab6bbeSdp-arm if (strncmp(p, "uuid=", strlen("uuid=")) == 0) { 662fcab6bbeSdp-arm p += strlen("uuid="); 663fcab6bbeSdp-arm uuid_from_str(uuid, p); 664fcab6bbeSdp-arm } else if (strncmp(p, "file=", strlen("file=")) == 0) { 665fcab6bbeSdp-arm p += strlen("file="); 666fcab6bbeSdp-arm snprintf(filename, len, "%s", p); 667fcab6bbeSdp-arm } 668fcab6bbeSdp-arm } 669fcab6bbeSdp-arm } 670fcab6bbeSdp-arm 671819281eeSdp-arm static int create_cmd(int argc, char *argv[]) 672819281eeSdp-arm { 673e0f083a0Sdp-arm struct option *opts = NULL; 674e0f083a0Sdp-arm size_t nr_opts = 0; 675819281eeSdp-arm unsigned long long toc_flags = 0; 6761c75d5dfSMasahiro Yamada unsigned long align = 1; 677819281eeSdp-arm 678819281eeSdp-arm if (argc < 2) 67985ee2778Sdp-arm create_usage(); 680819281eeSdp-arm 681e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 682e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 683819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 6841c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 685fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 686e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 687819281eeSdp-arm 688819281eeSdp-arm while (1) { 68920f87e78Sdp-arm int c, opt_index = 0; 690819281eeSdp-arm 691fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:", opts, &opt_index); 692819281eeSdp-arm if (c == -1) 693819281eeSdp-arm break; 694819281eeSdp-arm 695819281eeSdp-arm switch (c) { 696819281eeSdp-arm case OPT_TOC_ENTRY: { 697e0f083a0Sdp-arm image_desc_t *desc; 698819281eeSdp-arm 699e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 700d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 701819281eeSdp-arm break; 702819281eeSdp-arm } 703819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: 704819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 705819281eeSdp-arm break; 7061c75d5dfSMasahiro Yamada case OPT_ALIGN: 7071c75d5dfSMasahiro Yamada align = get_image_align(optarg); 7081c75d5dfSMasahiro Yamada break; 709fcab6bbeSdp-arm case 'b': { 710fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 711fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 712fcab6bbeSdp-arm uuid_t uuid = { 0 }; 713fcab6bbeSdp-arm image_desc_t *desc; 714fcab6bbeSdp-arm 715fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 716fcab6bbeSdp-arm filename, sizeof(filename)); 717fcab6bbeSdp-arm 718fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 719fcab6bbeSdp-arm filename[0] == '\0') 720fcab6bbeSdp-arm create_usage(); 721fcab6bbeSdp-arm 722fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 723d02fcebeSdp-arm if (desc == NULL) { 724fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 725fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 726fcab6bbeSdp-arm add_image_desc(desc); 727fcab6bbeSdp-arm } 728d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 729fcab6bbeSdp-arm break; 730fcab6bbeSdp-arm } 731819281eeSdp-arm default: 73285ee2778Sdp-arm create_usage(); 733819281eeSdp-arm } 734819281eeSdp-arm } 735819281eeSdp-arm argc -= optind; 736819281eeSdp-arm argv += optind; 737e0f083a0Sdp-arm free(opts); 738819281eeSdp-arm 739819281eeSdp-arm if (argc == 0) 74085ee2778Sdp-arm create_usage(); 741819281eeSdp-arm 742819281eeSdp-arm update_fip(); 743819281eeSdp-arm 7441c75d5dfSMasahiro Yamada pack_images(argv[0], toc_flags, align); 745819281eeSdp-arm return 0; 746819281eeSdp-arm } 747819281eeSdp-arm 748819281eeSdp-arm static void create_usage(void) 749819281eeSdp-arm { 750819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 751819281eeSdp-arm 752ee079320SMasahiro Yamada printf("fiptool create [opts] FIP_FILENAME\n"); 753ee079320SMasahiro Yamada printf("\n"); 754ee079320SMasahiro Yamada printf("Options:\n"); 7551c75d5dfSMasahiro Yamada printf(" --align <value>\t\tEach image is aligned to <value> (default: 1).\n"); 756802b42a0SMasahiro Yamada printf(" --blob uuid=...,file=...\tAdd an image with the given UUID pointed to by file.\n"); 757802b42a0SMasahiro Yamada printf(" --plat-toc-flags <value>\t16-bit platform specific flag field occupying bits 32-47 in 64-bit ToC header.\n"); 75895d2b268SMasahiro Yamada printf("\n"); 759819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 760819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 761819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 762819281eeSdp-arm toc_entry->name); 76385ee2778Sdp-arm exit(1); 764819281eeSdp-arm } 765819281eeSdp-arm 766819281eeSdp-arm static int update_cmd(int argc, char *argv[]) 767819281eeSdp-arm { 768e0f083a0Sdp-arm struct option *opts = NULL; 769e0f083a0Sdp-arm size_t nr_opts = 0; 770fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 771819281eeSdp-arm fip_toc_header_t toc_header = { 0 }; 772819281eeSdp-arm unsigned long long toc_flags = 0; 7731c75d5dfSMasahiro Yamada unsigned long align = 1; 774819281eeSdp-arm int pflag = 0; 775819281eeSdp-arm 776819281eeSdp-arm if (argc < 2) 77785ee2778Sdp-arm update_usage(); 778819281eeSdp-arm 779e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 7801c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 781fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 782e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 783e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 784819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 785e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 786819281eeSdp-arm 787819281eeSdp-arm while (1) { 78820f87e78Sdp-arm int c, opt_index = 0; 789819281eeSdp-arm 790fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:o:", opts, &opt_index); 791819281eeSdp-arm if (c == -1) 792819281eeSdp-arm break; 793819281eeSdp-arm 794819281eeSdp-arm switch (c) { 795819281eeSdp-arm case OPT_TOC_ENTRY: { 796e0f083a0Sdp-arm image_desc_t *desc; 797819281eeSdp-arm 798e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 799d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, optarg); 800819281eeSdp-arm break; 801819281eeSdp-arm } 802e0f083a0Sdp-arm case OPT_PLAT_TOC_FLAGS: 803819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 804819281eeSdp-arm pflag = 1; 805819281eeSdp-arm break; 806fcab6bbeSdp-arm case 'b': { 807fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 808fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 809fcab6bbeSdp-arm uuid_t uuid = { 0 }; 810fcab6bbeSdp-arm image_desc_t *desc; 811fcab6bbeSdp-arm 812fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 813fcab6bbeSdp-arm filename, sizeof(filename)); 814fcab6bbeSdp-arm 815fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 816fcab6bbeSdp-arm filename[0] == '\0') 817fcab6bbeSdp-arm update_usage(); 818fcab6bbeSdp-arm 819fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 820d02fcebeSdp-arm if (desc == NULL) { 821fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 822fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 823fcab6bbeSdp-arm add_image_desc(desc); 824fcab6bbeSdp-arm } 825d02fcebeSdp-arm set_image_desc_action(desc, DO_PACK, filename); 826fcab6bbeSdp-arm break; 827fcab6bbeSdp-arm } 8281c75d5dfSMasahiro Yamada case OPT_ALIGN: 8291c75d5dfSMasahiro Yamada align = get_image_align(optarg); 8301c75d5dfSMasahiro Yamada break; 831819281eeSdp-arm case 'o': 832819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 833819281eeSdp-arm break; 834819281eeSdp-arm default: 83585ee2778Sdp-arm update_usage(); 836819281eeSdp-arm } 837819281eeSdp-arm } 838819281eeSdp-arm argc -= optind; 839819281eeSdp-arm argv += optind; 840e0f083a0Sdp-arm free(opts); 841819281eeSdp-arm 842819281eeSdp-arm if (argc == 0) 84385ee2778Sdp-arm update_usage(); 844819281eeSdp-arm 845819281eeSdp-arm if (outfile[0] == '\0') 846819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 847819281eeSdp-arm 84840866aafSMasahiro Yamada if (access(argv[0], F_OK) == 0) 849819281eeSdp-arm parse_fip(argv[0], &toc_header); 850819281eeSdp-arm 851819281eeSdp-arm if (pflag) 852819281eeSdp-arm toc_header.flags &= ~(0xffffULL << 32); 853819281eeSdp-arm toc_flags = (toc_header.flags |= toc_flags); 854819281eeSdp-arm 855819281eeSdp-arm update_fip(); 856819281eeSdp-arm 8571c75d5dfSMasahiro Yamada pack_images(outfile, toc_flags, align); 858819281eeSdp-arm return 0; 859819281eeSdp-arm } 860819281eeSdp-arm 861819281eeSdp-arm static void update_usage(void) 862819281eeSdp-arm { 863819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 864819281eeSdp-arm 865ee079320SMasahiro Yamada printf("fiptool update [opts] FIP_FILENAME\n"); 866ee079320SMasahiro Yamada printf("\n"); 867ee079320SMasahiro Yamada printf("Options:\n"); 8681c75d5dfSMasahiro Yamada printf(" --align <value>\t\tEach image is aligned to <value> (default: 1).\n"); 869802b42a0SMasahiro Yamada printf(" --blob uuid=...,file=...\tAdd or update an image with the given UUID pointed to by file.\n"); 870819281eeSdp-arm printf(" --out FIP_FILENAME\t\tSet an alternative output FIP file.\n"); 871802b42a0SMasahiro Yamada printf(" --plat-toc-flags <value>\t16-bit platform specific flag field occupying bits 32-47 in 64-bit ToC header.\n"); 87295d2b268SMasahiro Yamada printf("\n"); 873819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 874819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 875819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 876819281eeSdp-arm toc_entry->name); 87785ee2778Sdp-arm exit(1); 878819281eeSdp-arm } 879819281eeSdp-arm 880819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]) 881819281eeSdp-arm { 882e0f083a0Sdp-arm struct option *opts = NULL; 883e0f083a0Sdp-arm size_t nr_opts = 0; 884fcab6bbeSdp-arm char outdir[PATH_MAX] = { 0 }; 885e0f083a0Sdp-arm image_desc_t *desc; 886819281eeSdp-arm int fflag = 0; 887819281eeSdp-arm int unpack_all = 1; 888819281eeSdp-arm 889819281eeSdp-arm if (argc < 2) 89085ee2778Sdp-arm unpack_usage(); 891819281eeSdp-arm 892e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 893fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 894e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 895e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 896e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 897819281eeSdp-arm 898819281eeSdp-arm while (1) { 89920f87e78Sdp-arm int c, opt_index = 0; 900819281eeSdp-arm 901fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 902819281eeSdp-arm if (c == -1) 903819281eeSdp-arm break; 904819281eeSdp-arm 905819281eeSdp-arm switch (c) { 906e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 907e0f083a0Sdp-arm image_desc_t *desc; 908e0f083a0Sdp-arm 909e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 910d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, optarg); 911e0f083a0Sdp-arm unpack_all = 0; 912819281eeSdp-arm break; 913e0f083a0Sdp-arm } 914fcab6bbeSdp-arm case 'b': { 915fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1]; 916fcab6bbeSdp-arm char filename[PATH_MAX] = { 0 }; 917fcab6bbeSdp-arm uuid_t uuid = { 0 }; 918fcab6bbeSdp-arm image_desc_t *desc; 919fcab6bbeSdp-arm 920fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 921fcab6bbeSdp-arm filename, sizeof(filename)); 922fcab6bbeSdp-arm 923fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 || 924fcab6bbeSdp-arm filename[0] == '\0') 925fcab6bbeSdp-arm unpack_usage(); 926fcab6bbeSdp-arm 927fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 928d02fcebeSdp-arm if (desc == NULL) { 929fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 930fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 931fcab6bbeSdp-arm add_image_desc(desc); 932fcab6bbeSdp-arm } 933d02fcebeSdp-arm set_image_desc_action(desc, DO_UNPACK, filename); 934fcab6bbeSdp-arm unpack_all = 0; 935fcab6bbeSdp-arm break; 936fcab6bbeSdp-arm } 937819281eeSdp-arm case 'f': 938819281eeSdp-arm fflag = 1; 939819281eeSdp-arm break; 940819281eeSdp-arm case 'o': 941819281eeSdp-arm snprintf(outdir, sizeof(outdir), "%s", optarg); 942819281eeSdp-arm break; 943819281eeSdp-arm default: 94485ee2778Sdp-arm unpack_usage(); 945819281eeSdp-arm } 946819281eeSdp-arm } 947819281eeSdp-arm argc -= optind; 948819281eeSdp-arm argv += optind; 949e0f083a0Sdp-arm free(opts); 950819281eeSdp-arm 951819281eeSdp-arm if (argc == 0) 95285ee2778Sdp-arm unpack_usage(); 953819281eeSdp-arm 954819281eeSdp-arm parse_fip(argv[0], NULL); 955819281eeSdp-arm 956819281eeSdp-arm if (outdir[0] != '\0') 957819281eeSdp-arm if (chdir(outdir) == -1) 958819281eeSdp-arm log_err("chdir %s", outdir); 959819281eeSdp-arm 960819281eeSdp-arm /* Unpack all specified images. */ 961e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 962fcab6bbeSdp-arm char file[PATH_MAX]; 963b9589fe5Sdp-arm image_t *image = desc->image; 964b04efcceSdp-arm 965e0f083a0Sdp-arm if (!unpack_all && desc->action != DO_UNPACK) 966819281eeSdp-arm continue; 967819281eeSdp-arm 968819281eeSdp-arm /* Build filename. */ 969e0f083a0Sdp-arm if (desc->action_arg == NULL) 970819281eeSdp-arm snprintf(file, sizeof(file), "%s.bin", 971e0f083a0Sdp-arm desc->cmdline_name); 972819281eeSdp-arm else 973819281eeSdp-arm snprintf(file, sizeof(file), "%s", 974e0f083a0Sdp-arm desc->action_arg); 975819281eeSdp-arm 976b04efcceSdp-arm if (image == NULL) { 977b04efcceSdp-arm if (!unpack_all) 978fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 979819281eeSdp-arm file, argv[0]); 980819281eeSdp-arm continue; 981819281eeSdp-arm } 982819281eeSdp-arm 983819281eeSdp-arm if (access(file, F_OK) != 0 || fflag) { 984819281eeSdp-arm if (verbose) 985819281eeSdp-arm log_dbgx("Unpacking %s", file); 986b04efcceSdp-arm write_image_to_file(image, file); 987819281eeSdp-arm } else { 988819281eeSdp-arm log_warnx("File %s already exists, use --force to overwrite it", 989819281eeSdp-arm file); 990819281eeSdp-arm } 991819281eeSdp-arm } 992819281eeSdp-arm 993819281eeSdp-arm return 0; 994819281eeSdp-arm } 995819281eeSdp-arm 996819281eeSdp-arm static void unpack_usage(void) 997819281eeSdp-arm { 998819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 999819281eeSdp-arm 1000ee079320SMasahiro Yamada printf("fiptool unpack [opts] FIP_FILENAME\n"); 1001ee079320SMasahiro Yamada printf("\n"); 1002ee079320SMasahiro Yamada printf("Options:\n"); 1003802b42a0SMasahiro Yamada printf(" --blob uuid=...,file=...\tUnpack an image with the given UUID to file.\n"); 1004802b42a0SMasahiro Yamada printf(" --force\t\t\tIf the output file already exists, use --force to overwrite it.\n"); 1005fcab6bbeSdp-arm printf(" --out path\t\t\tSet the output directory path.\n"); 100695d2b268SMasahiro Yamada printf("\n"); 1007819281eeSdp-arm printf("Specific images are unpacked with the following options:\n"); 1008819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1009819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 1010819281eeSdp-arm toc_entry->name); 101195d2b268SMasahiro Yamada printf("\n"); 1012819281eeSdp-arm printf("If no options are provided, all images will be unpacked.\n"); 101385ee2778Sdp-arm exit(1); 1014819281eeSdp-arm } 1015819281eeSdp-arm 1016819281eeSdp-arm static int remove_cmd(int argc, char *argv[]) 1017819281eeSdp-arm { 1018e0f083a0Sdp-arm struct option *opts = NULL; 1019e0f083a0Sdp-arm size_t nr_opts = 0; 1020fcab6bbeSdp-arm char outfile[PATH_MAX] = { 0 }; 1021819281eeSdp-arm fip_toc_header_t toc_header; 1022e0f083a0Sdp-arm image_desc_t *desc; 10231c75d5dfSMasahiro Yamada unsigned long align = 1; 1024819281eeSdp-arm int fflag = 0; 1025819281eeSdp-arm 1026819281eeSdp-arm if (argc < 2) 102785ee2778Sdp-arm remove_usage(); 1028819281eeSdp-arm 1029e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, no_argument); 10301c75d5dfSMasahiro Yamada opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN); 1031fcab6bbeSdp-arm opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b'); 1032e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 1033e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 1034e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 1035819281eeSdp-arm 1036819281eeSdp-arm while (1) { 103720f87e78Sdp-arm int c, opt_index = 0; 1038819281eeSdp-arm 1039fcab6bbeSdp-arm c = getopt_long(argc, argv, "b:fo:", opts, &opt_index); 1040819281eeSdp-arm if (c == -1) 1041819281eeSdp-arm break; 1042819281eeSdp-arm 1043819281eeSdp-arm switch (c) { 1044e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 1045e0f083a0Sdp-arm image_desc_t *desc; 1046e0f083a0Sdp-arm 1047e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 1048d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1049819281eeSdp-arm break; 1050e0f083a0Sdp-arm } 10511c75d5dfSMasahiro Yamada case OPT_ALIGN: 10521c75d5dfSMasahiro Yamada align = get_image_align(optarg); 10531c75d5dfSMasahiro Yamada break; 1054fcab6bbeSdp-arm case 'b': { 1055fcab6bbeSdp-arm char name[_UUID_STR_LEN + 1], filename[PATH_MAX]; 1056fcab6bbeSdp-arm uuid_t uuid = { 0 }; 1057fcab6bbeSdp-arm image_desc_t *desc; 1058fcab6bbeSdp-arm 1059fcab6bbeSdp-arm parse_blob_opt(optarg, &uuid, 1060fcab6bbeSdp-arm filename, sizeof(filename)); 1061fcab6bbeSdp-arm 1062fcab6bbeSdp-arm if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0) 1063fcab6bbeSdp-arm remove_usage(); 1064fcab6bbeSdp-arm 1065fcab6bbeSdp-arm desc = lookup_image_desc_from_uuid(&uuid); 1066d02fcebeSdp-arm if (desc == NULL) { 1067fcab6bbeSdp-arm uuid_to_str(name, sizeof(name), &uuid); 1068fcab6bbeSdp-arm desc = new_image_desc(&uuid, name, "blob"); 1069fcab6bbeSdp-arm add_image_desc(desc); 1070fcab6bbeSdp-arm } 1071d02fcebeSdp-arm set_image_desc_action(desc, DO_REMOVE, NULL); 1072fcab6bbeSdp-arm break; 1073fcab6bbeSdp-arm } 1074819281eeSdp-arm case 'f': 1075819281eeSdp-arm fflag = 1; 1076819281eeSdp-arm break; 1077819281eeSdp-arm case 'o': 1078819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 1079819281eeSdp-arm break; 1080819281eeSdp-arm default: 108185ee2778Sdp-arm remove_usage(); 1082819281eeSdp-arm } 1083819281eeSdp-arm } 1084819281eeSdp-arm argc -= optind; 1085819281eeSdp-arm argv += optind; 1086e0f083a0Sdp-arm free(opts); 1087819281eeSdp-arm 1088819281eeSdp-arm if (argc == 0) 108985ee2778Sdp-arm remove_usage(); 1090819281eeSdp-arm 1091819281eeSdp-arm if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag) 1092819281eeSdp-arm log_errx("File %s already exists, use --force to overwrite it", 1093819281eeSdp-arm outfile); 1094819281eeSdp-arm 1095819281eeSdp-arm if (outfile[0] == '\0') 1096819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 1097819281eeSdp-arm 1098819281eeSdp-arm parse_fip(argv[0], &toc_header); 1099819281eeSdp-arm 1100e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 1101e0f083a0Sdp-arm if (desc->action != DO_REMOVE) 1102819281eeSdp-arm continue; 1103fcab6bbeSdp-arm 1104b9589fe5Sdp-arm if (desc->image != NULL) { 1105819281eeSdp-arm if (verbose) 1106fcab6bbeSdp-arm log_dbgx("Removing %s", 1107e0f083a0Sdp-arm desc->cmdline_name); 1108b9589fe5Sdp-arm free(desc->image); 1109b9589fe5Sdp-arm desc->image = NULL; 1110819281eeSdp-arm } else { 1111fcab6bbeSdp-arm log_warnx("%s does not exist in %s", 1112e0f083a0Sdp-arm desc->cmdline_name, argv[0]); 1113819281eeSdp-arm } 1114819281eeSdp-arm } 1115819281eeSdp-arm 11161c75d5dfSMasahiro Yamada pack_images(outfile, toc_header.flags, align); 1117819281eeSdp-arm return 0; 1118819281eeSdp-arm } 1119819281eeSdp-arm 1120819281eeSdp-arm static void remove_usage(void) 1121819281eeSdp-arm { 1122819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 1123819281eeSdp-arm 1124ee079320SMasahiro Yamada printf("fiptool remove [opts] FIP_FILENAME\n"); 1125ee079320SMasahiro Yamada printf("\n"); 1126ee079320SMasahiro Yamada printf("Options:\n"); 11271c75d5dfSMasahiro Yamada printf(" --align <value>\tEach image is aligned to <value> (default: 1).\n"); 1128fcab6bbeSdp-arm printf(" --blob uuid=...\tRemove an image with the given UUID.\n"); 1129802b42a0SMasahiro Yamada printf(" --force\t\tIf the output FIP file already exists, use --force to overwrite it.\n"); 1130819281eeSdp-arm printf(" --out FIP_FILENAME\tSet an alternative output FIP file.\n"); 113195d2b268SMasahiro Yamada printf("\n"); 1132819281eeSdp-arm printf("Specific images are removed with the following options:\n"); 1133819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 1134819281eeSdp-arm printf(" --%-16s\t%s\n", toc_entry->cmdline_name, 1135819281eeSdp-arm toc_entry->name); 113685ee2778Sdp-arm exit(1); 1137819281eeSdp-arm } 1138819281eeSdp-arm 1139819281eeSdp-arm static int version_cmd(int argc, char *argv[]) 1140819281eeSdp-arm { 1141819281eeSdp-arm #ifdef VERSION 1142819281eeSdp-arm puts(VERSION); 1143819281eeSdp-arm #else 1144819281eeSdp-arm /* If built from fiptool directory, VERSION is not set. */ 1145819281eeSdp-arm puts("Unknown version"); 1146819281eeSdp-arm #endif 1147819281eeSdp-arm return 0; 1148819281eeSdp-arm } 1149819281eeSdp-arm 1150819281eeSdp-arm static void version_usage(void) 1151819281eeSdp-arm { 1152819281eeSdp-arm printf("fiptool version\n"); 115385ee2778Sdp-arm exit(1); 1154819281eeSdp-arm } 1155819281eeSdp-arm 1156819281eeSdp-arm static int help_cmd(int argc, char *argv[]) 1157819281eeSdp-arm { 1158819281eeSdp-arm int i; 1159819281eeSdp-arm 1160819281eeSdp-arm if (argc < 2) 1161819281eeSdp-arm usage(); 1162819281eeSdp-arm argc--, argv++; 1163819281eeSdp-arm 1164819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1165819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0 && 116685ee2778Sdp-arm cmds[i].usage != NULL) 1167819281eeSdp-arm cmds[i].usage(); 1168819281eeSdp-arm } 1169819281eeSdp-arm if (i == NELEM(cmds)) 1170819281eeSdp-arm printf("No help for subcommand '%s'\n", argv[0]); 1171819281eeSdp-arm return 0; 1172819281eeSdp-arm } 1173819281eeSdp-arm 1174819281eeSdp-arm static void usage(void) 1175819281eeSdp-arm { 11764f96a498SMasahiro Yamada printf("usage: fiptool [--verbose] <command> [<args>]\n"); 1177819281eeSdp-arm printf("Global options supported:\n"); 1178819281eeSdp-arm printf(" --verbose\tEnable verbose output for all commands.\n"); 117995d2b268SMasahiro Yamada printf("\n"); 1180819281eeSdp-arm printf("Commands supported:\n"); 1181819281eeSdp-arm printf(" info\t\tList images contained in FIP.\n"); 1182819281eeSdp-arm printf(" create\tCreate a new FIP with the given images.\n"); 1183819281eeSdp-arm printf(" update\tUpdate an existing FIP with the given images.\n"); 1184819281eeSdp-arm printf(" unpack\tUnpack images from FIP.\n"); 1185819281eeSdp-arm printf(" remove\tRemove images from FIP.\n"); 1186819281eeSdp-arm printf(" version\tShow fiptool version.\n"); 1187819281eeSdp-arm printf(" help\t\tShow help for given command.\n"); 1188819281eeSdp-arm exit(1); 1189819281eeSdp-arm } 1190819281eeSdp-arm 1191819281eeSdp-arm int main(int argc, char *argv[]) 1192819281eeSdp-arm { 1193819281eeSdp-arm int i, ret = 0; 1194819281eeSdp-arm 1195cc672bb2Sdp-arm while (1) { 1196cc672bb2Sdp-arm int c, opt_index = 0; 1197cc672bb2Sdp-arm static struct option opts[] = { 1198cc672bb2Sdp-arm { "verbose", no_argument, NULL, 'v' }, 1199cc672bb2Sdp-arm { NULL, no_argument, NULL, 0 } 1200cc672bb2Sdp-arm }; 1201819281eeSdp-arm 1202cc672bb2Sdp-arm /* 1203cc672bb2Sdp-arm * Set POSIX mode so getopt stops at the first non-option 1204cc672bb2Sdp-arm * which is the subcommand. 1205cc672bb2Sdp-arm */ 1206cc672bb2Sdp-arm c = getopt_long(argc, argv, "+v", opts, &opt_index); 1207cc672bb2Sdp-arm if (c == -1) 1208cc672bb2Sdp-arm break; 1209cc672bb2Sdp-arm 1210cc672bb2Sdp-arm switch (c) { 1211cc672bb2Sdp-arm case 'v': 1212819281eeSdp-arm verbose = 1; 1213cc672bb2Sdp-arm break; 1214cc672bb2Sdp-arm default: 1215c9cb4089SMasahiro Yamada usage(); 1216819281eeSdp-arm } 1217cc672bb2Sdp-arm } 1218cc672bb2Sdp-arm argc -= optind; 1219cc672bb2Sdp-arm argv += optind; 1220cc672bb2Sdp-arm /* Reset optind for subsequent getopt processing. */ 1221cc672bb2Sdp-arm optind = 0; 1222cc672bb2Sdp-arm 1223cc672bb2Sdp-arm if (argc == 0) 1224cc672bb2Sdp-arm usage(); 1225819281eeSdp-arm 1226e0f083a0Sdp-arm fill_image_descs(); 1227819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1228819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0) { 1229819281eeSdp-arm ret = cmds[i].handler(argc, argv); 1230819281eeSdp-arm break; 1231819281eeSdp-arm } 1232819281eeSdp-arm } 1233819281eeSdp-arm if (i == NELEM(cmds)) 1234819281eeSdp-arm usage(); 1235e0f083a0Sdp-arm free_image_descs(); 1236819281eeSdp-arm return ret; 1237819281eeSdp-arm } 1238