blob: 0c6dba89a099a32ff97ccfe04906e9344b6b3ab9 [file] [log] [blame]
wdenk5b1d7132002-11-03 00:07:02 +00001/*
Bartlomiej Siekad7816fb2008-03-11 12:34:47 +01002 * (C) Copyright 2008 Semihalf
3 *
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +05304 * (C) Copyright 2000-2009
wdenk5b1d7132002-11-03 00:07:02 +00005 * DENX Software Engineering
6 * Wolfgang Denk, wd@denx.de
wdenkdb82c8e2004-02-26 23:01:04 +00007 *
Wolfgang Denkd79de1d2013-07-08 09:37:19 +02008 * SPDX-License-Identifier: GPL-2.0+
wdenk5b1d7132002-11-03 00:07:02 +00009 */
10
Marian Balakowicz41d71ed2008-01-08 18:14:09 +010011#include "mkimage.h"
wdenk5b1d7132002-11-03 00:07:02 +000012#include <image.h>
Wolfgang Denke97ab722011-02-12 10:37:11 +010013#include <version.h>
wdenk5b1d7132002-11-03 00:07:02 +000014
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +053015static void copy_file(int, const char *, int);
wdenk5b1d7132002-11-03 00:07:02 +000016
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +053017/* parameters initialized by core will be used by the image type code */
Simon Glass52322d62016-02-22 22:55:38 -070018static struct image_tool_params params = {
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +053019 .os = IH_OS_LINUX,
20 .arch = IH_ARCH_PPC,
21 .type = IH_TYPE_KERNEL,
22 .comp = IH_COMP_GZIP,
23 .dtc = MKIMAGE_DEFAULT_DTC_OPTIONS,
Wolfgang Denk23416b82010-03-27 23:37:46 +010024 .imagename = "",
Shaohui Xieea65fd82012-08-10 02:49:35 +000025 .imagename2 = "",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +053026};
wdenk5b1d7132002-11-03 00:07:02 +000027
Simon Glass1d138b52016-06-30 10:52:17 -060028static enum ih_category cur_category;
29
30static int h_compare_category_name(const void *vtype1, const void *vtype2)
31{
32 const int *type1 = vtype1;
33 const int *type2 = vtype2;
34 const char *name1 = genimg_get_cat_short_name(cur_category, *type1);
35 const char *name2 = genimg_get_cat_short_name(cur_category, *type2);
36
37 return strcmp(name1, name2);
38}
39
Simon Glassbe421592016-06-30 10:52:18 -060040static int show_valid_options(enum ih_category category)
Simon Glass1d138b52016-06-30 10:52:17 -060041{
42 int *order;
43 int count;
44 int item;
45 int i;
46
47 count = genimg_get_cat_count(category);
48 order = calloc(count, sizeof(*order));
49 if (!order)
50 return -ENOMEM;
51
52 /* Sort the names in order of short name for easier reading */
53 for (item = 0; item < count; item++)
54 order[item] = item;
55 cur_category = category;
56 qsort(order, count, sizeof(int), h_compare_category_name);
57
58 fprintf(stderr, "\nInvalid %s, supported are:\n",
59 genimg_get_cat_desc(category));
60 for (i = 0; i < count; i++) {
61 item = order[i];
62 fprintf(stderr, "\t%-15s %s\n",
63 genimg_get_cat_short_name(category, item),
64 genimg_get_cat_name(category, item));
65 }
66 fprintf(stderr, "\n");
Simon Glassd435d9f2016-10-27 17:54:03 -060067 free(order);
Simon Glass1d138b52016-06-30 10:52:17 -060068
69 return 0;
70}
71
Simon Glass01168b32016-02-22 22:55:37 -070072static void usage(const char *msg)
Simon Glass99a7c262016-02-22 22:55:36 -070073{
Simon Glass01168b32016-02-22 22:55:37 -070074 fprintf(stderr, "Error: %s\n", msg);
Simon Glass99a7c262016-02-22 22:55:36 -070075 fprintf(stderr, "Usage: %s -l image\n"
76 " -l ==> list image header information\n",
77 params.cmdname);
78 fprintf(stderr,
79 " %s [-x] -A arch -O os -T type -C comp -a addr -e ep -n name -d data_file[:data_file...] image\n"
80 " -A ==> set architecture to 'arch'\n"
81 " -O ==> set operating system to 'os'\n"
82 " -T ==> set image type to 'type'\n"
83 " -C ==> set compression type 'comp'\n"
84 " -a ==> set load address to 'addr' (hex)\n"
85 " -e ==> set entry point to 'ep' (hex)\n"
86 " -n ==> set image name to 'name'\n"
87 " -d ==> use image data from 'datafile'\n"
88 " -x ==> set XIP (execute in place)\n",
89 params.cmdname);
90 fprintf(stderr,
Andreas Bießmannf4001582016-05-01 03:01:27 +020091 " %s [-D dtc_options] [-f fit-image.its|-f auto|-F] [-b <dtb> [-b <dtb>]] fit-image\n"
Vagrant Cascadian3e9e6d62016-10-23 20:45:17 -070092 " <dtb> file is used with -f auto, it may occur multiple times.\n",
Simon Glass99a7c262016-02-22 22:55:36 -070093 params.cmdname);
94 fprintf(stderr,
95 " -D => set all options for device tree compiler\n"
96 " -f => input filename for FIT source\n");
97#ifdef CONFIG_FIT_SIGNATURE
98 fprintf(stderr,
Teddy Reeda8457622016-06-09 19:38:02 -070099 "Signing / verified boot options: [-E] [-k keydir] [-K dtb] [ -c <comment>] [-p addr] [-r]\n"
100 " -E => place data outside of the FIT structure\n"
Simon Glass99a7c262016-02-22 22:55:36 -0700101 " -k => set directory containing private keys\n"
102 " -K => write public keys to this .dtb file\n"
103 " -c => add comment in signature node\n"
104 " -F => re-sign existing FIT image\n"
Teddy Reeda8457622016-06-09 19:38:02 -0700105 " -p => place external data at a static position\n"
Simon Glass99a7c262016-02-22 22:55:36 -0700106 " -r => mark keys used as 'required' in dtb\n");
107#else
108 fprintf(stderr,
109 "Signing / verified boot not supported (CONFIG_FIT_SIGNATURE undefined)\n");
110#endif
111 fprintf(stderr, " %s -V ==> print version information and exit\n",
112 params.cmdname);
113 fprintf(stderr, "Use -T to see a list of available image types\n");
114
115 exit(EXIT_FAILURE);
116}
117
Simon Glassbd8bc5d2016-02-22 22:55:52 -0700118static int add_content(int type, const char *fname)
119{
120 struct content_info *cont;
121
122 cont = calloc(1, sizeof(*cont));
123 if (!cont)
124 return -1;
125 cont->type = type;
126 cont->fname = fname;
127 if (params.content_tail)
128 params.content_tail->next = cont;
129 else
130 params.content_head = cont;
131 params.content_tail = cont;
132
133 return 0;
134}
135
Simon Glass7106bd42016-02-22 22:55:33 -0700136static void process_args(int argc, char **argv)
wdenk5b1d7132002-11-03 00:07:02 +0000137{
Peter Tyser632d9ca2010-04-04 22:36:03 -0500138 char *ptr;
Simon Glass04b46072016-02-22 22:55:48 -0700139 int type = IH_TYPE_INVALID;
140 char *datafile = NULL;
Simon Glasscf76cd22016-02-22 22:55:34 -0700141 int opt;
wdenk5b1d7132002-11-03 00:07:02 +0000142
Simon Glasscf76cd22016-02-22 22:55:34 -0700143 while ((opt = getopt(argc, argv,
Karl Beldan15bd26c2016-08-02 18:57:14 +0000144 "a:A:b:c:C:d:D:e:Ef:Fk:K:ln:p:O:rR:qsT:vVx")) != -1) {
Simon Glasscf76cd22016-02-22 22:55:34 -0700145 switch (opt) {
Simon Glassf6fe5f62016-02-22 22:55:35 -0700146 case 'a':
147 params.addr = strtoull(optarg, &ptr, 16);
148 if (*ptr) {
149 fprintf(stderr, "%s: invalid load address %s\n",
150 params.cmdname, optarg);
151 exit(EXIT_FAILURE);
152 }
Simon Glasscf76cd22016-02-22 22:55:34 -0700153 break;
154 case 'A':
155 params.arch = genimg_get_arch_id(optarg);
Simon Glasseaa25402016-06-30 10:52:19 -0600156 if (params.arch < 0) {
157 show_valid_options(IH_ARCH);
Simon Glass01168b32016-02-22 22:55:37 -0700158 usage("Invalid architecture");
Simon Glasseaa25402016-06-30 10:52:19 -0600159 }
Simon Glasscf76cd22016-02-22 22:55:34 -0700160 break;
Simon Glassbd8bc5d2016-02-22 22:55:52 -0700161 case 'b':
Andreas Bießmannddd34772016-05-03 15:17:03 +0200162 if (add_content(IH_TYPE_FLATDT, optarg)) {
Andreas Bießmannf4001582016-05-01 03:01:27 +0200163 fprintf(stderr,
164 "%s: Out of memory adding content '%s'",
165 params.cmdname, optarg);
166 exit(EXIT_FAILURE);
167 }
Simon Glassbd8bc5d2016-02-22 22:55:52 -0700168 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700169 case 'c':
170 params.comment = optarg;
171 break;
172 case 'C':
173 params.comp = genimg_get_comp_id(optarg);
Simon Glasseaa25402016-06-30 10:52:19 -0600174 if (params.comp < 0) {
175 show_valid_options(IH_COMP);
Simon Glass01168b32016-02-22 22:55:37 -0700176 usage("Invalid compression type");
Simon Glasseaa25402016-06-30 10:52:19 -0600177 }
Simon Glasscf76cd22016-02-22 22:55:34 -0700178 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700179 case 'd':
180 params.datafile = optarg;
181 params.dflag = 1;
182 break;
Simon Glassf6fe5f62016-02-22 22:55:35 -0700183 case 'D':
184 params.dtc = optarg;
185 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700186 case 'e':
187 params.ep = strtoull(optarg, &ptr, 16);
188 if (*ptr) {
189 fprintf(stderr, "%s: invalid entry point %s\n",
190 params.cmdname, optarg);
191 exit(EXIT_FAILURE);
wdenk5b1d7132002-11-03 00:07:02 +0000192 }
Simon Glasscf76cd22016-02-22 22:55:34 -0700193 params.eflag = 1;
194 break;
Simon Glassafd728c2016-02-22 22:55:53 -0700195 case 'E':
196 params.external_data = true;
197 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700198 case 'f':
Simon Glass88e31cb2016-02-22 22:55:51 -0700199 datafile = optarg;
200 params.auto_its = !strcmp(datafile, "auto");
Simon Glasscf76cd22016-02-22 22:55:34 -0700201 /* no break */
202 case 'F':
203 /*
204 * The flattened image tree (FIT) format
205 * requires a flattened device tree image type
206 */
207 params.type = IH_TYPE_FLATDT;
208 params.fflag = 1;
209 break;
210 case 'k':
211 params.keydir = optarg;
212 break;
213 case 'K':
214 params.keydest = optarg;
215 break;
Simon Glassf6fe5f62016-02-22 22:55:35 -0700216 case 'l':
217 params.lflag = 1;
218 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700219 case 'n':
220 params.imagename = optarg;
221 break;
Simon Glassf6fe5f62016-02-22 22:55:35 -0700222 case 'O':
223 params.os = genimg_get_os_id(optarg);
Simon Glasseaa25402016-06-30 10:52:19 -0600224 if (params.os < 0) {
225 show_valid_options(IH_OS);
Simon Glass01168b32016-02-22 22:55:37 -0700226 usage("Invalid operating system");
Simon Glasseaa25402016-06-30 10:52:19 -0600227 }
Simon Glassf6fe5f62016-02-22 22:55:35 -0700228 break;
Teddy Reeda8457622016-06-09 19:38:02 -0700229 case 'p':
230 params.external_offset = strtoull(optarg, &ptr, 16);
231 if (*ptr) {
232 fprintf(stderr, "%s: invalid offset size %s\n",
233 params.cmdname, optarg);
234 exit(EXIT_FAILURE);
235 }
Teddy Reed9e3cd3c2016-07-11 22:54:26 -0700236 break;
Simon Glassda1805f2016-05-01 13:55:38 -0600237 case 'q':
238 params.quiet = 1;
239 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700240 case 'r':
241 params.require_keys = 1;
242 break;
243 case 'R':
244 /*
245 * This entry is for the second configuration
246 * file, if only one is not enough.
247 */
248 params.imagename2 = optarg;
249 break;
250 case 's':
251 params.skipcpy = 1;
252 break;
Simon Glassf6fe5f62016-02-22 22:55:35 -0700253 case 'T':
Simon Glass04b46072016-02-22 22:55:48 -0700254 type = genimg_get_type_id(optarg);
255 if (type < 0) {
Simon Glassbe421592016-06-30 10:52:18 -0600256 show_valid_options(IH_TYPE);
Simon Glass01168b32016-02-22 22:55:37 -0700257 usage("Invalid image type");
Simon Glassf6fe5f62016-02-22 22:55:35 -0700258 }
259 break;
Simon Glasscf76cd22016-02-22 22:55:34 -0700260 case 'v':
261 params.vflag++;
262 break;
263 case 'V':
264 printf("mkimage version %s\n", PLAIN_VERSION);
265 exit(EXIT_SUCCESS);
266 case 'x':
267 params.xflag++;
268 break;
269 default:
Simon Glass01168b32016-02-22 22:55:37 -0700270 usage("Invalid option");
wdenk5b1d7132002-11-03 00:07:02 +0000271 }
wdenk5b1d7132002-11-03 00:07:02 +0000272 }
273
Andreas Bießmannddd34772016-05-03 15:17:03 +0200274 /* The last parameter is expected to be the imagefile */
275 if (optind < argc)
Andreas Bießmannf4001582016-05-01 03:01:27 +0200276 params.imagefile = argv[optind];
277
Simon Glass04b46072016-02-22 22:55:48 -0700278 /*
279 * For auto-generated FIT images we need to know the image type to put
280 * in the FIT, which is separate from the file's image type (which
281 * will always be IH_TYPE_FLATDT in this case).
282 */
283 if (params.type == IH_TYPE_FLATDT) {
Simon Glassa89857d2016-06-30 10:52:07 -0600284 params.fit_image_type = type ? type : IH_TYPE_KERNEL;
Simon Glass156c3492016-06-30 10:52:08 -0600285 /* For auto_its, datafile is always 'auto' */
Simon Glass88e31cb2016-02-22 22:55:51 -0700286 if (!params.auto_its)
287 params.datafile = datafile;
Simon Glass79b5ee52016-06-30 10:52:09 -0600288 else if (!params.datafile)
289 usage("Missing data file for auto-FIT (use -d)");
Simon Glass04b46072016-02-22 22:55:48 -0700290 } else if (type != IH_TYPE_INVALID) {
291 params.type = type;
292 }
293
294 if (!params.imagefile)
Simon Glass01168b32016-02-22 22:55:37 -0700295 usage("Missing output filename");
Simon Glass7106bd42016-02-22 22:55:33 -0700296}
297
Simon Glass7106bd42016-02-22 22:55:33 -0700298int main(int argc, char **argv)
299{
300 int ifd = -1;
301 struct stat sbuf;
302 char *ptr;
303 int retval = 0;
304 struct image_type_params *tparams = NULL;
305 int pad_len = 0;
306 int dfd;
307
308 params.cmdname = *argv;
309 params.addr = 0;
310 params.ep = 0;
311
312 process_args(argc, argv);
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530313
314 /* set tparams as per input type_id */
Guilherme Maciel Ferreira28be1cf2015-01-15 02:48:07 -0200315 tparams = imagetool_get_type(params.type);
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530316 if (tparams == NULL) {
317 fprintf (stderr, "%s: unsupported type %s\n",
318 params.cmdname, genimg_get_type_name(params.type));
319 exit (EXIT_FAILURE);
320 }
321
322 /*
323 * check the passed arguments parameters meets the requirements
324 * as per image type to be generated/listed
325 */
326 if (tparams->check_params)
327 if (tparams->check_params (&params))
Simon Glass01168b32016-02-22 22:55:37 -0700328 usage("Bad parameters for image type");
wdenk5b1d7132002-11-03 00:07:02 +0000329
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530330 if (!params.eflag) {
331 params.ep = params.addr;
wdenk5b1d7132002-11-03 00:07:02 +0000332 /* If XIP, entry point must be after the U-Boot header */
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530333 if (params.xflag)
334 params.ep += tparams->header_size;
wdenk5b1d7132002-11-03 00:07:02 +0000335 }
336
Peter Tyserb2f8b942009-11-24 16:42:10 -0600337 if (params.fflag){
338 if (tparams->fflag_handle)
339 /*
340 * in some cases, some additional processing needs
341 * to be done if fflag is defined
342 *
343 * For ex. fit_handle_file for Fit file support
344 */
345 retval = tparams->fflag_handle(&params);
wdenk5b1d7132002-11-03 00:07:02 +0000346
Peter Tyserb2f8b942009-11-24 16:42:10 -0600347 if (retval != EXIT_SUCCESS)
348 exit (retval);
wdenk5b1d7132002-11-03 00:07:02 +0000349 }
350
Peter Tyserb2f8b942009-11-24 16:42:10 -0600351 if (params.lflag || params.fflag) {
352 ifd = open (params.imagefile, O_RDONLY|O_BINARY);
353 } else {
354 ifd = open (params.imagefile,
355 O_RDWR|O_CREAT|O_TRUNC|O_BINARY, 0666);
356 }
357
358 if (ifd < 0) {
359 fprintf (stderr, "%s: Can't open %s: %s\n",
360 params.cmdname, params.imagefile,
361 strerror(errno));
362 exit (EXIT_FAILURE);
363 }
364
365 if (params.lflag || params.fflag) {
wdenk5b1d7132002-11-03 00:07:02 +0000366 /*
367 * list header information of existing image
368 */
369 if (fstat(ifd, &sbuf) < 0) {
370 fprintf (stderr, "%s: Can't stat %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530371 params.cmdname, params.imagefile,
372 strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000373 exit (EXIT_FAILURE);
374 }
375
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530376 if ((unsigned)sbuf.st_size < tparams->header_size) {
wdenk5b1d7132002-11-03 00:07:02 +0000377 fprintf (stderr,
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530378 "%s: Bad size: \"%s\" is not valid image\n",
379 params.cmdname, params.imagefile);
wdenk5b1d7132002-11-03 00:07:02 +0000380 exit (EXIT_FAILURE);
381 }
382
Mike Frysingercd7e1a32008-05-01 04:13:05 -0400383 ptr = mmap(0, sbuf.st_size, PROT_READ, MAP_SHARED, ifd, 0);
384 if (ptr == MAP_FAILED) {
wdenk5b1d7132002-11-03 00:07:02 +0000385 fprintf (stderr, "%s: Can't read %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530386 params.cmdname, params.imagefile,
387 strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000388 exit (EXIT_FAILURE);
389 }
390
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530391 /*
392 * scan through mkimage registry for all supported image types
393 * and verify the input image file header for match
394 * Print the image information for matched image type
395 * Returns the error code if not matched
396 */
Guilherme Maciel Ferreira0b338032015-01-15 02:48:05 -0200397 retval = imagetool_verify_print_header(ptr, &sbuf,
398 tparams, &params);
wdenk5b1d7132002-11-03 00:07:02 +0000399
wdenk5b1d7132002-11-03 00:07:02 +0000400 (void) munmap((void *)ptr, sbuf.st_size);
401 (void) close (ifd);
402
Prafulla Wadaskar139cb082009-08-10 18:49:37 +0530403 exit (retval);
wdenk5b1d7132002-11-03 00:07:02 +0000404 }
405
Marek Vasut4cc86272015-12-07 18:01:54 +0100406 if ((params.type != IH_TYPE_MULTI) && (params.type != IH_TYPE_SCRIPT)) {
Philippe De Swert3edd0412015-12-04 00:11:23 +0200407 dfd = open(params.datafile, O_RDONLY | O_BINARY);
408 if (dfd < 0) {
409 fprintf(stderr, "%s: Can't open %s: %s\n",
410 params.cmdname, params.datafile,
411 strerror(errno));
412 exit(EXIT_FAILURE);
413 }
Simon Glass139651f2015-06-23 15:39:12 -0600414
Philippe De Swert3edd0412015-12-04 00:11:23 +0200415 if (fstat(dfd, &sbuf) < 0) {
416 fprintf(stderr, "%s: Can't stat %s: %s\n",
417 params.cmdname, params.datafile,
418 strerror(errno));
419 exit(EXIT_FAILURE);
420 }
Simon Glass139651f2015-06-23 15:39:12 -0600421
Philippe De Swert3edd0412015-12-04 00:11:23 +0200422 params.file_size = sbuf.st_size + tparams->header_size;
423 close(dfd);
424 }
Simon Glass139651f2015-06-23 15:39:12 -0600425
wdenk5b1d7132002-11-03 00:07:02 +0000426 /*
Stefano Babic4a963302011-09-15 23:50:16 +0000427 * In case there an header with a variable
428 * length will be added, the corresponding
429 * function is called. This is responsible to
430 * allocate memory for the header itself.
wdenk5b1d7132002-11-03 00:07:02 +0000431 */
Stefano Babic4a963302011-09-15 23:50:16 +0000432 if (tparams->vrec_header)
Stefano Babic6531bd92013-08-19 19:03:19 +0200433 pad_len = tparams->vrec_header(&params, tparams);
Stefano Babic4a963302011-09-15 23:50:16 +0000434 else
435 memset(tparams->hdr, 0, tparams->header_size);
wdenk5b1d7132002-11-03 00:07:02 +0000436
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530437 if (write(ifd, tparams->hdr, tparams->header_size)
438 != tparams->header_size) {
wdenk5b1d7132002-11-03 00:07:02 +0000439 fprintf (stderr, "%s: Write error on %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530440 params.cmdname, params.imagefile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000441 exit (EXIT_FAILURE);
442 }
443
Christian Rieschc14f2812011-12-09 09:47:38 +0000444 if (!params.skipcpy) {
445 if (params.type == IH_TYPE_MULTI ||
446 params.type == IH_TYPE_SCRIPT) {
447 char *file = params.datafile;
448 uint32_t size;
wdenk5b1d7132002-11-03 00:07:02 +0000449
Christian Rieschc14f2812011-12-09 09:47:38 +0000450 for (;;) {
451 char *sep = NULL;
wdenk5b1d7132002-11-03 00:07:02 +0000452
Christian Rieschc14f2812011-12-09 09:47:38 +0000453 if (file) {
454 if ((sep = strchr(file, ':')) != NULL) {
455 *sep = '\0';
456 }
457
458 if (stat (file, &sbuf) < 0) {
459 fprintf (stderr, "%s: Can't stat %s: %s\n",
460 params.cmdname, file, strerror(errno));
461 exit (EXIT_FAILURE);
462 }
463 size = cpu_to_uimage (sbuf.st_size);
464 } else {
465 size = 0;
wdenk5b1d7132002-11-03 00:07:02 +0000466 }
467
Christian Rieschc14f2812011-12-09 09:47:38 +0000468 if (write(ifd, (char *)&size, sizeof(size)) != sizeof(size)) {
469 fprintf (stderr, "%s: Write error on %s: %s\n",
470 params.cmdname, params.imagefile,
471 strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000472 exit (EXIT_FAILURE);
473 }
wdenk5b1d7132002-11-03 00:07:02 +0000474
Christian Rieschc14f2812011-12-09 09:47:38 +0000475 if (!file) {
476 break;
477 }
wdenk5b1d7132002-11-03 00:07:02 +0000478
Christian Rieschc14f2812011-12-09 09:47:38 +0000479 if (sep) {
480 *sep = ':';
481 file = sep + 1;
482 } else {
483 file = NULL;
484 }
wdenk5b1d7132002-11-03 00:07:02 +0000485 }
wdenk5b1d7132002-11-03 00:07:02 +0000486
Christian Rieschc14f2812011-12-09 09:47:38 +0000487 file = params.datafile;
wdenk5b1d7132002-11-03 00:07:02 +0000488
Christian Rieschc14f2812011-12-09 09:47:38 +0000489 for (;;) {
490 char *sep = strchr(file, ':');
491 if (sep) {
492 *sep = '\0';
493 copy_file (ifd, file, 1);
494 *sep++ = ':';
495 file = sep;
496 } else {
497 copy_file (ifd, file, 0);
498 break;
499 }
wdenk5b1d7132002-11-03 00:07:02 +0000500 }
Shaohui Xieea65fd82012-08-10 02:49:35 +0000501 } else if (params.type == IH_TYPE_PBLIMAGE) {
502 /* PBL has special Image format, implements its' own */
503 pbl_load_uboot(ifd, &params);
Christian Rieschc14f2812011-12-09 09:47:38 +0000504 } else {
Stefano Babic6531bd92013-08-19 19:03:19 +0200505 copy_file(ifd, params.datafile, pad_len);
wdenk5b1d7132002-11-03 00:07:02 +0000506 }
wdenk5b1d7132002-11-03 00:07:02 +0000507 }
508
509 /* We're a bit of paranoid */
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530510#if defined(_POSIX_SYNCHRONIZED_IO) && \
511 !defined(__sun__) && \
512 !defined(__FreeBSD__) && \
Luka Perkov98c689b2014-08-09 18:17:47 +0200513 !defined(__OpenBSD__) && \
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530514 !defined(__APPLE__)
wdenk5b1d7132002-11-03 00:07:02 +0000515 (void) fdatasync (ifd);
516#else
517 (void) fsync (ifd);
518#endif
519
520 if (fstat(ifd, &sbuf) < 0) {
521 fprintf (stderr, "%s: Can't stat %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530522 params.cmdname, params.imagefile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000523 exit (EXIT_FAILURE);
524 }
Simon Glass139651f2015-06-23 15:39:12 -0600525 params.file_size = sbuf.st_size;
wdenk5b1d7132002-11-03 00:07:02 +0000526
Mike Frysingercd7e1a32008-05-01 04:13:05 -0400527 ptr = mmap(0, sbuf.st_size, PROT_READ|PROT_WRITE, MAP_SHARED, ifd, 0);
528 if (ptr == MAP_FAILED) {
wdenk5b1d7132002-11-03 00:07:02 +0000529 fprintf (stderr, "%s: Can't map %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530530 params.cmdname, params.imagefile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000531 exit (EXIT_FAILURE);
532 }
533
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530534 /* Setup the image header as per input image type*/
535 if (tparams->set_header)
536 tparams->set_header (ptr, &sbuf, ifd, &params);
537 else {
538 fprintf (stderr, "%s: Can't set header for %s: %s\n",
539 params.cmdname, tparams->name, strerror(errno));
540 exit (EXIT_FAILURE);
541 }
wdenk5b1d7132002-11-03 00:07:02 +0000542
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530543 /* Print the image information by processing image header */
544 if (tparams->print_header)
545 tparams->print_header (ptr);
546 else {
547 fprintf (stderr, "%s: Can't print header for %s: %s\n",
548 params.cmdname, tparams->name, strerror(errno));
549 exit (EXIT_FAILURE);
550 }
wdenk5b1d7132002-11-03 00:07:02 +0000551
552 (void) munmap((void *)ptr, sbuf.st_size);
553
554 /* We're a bit of paranoid */
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530555#if defined(_POSIX_SYNCHRONIZED_IO) && \
556 !defined(__sun__) && \
557 !defined(__FreeBSD__) && \
Luka Perkov98c689b2014-08-09 18:17:47 +0200558 !defined(__OpenBSD__) && \
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530559 !defined(__APPLE__)
wdenk5b1d7132002-11-03 00:07:02 +0000560 (void) fdatasync (ifd);
561#else
562 (void) fsync (ifd);
563#endif
564
565 if (close(ifd)) {
566 fprintf (stderr, "%s: Write error on %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530567 params.cmdname, params.imagefile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000568 exit (EXIT_FAILURE);
569 }
570
571 exit (EXIT_SUCCESS);
572}
573
574static void
575copy_file (int ifd, const char *datafile, int pad)
576{
577 int dfd;
578 struct stat sbuf;
579 unsigned char *ptr;
580 int tail;
581 int zero = 0;
Stefano Babic6531bd92013-08-19 19:03:19 +0200582 uint8_t zeros[4096];
wdenk5b1d7132002-11-03 00:07:02 +0000583 int offset = 0;
584 int size;
Guilherme Maciel Ferreira28be1cf2015-01-15 02:48:07 -0200585 struct image_type_params *tparams = imagetool_get_type(params.type);
wdenk5b1d7132002-11-03 00:07:02 +0000586
Stefano Babic6531bd92013-08-19 19:03:19 +0200587 memset(zeros, 0, sizeof(zeros));
588
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530589 if (params.vflag) {
wdenk5b1d7132002-11-03 00:07:02 +0000590 fprintf (stderr, "Adding Image %s\n", datafile);
591 }
592
wdenk1ee550e2003-10-08 22:14:02 +0000593 if ((dfd = open(datafile, O_RDONLY|O_BINARY)) < 0) {
wdenk5b1d7132002-11-03 00:07:02 +0000594 fprintf (stderr, "%s: Can't open %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530595 params.cmdname, datafile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000596 exit (EXIT_FAILURE);
597 }
598
599 if (fstat(dfd, &sbuf) < 0) {
600 fprintf (stderr, "%s: Can't stat %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530601 params.cmdname, datafile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000602 exit (EXIT_FAILURE);
603 }
604
Mike Frysingercd7e1a32008-05-01 04:13:05 -0400605 ptr = mmap(0, sbuf.st_size, PROT_READ, MAP_SHARED, dfd, 0);
606 if (ptr == MAP_FAILED) {
wdenk5b1d7132002-11-03 00:07:02 +0000607 fprintf (stderr, "%s: Can't read %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530608 params.cmdname, datafile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000609 exit (EXIT_FAILURE);
610 }
611
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530612 if (params.xflag) {
wdenk5b1d7132002-11-03 00:07:02 +0000613 unsigned char *p = NULL;
614 /*
615 * XIP: do not append the image_header_t at the
616 * beginning of the file, but consume the space
617 * reserved for it.
618 */
619
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530620 if ((unsigned)sbuf.st_size < tparams->header_size) {
wdenk5b1d7132002-11-03 00:07:02 +0000621 fprintf (stderr,
622 "%s: Bad size: \"%s\" is too small for XIP\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530623 params.cmdname, datafile);
wdenk5b1d7132002-11-03 00:07:02 +0000624 exit (EXIT_FAILURE);
625 }
626
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530627 for (p = ptr; p < ptr + tparams->header_size; p++) {
wdenk5b1d7132002-11-03 00:07:02 +0000628 if ( *p != 0xff ) {
629 fprintf (stderr,
630 "%s: Bad file: \"%s\" has invalid buffer for XIP\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530631 params.cmdname, datafile);
wdenk5b1d7132002-11-03 00:07:02 +0000632 exit (EXIT_FAILURE);
633 }
634 }
635
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530636 offset = tparams->header_size;
wdenk5b1d7132002-11-03 00:07:02 +0000637 }
638
639 size = sbuf.st_size - offset;
640 if (write(ifd, ptr + offset, size) != size) {
641 fprintf (stderr, "%s: Write error on %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530642 params.cmdname, params.imagefile, strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000643 exit (EXIT_FAILURE);
644 }
645
Stefano Babic6531bd92013-08-19 19:03:19 +0200646 tail = size % 4;
647 if ((pad == 1) && (tail != 0)) {
wdenk5b1d7132002-11-03 00:07:02 +0000648
649 if (write(ifd, (char *)&zero, 4-tail) != 4-tail) {
650 fprintf (stderr, "%s: Write error on %s: %s\n",
Prafulla Wadaskarfabf3cf2009-08-19 17:36:46 +0530651 params.cmdname, params.imagefile,
652 strerror(errno));
wdenk5b1d7132002-11-03 00:07:02 +0000653 exit (EXIT_FAILURE);
654 }
Stefano Babic6531bd92013-08-19 19:03:19 +0200655 } else if (pad > 1) {
Simon Glass46409cc2015-08-30 16:55:22 -0600656 while (pad > 0) {
657 int todo = sizeof(zeros);
658
659 if (todo > pad)
660 todo = pad;
661 if (write(ifd, (char *)&zeros, todo) != todo) {
662 fprintf(stderr, "%s: Write error on %s: %s\n",
663 params.cmdname, params.imagefile,
664 strerror(errno));
665 exit(EXIT_FAILURE);
666 }
667 pad -= todo;
Stefano Babic6531bd92013-08-19 19:03:19 +0200668 }
wdenk5b1d7132002-11-03 00:07:02 +0000669 }
670
671 (void) munmap((void *)ptr, sbuf.st_size);
672 (void) close (dfd);
673}