xref: /rk3399_ARM-atf/tools/fiptool/fiptool.c (revision 06e69f7c94637c693ea5eb26038096c196d10f07)
1819281eeSdp-arm /*
2*06e69f7cSAntonio Borneo  * Copyright (c) 2016-2023, ARM Limited and Contributors. All rights reserved.
3819281eeSdp-arm  *
482cb2c1aSdp-arm  * SPDX-License-Identifier: BSD-3-Clause
5819281eeSdp-arm  */
6819281eeSdp-arm 
7*06e69f7cSAntonio Borneo #ifndef _MSC_VER
8*06e69f7cSAntonio Borneo #include <sys/mount.h>
9*06e69f7cSAntonio Borneo #endif
10819281eeSdp-arm #include <sys/types.h>
11819281eeSdp-arm #include <sys/stat.h>
12819281eeSdp-arm 
13819281eeSdp-arm #include <assert.h>
14819281eeSdp-arm #include <errno.h>
15819281eeSdp-arm #include <limits.h>
16819281eeSdp-arm #include <stdarg.h>
17819281eeSdp-arm #include <stdint.h>
18819281eeSdp-arm #include <stdio.h>
19819281eeSdp-arm #include <stdlib.h>
20819281eeSdp-arm #include <string.h>
212a6c1a8fSMasahiro Yamada 
22819281eeSdp-arm #include "fiptool.h"
23819281eeSdp-arm #include "tbbr_config.h"
24819281eeSdp-arm 
25819281eeSdp-arm #define OPT_TOC_ENTRY 0
26819281eeSdp-arm #define OPT_PLAT_TOC_FLAGS 1
271c75d5dfSMasahiro Yamada #define OPT_ALIGN 2
28819281eeSdp-arm 
29819281eeSdp-arm static int info_cmd(int argc, char *argv[]);
304e500525SLeonardo Sandoval static void info_usage(int);
31819281eeSdp-arm static int create_cmd(int argc, char *argv[]);
324e500525SLeonardo Sandoval static void create_usage(int);
33819281eeSdp-arm static int update_cmd(int argc, char *argv[]);
344e500525SLeonardo Sandoval static void update_usage(int);
35819281eeSdp-arm static int unpack_cmd(int argc, char *argv[]);
364e500525SLeonardo Sandoval static void unpack_usage(int);
37819281eeSdp-arm static int remove_cmd(int argc, char *argv[]);
384e500525SLeonardo Sandoval static void remove_usage(int);
39819281eeSdp-arm static int version_cmd(int argc, char *argv[]);
404e500525SLeonardo Sandoval static void version_usage(int);
41819281eeSdp-arm static int help_cmd(int argc, char *argv[]);
42819281eeSdp-arm static void usage(void);
43819281eeSdp-arm 
44819281eeSdp-arm /* Available subcommands. */
45819281eeSdp-arm static cmd_t cmds[] = {
46819281eeSdp-arm 	{ .name = "info",    .handler = info_cmd,    .usage = info_usage    },
47819281eeSdp-arm 	{ .name = "create",  .handler = create_cmd,  .usage = create_usage  },
48819281eeSdp-arm 	{ .name = "update",  .handler = update_cmd,  .usage = update_usage  },
49819281eeSdp-arm 	{ .name = "unpack",  .handler = unpack_cmd,  .usage = unpack_usage  },
50819281eeSdp-arm 	{ .name = "remove",  .handler = remove_cmd,  .usage = remove_usage  },
51819281eeSdp-arm 	{ .name = "version", .handler = version_cmd, .usage = version_usage },
52819281eeSdp-arm 	{ .name = "help",    .handler = help_cmd,    .usage = NULL          },
53819281eeSdp-arm };
54819281eeSdp-arm 
55e0f083a0Sdp-arm static image_desc_t *image_desc_head;
56e0f083a0Sdp-arm static size_t nr_image_descs;
5703364865SRoberto Vargas static const uuid_t uuid_null;
58819281eeSdp-arm static int verbose;
59819281eeSdp-arm 
6060b499feSdp-arm static void vlog(int prio, const char *msg, va_list ap)
61819281eeSdp-arm {
62819281eeSdp-arm 	char *prefix[] = { "DEBUG", "WARN", "ERROR" };
63819281eeSdp-arm 
64819281eeSdp-arm 	fprintf(stderr, "%s: ", prefix[prio]);
65819281eeSdp-arm 	vfprintf(stderr, msg, ap);
66819281eeSdp-arm 	fputc('\n', stderr);
67819281eeSdp-arm }
68819281eeSdp-arm 
6960b499feSdp-arm static void log_dbgx(const char *msg, ...)
70819281eeSdp-arm {
71819281eeSdp-arm 	va_list ap;
72819281eeSdp-arm 
73819281eeSdp-arm 	va_start(ap, msg);
74819281eeSdp-arm 	vlog(LOG_DBG, msg, ap);
75819281eeSdp-arm 	va_end(ap);
76819281eeSdp-arm }
77819281eeSdp-arm 
7860b499feSdp-arm static void log_warnx(const char *msg, ...)
79819281eeSdp-arm {
80819281eeSdp-arm 	va_list ap;
81819281eeSdp-arm 
82819281eeSdp-arm 	va_start(ap, msg);
83819281eeSdp-arm 	vlog(LOG_WARN, msg, ap);
84819281eeSdp-arm 	va_end(ap);
85819281eeSdp-arm }
86819281eeSdp-arm 
8760b499feSdp-arm static void log_err(const char *msg, ...)
88819281eeSdp-arm {
89819281eeSdp-arm 	char buf[512];
90819281eeSdp-arm 	va_list ap;
91819281eeSdp-arm 
92819281eeSdp-arm 	va_start(ap, msg);
93819281eeSdp-arm 	snprintf(buf, sizeof(buf), "%s: %s", msg, strerror(errno));
94819281eeSdp-arm 	vlog(LOG_ERR, buf, ap);
95819281eeSdp-arm 	va_end(ap);
96819281eeSdp-arm 	exit(1);
97819281eeSdp-arm }
98819281eeSdp-arm 
9960b499feSdp-arm static void log_errx(const char *msg, ...)
100819281eeSdp-arm {
101819281eeSdp-arm 	va_list ap;
102819281eeSdp-arm 
103819281eeSdp-arm 	va_start(ap, msg);
104819281eeSdp-arm 	vlog(LOG_ERR, msg, ap);
105819281eeSdp-arm 	va_end(ap);
106819281eeSdp-arm 	exit(1);
107819281eeSdp-arm }
108819281eeSdp-arm 
109a22f6285Sdp-arm static char *xstrdup(const char *s, const char *msg)
110a22f6285Sdp-arm {
111a22f6285Sdp-arm 	char *d;
112a22f6285Sdp-arm 
113a22f6285Sdp-arm 	d = strdup(s);
114a22f6285Sdp-arm 	if (d == NULL)
1159fc9ff1fSdp-arm 		log_errx("strdup: %s", msg);
116a22f6285Sdp-arm 	return d;
117a22f6285Sdp-arm }
118a22f6285Sdp-arm 
119a22f6285Sdp-arm static void *xmalloc(size_t size, const char *msg)
120a22f6285Sdp-arm {
121a22f6285Sdp-arm 	void *d;
122a22f6285Sdp-arm 
123a22f6285Sdp-arm 	d = malloc(size);
124a22f6285Sdp-arm 	if (d == NULL)
1259fc9ff1fSdp-arm 		log_errx("malloc: %s", msg);
126a22f6285Sdp-arm 	return d;
127a22f6285Sdp-arm }
128a22f6285Sdp-arm 
129696ccba6SMasahiro Yamada static void *xzalloc(size_t size, const char *msg)
130696ccba6SMasahiro Yamada {
131696ccba6SMasahiro Yamada 	return memset(xmalloc(size, msg), 0, size);
132696ccba6SMasahiro Yamada }
133696ccba6SMasahiro Yamada 
134a1da83f5SMasahiro Yamada static void xfwrite(void *buf, size_t size, FILE *fp, const char *filename)
135a1da83f5SMasahiro Yamada {
136a1da83f5SMasahiro Yamada 	if (fwrite(buf, 1, size, fp) != size)
137a1da83f5SMasahiro Yamada 		log_errx("Failed to write %s", filename);
138a1da83f5SMasahiro Yamada }
139a1da83f5SMasahiro Yamada 
140e0f083a0Sdp-arm static image_desc_t *new_image_desc(const uuid_t *uuid,
141e0f083a0Sdp-arm     const char *name, const char *cmdline_name)
142e0f083a0Sdp-arm {
143e0f083a0Sdp-arm 	image_desc_t *desc;
144e0f083a0Sdp-arm 
145696ccba6SMasahiro Yamada 	desc = xzalloc(sizeof(*desc),
146e0f083a0Sdp-arm 	    "failed to allocate memory for image descriptor");
147e0f083a0Sdp-arm 	memcpy(&desc->uuid, uuid, sizeof(uuid_t));
148e0f083a0Sdp-arm 	desc->name = xstrdup(name,
149e0f083a0Sdp-arm 	    "failed to allocate memory for image name");
150e0f083a0Sdp-arm 	desc->cmdline_name = xstrdup(cmdline_name,
151e0f083a0Sdp-arm 	    "failed to allocate memory for image command line name");
152e0f083a0Sdp-arm 	desc->action = DO_UNSPEC;
153e0f083a0Sdp-arm 	return desc;
154e0f083a0Sdp-arm }
155e0f083a0Sdp-arm 
156d02fcebeSdp-arm static void set_image_desc_action(image_desc_t *desc, int action,
157d02fcebeSdp-arm     const char *arg)
158d02fcebeSdp-arm {
159d02fcebeSdp-arm 	assert(desc != NULL);
160d02fcebeSdp-arm 
16196851114SEvan Lloyd 	if (desc->action_arg != (char *)DO_UNSPEC)
162d02fcebeSdp-arm 		free(desc->action_arg);
163d02fcebeSdp-arm 	desc->action = action;
164d02fcebeSdp-arm 	desc->action_arg = NULL;
165d02fcebeSdp-arm 	if (arg != NULL)
166d02fcebeSdp-arm 		desc->action_arg = xstrdup(arg,
167d02fcebeSdp-arm 		    "failed to allocate memory for argument");
168d02fcebeSdp-arm }
169d02fcebeSdp-arm 
170e0f083a0Sdp-arm static void free_image_desc(image_desc_t *desc)
171e0f083a0Sdp-arm {
172e0f083a0Sdp-arm 	free(desc->name);
173e0f083a0Sdp-arm 	free(desc->cmdline_name);
174e0f083a0Sdp-arm 	free(desc->action_arg);
1758d283231SJonathan Wright 	if (desc->image) {
1768d283231SJonathan Wright 		free(desc->image->buffer);
177b9589fe5Sdp-arm 		free(desc->image);
1788d283231SJonathan Wright 	}
179e0f083a0Sdp-arm 	free(desc);
180e0f083a0Sdp-arm }
181e0f083a0Sdp-arm 
182e0f083a0Sdp-arm static void add_image_desc(image_desc_t *desc)
183e0f083a0Sdp-arm {
18411c0a4ffSMasahiro Yamada 	image_desc_t **p = &image_desc_head;
18511c0a4ffSMasahiro Yamada 
18611c0a4ffSMasahiro Yamada 	while (*p)
18711c0a4ffSMasahiro Yamada 		p = &(*p)->next;
18811c0a4ffSMasahiro Yamada 
189e9e0d287SMasahiro Yamada 	assert(*p == NULL);
19011c0a4ffSMasahiro Yamada 	*p = desc;
191e0f083a0Sdp-arm 	nr_image_descs++;
192e0f083a0Sdp-arm }
193e0f083a0Sdp-arm 
194e0f083a0Sdp-arm static void free_image_descs(void)
195e0f083a0Sdp-arm {
196e0f083a0Sdp-arm 	image_desc_t *desc = image_desc_head, *tmp;
197e0f083a0Sdp-arm 
198e0f083a0Sdp-arm 	while (desc != NULL) {
199e0f083a0Sdp-arm 		tmp = desc->next;
200e0f083a0Sdp-arm 		free_image_desc(desc);
201e0f083a0Sdp-arm 		desc = tmp;
202e0f083a0Sdp-arm 		nr_image_descs--;
203e0f083a0Sdp-arm 	}
204e0f083a0Sdp-arm 	assert(nr_image_descs == 0);
205e0f083a0Sdp-arm }
206e0f083a0Sdp-arm 
207e0f083a0Sdp-arm static void fill_image_descs(void)
208e0f083a0Sdp-arm {
209e0f083a0Sdp-arm 	toc_entry_t *toc_entry;
210e0f083a0Sdp-arm 
211e0f083a0Sdp-arm 	for (toc_entry = toc_entries;
212e0f083a0Sdp-arm 	     toc_entry->cmdline_name != NULL;
213e0f083a0Sdp-arm 	     toc_entry++) {
214e0f083a0Sdp-arm 		image_desc_t *desc;
215e0f083a0Sdp-arm 
216e0f083a0Sdp-arm 		desc = new_image_desc(&toc_entry->uuid,
217e0f083a0Sdp-arm 		    toc_entry->name,
218e0f083a0Sdp-arm 		    toc_entry->cmdline_name);
219e0f083a0Sdp-arm 		add_image_desc(desc);
220e0f083a0Sdp-arm 	}
2213527d6d2SPankaj Gupta #ifdef PLAT_DEF_FIP_UUID
2223527d6d2SPankaj Gupta 	for (toc_entry = plat_def_toc_entries;
2233527d6d2SPankaj Gupta 	     toc_entry->cmdline_name != NULL;
2243527d6d2SPankaj Gupta 	     toc_entry++) {
2253527d6d2SPankaj Gupta 		image_desc_t *desc;
2263527d6d2SPankaj Gupta 
2273527d6d2SPankaj Gupta 		desc = new_image_desc(&toc_entry->uuid,
2283527d6d2SPankaj Gupta 		    toc_entry->name,
2293527d6d2SPankaj Gupta 		    toc_entry->cmdline_name);
2303527d6d2SPankaj Gupta 		add_image_desc(desc);
2313527d6d2SPankaj Gupta 	}
2323527d6d2SPankaj Gupta #endif
233e0f083a0Sdp-arm }
234e0f083a0Sdp-arm 
235e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_uuid(const uuid_t *uuid)
236819281eeSdp-arm {
237e0f083a0Sdp-arm 	image_desc_t *desc;
238819281eeSdp-arm 
239e0f083a0Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next)
240e0f083a0Sdp-arm 		if (memcmp(&desc->uuid, uuid, sizeof(uuid_t)) == 0)
241e0f083a0Sdp-arm 			return desc;
242819281eeSdp-arm 	return NULL;
243819281eeSdp-arm }
244819281eeSdp-arm 
245e0f083a0Sdp-arm static image_desc_t *lookup_image_desc_from_opt(const char *opt)
246e0f083a0Sdp-arm {
247e0f083a0Sdp-arm 	image_desc_t *desc;
248e0f083a0Sdp-arm 
249e0f083a0Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next)
250e0f083a0Sdp-arm 		if (strcmp(desc->cmdline_name, opt) == 0)
251e0f083a0Sdp-arm 			return desc;
252e0f083a0Sdp-arm 	return NULL;
253e0f083a0Sdp-arm }
254e0f083a0Sdp-arm 
255fcab6bbeSdp-arm static void uuid_to_str(char *s, size_t len, const uuid_t *u)
256fcab6bbeSdp-arm {
257fcab6bbeSdp-arm 	assert(len >= (_UUID_STR_LEN + 1));
258fcab6bbeSdp-arm 
25903364865SRoberto Vargas 	snprintf(s, len,
26003364865SRoberto Vargas 	    "%02X%02X%02X%02X-%02X%02X-%02X%02X-%04X-%04X%04X%04X",
26103364865SRoberto Vargas 	    u->time_low[0], u->time_low[1], u->time_low[2], u->time_low[3],
26203364865SRoberto Vargas 	    u->time_mid[0], u->time_mid[1],
26303364865SRoberto Vargas 	    u->time_hi_and_version[0], u->time_hi_and_version[1],
26403364865SRoberto Vargas 	    (u->clock_seq_hi_and_reserved << 8) | u->clock_seq_low,
26503364865SRoberto Vargas 	    (u->node[0] << 8) | u->node[1],
26603364865SRoberto Vargas 	    (u->node[2] << 8) | u->node[3],
26703364865SRoberto Vargas 	    (u->node[4] << 8) | u->node[5]);
268fcab6bbeSdp-arm }
269fcab6bbeSdp-arm 
270fcab6bbeSdp-arm static void uuid_from_str(uuid_t *u, const char *s)
271fcab6bbeSdp-arm {
272fcab6bbeSdp-arm 	int n;
273fcab6bbeSdp-arm 
274fcab6bbeSdp-arm 	if (s == NULL)
275fcab6bbeSdp-arm 		log_errx("UUID cannot be NULL");
276fcab6bbeSdp-arm 	if (strlen(s) != _UUID_STR_LEN)
277fcab6bbeSdp-arm 		log_errx("Invalid UUID: %s", s);
278fcab6bbeSdp-arm 
279fcab6bbeSdp-arm 	n = sscanf(s,
28003364865SRoberto Vargas 	    "%2hhx%2hhx%2hhx%2hhx-%2hhx%2hhx-%2hhx%2hhx-%2hhx%2hhx-%2hhx%2hhx%2hhx%2hhx%2hhx%2hhx",
28103364865SRoberto Vargas 	    &u->time_low[0], &u->time_low[1], &u->time_low[2], &u->time_low[3],
28203364865SRoberto Vargas 	    &u->time_mid[0], &u->time_mid[1],
28303364865SRoberto Vargas 	    &u->time_hi_and_version[0], &u->time_hi_and_version[1],
28403364865SRoberto Vargas 	    &u->clock_seq_hi_and_reserved, &u->clock_seq_low,
28503364865SRoberto Vargas 	    &u->node[0], &u->node[1],
28603364865SRoberto Vargas 	    &u->node[2], &u->node[3],
28703364865SRoberto Vargas 	    &u->node[4], &u->node[5]);
288fcab6bbeSdp-arm 	/*
289e56b8dc8SAndre Przywara 	 * Given the format specifier above, we expect 16 items to be scanned
290fcab6bbeSdp-arm 	 * for a properly formatted UUID.
291fcab6bbeSdp-arm 	 */
292e56b8dc8SAndre Przywara 	if (n != 16)
293fcab6bbeSdp-arm 		log_errx("Invalid UUID: %s", s);
294fcab6bbeSdp-arm }
295fcab6bbeSdp-arm 
29660b499feSdp-arm static int parse_fip(const char *filename, fip_toc_header_t *toc_header_out)
297819281eeSdp-arm {
29896851114SEvan Lloyd 	struct BLD_PLAT_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 	int terminated = 0;
304*06e69f7cSAntonio Borneo 	size_t st_size;
305819281eeSdp-arm 
30655745deaSEvan Lloyd 	fp = fopen(filename, "rb");
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 
313*06e69f7cSAntonio Borneo 	st_size = st.st_size;
314*06e69f7cSAntonio Borneo 
315*06e69f7cSAntonio Borneo #ifdef BLKGETSIZE64
316*06e69f7cSAntonio Borneo 	if ((st.st_mode & S_IFBLK) != 0)
317*06e69f7cSAntonio Borneo 		if (ioctl(fileno(fp), BLKGETSIZE64, &st_size) == -1)
318*06e69f7cSAntonio Borneo 			log_err("ioctl %s", filename);
319*06e69f7cSAntonio Borneo #endif
320*06e69f7cSAntonio Borneo 
321*06e69f7cSAntonio Borneo 	buf = xmalloc(st_size, "failed to load file into memory");
322*06e69f7cSAntonio Borneo 	if (fread(buf, 1, st_size, fp) != st_size)
323819281eeSdp-arm 		log_errx("Failed to read %s", filename);
324*06e69f7cSAntonio Borneo 	bufend = buf + st_size;
325819281eeSdp-arm 	fclose(fp);
326819281eeSdp-arm 
327*06e69f7cSAntonio Borneo 	if (st_size < sizeof(fip_toc_header_t))
328819281eeSdp-arm 		log_errx("FIP %s is truncated", filename);
329819281eeSdp-arm 
330819281eeSdp-arm 	toc_header = (fip_toc_header_t *)buf;
331819281eeSdp-arm 	toc_entry = (fip_toc_entry_t *)(toc_header + 1);
332819281eeSdp-arm 
333819281eeSdp-arm 	if (toc_header->name != TOC_HEADER_NAME)
334819281eeSdp-arm 		log_errx("%s is not a FIP file", filename);
335819281eeSdp-arm 
336819281eeSdp-arm 	/* Return the ToC header if the caller wants it. */
337819281eeSdp-arm 	if (toc_header_out != NULL)
338819281eeSdp-arm 		*toc_header_out = *toc_header;
339819281eeSdp-arm 
340819281eeSdp-arm 	/* Walk through each ToC entry in the file. */
341819281eeSdp-arm 	while ((char *)toc_entry + sizeof(*toc_entry) - 1 < bufend) {
342fcab6bbeSdp-arm 		image_t *image;
343fcab6bbeSdp-arm 		image_desc_t *desc;
344fcab6bbeSdp-arm 
345819281eeSdp-arm 		/* Found the ToC terminator, we are done. */
346819281eeSdp-arm 		if (memcmp(&toc_entry->uuid, &uuid_null, sizeof(uuid_t)) == 0) {
347819281eeSdp-arm 			terminated = 1;
348819281eeSdp-arm 			break;
349819281eeSdp-arm 		}
350819281eeSdp-arm 
351819281eeSdp-arm 		/*
352819281eeSdp-arm 		 * Build a new image out of the ToC entry and add it to the
353819281eeSdp-arm 		 * table of images.
354819281eeSdp-arm 		 */
35511c0a4ffSMasahiro Yamada 		image = xzalloc(sizeof(*image),
356a22f6285Sdp-arm 		    "failed to allocate memory for image");
35765caa3d0SMasahiro Yamada 		image->toc_e = *toc_entry;
358a22f6285Sdp-arm 		image->buffer = xmalloc(toc_entry->size,
359a22f6285Sdp-arm 		    "failed to allocate image buffer, is FIP file corrupted?");
360819281eeSdp-arm 		/* Overflow checks before memory copy. */
361819281eeSdp-arm 		if (toc_entry->size > (uint64_t)-1 - toc_entry->offset_address)
362*06e69f7cSAntonio Borneo 			log_errx("FIP %s is corrupted: entry size exceeds 64 bit address space",
363*06e69f7cSAntonio Borneo 				filename);
364*06e69f7cSAntonio Borneo 		if (toc_entry->size + toc_entry->offset_address > st_size)
365*06e69f7cSAntonio Borneo 			log_errx("FIP %s is corrupted: entry size exceeds FIP file size",
366*06e69f7cSAntonio Borneo 				filename);
367819281eeSdp-arm 
368819281eeSdp-arm 		memcpy(image->buffer, buf + toc_entry->offset_address,
369819281eeSdp-arm 		    toc_entry->size);
370819281eeSdp-arm 
371fcab6bbeSdp-arm 		/* If this is an unknown image, create a descriptor for it. */
37265caa3d0SMasahiro Yamada 		desc = lookup_image_desc_from_uuid(&toc_entry->uuid);
373fcab6bbeSdp-arm 		if (desc == NULL) {
374fcab6bbeSdp-arm 			char name[_UUID_STR_LEN + 1], filename[PATH_MAX];
375fcab6bbeSdp-arm 
37665caa3d0SMasahiro Yamada 			uuid_to_str(name, sizeof(name), &toc_entry->uuid);
377fcab6bbeSdp-arm 			snprintf(filename, sizeof(filename), "%s%s",
378fcab6bbeSdp-arm 			    name, ".bin");
37965caa3d0SMasahiro Yamada 			desc = new_image_desc(&toc_entry->uuid, name, "blob");
380fcab6bbeSdp-arm 			desc->action = DO_UNPACK;
381fcab6bbeSdp-arm 			desc->action_arg = xstrdup(filename,
382fcab6bbeSdp-arm 			    "failed to allocate memory for blob filename");
383fcab6bbeSdp-arm 			add_image_desc(desc);
384fcab6bbeSdp-arm 		}
385fcab6bbeSdp-arm 
386b9589fe5Sdp-arm 		assert(desc->image == NULL);
387b9589fe5Sdp-arm 		desc->image = image;
388819281eeSdp-arm 
389819281eeSdp-arm 		toc_entry++;
390819281eeSdp-arm 	}
391819281eeSdp-arm 
392819281eeSdp-arm 	if (terminated == 0)
393819281eeSdp-arm 		log_errx("FIP %s does not have a ToC terminator entry",
394819281eeSdp-arm 		    filename);
395819281eeSdp-arm 	free(buf);
396819281eeSdp-arm 	return 0;
397819281eeSdp-arm }
398819281eeSdp-arm 
39960b499feSdp-arm static image_t *read_image_from_file(const uuid_t *uuid, const char *filename)
400819281eeSdp-arm {
40196851114SEvan Lloyd 	struct BLD_PLAT_STAT st;
402819281eeSdp-arm 	image_t *image;
403819281eeSdp-arm 	FILE *fp;
404819281eeSdp-arm 
405b04efcceSdp-arm 	assert(uuid != NULL);
40696851114SEvan Lloyd 	assert(filename != NULL);
407b04efcceSdp-arm 
40855745deaSEvan Lloyd 	fp = fopen(filename, "rb");
409819281eeSdp-arm 	if (fp == NULL)
410819281eeSdp-arm 		log_err("fopen %s", filename);
411819281eeSdp-arm 
412819281eeSdp-arm 	if (fstat(fileno(fp), &st) == -1)
413819281eeSdp-arm 		log_errx("fstat %s", filename);
414819281eeSdp-arm 
41511c0a4ffSMasahiro Yamada 	image = xzalloc(sizeof(*image), "failed to allocate memory for image");
41665caa3d0SMasahiro Yamada 	image->toc_e.uuid = *uuid;
417a22f6285Sdp-arm 	image->buffer = xmalloc(st.st_size, "failed to allocate image buffer");
418819281eeSdp-arm 	if (fread(image->buffer, 1, st.st_size, fp) != st.st_size)
419819281eeSdp-arm 		log_errx("Failed to read %s", filename);
42065caa3d0SMasahiro Yamada 	image->toc_e.size = st.st_size;
421819281eeSdp-arm 
422819281eeSdp-arm 	fclose(fp);
423819281eeSdp-arm 	return image;
424819281eeSdp-arm }
425819281eeSdp-arm 
42660b499feSdp-arm static int write_image_to_file(const image_t *image, const char *filename)
427819281eeSdp-arm {
428819281eeSdp-arm 	FILE *fp;
429819281eeSdp-arm 
43055745deaSEvan Lloyd 	fp = fopen(filename, "wb");
431819281eeSdp-arm 	if (fp == NULL)
432819281eeSdp-arm 		log_err("fopen");
43365caa3d0SMasahiro Yamada 	xfwrite(image->buffer, image->toc_e.size, fp, filename);
434819281eeSdp-arm 	fclose(fp);
435819281eeSdp-arm 	return 0;
436819281eeSdp-arm }
437819281eeSdp-arm 
438e0f083a0Sdp-arm static struct option *add_opt(struct option *opts, size_t *nr_opts,
439e0f083a0Sdp-arm     const char *name, int has_arg, int val)
440819281eeSdp-arm {
441e0f083a0Sdp-arm 	opts = realloc(opts, (*nr_opts + 1) * sizeof(*opts));
442e0f083a0Sdp-arm 	if (opts == NULL)
443e0f083a0Sdp-arm 		log_err("realloc");
444e0f083a0Sdp-arm 	opts[*nr_opts].name = name;
445e0f083a0Sdp-arm 	opts[*nr_opts].has_arg = has_arg;
446e0f083a0Sdp-arm 	opts[*nr_opts].flag = NULL;
447e0f083a0Sdp-arm 	opts[*nr_opts].val = val;
448e0f083a0Sdp-arm 	++*nr_opts;
449e0f083a0Sdp-arm 	return opts;
450819281eeSdp-arm }
451819281eeSdp-arm 
452e0f083a0Sdp-arm static struct option *fill_common_opts(struct option *opts, size_t *nr_opts,
453e0f083a0Sdp-arm     int has_arg)
454819281eeSdp-arm {
455e0f083a0Sdp-arm 	image_desc_t *desc;
456e0f083a0Sdp-arm 
457e0f083a0Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next)
458e0f083a0Sdp-arm 		opts = add_opt(opts, nr_opts, desc->cmdline_name, has_arg,
459e0f083a0Sdp-arm 		    OPT_TOC_ENTRY);
460e0f083a0Sdp-arm 	return opts;
461819281eeSdp-arm }
462819281eeSdp-arm 
463e0f083a0Sdp-arm static void md_print(const unsigned char *md, size_t len)
4649df69ba3Sdp-arm {
4659df69ba3Sdp-arm 	size_t i;
4669df69ba3Sdp-arm 
4679df69ba3Sdp-arm 	for (i = 0; i < len; i++)
4689df69ba3Sdp-arm 		printf("%02x", md[i]);
4699df69ba3Sdp-arm }
4709df69ba3Sdp-arm 
471819281eeSdp-arm static int info_cmd(int argc, char *argv[])
472819281eeSdp-arm {
473b9589fe5Sdp-arm 	image_desc_t *desc;
474819281eeSdp-arm 	fip_toc_header_t toc_header;
475819281eeSdp-arm 
476819281eeSdp-arm 	if (argc != 2)
4774e500525SLeonardo Sandoval 		info_usage(EXIT_FAILURE);
478819281eeSdp-arm 	argc--, argv++;
479819281eeSdp-arm 
480819281eeSdp-arm 	parse_fip(argv[0], &toc_header);
481819281eeSdp-arm 
482819281eeSdp-arm 	if (verbose) {
483819281eeSdp-arm 		log_dbgx("toc_header[name]: 0x%llX",
484819281eeSdp-arm 		    (unsigned long long)toc_header.name);
485819281eeSdp-arm 		log_dbgx("toc_header[serial_number]: 0x%llX",
486819281eeSdp-arm 		    (unsigned long long)toc_header.serial_number);
487819281eeSdp-arm 		log_dbgx("toc_header[flags]: 0x%llX",
488819281eeSdp-arm 		    (unsigned long long)toc_header.flags);
489819281eeSdp-arm 	}
490819281eeSdp-arm 
491b9589fe5Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next) {
492b9589fe5Sdp-arm 		image_t *image = desc->image;
493b04efcceSdp-arm 
494b9589fe5Sdp-arm 		if (image == NULL)
495b9589fe5Sdp-arm 			continue;
49665caa3d0SMasahiro Yamada 		printf("%s: offset=0x%llX, size=0x%llX, cmdline=\"--%s\"",
49765caa3d0SMasahiro Yamada 		       desc->name,
49865caa3d0SMasahiro Yamada 		       (unsigned long long)image->toc_e.offset_address,
49965caa3d0SMasahiro Yamada 		       (unsigned long long)image->toc_e.size,
500e0f083a0Sdp-arm 		       desc->cmdline_name);
50196851114SEvan Lloyd #ifndef _MSC_VER	/* We don't have SHA256 for Visual Studio. */
5029df69ba3Sdp-arm 		if (verbose) {
5039df69ba3Sdp-arm 			unsigned char md[SHA256_DIGEST_LENGTH];
5049df69ba3Sdp-arm 
50565caa3d0SMasahiro Yamada 			SHA256(image->buffer, image->toc_e.size, md);
5069df69ba3Sdp-arm 			printf(", sha256=");
5079df69ba3Sdp-arm 			md_print(md, sizeof(md));
5089df69ba3Sdp-arm 		}
50996851114SEvan Lloyd #endif
510819281eeSdp-arm 		putchar('\n');
511819281eeSdp-arm 	}
512819281eeSdp-arm 
513819281eeSdp-arm 	return 0;
514819281eeSdp-arm }
515819281eeSdp-arm 
5164e500525SLeonardo Sandoval static void info_usage(int exit_status)
517819281eeSdp-arm {
518819281eeSdp-arm 	printf("fiptool info FIP_FILENAME\n");
5194e500525SLeonardo Sandoval 	exit(exit_status);
520819281eeSdp-arm }
521819281eeSdp-arm 
5221c75d5dfSMasahiro Yamada static int pack_images(const char *filename, uint64_t toc_flags, unsigned long align)
523819281eeSdp-arm {
524819281eeSdp-arm 	FILE *fp;
525b9589fe5Sdp-arm 	image_desc_t *desc;
526819281eeSdp-arm 	fip_toc_header_t *toc_header;
527819281eeSdp-arm 	fip_toc_entry_t *toc_entry;
528819281eeSdp-arm 	char *buf;
529880b9e8bSRoberto Vargas 	uint64_t entry_offset, buf_size, payload_size = 0, pad_size;
530b9589fe5Sdp-arm 	size_t nr_images = 0;
531b9589fe5Sdp-arm 
532b9589fe5Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next)
533b9589fe5Sdp-arm 		if (desc->image != NULL)
534b9589fe5Sdp-arm 			nr_images++;
535819281eeSdp-arm 
536819281eeSdp-arm 	buf_size = sizeof(fip_toc_header_t) +
537819281eeSdp-arm 	    sizeof(fip_toc_entry_t) * (nr_images + 1);
538819281eeSdp-arm 	buf = calloc(1, buf_size);
539819281eeSdp-arm 	if (buf == NULL)
540819281eeSdp-arm 		log_err("calloc");
541819281eeSdp-arm 
542819281eeSdp-arm 	/* Build up header and ToC entries from the image table. */
543819281eeSdp-arm 	toc_header = (fip_toc_header_t *)buf;
544819281eeSdp-arm 	toc_header->name = TOC_HEADER_NAME;
545819281eeSdp-arm 	toc_header->serial_number = TOC_HEADER_SERIAL_NUMBER;
546819281eeSdp-arm 	toc_header->flags = toc_flags;
547819281eeSdp-arm 
548819281eeSdp-arm 	toc_entry = (fip_toc_entry_t *)(toc_header + 1);
549819281eeSdp-arm 
550819281eeSdp-arm 	entry_offset = buf_size;
551b9589fe5Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next) {
552b9589fe5Sdp-arm 		image_t *image = desc->image;
553b9589fe5Sdp-arm 
554ab556c9cSManish V Badarkhe 		if (image == NULL || (image->toc_e.size == 0ULL))
555b9589fe5Sdp-arm 			continue;
55665caa3d0SMasahiro Yamada 		payload_size += image->toc_e.size;
5571c75d5dfSMasahiro Yamada 		entry_offset = (entry_offset + align - 1) & ~(align - 1);
55865caa3d0SMasahiro Yamada 		image->toc_e.offset_address = entry_offset;
55965caa3d0SMasahiro Yamada 		*toc_entry++ = image->toc_e;
56065caa3d0SMasahiro Yamada 		entry_offset += image->toc_e.size;
561819281eeSdp-arm 	}
562819281eeSdp-arm 
563880b9e8bSRoberto Vargas 	/*
564880b9e8bSRoberto Vargas 	 * Append a null uuid entry to mark the end of ToC entries.
565880b9e8bSRoberto Vargas 	 * NOTE the offset address for the last toc_entry must match the fip
566880b9e8bSRoberto Vargas 	 * size.
567880b9e8bSRoberto Vargas 	 */
56865caa3d0SMasahiro Yamada 	memset(toc_entry, 0, sizeof(*toc_entry));
569880b9e8bSRoberto Vargas 	toc_entry->offset_address = (entry_offset + align - 1) & ~(align - 1);
570819281eeSdp-arm 
571819281eeSdp-arm 	/* Generate the FIP file. */
57255745deaSEvan Lloyd 	fp = fopen(filename, "wb");
573819281eeSdp-arm 	if (fp == NULL)
574819281eeSdp-arm 		log_err("fopen %s", filename);
575819281eeSdp-arm 
576819281eeSdp-arm 	if (verbose)
577819281eeSdp-arm 		log_dbgx("Metadata size: %zu bytes", buf_size);
578819281eeSdp-arm 
579a1da83f5SMasahiro Yamada 	xfwrite(buf, buf_size, fp, filename);
580819281eeSdp-arm 
581819281eeSdp-arm 	if (verbose)
582819281eeSdp-arm 		log_dbgx("Payload size: %zu bytes", payload_size);
583819281eeSdp-arm 
584b9589fe5Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next) {
585b9589fe5Sdp-arm 		image_t *image = desc->image;
586b9589fe5Sdp-arm 
587b9589fe5Sdp-arm 		if (image == NULL)
588b9589fe5Sdp-arm 			continue;
5891c75d5dfSMasahiro Yamada 		if (fseek(fp, image->toc_e.offset_address, SEEK_SET))
5901c75d5dfSMasahiro Yamada 			log_errx("Failed to set file position");
5911c75d5dfSMasahiro Yamada 
59265caa3d0SMasahiro Yamada 		xfwrite(image->buffer, image->toc_e.size, fp, filename);
5931c75d5dfSMasahiro Yamada 	}
594819281eeSdp-arm 
595880b9e8bSRoberto Vargas 	if (fseek(fp, entry_offset, SEEK_SET))
596880b9e8bSRoberto Vargas 		log_errx("Failed to set file position");
597880b9e8bSRoberto Vargas 
598880b9e8bSRoberto Vargas 	pad_size = toc_entry->offset_address - entry_offset;
599880b9e8bSRoberto Vargas 	while (pad_size--)
600880b9e8bSRoberto Vargas 		fputc(0x0, fp);
601880b9e8bSRoberto Vargas 
6028e4cdd22SAndreas Färber 	free(buf);
603819281eeSdp-arm 	fclose(fp);
604819281eeSdp-arm 	return 0;
605819281eeSdp-arm }
606819281eeSdp-arm 
607819281eeSdp-arm /*
608819281eeSdp-arm  * This function is shared between the create and update subcommands.
609819281eeSdp-arm  * The difference between the two subcommands is that when the FIP file
610819281eeSdp-arm  * is created, the parsing of an existing FIP is skipped.  This results
611819281eeSdp-arm  * in update_fip() creating the new FIP file from scratch because the
612819281eeSdp-arm  * internal image table is not populated.
613819281eeSdp-arm  */
614819281eeSdp-arm static void update_fip(void)
615819281eeSdp-arm {
616e0f083a0Sdp-arm 	image_desc_t *desc;
617819281eeSdp-arm 
618819281eeSdp-arm 	/* Add or replace images in the FIP file. */
619e0f083a0Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next) {
620b9589fe5Sdp-arm 		image_t *image;
621fcab6bbeSdp-arm 
622e0f083a0Sdp-arm 		if (desc->action != DO_PACK)
623819281eeSdp-arm 			continue;
624819281eeSdp-arm 
625b9589fe5Sdp-arm 		image = read_image_from_file(&desc->uuid,
626e0f083a0Sdp-arm 		    desc->action_arg);
627b9589fe5Sdp-arm 		if (desc->image != NULL) {
628e0f083a0Sdp-arm 			if (verbose) {
629fcab6bbeSdp-arm 				log_dbgx("Replacing %s with %s",
630e0f083a0Sdp-arm 				    desc->cmdline_name,
631e0f083a0Sdp-arm 				    desc->action_arg);
632e0f083a0Sdp-arm 			}
633b9589fe5Sdp-arm 			free(desc->image);
634b9589fe5Sdp-arm 			desc->image = image;
635819281eeSdp-arm 		} else {
636819281eeSdp-arm 			if (verbose)
637819281eeSdp-arm 				log_dbgx("Adding image %s",
638e0f083a0Sdp-arm 				    desc->action_arg);
639b9589fe5Sdp-arm 			desc->image = image;
640819281eeSdp-arm 		}
641819281eeSdp-arm 	}
642819281eeSdp-arm }
643819281eeSdp-arm 
644e0f083a0Sdp-arm static void parse_plat_toc_flags(const char *arg, unsigned long long *toc_flags)
645819281eeSdp-arm {
646819281eeSdp-arm 	unsigned long long flags;
647819281eeSdp-arm 	char *endptr;
648819281eeSdp-arm 
649819281eeSdp-arm 	errno = 0;
650819281eeSdp-arm 	flags = strtoull(arg, &endptr, 16);
651819281eeSdp-arm 	if (*endptr != '\0' || flags > UINT16_MAX || errno != 0)
652819281eeSdp-arm 		log_errx("Invalid platform ToC flags: %s", arg);
653819281eeSdp-arm 	/* Platform ToC flags is a 16-bit field occupying bits [32-47]. */
654819281eeSdp-arm 	*toc_flags |= flags << 32;
655819281eeSdp-arm }
656819281eeSdp-arm 
6571c75d5dfSMasahiro Yamada static int is_power_of_2(unsigned long x)
6581c75d5dfSMasahiro Yamada {
6591c75d5dfSMasahiro Yamada 	return x && !(x & (x - 1));
6601c75d5dfSMasahiro Yamada }
6611c75d5dfSMasahiro Yamada 
6621c75d5dfSMasahiro Yamada static unsigned long get_image_align(char *arg)
6631c75d5dfSMasahiro Yamada {
6641c75d5dfSMasahiro Yamada 	char *endptr;
6651c75d5dfSMasahiro Yamada 	unsigned long align;
6661c75d5dfSMasahiro Yamada 
6671c75d5dfSMasahiro Yamada 	errno = 0;
668fb5f7949SAndreas Färber 	align = strtoul(arg, &endptr, 0);
6691c75d5dfSMasahiro Yamada 	if (*endptr != '\0' || !is_power_of_2(align) || errno != 0)
6701c75d5dfSMasahiro Yamada 		log_errx("Invalid alignment: %s", arg);
6711c75d5dfSMasahiro Yamada 
6721c75d5dfSMasahiro Yamada 	return align;
6731c75d5dfSMasahiro Yamada }
6741c75d5dfSMasahiro Yamada 
675fcab6bbeSdp-arm static void parse_blob_opt(char *arg, uuid_t *uuid, char *filename, size_t len)
676fcab6bbeSdp-arm {
677fcab6bbeSdp-arm 	char *p;
678fcab6bbeSdp-arm 
679fcab6bbeSdp-arm 	for (p = strtok(arg, ","); p != NULL; p = strtok(NULL, ",")) {
680fcab6bbeSdp-arm 		if (strncmp(p, "uuid=", strlen("uuid=")) == 0) {
681fcab6bbeSdp-arm 			p += strlen("uuid=");
682fcab6bbeSdp-arm 			uuid_from_str(uuid, p);
683fcab6bbeSdp-arm 		} else if (strncmp(p, "file=", strlen("file=")) == 0) {
684fcab6bbeSdp-arm 			p += strlen("file=");
685fcab6bbeSdp-arm 			snprintf(filename, len, "%s", p);
686fcab6bbeSdp-arm 		}
687fcab6bbeSdp-arm 	}
688fcab6bbeSdp-arm }
689fcab6bbeSdp-arm 
690819281eeSdp-arm static int create_cmd(int argc, char *argv[])
691819281eeSdp-arm {
692e0f083a0Sdp-arm 	struct option *opts = NULL;
693e0f083a0Sdp-arm 	size_t nr_opts = 0;
694819281eeSdp-arm 	unsigned long long toc_flags = 0;
6951c75d5dfSMasahiro Yamada 	unsigned long align = 1;
696819281eeSdp-arm 
697819281eeSdp-arm 	if (argc < 2)
6984e500525SLeonardo Sandoval 		create_usage(EXIT_FAILURE);
699819281eeSdp-arm 
700e0f083a0Sdp-arm 	opts = fill_common_opts(opts, &nr_opts, required_argument);
701e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument,
702819281eeSdp-arm 	    OPT_PLAT_TOC_FLAGS);
7031c75d5dfSMasahiro Yamada 	opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN);
704fcab6bbeSdp-arm 	opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b');
705e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, NULL, 0, 0);
706819281eeSdp-arm 
707819281eeSdp-arm 	while (1) {
70820f87e78Sdp-arm 		int c, opt_index = 0;
709819281eeSdp-arm 
710fcab6bbeSdp-arm 		c = getopt_long(argc, argv, "b:", opts, &opt_index);
711819281eeSdp-arm 		if (c == -1)
712819281eeSdp-arm 			break;
713819281eeSdp-arm 
714819281eeSdp-arm 		switch (c) {
715819281eeSdp-arm 		case OPT_TOC_ENTRY: {
716e0f083a0Sdp-arm 			image_desc_t *desc;
717819281eeSdp-arm 
718e0f083a0Sdp-arm 			desc = lookup_image_desc_from_opt(opts[opt_index].name);
719d02fcebeSdp-arm 			set_image_desc_action(desc, DO_PACK, optarg);
720819281eeSdp-arm 			break;
721819281eeSdp-arm 		}
722819281eeSdp-arm 		case OPT_PLAT_TOC_FLAGS:
723819281eeSdp-arm 			parse_plat_toc_flags(optarg, &toc_flags);
724819281eeSdp-arm 			break;
7251c75d5dfSMasahiro Yamada 		case OPT_ALIGN:
7261c75d5dfSMasahiro Yamada 			align = get_image_align(optarg);
7271c75d5dfSMasahiro Yamada 			break;
728fcab6bbeSdp-arm 		case 'b': {
729fcab6bbeSdp-arm 			char name[_UUID_STR_LEN + 1];
730fcab6bbeSdp-arm 			char filename[PATH_MAX] = { 0 };
73103364865SRoberto Vargas 			uuid_t uuid = uuid_null;
732fcab6bbeSdp-arm 			image_desc_t *desc;
733fcab6bbeSdp-arm 
734fcab6bbeSdp-arm 			parse_blob_opt(optarg, &uuid,
735fcab6bbeSdp-arm 			    filename, sizeof(filename));
736fcab6bbeSdp-arm 
737fcab6bbeSdp-arm 			if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 ||
738fcab6bbeSdp-arm 			    filename[0] == '\0')
7394e500525SLeonardo Sandoval 				create_usage(EXIT_FAILURE);
740fcab6bbeSdp-arm 
741fcab6bbeSdp-arm 			desc = lookup_image_desc_from_uuid(&uuid);
742d02fcebeSdp-arm 			if (desc == NULL) {
743fcab6bbeSdp-arm 				uuid_to_str(name, sizeof(name), &uuid);
744fcab6bbeSdp-arm 				desc = new_image_desc(&uuid, name, "blob");
745fcab6bbeSdp-arm 				add_image_desc(desc);
746fcab6bbeSdp-arm 			}
747d02fcebeSdp-arm 			set_image_desc_action(desc, DO_PACK, filename);
748fcab6bbeSdp-arm 			break;
749fcab6bbeSdp-arm 		}
750819281eeSdp-arm 		default:
7514e500525SLeonardo Sandoval 			create_usage(EXIT_FAILURE);
752819281eeSdp-arm 		}
753819281eeSdp-arm 	}
754819281eeSdp-arm 	argc -= optind;
755819281eeSdp-arm 	argv += optind;
756e0f083a0Sdp-arm 	free(opts);
757819281eeSdp-arm 
758819281eeSdp-arm 	if (argc == 0)
7594e500525SLeonardo Sandoval 		create_usage(EXIT_SUCCESS);
760819281eeSdp-arm 
761819281eeSdp-arm 	update_fip();
762819281eeSdp-arm 
7631c75d5dfSMasahiro Yamada 	pack_images(argv[0], toc_flags, align);
764819281eeSdp-arm 	return 0;
765819281eeSdp-arm }
766819281eeSdp-arm 
7674e500525SLeonardo Sandoval static void create_usage(int exit_status)
768819281eeSdp-arm {
769819281eeSdp-arm 	toc_entry_t *toc_entry = toc_entries;
770819281eeSdp-arm 
771ee079320SMasahiro Yamada 	printf("fiptool create [opts] FIP_FILENAME\n");
772ee079320SMasahiro Yamada 	printf("\n");
773ee079320SMasahiro Yamada 	printf("Options:\n");
7741c75d5dfSMasahiro Yamada 	printf("  --align <value>\t\tEach image is aligned to <value> (default: 1).\n");
775802b42a0SMasahiro Yamada 	printf("  --blob uuid=...,file=...\tAdd an image with the given UUID pointed to by file.\n");
776802b42a0SMasahiro Yamada 	printf("  --plat-toc-flags <value>\t16-bit platform specific flag field occupying bits 32-47 in 64-bit ToC header.\n");
77795d2b268SMasahiro Yamada 	printf("\n");
778819281eeSdp-arm 	printf("Specific images are packed with the following options:\n");
779819281eeSdp-arm 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
780819281eeSdp-arm 		printf("  --%-16s FILENAME\t%s\n", toc_entry->cmdline_name,
781819281eeSdp-arm 		    toc_entry->name);
7823527d6d2SPankaj Gupta #ifdef PLAT_DEF_FIP_UUID
7833527d6d2SPankaj Gupta 	toc_entry = plat_def_toc_entries;
7843527d6d2SPankaj Gupta 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
7853527d6d2SPankaj Gupta 		printf("  --%-16s FILENAME\t%s\n", toc_entry->cmdline_name,
7863527d6d2SPankaj Gupta 		    toc_entry->name);
7873527d6d2SPankaj Gupta #endif
7884e500525SLeonardo Sandoval 	exit(exit_status);
789819281eeSdp-arm }
790819281eeSdp-arm 
791819281eeSdp-arm static int update_cmd(int argc, char *argv[])
792819281eeSdp-arm {
793e0f083a0Sdp-arm 	struct option *opts = NULL;
794e0f083a0Sdp-arm 	size_t nr_opts = 0;
795fcab6bbeSdp-arm 	char outfile[PATH_MAX] = { 0 };
796819281eeSdp-arm 	fip_toc_header_t toc_header = { 0 };
797819281eeSdp-arm 	unsigned long long toc_flags = 0;
7981c75d5dfSMasahiro Yamada 	unsigned long align = 1;
799819281eeSdp-arm 	int pflag = 0;
800819281eeSdp-arm 
801819281eeSdp-arm 	if (argc < 2)
8024e500525SLeonardo Sandoval 		update_usage(EXIT_FAILURE);
803819281eeSdp-arm 
804e0f083a0Sdp-arm 	opts = fill_common_opts(opts, &nr_opts, required_argument);
8051c75d5dfSMasahiro Yamada 	opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN);
806fcab6bbeSdp-arm 	opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b');
807e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "out", required_argument, 'o');
808e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "plat-toc-flags", required_argument,
809819281eeSdp-arm 	    OPT_PLAT_TOC_FLAGS);
810e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, NULL, 0, 0);
811819281eeSdp-arm 
812819281eeSdp-arm 	while (1) {
81320f87e78Sdp-arm 		int c, opt_index = 0;
814819281eeSdp-arm 
815fcab6bbeSdp-arm 		c = getopt_long(argc, argv, "b:o:", opts, &opt_index);
816819281eeSdp-arm 		if (c == -1)
817819281eeSdp-arm 			break;
818819281eeSdp-arm 
819819281eeSdp-arm 		switch (c) {
820819281eeSdp-arm 		case OPT_TOC_ENTRY: {
821e0f083a0Sdp-arm 			image_desc_t *desc;
822819281eeSdp-arm 
823e0f083a0Sdp-arm 			desc = lookup_image_desc_from_opt(opts[opt_index].name);
824d02fcebeSdp-arm 			set_image_desc_action(desc, DO_PACK, optarg);
825819281eeSdp-arm 			break;
826819281eeSdp-arm 		}
827e0f083a0Sdp-arm 		case OPT_PLAT_TOC_FLAGS:
828819281eeSdp-arm 			parse_plat_toc_flags(optarg, &toc_flags);
829819281eeSdp-arm 			pflag = 1;
830819281eeSdp-arm 			break;
831fcab6bbeSdp-arm 		case 'b': {
832fcab6bbeSdp-arm 			char name[_UUID_STR_LEN + 1];
833fcab6bbeSdp-arm 			char filename[PATH_MAX] = { 0 };
83403364865SRoberto Vargas 			uuid_t uuid = uuid_null;
835fcab6bbeSdp-arm 			image_desc_t *desc;
836fcab6bbeSdp-arm 
837fcab6bbeSdp-arm 			parse_blob_opt(optarg, &uuid,
838fcab6bbeSdp-arm 			    filename, sizeof(filename));
839fcab6bbeSdp-arm 
840fcab6bbeSdp-arm 			if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 ||
841fcab6bbeSdp-arm 			    filename[0] == '\0')
8424e500525SLeonardo Sandoval 				update_usage(EXIT_FAILURE);
843fcab6bbeSdp-arm 
844fcab6bbeSdp-arm 			desc = lookup_image_desc_from_uuid(&uuid);
845d02fcebeSdp-arm 			if (desc == NULL) {
846fcab6bbeSdp-arm 				uuid_to_str(name, sizeof(name), &uuid);
847fcab6bbeSdp-arm 				desc = new_image_desc(&uuid, name, "blob");
848fcab6bbeSdp-arm 				add_image_desc(desc);
849fcab6bbeSdp-arm 			}
850d02fcebeSdp-arm 			set_image_desc_action(desc, DO_PACK, filename);
851fcab6bbeSdp-arm 			break;
852fcab6bbeSdp-arm 		}
8531c75d5dfSMasahiro Yamada 		case OPT_ALIGN:
8541c75d5dfSMasahiro Yamada 			align = get_image_align(optarg);
8551c75d5dfSMasahiro Yamada 			break;
856819281eeSdp-arm 		case 'o':
857819281eeSdp-arm 			snprintf(outfile, sizeof(outfile), "%s", optarg);
858819281eeSdp-arm 			break;
859819281eeSdp-arm 		default:
8604e500525SLeonardo Sandoval 			update_usage(EXIT_FAILURE);
861819281eeSdp-arm 		}
862819281eeSdp-arm 	}
863819281eeSdp-arm 	argc -= optind;
864819281eeSdp-arm 	argv += optind;
865e0f083a0Sdp-arm 	free(opts);
866819281eeSdp-arm 
867819281eeSdp-arm 	if (argc == 0)
8684e500525SLeonardo Sandoval 		update_usage(EXIT_SUCCESS);
869819281eeSdp-arm 
870819281eeSdp-arm 	if (outfile[0] == '\0')
871819281eeSdp-arm 		snprintf(outfile, sizeof(outfile), "%s", argv[0]);
872819281eeSdp-arm 
87340866aafSMasahiro Yamada 	if (access(argv[0], F_OK) == 0)
874819281eeSdp-arm 		parse_fip(argv[0], &toc_header);
875819281eeSdp-arm 
876819281eeSdp-arm 	if (pflag)
877819281eeSdp-arm 		toc_header.flags &= ~(0xffffULL << 32);
878819281eeSdp-arm 	toc_flags = (toc_header.flags |= toc_flags);
879819281eeSdp-arm 
880819281eeSdp-arm 	update_fip();
881819281eeSdp-arm 
8821c75d5dfSMasahiro Yamada 	pack_images(outfile, toc_flags, align);
883819281eeSdp-arm 	return 0;
884819281eeSdp-arm }
885819281eeSdp-arm 
8864e500525SLeonardo Sandoval static void update_usage(int exit_status)
887819281eeSdp-arm {
888819281eeSdp-arm 	toc_entry_t *toc_entry = toc_entries;
889819281eeSdp-arm 
890ee079320SMasahiro Yamada 	printf("fiptool update [opts] FIP_FILENAME\n");
891ee079320SMasahiro Yamada 	printf("\n");
892ee079320SMasahiro Yamada 	printf("Options:\n");
8931c75d5dfSMasahiro Yamada 	printf("  --align <value>\t\tEach image is aligned to <value> (default: 1).\n");
894802b42a0SMasahiro Yamada 	printf("  --blob uuid=...,file=...\tAdd or update an image with the given UUID pointed to by file.\n");
895819281eeSdp-arm 	printf("  --out FIP_FILENAME\t\tSet an alternative output FIP file.\n");
896802b42a0SMasahiro Yamada 	printf("  --plat-toc-flags <value>\t16-bit platform specific flag field occupying bits 32-47 in 64-bit ToC header.\n");
89795d2b268SMasahiro Yamada 	printf("\n");
898819281eeSdp-arm 	printf("Specific images are packed with the following options:\n");
899819281eeSdp-arm 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
900819281eeSdp-arm 		printf("  --%-16s FILENAME\t%s\n", toc_entry->cmdline_name,
901819281eeSdp-arm 		    toc_entry->name);
9023527d6d2SPankaj Gupta #ifdef PLAT_DEF_FIP_UUID
9033527d6d2SPankaj Gupta 	toc_entry = plat_def_toc_entries;
9043527d6d2SPankaj Gupta 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
9053527d6d2SPankaj Gupta 		printf("  --%-16s FILENAME\t%s\n", toc_entry->cmdline_name,
9063527d6d2SPankaj Gupta 		    toc_entry->name);
9073527d6d2SPankaj Gupta #endif
9084e500525SLeonardo Sandoval 	exit(exit_status);
909819281eeSdp-arm }
910819281eeSdp-arm 
911819281eeSdp-arm static int unpack_cmd(int argc, char *argv[])
912819281eeSdp-arm {
913e0f083a0Sdp-arm 	struct option *opts = NULL;
914e0f083a0Sdp-arm 	size_t nr_opts = 0;
915fcab6bbeSdp-arm 	char outdir[PATH_MAX] = { 0 };
916e0f083a0Sdp-arm 	image_desc_t *desc;
917819281eeSdp-arm 	int fflag = 0;
918819281eeSdp-arm 	int unpack_all = 1;
919819281eeSdp-arm 
920819281eeSdp-arm 	if (argc < 2)
9214e500525SLeonardo Sandoval 		unpack_usage(EXIT_FAILURE);
922819281eeSdp-arm 
923e0f083a0Sdp-arm 	opts = fill_common_opts(opts, &nr_opts, required_argument);
924fcab6bbeSdp-arm 	opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b');
925e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "force", no_argument, 'f');
926e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "out", required_argument, 'o');
927e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, NULL, 0, 0);
928819281eeSdp-arm 
929819281eeSdp-arm 	while (1) {
93020f87e78Sdp-arm 		int c, opt_index = 0;
931819281eeSdp-arm 
932fcab6bbeSdp-arm 		c = getopt_long(argc, argv, "b:fo:", opts, &opt_index);
933819281eeSdp-arm 		if (c == -1)
934819281eeSdp-arm 			break;
935819281eeSdp-arm 
936819281eeSdp-arm 		switch (c) {
937e0f083a0Sdp-arm 		case OPT_TOC_ENTRY: {
938e0f083a0Sdp-arm 			image_desc_t *desc;
939e0f083a0Sdp-arm 
940e0f083a0Sdp-arm 			desc = lookup_image_desc_from_opt(opts[opt_index].name);
941d02fcebeSdp-arm 			set_image_desc_action(desc, DO_UNPACK, optarg);
942e0f083a0Sdp-arm 			unpack_all = 0;
943819281eeSdp-arm 			break;
944e0f083a0Sdp-arm 		}
945fcab6bbeSdp-arm 		case 'b': {
946fcab6bbeSdp-arm 			char name[_UUID_STR_LEN + 1];
947fcab6bbeSdp-arm 			char filename[PATH_MAX] = { 0 };
94803364865SRoberto Vargas 			uuid_t uuid = uuid_null;
949fcab6bbeSdp-arm 			image_desc_t *desc;
950fcab6bbeSdp-arm 
951fcab6bbeSdp-arm 			parse_blob_opt(optarg, &uuid,
952fcab6bbeSdp-arm 			    filename, sizeof(filename));
953fcab6bbeSdp-arm 
954fcab6bbeSdp-arm 			if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0 ||
955fcab6bbeSdp-arm 			    filename[0] == '\0')
9564e500525SLeonardo Sandoval 				unpack_usage(EXIT_FAILURE);
957fcab6bbeSdp-arm 
958fcab6bbeSdp-arm 			desc = lookup_image_desc_from_uuid(&uuid);
959d02fcebeSdp-arm 			if (desc == NULL) {
960fcab6bbeSdp-arm 				uuid_to_str(name, sizeof(name), &uuid);
961fcab6bbeSdp-arm 				desc = new_image_desc(&uuid, name, "blob");
962fcab6bbeSdp-arm 				add_image_desc(desc);
963fcab6bbeSdp-arm 			}
964d02fcebeSdp-arm 			set_image_desc_action(desc, DO_UNPACK, filename);
965fcab6bbeSdp-arm 			unpack_all = 0;
966fcab6bbeSdp-arm 			break;
967fcab6bbeSdp-arm 		}
968819281eeSdp-arm 		case 'f':
969819281eeSdp-arm 			fflag = 1;
970819281eeSdp-arm 			break;
971819281eeSdp-arm 		case 'o':
972819281eeSdp-arm 			snprintf(outdir, sizeof(outdir), "%s", optarg);
973819281eeSdp-arm 			break;
974819281eeSdp-arm 		default:
9754e500525SLeonardo Sandoval 			unpack_usage(EXIT_FAILURE);
976819281eeSdp-arm 		}
977819281eeSdp-arm 	}
978819281eeSdp-arm 	argc -= optind;
979819281eeSdp-arm 	argv += optind;
980e0f083a0Sdp-arm 	free(opts);
981819281eeSdp-arm 
982819281eeSdp-arm 	if (argc == 0)
9834e500525SLeonardo Sandoval 		unpack_usage(EXIT_SUCCESS);
984819281eeSdp-arm 
985819281eeSdp-arm 	parse_fip(argv[0], NULL);
986819281eeSdp-arm 
987819281eeSdp-arm 	if (outdir[0] != '\0')
988819281eeSdp-arm 		if (chdir(outdir) == -1)
989819281eeSdp-arm 			log_err("chdir %s", outdir);
990819281eeSdp-arm 
991819281eeSdp-arm 	/* Unpack all specified images. */
992e0f083a0Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next) {
993fcab6bbeSdp-arm 		char file[PATH_MAX];
994b9589fe5Sdp-arm 		image_t *image = desc->image;
995b04efcceSdp-arm 
996e0f083a0Sdp-arm 		if (!unpack_all && desc->action != DO_UNPACK)
997819281eeSdp-arm 			continue;
998819281eeSdp-arm 
999819281eeSdp-arm 		/* Build filename. */
1000e0f083a0Sdp-arm 		if (desc->action_arg == NULL)
1001819281eeSdp-arm 			snprintf(file, sizeof(file), "%s.bin",
1002e0f083a0Sdp-arm 			    desc->cmdline_name);
1003819281eeSdp-arm 		else
1004819281eeSdp-arm 			snprintf(file, sizeof(file), "%s",
1005e0f083a0Sdp-arm 			    desc->action_arg);
1006819281eeSdp-arm 
1007b04efcceSdp-arm 		if (image == NULL) {
1008b04efcceSdp-arm 			if (!unpack_all)
1009fcab6bbeSdp-arm 				log_warnx("%s does not exist in %s",
1010819281eeSdp-arm 				    file, argv[0]);
1011819281eeSdp-arm 			continue;
1012819281eeSdp-arm 		}
1013819281eeSdp-arm 
1014819281eeSdp-arm 		if (access(file, F_OK) != 0 || fflag) {
1015819281eeSdp-arm 			if (verbose)
1016819281eeSdp-arm 				log_dbgx("Unpacking %s", file);
1017b04efcceSdp-arm 			write_image_to_file(image, file);
1018819281eeSdp-arm 		} else {
1019819281eeSdp-arm 			log_warnx("File %s already exists, use --force to overwrite it",
1020819281eeSdp-arm 			    file);
1021819281eeSdp-arm 		}
1022819281eeSdp-arm 	}
1023819281eeSdp-arm 
1024819281eeSdp-arm 	return 0;
1025819281eeSdp-arm }
1026819281eeSdp-arm 
10274e500525SLeonardo Sandoval static void unpack_usage(int exit_status)
1028819281eeSdp-arm {
1029819281eeSdp-arm 	toc_entry_t *toc_entry = toc_entries;
1030819281eeSdp-arm 
1031ee079320SMasahiro Yamada 	printf("fiptool unpack [opts] FIP_FILENAME\n");
1032ee079320SMasahiro Yamada 	printf("\n");
1033ee079320SMasahiro Yamada 	printf("Options:\n");
1034802b42a0SMasahiro Yamada 	printf("  --blob uuid=...,file=...\tUnpack an image with the given UUID to file.\n");
1035802b42a0SMasahiro Yamada 	printf("  --force\t\t\tIf the output file already exists, use --force to overwrite it.\n");
1036fcab6bbeSdp-arm 	printf("  --out path\t\t\tSet the output directory path.\n");
103795d2b268SMasahiro Yamada 	printf("\n");
1038819281eeSdp-arm 	printf("Specific images are unpacked with the following options:\n");
1039819281eeSdp-arm 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
1040819281eeSdp-arm 		printf("  --%-16s FILENAME\t%s\n", toc_entry->cmdline_name,
1041819281eeSdp-arm 		    toc_entry->name);
10423527d6d2SPankaj Gupta #ifdef PLAT_DEF_FIP_UUID
10433527d6d2SPankaj Gupta 	toc_entry = plat_def_toc_entries;
10443527d6d2SPankaj Gupta 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
10453527d6d2SPankaj Gupta 		printf("  --%-16s FILENAME\t%s\n", toc_entry->cmdline_name,
10463527d6d2SPankaj Gupta 		    toc_entry->name);
10473527d6d2SPankaj Gupta #endif
104895d2b268SMasahiro Yamada 	printf("\n");
1049819281eeSdp-arm 	printf("If no options are provided, all images will be unpacked.\n");
10504e500525SLeonardo Sandoval 	exit(exit_status);
1051819281eeSdp-arm }
1052819281eeSdp-arm 
1053819281eeSdp-arm static int remove_cmd(int argc, char *argv[])
1054819281eeSdp-arm {
1055e0f083a0Sdp-arm 	struct option *opts = NULL;
1056e0f083a0Sdp-arm 	size_t nr_opts = 0;
1057fcab6bbeSdp-arm 	char outfile[PATH_MAX] = { 0 };
1058819281eeSdp-arm 	fip_toc_header_t toc_header;
1059e0f083a0Sdp-arm 	image_desc_t *desc;
10601c75d5dfSMasahiro Yamada 	unsigned long align = 1;
1061819281eeSdp-arm 	int fflag = 0;
1062819281eeSdp-arm 
1063819281eeSdp-arm 	if (argc < 2)
10644e500525SLeonardo Sandoval 		remove_usage(EXIT_FAILURE);
1065819281eeSdp-arm 
1066e0f083a0Sdp-arm 	opts = fill_common_opts(opts, &nr_opts, no_argument);
10671c75d5dfSMasahiro Yamada 	opts = add_opt(opts, &nr_opts, "align", required_argument, OPT_ALIGN);
1068fcab6bbeSdp-arm 	opts = add_opt(opts, &nr_opts, "blob", required_argument, 'b');
1069e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "force", no_argument, 'f');
1070e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, "out", required_argument, 'o');
1071e0f083a0Sdp-arm 	opts = add_opt(opts, &nr_opts, NULL, 0, 0);
1072819281eeSdp-arm 
1073819281eeSdp-arm 	while (1) {
107420f87e78Sdp-arm 		int c, opt_index = 0;
1075819281eeSdp-arm 
1076fcab6bbeSdp-arm 		c = getopt_long(argc, argv, "b:fo:", opts, &opt_index);
1077819281eeSdp-arm 		if (c == -1)
1078819281eeSdp-arm 			break;
1079819281eeSdp-arm 
1080819281eeSdp-arm 		switch (c) {
1081e0f083a0Sdp-arm 		case OPT_TOC_ENTRY: {
1082e0f083a0Sdp-arm 			image_desc_t *desc;
1083e0f083a0Sdp-arm 
1084e0f083a0Sdp-arm 			desc = lookup_image_desc_from_opt(opts[opt_index].name);
1085d02fcebeSdp-arm 			set_image_desc_action(desc, DO_REMOVE, NULL);
1086819281eeSdp-arm 			break;
1087e0f083a0Sdp-arm 		}
10881c75d5dfSMasahiro Yamada 		case OPT_ALIGN:
10891c75d5dfSMasahiro Yamada 			align = get_image_align(optarg);
10901c75d5dfSMasahiro Yamada 			break;
1091fcab6bbeSdp-arm 		case 'b': {
1092fcab6bbeSdp-arm 			char name[_UUID_STR_LEN + 1], filename[PATH_MAX];
109303364865SRoberto Vargas 			uuid_t uuid = uuid_null;
1094fcab6bbeSdp-arm 			image_desc_t *desc;
1095fcab6bbeSdp-arm 
1096fcab6bbeSdp-arm 			parse_blob_opt(optarg, &uuid,
1097fcab6bbeSdp-arm 			    filename, sizeof(filename));
1098fcab6bbeSdp-arm 
1099fcab6bbeSdp-arm 			if (memcmp(&uuid, &uuid_null, sizeof(uuid_t)) == 0)
11004e500525SLeonardo Sandoval 				remove_usage(EXIT_FAILURE);
1101fcab6bbeSdp-arm 
1102fcab6bbeSdp-arm 			desc = lookup_image_desc_from_uuid(&uuid);
1103d02fcebeSdp-arm 			if (desc == NULL) {
1104fcab6bbeSdp-arm 				uuid_to_str(name, sizeof(name), &uuid);
1105fcab6bbeSdp-arm 				desc = new_image_desc(&uuid, name, "blob");
1106fcab6bbeSdp-arm 				add_image_desc(desc);
1107fcab6bbeSdp-arm 			}
1108d02fcebeSdp-arm 			set_image_desc_action(desc, DO_REMOVE, NULL);
1109fcab6bbeSdp-arm 			break;
1110fcab6bbeSdp-arm 		}
1111819281eeSdp-arm 		case 'f':
1112819281eeSdp-arm 			fflag = 1;
1113819281eeSdp-arm 			break;
1114819281eeSdp-arm 		case 'o':
1115819281eeSdp-arm 			snprintf(outfile, sizeof(outfile), "%s", optarg);
1116819281eeSdp-arm 			break;
1117819281eeSdp-arm 		default:
11184e500525SLeonardo Sandoval 			remove_usage(EXIT_FAILURE);
1119819281eeSdp-arm 		}
1120819281eeSdp-arm 	}
1121819281eeSdp-arm 	argc -= optind;
1122819281eeSdp-arm 	argv += optind;
1123e0f083a0Sdp-arm 	free(opts);
1124819281eeSdp-arm 
1125819281eeSdp-arm 	if (argc == 0)
11264e500525SLeonardo Sandoval 		remove_usage(EXIT_SUCCESS);
1127819281eeSdp-arm 
1128819281eeSdp-arm 	if (outfile[0] != '\0' && access(outfile, F_OK) == 0 && !fflag)
1129819281eeSdp-arm 		log_errx("File %s already exists, use --force to overwrite it",
1130819281eeSdp-arm 		    outfile);
1131819281eeSdp-arm 
1132819281eeSdp-arm 	if (outfile[0] == '\0')
1133819281eeSdp-arm 		snprintf(outfile, sizeof(outfile), "%s", argv[0]);
1134819281eeSdp-arm 
1135819281eeSdp-arm 	parse_fip(argv[0], &toc_header);
1136819281eeSdp-arm 
1137e0f083a0Sdp-arm 	for (desc = image_desc_head; desc != NULL; desc = desc->next) {
1138e0f083a0Sdp-arm 		if (desc->action != DO_REMOVE)
1139819281eeSdp-arm 			continue;
1140fcab6bbeSdp-arm 
1141b9589fe5Sdp-arm 		if (desc->image != NULL) {
1142819281eeSdp-arm 			if (verbose)
1143fcab6bbeSdp-arm 				log_dbgx("Removing %s",
1144e0f083a0Sdp-arm 				    desc->cmdline_name);
1145b9589fe5Sdp-arm 			free(desc->image);
1146b9589fe5Sdp-arm 			desc->image = NULL;
1147819281eeSdp-arm 		} else {
1148fcab6bbeSdp-arm 			log_warnx("%s does not exist in %s",
1149e0f083a0Sdp-arm 			    desc->cmdline_name, argv[0]);
1150819281eeSdp-arm 		}
1151819281eeSdp-arm 	}
1152819281eeSdp-arm 
11531c75d5dfSMasahiro Yamada 	pack_images(outfile, toc_header.flags, align);
1154819281eeSdp-arm 	return 0;
1155819281eeSdp-arm }
1156819281eeSdp-arm 
11574e500525SLeonardo Sandoval static void remove_usage(int exit_status)
1158819281eeSdp-arm {
1159819281eeSdp-arm 	toc_entry_t *toc_entry = toc_entries;
1160819281eeSdp-arm 
1161ee079320SMasahiro Yamada 	printf("fiptool remove [opts] FIP_FILENAME\n");
1162ee079320SMasahiro Yamada 	printf("\n");
1163ee079320SMasahiro Yamada 	printf("Options:\n");
11641c75d5dfSMasahiro Yamada 	printf("  --align <value>\tEach image is aligned to <value> (default: 1).\n");
1165fcab6bbeSdp-arm 	printf("  --blob uuid=...\tRemove an image with the given UUID.\n");
1166802b42a0SMasahiro Yamada 	printf("  --force\t\tIf the output FIP file already exists, use --force to overwrite it.\n");
1167819281eeSdp-arm 	printf("  --out FIP_FILENAME\tSet an alternative output FIP file.\n");
116895d2b268SMasahiro Yamada 	printf("\n");
1169819281eeSdp-arm 	printf("Specific images are removed with the following options:\n");
1170819281eeSdp-arm 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
1171819281eeSdp-arm 		printf("  --%-16s\t%s\n", toc_entry->cmdline_name,
1172819281eeSdp-arm 		    toc_entry->name);
11733527d6d2SPankaj Gupta #ifdef PLAT_DEF_FIP_UUID
11743527d6d2SPankaj Gupta 	toc_entry = plat_def_toc_entries;
11753527d6d2SPankaj Gupta 	for (; toc_entry->cmdline_name != NULL; toc_entry++)
11763527d6d2SPankaj Gupta 		printf("  --%-16s\t%s\n", toc_entry->cmdline_name,
11773527d6d2SPankaj Gupta 		    toc_entry->name);
11783527d6d2SPankaj Gupta #endif
11794e500525SLeonardo Sandoval 	exit(exit_status);
1180819281eeSdp-arm }
1181819281eeSdp-arm 
1182819281eeSdp-arm static int version_cmd(int argc, char *argv[])
1183819281eeSdp-arm {
1184819281eeSdp-arm #ifdef VERSION
1185819281eeSdp-arm 	puts(VERSION);
1186819281eeSdp-arm #else
1187819281eeSdp-arm 	/* If built from fiptool directory, VERSION is not set. */
1188819281eeSdp-arm 	puts("Unknown version");
1189819281eeSdp-arm #endif
1190819281eeSdp-arm 	return 0;
1191819281eeSdp-arm }
1192819281eeSdp-arm 
11934e500525SLeonardo Sandoval static void version_usage(int exit_status)
1194819281eeSdp-arm {
1195819281eeSdp-arm 	printf("fiptool version\n");
11964e500525SLeonardo Sandoval 	exit(exit_status);
1197819281eeSdp-arm }
1198819281eeSdp-arm 
1199819281eeSdp-arm static int help_cmd(int argc, char *argv[])
1200819281eeSdp-arm {
1201819281eeSdp-arm 	int i;
1202819281eeSdp-arm 
1203819281eeSdp-arm 	if (argc < 2)
1204819281eeSdp-arm 		usage();
1205819281eeSdp-arm 	argc--, argv++;
1206819281eeSdp-arm 
1207819281eeSdp-arm 	for (i = 0; i < NELEM(cmds); i++) {
1208819281eeSdp-arm 		if (strcmp(cmds[i].name, argv[0]) == 0 &&
120985ee2778Sdp-arm 		    cmds[i].usage != NULL)
12104e500525SLeonardo Sandoval 			cmds[i].usage(EXIT_SUCCESS);
1211819281eeSdp-arm 	}
1212819281eeSdp-arm 	if (i == NELEM(cmds))
1213819281eeSdp-arm 		printf("No help for subcommand '%s'\n", argv[0]);
1214819281eeSdp-arm 	return 0;
1215819281eeSdp-arm }
1216819281eeSdp-arm 
1217819281eeSdp-arm static void usage(void)
1218819281eeSdp-arm {
12194f96a498SMasahiro Yamada 	printf("usage: fiptool [--verbose] <command> [<args>]\n");
1220819281eeSdp-arm 	printf("Global options supported:\n");
1221819281eeSdp-arm 	printf("  --verbose\tEnable verbose output for all commands.\n");
122295d2b268SMasahiro Yamada 	printf("\n");
1223819281eeSdp-arm 	printf("Commands supported:\n");
1224819281eeSdp-arm 	printf("  info\t\tList images contained in FIP.\n");
1225819281eeSdp-arm 	printf("  create\tCreate a new FIP with the given images.\n");
1226819281eeSdp-arm 	printf("  update\tUpdate an existing FIP with the given images.\n");
1227819281eeSdp-arm 	printf("  unpack\tUnpack images from FIP.\n");
1228819281eeSdp-arm 	printf("  remove\tRemove images from FIP.\n");
1229819281eeSdp-arm 	printf("  version\tShow fiptool version.\n");
1230819281eeSdp-arm 	printf("  help\t\tShow help for given command.\n");
12314e500525SLeonardo Sandoval 	exit(EXIT_SUCCESS);
1232819281eeSdp-arm }
1233819281eeSdp-arm 
1234819281eeSdp-arm int main(int argc, char *argv[])
1235819281eeSdp-arm {
1236819281eeSdp-arm 	int i, ret = 0;
1237819281eeSdp-arm 
1238cc672bb2Sdp-arm 	while (1) {
1239cc672bb2Sdp-arm 		int c, opt_index = 0;
1240cc672bb2Sdp-arm 		static struct option opts[] = {
1241cc672bb2Sdp-arm 			{ "verbose", no_argument, NULL, 'v' },
1242cc672bb2Sdp-arm 			{ NULL, no_argument, NULL, 0 }
1243cc672bb2Sdp-arm 		};
1244819281eeSdp-arm 
1245cc672bb2Sdp-arm 		/*
1246cc672bb2Sdp-arm 		 * Set POSIX mode so getopt stops at the first non-option
1247cc672bb2Sdp-arm 		 * which is the subcommand.
1248cc672bb2Sdp-arm 		 */
1249cc672bb2Sdp-arm 		c = getopt_long(argc, argv, "+v", opts, &opt_index);
1250cc672bb2Sdp-arm 		if (c == -1)
1251cc672bb2Sdp-arm 			break;
1252cc672bb2Sdp-arm 
1253cc672bb2Sdp-arm 		switch (c) {
1254cc672bb2Sdp-arm 		case 'v':
1255819281eeSdp-arm 			verbose = 1;
1256cc672bb2Sdp-arm 			break;
1257cc672bb2Sdp-arm 		default:
1258c9cb4089SMasahiro Yamada 			usage();
1259819281eeSdp-arm 		}
1260cc672bb2Sdp-arm 	}
1261cc672bb2Sdp-arm 	argc -= optind;
1262cc672bb2Sdp-arm 	argv += optind;
1263cc672bb2Sdp-arm 	/* Reset optind for subsequent getopt processing. */
1264cc672bb2Sdp-arm 	optind = 0;
1265cc672bb2Sdp-arm 
1266cc672bb2Sdp-arm 	if (argc == 0)
1267cc672bb2Sdp-arm 		usage();
1268819281eeSdp-arm 
1269e0f083a0Sdp-arm 	fill_image_descs();
1270819281eeSdp-arm 	for (i = 0; i < NELEM(cmds); i++) {
1271819281eeSdp-arm 		if (strcmp(cmds[i].name, argv[0]) == 0) {
1272819281eeSdp-arm 			ret = cmds[i].handler(argc, argv);
1273819281eeSdp-arm 			break;
1274819281eeSdp-arm 		}
1275819281eeSdp-arm 	}
1276819281eeSdp-arm 	if (i == NELEM(cmds))
1277819281eeSdp-arm 		usage();
1278e0f083a0Sdp-arm 	free_image_descs();
1279819281eeSdp-arm 	return ret;
1280819281eeSdp-arm }
1281