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 54*e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid); 55*e0f083a0Sdp-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 82*e0f083a0Sdp-arm static image_desc_t *image_desc_head; 83*e0f083a0Sdp-arm static size_t nr_image_descs; 84*e0f083a0Sdp-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) 144a22f6285Sdp-arm log_errx("strdup: ", 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) 154a22f6285Sdp-arm log_errx("malloc: ", msg); 155a22f6285Sdp-arm return d; 156a22f6285Sdp-arm } 157a22f6285Sdp-arm 158*e0f083a0Sdp-arm static image_desc_t *new_image_desc(const uuid_t *uuid, 159*e0f083a0Sdp-arm const char *name, const char *cmdline_name) 160*e0f083a0Sdp-arm { 161*e0f083a0Sdp-arm image_desc_t *desc; 162*e0f083a0Sdp-arm 163*e0f083a0Sdp-arm desc = xmalloc(sizeof(*desc), 164*e0f083a0Sdp-arm "failed to allocate memory for image descriptor"); 165*e0f083a0Sdp-arm memcpy(&desc->uuid, uuid, sizeof(uuid_t)); 166*e0f083a0Sdp-arm desc->name = xstrdup(name, 167*e0f083a0Sdp-arm "failed to allocate memory for image name"); 168*e0f083a0Sdp-arm desc->cmdline_name = xstrdup(cmdline_name, 169*e0f083a0Sdp-arm "failed to allocate memory for image command line name"); 170*e0f083a0Sdp-arm desc->action = DO_UNSPEC; 171*e0f083a0Sdp-arm desc->action_arg = NULL; 172*e0f083a0Sdp-arm return desc; 173*e0f083a0Sdp-arm } 174*e0f083a0Sdp-arm 175*e0f083a0Sdp-arm static void free_image_desc(image_desc_t *desc) 176*e0f083a0Sdp-arm { 177*e0f083a0Sdp-arm free(desc->name); 178*e0f083a0Sdp-arm free(desc->cmdline_name); 179*e0f083a0Sdp-arm free(desc->action_arg); 180*e0f083a0Sdp-arm free(desc); 181*e0f083a0Sdp-arm } 182*e0f083a0Sdp-arm 183*e0f083a0Sdp-arm static void add_image_desc(image_desc_t *desc) 184*e0f083a0Sdp-arm { 185*e0f083a0Sdp-arm assert(lookup_image_desc_from_uuid(&desc->uuid) == NULL); 186*e0f083a0Sdp-arm desc->next = image_desc_head; 187*e0f083a0Sdp-arm image_desc_head = desc; 188*e0f083a0Sdp-arm nr_image_descs++; 189*e0f083a0Sdp-arm } 190*e0f083a0Sdp-arm 191*e0f083a0Sdp-arm static void free_image_descs(void) 192*e0f083a0Sdp-arm { 193*e0f083a0Sdp-arm image_desc_t *desc = image_desc_head, *tmp; 194*e0f083a0Sdp-arm 195*e0f083a0Sdp-arm while (desc != NULL) { 196*e0f083a0Sdp-arm tmp = desc->next; 197*e0f083a0Sdp-arm free_image_desc(desc); 198*e0f083a0Sdp-arm desc = tmp; 199*e0f083a0Sdp-arm nr_image_descs--; 200*e0f083a0Sdp-arm } 201*e0f083a0Sdp-arm assert(nr_image_descs == 0); 202*e0f083a0Sdp-arm } 203*e0f083a0Sdp-arm 204*e0f083a0Sdp-arm static void fill_image_descs(void) 205*e0f083a0Sdp-arm { 206*e0f083a0Sdp-arm toc_entry_t *toc_entry; 207*e0f083a0Sdp-arm 208*e0f083a0Sdp-arm for (toc_entry = toc_entries; 209*e0f083a0Sdp-arm toc_entry->cmdline_name != NULL; 210*e0f083a0Sdp-arm toc_entry++) { 211*e0f083a0Sdp-arm image_desc_t *desc; 212*e0f083a0Sdp-arm 213*e0f083a0Sdp-arm desc = new_image_desc(&toc_entry->uuid, 214*e0f083a0Sdp-arm toc_entry->name, 215*e0f083a0Sdp-arm toc_entry->cmdline_name); 216*e0f083a0Sdp-arm add_image_desc(desc); 217*e0f083a0Sdp-arm } 218*e0f083a0Sdp-arm } 219*e0f083a0Sdp-arm 220819281eeSdp-arm static void add_image(image_t *image) 221819281eeSdp-arm { 222*e0f083a0Sdp-arm assert(lookup_image_from_uuid(&image->uuid) == NULL); 223*e0f083a0Sdp-arm image->next = image_head; 224*e0f083a0Sdp-arm image_head = image; 225*e0f083a0Sdp-arm nr_images++; 226819281eeSdp-arm } 227819281eeSdp-arm 228819281eeSdp-arm static void free_image(image_t *image) 229819281eeSdp-arm { 230819281eeSdp-arm free(image->buffer); 231819281eeSdp-arm free(image); 232819281eeSdp-arm } 233819281eeSdp-arm 234819281eeSdp-arm static void remove_image(image_t *image) 235819281eeSdp-arm { 236*e0f083a0Sdp-arm image_t *tmp = image_head, *prev; 237819281eeSdp-arm 238*e0f083a0Sdp-arm if (tmp == image) { 239*e0f083a0Sdp-arm image_head = tmp->next; 240*e0f083a0Sdp-arm free_image(tmp); 241*e0f083a0Sdp-arm } else { 242*e0f083a0Sdp-arm while (tmp != NULL && tmp != image) { 243*e0f083a0Sdp-arm prev = tmp; 244*e0f083a0Sdp-arm tmp = tmp->next; 245819281eeSdp-arm } 246*e0f083a0Sdp-arm assert(tmp != NULL); 247*e0f083a0Sdp-arm prev->next = tmp->next; 248*e0f083a0Sdp-arm free_image(tmp); 249819281eeSdp-arm } 250819281eeSdp-arm nr_images--; 251819281eeSdp-arm } 252819281eeSdp-arm 253819281eeSdp-arm static void free_images(void) 254819281eeSdp-arm { 255*e0f083a0Sdp-arm image_t *image = image_head, *tmp; 256819281eeSdp-arm 257*e0f083a0Sdp-arm while (image != NULL) { 258*e0f083a0Sdp-arm tmp = image->next; 259*e0f083a0Sdp-arm free_image(image); 260*e0f083a0Sdp-arm image = tmp; 261*e0f083a0Sdp-arm nr_images--; 262819281eeSdp-arm } 263*e0f083a0Sdp-arm assert(nr_images == 0); 264819281eeSdp-arm } 265819281eeSdp-arm 266*e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid) 267819281eeSdp-arm { 268*e0f083a0Sdp-arm image_desc_t *desc; 269819281eeSdp-arm 270*e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 271*e0f083a0Sdp-arm if (memcmp(&desc->uuid, uuid, sizeof(uuid_t)) == 0) 272*e0f083a0Sdp-arm return desc; 273819281eeSdp-arm return NULL; 274819281eeSdp-arm } 275819281eeSdp-arm 276*e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_opt(const char *opt) 277*e0f083a0Sdp-arm { 278*e0f083a0Sdp-arm image_desc_t *desc; 279*e0f083a0Sdp-arm 280*e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 281*e0f083a0Sdp-arm if (strcmp(desc->cmdline_name, opt) == 0) 282*e0f083a0Sdp-arm return desc; 283*e0f083a0Sdp-arm return NULL; 284*e0f083a0Sdp-arm } 285*e0f083a0Sdp-arm 286*e0f083a0Sdp-arm static image_t *lookup_image_from_uuid(const uuid_t *uuid) 287b04efcceSdp-arm { 288b04efcceSdp-arm image_t *image; 289b04efcceSdp-arm 290*e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 291b04efcceSdp-arm if (memcmp(&image->uuid, uuid, sizeof(uuid_t)) == 0) 292b04efcceSdp-arm return image; 293b04efcceSdp-arm return NULL; 294b04efcceSdp-arm } 295b04efcceSdp-arm 29660b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out) 297819281eeSdp-arm { 298819281eeSdp-arm struct stat st; 299819281eeSdp-arm FILE *fp; 300819281eeSdp-arm char *buf, *bufend; 301819281eeSdp-arm fip_toc_header_t *toc_header; 302819281eeSdp-arm fip_toc_entry_t *toc_entry; 303819281eeSdp-arm image_t *image; 304819281eeSdp-arm int terminated = 0; 305819281eeSdp-arm 306819281eeSdp-arm fp = fopen(filename, "r"); 307819281eeSdp-arm if (fp == NULL) 308819281eeSdp-arm log_err("fopen %s", filename); 309819281eeSdp-arm 310819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 311819281eeSdp-arm log_err("fstat %s", filename); 312819281eeSdp-arm 313a22f6285Sdp-arm buf = xmalloc(st.st_size, "failed to load file into memory"); 314819281eeSdp-arm if (fread(buf, 1, st.st_size, fp) != st.st_size) 315819281eeSdp-arm log_errx("Failed to read %s", filename); 316819281eeSdp-arm bufend = buf + st.st_size; 317819281eeSdp-arm fclose(fp); 318819281eeSdp-arm 319819281eeSdp-arm if (st.st_size < sizeof(fip_toc_header_t)) 320819281eeSdp-arm log_errx("FIP %s is truncated", filename); 321819281eeSdp-arm 322819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 323819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 324819281eeSdp-arm 325819281eeSdp-arm if (toc_header->name != TOC_HEADER_NAME) 326819281eeSdp-arm log_errx("%s is not a FIP file", filename); 327819281eeSdp-arm 328819281eeSdp-arm /* Return the ToC header if the caller wants it. */ 329819281eeSdp-arm if (toc_header_out != NULL) 330819281eeSdp-arm *toc_header_out = *toc_header; 331819281eeSdp-arm 332819281eeSdp-arm /* Walk through each ToC entry in the file. */ 333819281eeSdp-arm while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) { 334819281eeSdp-arm /* Found the ToC terminator, we are done. */ 335819281eeSdp-arm if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) { 336819281eeSdp-arm terminated = 1; 337819281eeSdp-arm break; 338819281eeSdp-arm } 339819281eeSdp-arm 340819281eeSdp-arm /* 341819281eeSdp-arm * Build a new image out of the ToC entry and add it to the 342819281eeSdp-arm * table of images. 343819281eeSdp-arm */ 344a22f6285Sdp-arm image = xmalloc(sizeof(*image), 345a22f6285Sdp-arm "failed to allocate memory for image"); 346819281eeSdp-arm memcpy(&image->uuid, &toc_entry->uuid, sizeof(uuid_t)); 347a22f6285Sdp-arm image->buffer = xmalloc(toc_entry->size, 348a22f6285Sdp-arm "failed to allocate image buffer, is FIP file corrupted?"); 349819281eeSdp-arm /* Overflow checks before memory copy. */ 350819281eeSdp-arm if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address) 351819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 352819281eeSdp-arm if (toc_entry->size + toc_entry->offset_address > st.st_size) 353819281eeSdp-arm log_errx("FIP %s is corrupted", filename); 354819281eeSdp-arm 355819281eeSdp-arm memcpy(image->buffer, buf + toc_entry->offset_address, 356819281eeSdp-arm toc_entry->size); 357819281eeSdp-arm image->size = toc_entry->size; 358819281eeSdp-arm 359819281eeSdp-arm add_image(image); 360819281eeSdp-arm 361819281eeSdp-arm toc_entry++; 362819281eeSdp-arm } 363819281eeSdp-arm 364819281eeSdp-arm if (terminated == 0) 365819281eeSdp-arm log_errx("FIP %s does not have a ToC terminator entry", 366819281eeSdp-arm filename); 367819281eeSdp-arm free(buf); 368819281eeSdp-arm return 0; 369819281eeSdp-arm } 370819281eeSdp-arm 37160b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename) 372819281eeSdp-arm { 373819281eeSdp-arm struct stat st; 374819281eeSdp-arm image_t *image; 375819281eeSdp-arm FILE *fp; 376819281eeSdp-arm 377b04efcceSdp-arm assert(uuid != NULL); 378b04efcceSdp-arm 379819281eeSdp-arm fp = fopen(filename, "r"); 380819281eeSdp-arm if (fp == NULL) 381819281eeSdp-arm log_err("fopen %s", filename); 382819281eeSdp-arm 383819281eeSdp-arm if (fstat(fileno(fp), &st) == -1) 384819281eeSdp-arm log_errx("fstat %s", filename); 385819281eeSdp-arm 386a22f6285Sdp-arm image = xmalloc(sizeof(*image), "failed to allocate memory for image"); 387b04efcceSdp-arm memcpy(&image->uuid, uuid, sizeof(uuid_t)); 388a22f6285Sdp-arm image->buffer = xmalloc(st.st_size, "failed to allocate image buffer"); 389819281eeSdp-arm if (fread(image->buffer, 1, st.st_size, fp) != st.st_size) 390819281eeSdp-arm log_errx("Failed to read %s", filename); 391819281eeSdp-arm image->size = st.st_size; 392819281eeSdp-arm 393819281eeSdp-arm fclose(fp); 394819281eeSdp-arm return image; 395819281eeSdp-arm } 396819281eeSdp-arm 39760b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename) 398819281eeSdp-arm { 399819281eeSdp-arm FILE *fp; 400819281eeSdp-arm 401819281eeSdp-arm fp = fopen(filename, "w"); 402819281eeSdp-arm if (fp == NULL) 403819281eeSdp-arm log_err("fopen"); 404819281eeSdp-arm if (fwrite(image->buffer, 1, image->size, fp) != image->size) 405819281eeSdp-arm log_errx("Failed to write %s", filename); 406819281eeSdp-arm fclose(fp); 407819281eeSdp-arm return 0; 408819281eeSdp-arm } 409819281eeSdp-arm 410*e0f083a0Sdp-arm static struct option *add_opt(struct option *opts, size_t *nr_opts, 411*e0f083a0Sdp-arm const char *name, int has_arg, int val) 412819281eeSdp-arm { 413*e0f083a0Sdp-arm opts = realloc(opts, (*nr_opts + 1) * sizeof(*opts)); 414*e0f083a0Sdp-arm if (opts == NULL) 415*e0f083a0Sdp-arm log_err("realloc"); 416*e0f083a0Sdp-arm opts[*nr_opts].name = name; 417*e0f083a0Sdp-arm opts[*nr_opts].has_arg = has_arg; 418*e0f083a0Sdp-arm opts[*nr_opts].flag = NULL; 419*e0f083a0Sdp-arm opts[*nr_opts].val = val; 420*e0f083a0Sdp-arm ++*nr_opts; 421*e0f083a0Sdp-arm return opts; 422819281eeSdp-arm } 423819281eeSdp-arm 424*e0f083a0Sdp-arm static struct option *fill_common_opts(struct option *opts, size_t *nr_opts, 425*e0f083a0Sdp-arm int has_arg) 426819281eeSdp-arm { 427*e0f083a0Sdp-arm image_desc_t *desc; 428*e0f083a0Sdp-arm 429*e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) 430*e0f083a0Sdp-arm opts = add_opt(opts, nr_opts, desc->cmdline_name, has_arg, 431*e0f083a0Sdp-arm OPT_TOC_ENTRY); 432*e0f083a0Sdp-arm return opts; 433819281eeSdp-arm } 434819281eeSdp-arm 435*e0f083a0Sdp-arm static void md_print(const unsigned char *md, size_t len) 4369df69ba3Sdp-arm { 4379df69ba3Sdp-arm size_t i; 4389df69ba3Sdp-arm 4399df69ba3Sdp-arm for (i = 0; i < len; i++) 4409df69ba3Sdp-arm printf("%02x", md[i]); 4419df69ba3Sdp-arm } 4429df69ba3Sdp-arm 443819281eeSdp-arm static int info_cmd(int argc, char *argv[]) 444819281eeSdp-arm { 445819281eeSdp-arm image_t *image; 446819281eeSdp-arm uint64_t image_offset; 447819281eeSdp-arm uint64_t image_size = 0; 448819281eeSdp-arm fip_toc_header_t toc_header; 449819281eeSdp-arm 450819281eeSdp-arm if (argc != 2) 45185ee2778Sdp-arm info_usage(); 452819281eeSdp-arm argc--, argv++; 453819281eeSdp-arm 454819281eeSdp-arm parse_fip(argv[0], &toc_header); 455819281eeSdp-arm 456819281eeSdp-arm if (verbose) { 457819281eeSdp-arm log_dbgx("toc_header[name]: 0x%llX", 458819281eeSdp-arm (unsigned long long)toc_header.name); 459819281eeSdp-arm log_dbgx("toc_header[serial_number]: 0x%llX", 460819281eeSdp-arm (unsigned long long)toc_header.serial_number); 461819281eeSdp-arm log_dbgx("toc_header[flags]: 0x%llX", 462819281eeSdp-arm (unsigned long long)toc_header.flags); 463819281eeSdp-arm } 464819281eeSdp-arm 465819281eeSdp-arm image_offset = sizeof(fip_toc_header_t) + 466819281eeSdp-arm (sizeof(fip_toc_entry_t) * (nr_images + 1)); 467819281eeSdp-arm 468*e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) { 469*e0f083a0Sdp-arm image_desc_t *desc; 470b04efcceSdp-arm 471*e0f083a0Sdp-arm desc = lookup_image_desc_from_uuid(&image->uuid); 472*e0f083a0Sdp-arm if (desc != NULL) 473*e0f083a0Sdp-arm printf("%s: ", desc->name); 474819281eeSdp-arm else 475819281eeSdp-arm printf("Unknown entry: "); 476819281eeSdp-arm image_size = image->size; 477819281eeSdp-arm printf("offset=0x%llX, size=0x%llX", 478819281eeSdp-arm (unsigned long long)image_offset, 479819281eeSdp-arm (unsigned long long)image_size); 480*e0f083a0Sdp-arm if (desc != NULL) 4819df69ba3Sdp-arm printf(", cmdline=\"--%s\"", 482*e0f083a0Sdp-arm desc->cmdline_name); 4839df69ba3Sdp-arm if (verbose) { 4849df69ba3Sdp-arm unsigned char md[SHA256_DIGEST_LENGTH]; 4859df69ba3Sdp-arm 4869df69ba3Sdp-arm SHA256(image->buffer, image_size, md); 4879df69ba3Sdp-arm printf(", sha256="); 4889df69ba3Sdp-arm md_print(md, sizeof(md)); 4899df69ba3Sdp-arm } 490819281eeSdp-arm putchar('\n'); 491819281eeSdp-arm image_offset += image_size; 492819281eeSdp-arm } 493819281eeSdp-arm 494819281eeSdp-arm free_images(); 495819281eeSdp-arm return 0; 496819281eeSdp-arm } 497819281eeSdp-arm 498819281eeSdp-arm static void info_usage(void) 499819281eeSdp-arm { 500819281eeSdp-arm printf("fiptool info FIP_FILENAME\n"); 50185ee2778Sdp-arm exit(1); 502819281eeSdp-arm } 503819281eeSdp-arm 504*e0f083a0Sdp-arm static int pack_images(const char *filename, uint64_t toc_flags) 505819281eeSdp-arm { 506819281eeSdp-arm FILE *fp; 507819281eeSdp-arm image_t *image; 508819281eeSdp-arm fip_toc_header_t *toc_header; 509819281eeSdp-arm fip_toc_entry_t *toc_entry; 510819281eeSdp-arm char *buf; 511819281eeSdp-arm uint64_t entry_offset, buf_size, payload_size; 512819281eeSdp-arm 513819281eeSdp-arm /* Calculate total payload size and allocate scratch buffer. */ 514819281eeSdp-arm payload_size = 0; 515*e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 516*e0f083a0Sdp-arm payload_size += image->size; 517819281eeSdp-arm 518819281eeSdp-arm buf_size = sizeof(fip_toc_header_t) + 519819281eeSdp-arm sizeof(fip_toc_entry_t) * (nr_images + 1); 520819281eeSdp-arm buf = calloc(1, buf_size); 521819281eeSdp-arm if (buf == NULL) 522819281eeSdp-arm log_err("calloc"); 523819281eeSdp-arm 524819281eeSdp-arm /* Build up header and ToC entries from the image table. */ 525819281eeSdp-arm toc_header = (fip_toc_header_t *)buf; 526819281eeSdp-arm toc_header->name = TOC_HEADER_NAME; 527819281eeSdp-arm toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER; 528819281eeSdp-arm toc_header->flags = toc_flags; 529819281eeSdp-arm 530819281eeSdp-arm toc_entry = (fip_toc_entry_t *)(toc_header + 1); 531819281eeSdp-arm 532819281eeSdp-arm entry_offset = buf_size; 533*e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) { 534819281eeSdp-arm memcpy(&toc_entry->uuid, &image->uuid, sizeof(uuid_t)); 535819281eeSdp-arm toc_entry->offset_address = entry_offset; 536819281eeSdp-arm toc_entry->size = image->size; 537819281eeSdp-arm toc_entry->flags = 0; 538819281eeSdp-arm entry_offset += toc_entry->size; 539819281eeSdp-arm toc_entry++; 540819281eeSdp-arm } 541819281eeSdp-arm 542819281eeSdp-arm /* Append a null uuid entry to mark the end of ToC entries. */ 543819281eeSdp-arm memcpy(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)); 544819281eeSdp-arm toc_entry->offset_address = entry_offset; 545819281eeSdp-arm toc_entry->size = 0; 546819281eeSdp-arm toc_entry->flags = 0; 547819281eeSdp-arm 548819281eeSdp-arm /* Generate the FIP file. */ 549819281eeSdp-arm fp = fopen(filename, "w"); 550819281eeSdp-arm if (fp == NULL) 551819281eeSdp-arm log_err("fopen %s", filename); 552819281eeSdp-arm 553819281eeSdp-arm if (verbose) 554819281eeSdp-arm log_dbgx("Metadata size: %zu bytes", buf_size); 555819281eeSdp-arm 556819281eeSdp-arm if (fwrite(buf, 1, buf_size, fp) != buf_size) 557819281eeSdp-arm log_errx("Failed to write image to %s", filename); 558819281eeSdp-arm free(buf); 559819281eeSdp-arm 560819281eeSdp-arm if (verbose) 561819281eeSdp-arm log_dbgx("Payload size: %zu bytes", payload_size); 562819281eeSdp-arm 563*e0f083a0Sdp-arm for (image = image_head; image != NULL; image = image->next) 564819281eeSdp-arm if (fwrite(image->buffer, 1, image->size, fp) != image->size) 565819281eeSdp-arm log_errx("Failed to write image to %s", filename); 566819281eeSdp-arm 567819281eeSdp-arm fclose(fp); 568819281eeSdp-arm return 0; 569819281eeSdp-arm } 570819281eeSdp-arm 571819281eeSdp-arm /* 572819281eeSdp-arm * This function is shared between the create and update subcommands. 573819281eeSdp-arm * The difference between the two subcommands is that when the FIP file 574819281eeSdp-arm * is created, the parsing of an existing FIP is skipped. This results 575819281eeSdp-arm * in update_fip() creating the new FIP file from scratch because the 576819281eeSdp-arm * internal image table is not populated. 577819281eeSdp-arm */ 578819281eeSdp-arm static void update_fip(void) 579819281eeSdp-arm { 580*e0f083a0Sdp-arm image_desc_t *desc; 581b04efcceSdp-arm image_t *new_image, *old_image; 582819281eeSdp-arm 583819281eeSdp-arm /* Add or replace images in the FIP file. */ 584*e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 585*e0f083a0Sdp-arm if (desc->action != DO_PACK) 586819281eeSdp-arm continue; 587819281eeSdp-arm 588*e0f083a0Sdp-arm new_image = read_image_from_file(&desc->uuid, 589*e0f083a0Sdp-arm desc->action_arg); 590*e0f083a0Sdp-arm old_image = lookup_image_from_uuid(&desc->uuid); 591b04efcceSdp-arm if (old_image != NULL) { 592*e0f083a0Sdp-arm if (verbose) { 593819281eeSdp-arm log_dbgx("Replacing image %s.bin with %s", 594*e0f083a0Sdp-arm desc->cmdline_name, 595*e0f083a0Sdp-arm desc->action_arg); 596*e0f083a0Sdp-arm } 597*e0f083a0Sdp-arm remove_image(old_image); 598*e0f083a0Sdp-arm add_image(new_image); 599819281eeSdp-arm } else { 600819281eeSdp-arm if (verbose) 601819281eeSdp-arm log_dbgx("Adding image %s", 602*e0f083a0Sdp-arm desc->action_arg); 603b04efcceSdp-arm add_image(new_image); 604819281eeSdp-arm } 605819281eeSdp-arm } 606819281eeSdp-arm } 607819281eeSdp-arm 608*e0f083a0Sdp-arm static void parse_plat_toc_flags(const char *arg, unsigned long long *toc_flags) 609819281eeSdp-arm { 610819281eeSdp-arm unsigned long long flags; 611819281eeSdp-arm char *endptr; 612819281eeSdp-arm 613819281eeSdp-arm errno = 0; 614819281eeSdp-arm flags = strtoull(arg, &endptr, 16); 615819281eeSdp-arm if (*endptr != '\0' || flags > UINT16_MAX || errno != 0) 616819281eeSdp-arm log_errx("Invalid platform ToC flags: %s", arg); 617819281eeSdp-arm /* Platform ToC flags is a 16-bit field occupying bits [32-47]. */ 618819281eeSdp-arm *toc_flags |= flags << 32; 619819281eeSdp-arm } 620819281eeSdp-arm 621819281eeSdp-arm static int create_cmd(int argc, char *argv[]) 622819281eeSdp-arm { 623*e0f083a0Sdp-arm struct option *opts = NULL; 624*e0f083a0Sdp-arm size_t nr_opts = 0; 625819281eeSdp-arm unsigned long long toc_flags = 0; 626819281eeSdp-arm 627819281eeSdp-arm if (argc < 2) 62885ee2778Sdp-arm create_usage(); 629819281eeSdp-arm 630*e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 631*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 632819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 633*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 634819281eeSdp-arm 635819281eeSdp-arm while (1) { 63620f87e78Sdp-arm int c, opt_index = 0; 637819281eeSdp-arm 63860b499feSdp-arm c = getopt_long(argc, argv, "", opts, &opt_index); 639819281eeSdp-arm if (c == -1) 640819281eeSdp-arm break; 641819281eeSdp-arm 642819281eeSdp-arm switch (c) { 643819281eeSdp-arm case OPT_TOC_ENTRY: { 644*e0f083a0Sdp-arm image_desc_t *desc; 645819281eeSdp-arm 646*e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 647*e0f083a0Sdp-arm assert(desc != NULL); 648*e0f083a0Sdp-arm if (desc->action != DO_UNSPEC) 649*e0f083a0Sdp-arm free(desc->action_arg); 650*e0f083a0Sdp-arm desc->action = DO_PACK; 651*e0f083a0Sdp-arm desc->action_arg = xstrdup(optarg, 652a22f6285Sdp-arm "failed to allocate memory for argument"); 653819281eeSdp-arm break; 654819281eeSdp-arm } 655819281eeSdp-arm case OPT_PLAT_TOC_FLAGS: 656819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 657819281eeSdp-arm break; 658819281eeSdp-arm default: 65985ee2778Sdp-arm create_usage(); 660819281eeSdp-arm } 661819281eeSdp-arm } 662819281eeSdp-arm argc -= optind; 663819281eeSdp-arm argv += optind; 664*e0f083a0Sdp-arm free(opts); 665819281eeSdp-arm 666819281eeSdp-arm if (argc == 0) 66785ee2778Sdp-arm create_usage(); 668819281eeSdp-arm 669819281eeSdp-arm update_fip(); 670819281eeSdp-arm 671819281eeSdp-arm pack_images(argv[0], toc_flags); 672819281eeSdp-arm free_images(); 673819281eeSdp-arm return 0; 674819281eeSdp-arm } 675819281eeSdp-arm 676819281eeSdp-arm static void create_usage(void) 677819281eeSdp-arm { 678819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 679819281eeSdp-arm 68023fcb90dSdp-arm printf("fiptool create [--plat-toc-flags <value>] [opts] FIP_FILENAME\n"); 681819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 682819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 683819281eeSdp-arm fputc('\n', stderr); 684819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 685819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 686819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 687819281eeSdp-arm toc_entry->name); 68885ee2778Sdp-arm exit(1); 689819281eeSdp-arm } 690819281eeSdp-arm 691819281eeSdp-arm static int update_cmd(int argc, char *argv[]) 692819281eeSdp-arm { 693*e0f083a0Sdp-arm struct option *opts = NULL; 694*e0f083a0Sdp-arm size_t nr_opts = 0; 695819281eeSdp-arm char outfile[FILENAME_MAX] = { 0 }; 696819281eeSdp-arm fip_toc_header_t toc_header = { 0 }; 697819281eeSdp-arm unsigned long long toc_flags = 0; 698819281eeSdp-arm int pflag = 0; 699819281eeSdp-arm 700819281eeSdp-arm if (argc < 2) 70185ee2778Sdp-arm update_usage(); 702819281eeSdp-arm 703*e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 704*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 705*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument, 706819281eeSdp-arm OPT_PLAT_TOC_FLAGS); 707*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 708819281eeSdp-arm 709819281eeSdp-arm while (1) { 71020f87e78Sdp-arm int c, opt_index = 0; 711819281eeSdp-arm 712819281eeSdp-arm c = getopt_long(argc, argv, "o:", opts, &opt_index); 713819281eeSdp-arm if (c == -1) 714819281eeSdp-arm break; 715819281eeSdp-arm 716819281eeSdp-arm switch (c) { 717819281eeSdp-arm case OPT_TOC_ENTRY: { 718*e0f083a0Sdp-arm image_desc_t *desc; 719819281eeSdp-arm 720*e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 721*e0f083a0Sdp-arm assert(desc != NULL); 722*e0f083a0Sdp-arm if (desc->action != DO_UNSPEC) 723*e0f083a0Sdp-arm free(desc->action_arg); 724*e0f083a0Sdp-arm desc->action = DO_PACK; 725*e0f083a0Sdp-arm desc->action_arg = xstrdup(optarg, 726a22f6285Sdp-arm "failed to allocate memory for argument"); 727819281eeSdp-arm break; 728819281eeSdp-arm } 729*e0f083a0Sdp-arm case OPT_PLAT_TOC_FLAGS: 730819281eeSdp-arm parse_plat_toc_flags(optarg, &toc_flags); 731819281eeSdp-arm pflag = 1; 732819281eeSdp-arm break; 733819281eeSdp-arm case 'o': 734819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 735819281eeSdp-arm break; 736819281eeSdp-arm default: 73785ee2778Sdp-arm update_usage(); 738819281eeSdp-arm } 739819281eeSdp-arm } 740819281eeSdp-arm argc -= optind; 741819281eeSdp-arm argv += optind; 742*e0f083a0Sdp-arm free(opts); 743819281eeSdp-arm 744819281eeSdp-arm if (argc == 0) 74585ee2778Sdp-arm update_usage(); 746819281eeSdp-arm 747819281eeSdp-arm if (outfile[0] == '\0') 748819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 749819281eeSdp-arm 750819281eeSdp-arm if (access(outfile, F_OK) == 0) 751819281eeSdp-arm parse_fip(argv[0], &toc_header); 752819281eeSdp-arm 753819281eeSdp-arm if (pflag) 754819281eeSdp-arm toc_header.flags &= ~(0xffffULL << 32); 755819281eeSdp-arm toc_flags = (toc_header.flags |= toc_flags); 756819281eeSdp-arm 757819281eeSdp-arm update_fip(); 758819281eeSdp-arm 759819281eeSdp-arm pack_images(outfile, toc_flags); 760819281eeSdp-arm free_images(); 761819281eeSdp-arm return 0; 762819281eeSdp-arm } 763819281eeSdp-arm 764819281eeSdp-arm static void update_usage(void) 765819281eeSdp-arm { 766819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 767819281eeSdp-arm 76823fcb90dSdp-arm printf("fiptool update [--out FIP_FILENAME] " 769819281eeSdp-arm "[--plat-toc-flags <value>] [opts] FIP_FILENAME\n"); 770819281eeSdp-arm printf(" --out FIP_FILENAME\t\tSet an alternative output FIP file.\n"); 771819281eeSdp-arm printf(" --plat-toc-flags <value>\t16-bit platform specific flag field " 772819281eeSdp-arm "occupying bits 32-47 in 64-bit ToC header.\n"); 773819281eeSdp-arm fputc('\n', stderr); 774819281eeSdp-arm printf("Specific images are packed with the following options:\n"); 775819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 776819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 777819281eeSdp-arm toc_entry->name); 77885ee2778Sdp-arm exit(1); 779819281eeSdp-arm } 780819281eeSdp-arm 781819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]) 782819281eeSdp-arm { 783*e0f083a0Sdp-arm struct option *opts = NULL; 784*e0f083a0Sdp-arm size_t nr_opts = 0; 785819281eeSdp-arm char file[FILENAME_MAX], outdir[PATH_MAX] = { 0 }; 786*e0f083a0Sdp-arm image_desc_t *desc; 787819281eeSdp-arm int fflag = 0; 788819281eeSdp-arm int unpack_all = 1; 789819281eeSdp-arm 790819281eeSdp-arm if (argc < 2) 79185ee2778Sdp-arm unpack_usage(); 792819281eeSdp-arm 793*e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, required_argument); 794*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 795*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 796*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 797819281eeSdp-arm 798819281eeSdp-arm while (1) { 79920f87e78Sdp-arm int c, opt_index = 0; 800819281eeSdp-arm 801819281eeSdp-arm c = getopt_long(argc, argv, "fo:", opts, &opt_index); 802819281eeSdp-arm if (c == -1) 803819281eeSdp-arm break; 804819281eeSdp-arm 805819281eeSdp-arm switch (c) { 806*e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 807*e0f083a0Sdp-arm image_desc_t *desc; 808*e0f083a0Sdp-arm 809*e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 810*e0f083a0Sdp-arm assert(desc != NULL); 811*e0f083a0Sdp-arm if (desc->action != DO_UNSPEC) 812*e0f083a0Sdp-arm free(desc->action_arg); 813*e0f083a0Sdp-arm desc->action = DO_UNPACK; 814*e0f083a0Sdp-arm desc->action_arg = xstrdup(optarg, 815a22f6285Sdp-arm "failed to allocate memory for argument"); 816*e0f083a0Sdp-arm unpack_all = 0; 817819281eeSdp-arm break; 818*e0f083a0Sdp-arm } 819819281eeSdp-arm case 'f': 820819281eeSdp-arm fflag = 1; 821819281eeSdp-arm break; 822819281eeSdp-arm case 'o': 823819281eeSdp-arm snprintf(outdir, sizeof(outdir), "%s", optarg); 824819281eeSdp-arm break; 825819281eeSdp-arm default: 82685ee2778Sdp-arm unpack_usage(); 827819281eeSdp-arm } 828819281eeSdp-arm } 829819281eeSdp-arm argc -= optind; 830819281eeSdp-arm argv += optind; 831*e0f083a0Sdp-arm free(opts); 832819281eeSdp-arm 833819281eeSdp-arm if (argc == 0) 83485ee2778Sdp-arm unpack_usage(); 835819281eeSdp-arm 836819281eeSdp-arm parse_fip(argv[0], NULL); 837819281eeSdp-arm 838819281eeSdp-arm if (outdir[0] != '\0') 839819281eeSdp-arm if (chdir(outdir) == -1) 840819281eeSdp-arm log_err("chdir %s", outdir); 841819281eeSdp-arm 842819281eeSdp-arm /* Unpack all specified images. */ 843*e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 844b04efcceSdp-arm image_t *image; 845b04efcceSdp-arm 846*e0f083a0Sdp-arm if (!unpack_all && desc->action != DO_UNPACK) 847819281eeSdp-arm continue; 848819281eeSdp-arm 849819281eeSdp-arm /* Build filename. */ 850*e0f083a0Sdp-arm if (desc->action_arg == NULL) 851819281eeSdp-arm snprintf(file, sizeof(file), "%s.bin", 852*e0f083a0Sdp-arm desc->cmdline_name); 853819281eeSdp-arm else 854819281eeSdp-arm snprintf(file, sizeof(file), "%s", 855*e0f083a0Sdp-arm desc->action_arg); 856819281eeSdp-arm 857*e0f083a0Sdp-arm image = lookup_image_from_uuid(&desc->uuid); 858b04efcceSdp-arm if (image == NULL) { 859b04efcceSdp-arm if (!unpack_all) 860819281eeSdp-arm log_warnx("Requested image %s is not in %s", 861819281eeSdp-arm file, argv[0]); 862819281eeSdp-arm continue; 863819281eeSdp-arm } 864819281eeSdp-arm 865819281eeSdp-arm if (access(file, F_OK) != 0 || fflag) { 866819281eeSdp-arm if (verbose) 867819281eeSdp-arm log_dbgx("Unpacking %s", file); 868b04efcceSdp-arm write_image_to_file(image, file); 869819281eeSdp-arm } else { 870819281eeSdp-arm log_warnx("File %s already exists, use --force to overwrite it", 871819281eeSdp-arm file); 872819281eeSdp-arm } 873819281eeSdp-arm } 874819281eeSdp-arm 875819281eeSdp-arm free_images(); 876819281eeSdp-arm return 0; 877819281eeSdp-arm } 878819281eeSdp-arm 879819281eeSdp-arm static void unpack_usage(void) 880819281eeSdp-arm { 881819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 882819281eeSdp-arm 883819281eeSdp-arm printf("fiptool unpack [--force] [--out <path>] [opts] FIP_FILENAME\n"); 884819281eeSdp-arm printf(" --force\tIf the output file already exists, use --force to " 885819281eeSdp-arm "overwrite it.\n"); 886819281eeSdp-arm printf(" --out path\tSet the output directory path.\n"); 887819281eeSdp-arm fputc('\n', stderr); 888819281eeSdp-arm printf("Specific images are unpacked with the following options:\n"); 889819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 890819281eeSdp-arm printf(" --%-16s FILENAME\t%s\n", toc_entry->cmdline_name, 891819281eeSdp-arm toc_entry->name); 892819281eeSdp-arm fputc('\n', stderr); 893819281eeSdp-arm printf("If no options are provided, all images will be unpacked.\n"); 89485ee2778Sdp-arm exit(1); 895819281eeSdp-arm } 896819281eeSdp-arm 897819281eeSdp-arm static int remove_cmd(int argc, char *argv[]) 898819281eeSdp-arm { 899*e0f083a0Sdp-arm struct option *opts = NULL; 900*e0f083a0Sdp-arm size_t nr_opts = 0; 901819281eeSdp-arm char outfile[FILENAME_MAX] = { 0 }; 902819281eeSdp-arm fip_toc_header_t toc_header; 903*e0f083a0Sdp-arm image_desc_t *desc; 904819281eeSdp-arm int fflag = 0; 905819281eeSdp-arm 906819281eeSdp-arm if (argc < 2) 90785ee2778Sdp-arm remove_usage(); 908819281eeSdp-arm 909*e0f083a0Sdp-arm opts = fill_common_opts(opts, &nr_opts, no_argument); 910*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "force", no_argument, 'f'); 911*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, "out", required_argument, 'o'); 912*e0f083a0Sdp-arm opts = add_opt(opts, &nr_opts, NULL, 0, 0); 913819281eeSdp-arm 914819281eeSdp-arm while (1) { 91520f87e78Sdp-arm int c, opt_index = 0; 916819281eeSdp-arm 917819281eeSdp-arm c = getopt_long(argc, argv, "fo:", opts, &opt_index); 918819281eeSdp-arm if (c == -1) 919819281eeSdp-arm break; 920819281eeSdp-arm 921819281eeSdp-arm switch (c) { 922*e0f083a0Sdp-arm case OPT_TOC_ENTRY: { 923*e0f083a0Sdp-arm image_desc_t *desc; 924*e0f083a0Sdp-arm 925*e0f083a0Sdp-arm desc = lookup_image_desc_from_opt(opts[opt_index].name); 926*e0f083a0Sdp-arm assert(desc != NULL); 927*e0f083a0Sdp-arm desc->action = DO_REMOVE; 928*e0f083a0Sdp-arm desc->action_arg = NULL; 929819281eeSdp-arm break; 930*e0f083a0Sdp-arm } 931819281eeSdp-arm case 'f': 932819281eeSdp-arm fflag = 1; 933819281eeSdp-arm break; 934819281eeSdp-arm case 'o': 935819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", optarg); 936819281eeSdp-arm break; 937819281eeSdp-arm default: 93885ee2778Sdp-arm remove_usage(); 939819281eeSdp-arm } 940819281eeSdp-arm } 941819281eeSdp-arm argc -= optind; 942819281eeSdp-arm argv += optind; 943*e0f083a0Sdp-arm free(opts); 944819281eeSdp-arm 945819281eeSdp-arm if (argc == 0) 94685ee2778Sdp-arm remove_usage(); 947819281eeSdp-arm 948819281eeSdp-arm if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag) 949819281eeSdp-arm log_errx("File %s already exists, use --force to overwrite it", 950819281eeSdp-arm outfile); 951819281eeSdp-arm 952819281eeSdp-arm if (outfile[0] == '\0') 953819281eeSdp-arm snprintf(outfile, sizeof(outfile), "%s", argv[0]); 954819281eeSdp-arm 955819281eeSdp-arm parse_fip(argv[0], &toc_header); 956819281eeSdp-arm 957*e0f083a0Sdp-arm for (desc = image_desc_head; desc != NULL; desc = desc->next) { 958b04efcceSdp-arm image_t *image; 959b04efcceSdp-arm 960*e0f083a0Sdp-arm if (desc->action != DO_REMOVE) 961819281eeSdp-arm continue; 962*e0f083a0Sdp-arm image = lookup_image_from_uuid(&desc->uuid); 963b04efcceSdp-arm if (image != NULL) { 964819281eeSdp-arm if (verbose) 965819281eeSdp-arm log_dbgx("Removing %s.bin", 966*e0f083a0Sdp-arm desc->cmdline_name); 967b04efcceSdp-arm remove_image(image); 968819281eeSdp-arm } else { 969819281eeSdp-arm log_warnx("Requested image %s.bin is not in %s", 970*e0f083a0Sdp-arm desc->cmdline_name, argv[0]); 971819281eeSdp-arm } 972819281eeSdp-arm } 973819281eeSdp-arm 974819281eeSdp-arm pack_images(outfile, toc_header.flags); 975819281eeSdp-arm free_images(); 976819281eeSdp-arm return 0; 977819281eeSdp-arm } 978819281eeSdp-arm 979819281eeSdp-arm static void remove_usage(void) 980819281eeSdp-arm { 981819281eeSdp-arm toc_entry_t *toc_entry = toc_entries; 982819281eeSdp-arm 983819281eeSdp-arm printf("fiptool remove [--force] [--out FIP_FILENAME] [opts] FIP_FILENAME\n"); 984819281eeSdp-arm printf(" --force\t\tIf the output FIP file already exists, use --force to " 985819281eeSdp-arm "overwrite it.\n"); 986819281eeSdp-arm printf(" --out FIP_FILENAME\tSet an alternative output FIP file.\n"); 987819281eeSdp-arm fputc('\n', stderr); 988819281eeSdp-arm printf("Specific images are removed with the following options:\n"); 989819281eeSdp-arm for (; toc_entry->cmdline_name != NULL; toc_entry++) 990819281eeSdp-arm printf(" --%-16s\t%s\n", toc_entry->cmdline_name, 991819281eeSdp-arm toc_entry->name); 99285ee2778Sdp-arm exit(1); 993819281eeSdp-arm } 994819281eeSdp-arm 995819281eeSdp-arm static int version_cmd(int argc, char *argv[]) 996819281eeSdp-arm { 997819281eeSdp-arm #ifdef VERSION 998819281eeSdp-arm puts(VERSION); 999819281eeSdp-arm #else 1000819281eeSdp-arm /* If built from fiptool directory, VERSION is not set. */ 1001819281eeSdp-arm puts("Unknown version"); 1002819281eeSdp-arm #endif 1003819281eeSdp-arm return 0; 1004819281eeSdp-arm } 1005819281eeSdp-arm 1006819281eeSdp-arm static void version_usage(void) 1007819281eeSdp-arm { 1008819281eeSdp-arm printf("fiptool version\n"); 100985ee2778Sdp-arm exit(1); 1010819281eeSdp-arm } 1011819281eeSdp-arm 1012819281eeSdp-arm static int help_cmd(int argc, char *argv[]) 1013819281eeSdp-arm { 1014819281eeSdp-arm int i; 1015819281eeSdp-arm 1016819281eeSdp-arm if (argc < 2) 1017819281eeSdp-arm usage(); 1018819281eeSdp-arm argc--, argv++; 1019819281eeSdp-arm 1020819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1021819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0 && 102285ee2778Sdp-arm cmds[i].usage != NULL) 1023819281eeSdp-arm cmds[i].usage(); 1024819281eeSdp-arm } 1025819281eeSdp-arm if (i == NELEM(cmds)) 1026819281eeSdp-arm printf("No help for subcommand '%s'\n", argv[0]); 1027819281eeSdp-arm return 0; 1028819281eeSdp-arm } 1029819281eeSdp-arm 1030819281eeSdp-arm static void usage(void) 1031819281eeSdp-arm { 1032819281eeSdp-arm printf("usage: [--verbose] fiptool <command> [<args>]\n"); 1033819281eeSdp-arm printf("Global options supported:\n"); 1034819281eeSdp-arm printf(" --verbose\tEnable verbose output for all commands.\n"); 1035819281eeSdp-arm fputc('\n', stderr); 1036819281eeSdp-arm printf("Commands supported:\n"); 1037819281eeSdp-arm printf(" info\t\tList images contained in FIP.\n"); 1038819281eeSdp-arm printf(" create\tCreate a new FIP with the given images.\n"); 1039819281eeSdp-arm printf(" update\tUpdate an existing FIP with the given images.\n"); 1040819281eeSdp-arm printf(" unpack\tUnpack images from FIP.\n"); 1041819281eeSdp-arm printf(" remove\tRemove images from FIP.\n"); 1042819281eeSdp-arm printf(" version\tShow fiptool version.\n"); 1043819281eeSdp-arm printf(" help\t\tShow help for given command.\n"); 1044819281eeSdp-arm exit(1); 1045819281eeSdp-arm } 1046819281eeSdp-arm 1047819281eeSdp-arm int main(int argc, char *argv[]) 1048819281eeSdp-arm { 1049819281eeSdp-arm int i, ret = 0; 1050819281eeSdp-arm 1051cc672bb2Sdp-arm while (1) { 1052cc672bb2Sdp-arm int c, opt_index = 0; 1053cc672bb2Sdp-arm static struct option opts[] = { 1054cc672bb2Sdp-arm { "verbose", no_argument, NULL, 'v' }, 1055cc672bb2Sdp-arm { NULL, no_argument, NULL, 0 } 1056cc672bb2Sdp-arm }; 1057819281eeSdp-arm 1058cc672bb2Sdp-arm /* 1059cc672bb2Sdp-arm * Set POSIX mode so getopt stops at the first non-option 1060cc672bb2Sdp-arm * which is the subcommand. 1061cc672bb2Sdp-arm */ 1062cc672bb2Sdp-arm c = getopt_long(argc, argv, "+v", opts, &opt_index); 1063cc672bb2Sdp-arm if (c == -1) 1064cc672bb2Sdp-arm break; 1065cc672bb2Sdp-arm 1066cc672bb2Sdp-arm switch (c) { 1067cc672bb2Sdp-arm case 'v': 1068819281eeSdp-arm verbose = 1; 1069cc672bb2Sdp-arm break; 1070cc672bb2Sdp-arm default: 1071c9cb4089SMasahiro Yamada usage(); 1072819281eeSdp-arm } 1073cc672bb2Sdp-arm } 1074cc672bb2Sdp-arm argc -= optind; 1075cc672bb2Sdp-arm argv += optind; 1076cc672bb2Sdp-arm /* Reset optind for subsequent getopt processing. */ 1077cc672bb2Sdp-arm optind = 0; 1078cc672bb2Sdp-arm 1079cc672bb2Sdp-arm if (argc == 0) 1080cc672bb2Sdp-arm usage(); 1081819281eeSdp-arm 1082*e0f083a0Sdp-arm fill_image_descs(); 1083819281eeSdp-arm for (i = 0; i < NELEM(cmds); i++) { 1084819281eeSdp-arm if (strcmp(cmds[i].name, argv[0]) == 0) { 1085819281eeSdp-arm ret = cmds[i].handler(argc, argv); 1086819281eeSdp-arm break; 1087819281eeSdp-arm } 1088819281eeSdp-arm } 1089819281eeSdp-arm if (i == NELEM(cmds)) 1090819281eeSdp-arm usage(); 1091*e0f083a0Sdp-arm free_image_descs(); 1092819281eeSdp-arm return ret; 1093819281eeSdp-arm } 1094