board: engicam: Attach i.MX6 common code
[platform/kernel/u-boot.git] / tools / zynqmpbif.c
1 /*
2  * Copyright (C) 2018 Alexander Graf <agraf@suse.de>
3  *
4  * SPDX-License-Identifier:     GPL-2.0+
5  */
6
7 #include "imagetool.h"
8 #include "mkimage.h"
9 #include "zynqmpimage.h"
10 #include <elf.h>
11 #include <image.h>
12
13 struct bif_entry {
14         const char *filename;
15         uint64_t flags;
16         uint64_t dest_cpu;
17         uint64_t exp_lvl;
18         uint64_t dest_dev;
19         uint64_t load;
20         uint64_t entry;
21         size_t offset;
22 };
23
24 enum bif_flag {
25         BIF_FLAG_AESKEYFILE,
26         BIF_FLAG_INIT,
27         BIF_FLAG_UDF_BH,
28         BIF_FLAG_HEADERSIGNATURE,
29         BIF_FLAG_PPKFILE,
30         BIF_FLAG_PSKFILE,
31         BIF_FLAG_SPKFILE,
32         BIF_FLAG_SSKFILE,
33         BIF_FLAG_SPKSIGNATURE,
34         BIF_FLAG_FSBL_CONFIG,
35         BIF_FLAG_AUTH_PARAMS,
36         BIF_FLAG_KEYSRC_ENCRYPTION,
37         BIF_FLAG_PMUFW_IMAGE,
38         BIF_FLAG_BOOTLOADER,
39         BIF_FLAG_TZ,
40         BIF_FLAG_BH_KEY_IV,
41         BIF_FLAG_BH_KEYFILE,
42         BIF_FLAG_PUF_FILE,
43         BIF_FLAG_AARCH32,
44         BIF_FLAG_PART_OWNER_UBOOT,
45
46         /* Internal flags */
47         BIF_FLAG_BIT_FILE,
48         BIF_FLAG_ELF_FILE,
49         BIF_FLAG_BIN_FILE,
50 };
51
52 struct bif_flags {
53         const char name[32];
54         uint64_t flag;
55         char *(*parse)(char *line, struct bif_entry *bf);
56 };
57
58 struct bif_file_type {
59         const char name[32];
60         uint32_t header;
61         int (*add)(struct bif_entry *bf);
62 };
63
64 struct bif_output {
65         size_t data_len;
66         char *data;
67         struct image_header_table *imgheader;
68         struct zynqmp_header *header;
69         struct partition_header *last_part;
70 };
71
72 struct bif_output bif_output;
73
74 static uint32_t zynqmp_csum(void *start, void *end)
75 {
76         uint32_t checksum = 0;
77         uint32_t *ptr32 = start;
78
79         while (ptr32 != end) {
80                 checksum += le32_to_cpu(*ptr32);
81                 ptr32++;
82         }
83
84         return ~checksum;
85 }
86
87 static int zynqmpbif_check_params(struct image_tool_params *params)
88 {
89         if (!params)
90                 return 0;
91
92         if (params->addr != 0x0) {
93                 fprintf(stderr, "Error: Load Address can not be specified.\n");
94                 return -1;
95         }
96
97         if (params->eflag) {
98                 fprintf(stderr, "Error: Entry Point can not be specified.\n");
99                 return -1;
100         }
101
102         return !(params->lflag || params->dflag);
103 }
104
105 static int zynqmpbif_check_image_types(uint8_t type)
106 {
107         return (type == IH_TYPE_ZYNQMPBIF) ? EXIT_SUCCESS : EXIT_FAILURE;
108 }
109
110 static char *parse_dest_cpu(char *line, struct bif_entry *bf)
111 {
112         uint64_t i;
113
114         for (i = 0; i < ARRAY_SIZE(dest_cpus); i++) {
115                 if (!strncmp(line, dest_cpus[i], strlen(dest_cpus[i]))) {
116                         bf->dest_cpu = i << PART_ATTR_DEST_CPU_SHIFT;
117                         return line + strlen(dest_cpus[i]);
118                 }
119
120                 /* a5x can also be written as a53 */
121                 if (!strncmp(dest_cpus[i], "a5x", 3)) {
122                         char a53[] = "a53-X";
123
124                         a53[4] = dest_cpus[i][4];
125                         if (!strncmp(line, a53, strlen(a53))) {
126                                 bf->dest_cpu = i << PART_ATTR_DEST_CPU_SHIFT;
127                                 return line + strlen(a53);
128                         }
129                 }
130         }
131
132         return line;
133 }
134
135 static char *parse_el(char *line, struct bif_entry *bf)
136 {
137         const char *dest_els[] = { "none", "el-0", "el-1", "el-2", "el-3" };
138         int i;
139
140         for (i = 0; i < ARRAY_SIZE(dest_els); i++) {
141                 if (!strncmp(line, dest_els[i], strlen(dest_els[i]))) {
142                         bf->exp_lvl = i;
143                         return line + strlen(dest_els[i]);
144                 }
145         }
146
147         return line;
148 }
149
150 static char *parse_load(char *line, struct bif_entry *bf)
151 {
152         char *endptr;
153
154         bf->load = strtoll(line, &endptr, 0);
155
156         return endptr;
157 }
158
159 static char *parse_entry(char *line, struct bif_entry *bf)
160 {
161         char *endptr;
162
163         bf->entry = strtoll(line, &endptr, 0);
164
165         return endptr;
166 }
167
168 static char *parse_offset(char *line, struct bif_entry *bf)
169 {
170         char *endptr;
171
172         bf->offset = strtoll(line, &endptr, 0);
173
174         return endptr;
175 }
176
177 static char *parse_partition_owner(char *line, struct bif_entry *bf)
178 {
179         char *endptr = NULL;
180
181         if (!strncmp(line, "fsbl", 4)) {
182                 endptr = line + 4;
183         } else if (!strncmp(line, "uboot", 5)) {
184                 bf->flags |= 1ULL << BIF_FLAG_PART_OWNER_UBOOT;
185                 endptr = line + 5;
186         } else {
187                 printf("ERROR: Unknown partition type '%s'\n", line);
188         }
189
190         return endptr;
191 }
192
193 static const struct bif_flags bif_flags[] = {
194         { "fsbl_config", BIF_FLAG_FSBL_CONFIG },
195         { "trustzone", BIF_FLAG_TZ },
196         { "pmufw_image", BIF_FLAG_PMUFW_IMAGE },
197         { "bootloader", BIF_FLAG_BOOTLOADER },
198         { "destination_cpu=", 0, parse_dest_cpu },
199         { "exception_level=", 0, parse_el },
200         { "load=", 0, parse_load },
201         { "startup=", 0, parse_entry },
202         { "offset=", 0, parse_offset },
203         { "partition_owner=", 0, parse_partition_owner },
204 };
205
206 static char *read_full_file(const char *filename, size_t *size)
207 {
208         char *buf, *bufp;
209         struct stat sbuf;
210         int len = 0, r, fd;
211
212         fd = open(filename, O_RDONLY);
213         if (fd < 0)
214                 return NULL;
215
216         if (fstat(fd, &sbuf) < 0)
217                 return NULL;
218
219         if (size)
220                 *size = sbuf.st_size;
221
222         buf = malloc(sbuf.st_size);
223         if (!buf)
224                 return NULL;
225
226         bufp = buf;
227         while (len < sbuf.st_size) {
228                 r = read(fd, bufp, sbuf.st_size - len);
229                 if (r < 0)
230                         return NULL;
231                 len += r;
232                 bufp += r;
233         }
234
235         close(fd);
236
237         return buf;
238 }
239
240 static int bif_add_blob(const void *data, size_t len, size_t *offset)
241 {
242         size_t new_size;
243         uintptr_t header_off;
244         uintptr_t last_part_off;
245         uintptr_t imgheader_off;
246         uintptr_t old_data = (uintptr_t)bif_output.data;
247         void *new_data;
248
249         header_off = (uintptr_t)bif_output.header - old_data;
250         last_part_off = (uintptr_t)bif_output.last_part - old_data;
251         imgheader_off = (uintptr_t)bif_output.imgheader - old_data;
252
253         if (offset && *offset) {
254                 /* Pad to a given offset */
255                 if (bif_output.data_len > *offset) {
256                         printf("Can not pad to offset %zx\n", *offset);
257                         return -1;
258                 }
259
260                 bif_output.data_len = *offset;
261         }
262
263         new_size = ROUND(bif_output.data_len + len, 64);
264         new_data = realloc(bif_output.data, new_size);
265         memcpy(new_data + bif_output.data_len, data, len);
266         if (offset)
267                 *offset = bif_output.data_len;
268         bif_output.data = new_data;
269         bif_output.data_len = new_size;
270
271         /* Readjust internal pointers */
272         if (bif_output.header)
273                 bif_output.header = new_data + header_off;
274         if (bif_output.last_part)
275                 bif_output.last_part = new_data + last_part_off;
276         if (bif_output.imgheader)
277                 bif_output.imgheader = new_data + imgheader_off;
278
279         return 0;
280 }
281
282 static int bif_init(void)
283 {
284         struct zynqmp_header header = { { 0 } };
285         int r;
286
287         zynqmpimage_default_header(&header);
288
289         r = bif_add_blob(&header, sizeof(header), NULL);
290         if (r)
291                 return r;
292
293         bif_output.header = (void *)bif_output.data;
294
295         return 0;
296 }
297
298 static int bif_add_pmufw(struct bif_entry *bf, const char *data, size_t len)
299 {
300         int r;
301
302         if (bif_output.header->image_offset) {
303                 printf("PMUFW expected before bootloader in your .bif file!\n");
304                 return -1;
305         }
306
307         r = bif_add_blob(data, len, &bf->offset);
308         if (r)
309                 return r;
310
311         len = ROUND(len, 64);
312         bif_output.header->pfw_image_length = cpu_to_le32(len);
313         bif_output.header->total_pfw_image_length = cpu_to_le32(len);
314         bif_output.header->image_offset = cpu_to_le32(bf->offset);
315
316         return 0;
317 }
318
319 static int bif_add_part(struct bif_entry *bf, const char *data, size_t len)
320 {
321         size_t parthdr_offset = 0;
322         size_t len_padded = ROUND(len, 4);
323
324         struct partition_header parthdr = {
325                 .len_enc = cpu_to_le32(len_padded / 4),
326                 .len_unenc = cpu_to_le32(len_padded / 4),
327                 .len = cpu_to_le32(len_padded / 4),
328                 .entry_point = cpu_to_le64(bf->entry),
329                 .load_address = cpu_to_le64(bf->load),
330         };
331         int r;
332         uint32_t csum;
333
334         if (len < len_padded) {
335                 char *newdata = malloc(len_padded);
336                 memcpy(newdata, data, len);
337                 memset(newdata + len, 0, len_padded - len);
338                 data = newdata;
339         }
340
341         if (bf->flags & (1ULL << BIF_FLAG_PMUFW_IMAGE))
342                 return bif_add_pmufw(bf, data, len);
343
344         r = bif_add_blob(data, len, &bf->offset);
345         if (r)
346                 return r;
347
348         parthdr.offset = cpu_to_le32(bf->offset / 4);
349
350         if (bf->flags & (1ULL << BIF_FLAG_BOOTLOADER)) {
351                 if (bif_output.last_part) {
352                         printf("ERROR: Bootloader expected before others\n");
353                         return -1;
354                 }
355
356                 parthdr.offset = cpu_to_le32(bif_output.header->image_offset);
357                 parthdr.len = cpu_to_le32((bf->offset + len -
358                         bif_output.header->image_offset) / 4);
359                 parthdr.len_enc = parthdr.len;
360                 parthdr.len_unenc = parthdr.len;
361         }
362
363         /* Normalize EL */
364         bf->exp_lvl = bf->exp_lvl ? bf->exp_lvl - 1 : 3;
365         parthdr.attributes |= bf->exp_lvl << PART_ATTR_TARGET_EL_SHIFT;
366         parthdr.attributes |= bf->dest_dev;
367         parthdr.attributes |= bf->dest_cpu;
368         if (bf->flags & (1ULL << BIF_FLAG_TZ))
369                 parthdr.attributes |= PART_ATTR_TZ_SECURE;
370         if (bf->flags & (1ULL << BIF_FLAG_PART_OWNER_UBOOT))
371                 parthdr.attributes |= PART_ATTR_PART_OWNER_UBOOT;
372         switch (bf->dest_cpu) {
373         case PART_ATTR_DEST_CPU_NONE:
374         case PART_ATTR_DEST_CPU_A53_0:
375         case PART_ATTR_DEST_CPU_A53_1:
376         case PART_ATTR_DEST_CPU_A53_2:
377         case PART_ATTR_DEST_CPU_A53_3:
378                 if (bf->flags & (1ULL << BIF_FLAG_AARCH32))
379                         parthdr.attributes |= PART_ATTR_A53_EXEC_AARCH32;
380         }
381
382         csum = zynqmp_csum(&parthdr, &parthdr.checksum);
383         parthdr.checksum = cpu_to_le32(csum);
384
385         r = bif_add_blob(&parthdr, sizeof(parthdr), &parthdr_offset);
386         if (r)
387                 return r;
388
389         /* Add image header table if not there yet */
390         if (!bif_output.imgheader) {
391                 size_t imghdr_off = 0;
392                 struct image_header_table imghdr = {
393                         .version = cpu_to_le32(0x01020000),
394                         .nr_parts = 0,
395                 };
396
397                 r = bif_add_blob(&imghdr, sizeof(imghdr), &imghdr_off);
398                 if (r)
399                         return r;
400
401                 bif_output.header->image_header_table_offset = imghdr_off;
402                 bif_output.imgheader = (void *)(bif_output.data + imghdr_off);
403         }
404
405         bif_output.imgheader->nr_parts = cpu_to_le32(le32_to_cpu(
406                 bif_output.imgheader->nr_parts) + 1);
407
408         /* Link to this partition header */
409         if (bif_output.last_part) {
410                 bif_output.last_part->next_partition_offset =
411                         cpu_to_le32(parthdr_offset / 4);
412
413                 /* Recalc checksum of last_part */
414                 csum = zynqmp_csum(bif_output.last_part,
415                                    &bif_output.last_part->checksum);
416                 bif_output.last_part->checksum = cpu_to_le32(csum);
417         } else {
418                 bif_output.imgheader->partition_header_offset =
419                         cpu_to_le32(parthdr_offset / 4);
420         }
421         bif_output.last_part = (void *)(bif_output.data + parthdr_offset);
422
423         if (bf->flags & (1ULL << BIF_FLAG_BOOTLOADER)) {
424                 bif_output.header->image_load = cpu_to_le32(bf->load);
425                 if (!bif_output.header->image_offset)
426                         bif_output.header->image_offset =
427                                 cpu_to_le32(bf->offset);
428                 bif_output.header->image_size = cpu_to_le32(len_padded);
429                 bif_output.header->image_stored_size = cpu_to_le32(len_padded);
430
431                 bif_output.header->image_attributes &= ~HEADER_CPU_SELECT_MASK;
432                 switch (bf->dest_cpu) {
433                 default:
434                 case PART_ATTR_DEST_CPU_A53_0:
435                         if (bf->flags & BIF_FLAG_AARCH32)
436                                 bif_output.header->image_attributes |=
437                                         HEADER_CPU_SELECT_A53_32BIT;
438                         else
439                                 bif_output.header->image_attributes |=
440                                         HEADER_CPU_SELECT_A53_64BIT;
441                         break;
442                 case PART_ATTR_DEST_CPU_R5_0:
443                         bif_output.header->image_attributes |=
444                                 HEADER_CPU_SELECT_R5_SINGLE;
445                         break;
446                 case PART_ATTR_DEST_CPU_R5_L:
447                         bif_output.header->image_attributes |=
448                                 HEADER_CPU_SELECT_R5_DUAL;
449                         break;
450                 }
451         }
452
453         return 0;
454 }
455
456 /* Add .bit bitstream */
457 static int bif_add_bit(struct bif_entry *bf)
458 {
459         char *bit = read_full_file(bf->filename, NULL);
460         char *bitbin;
461         uint8_t initial_header[] = { 0x00, 0x09, 0x0f, 0xf0, 0x0f, 0xf0, 0x0f,
462                                      0xf0, 0x0f, 0xf0, 0x00, 0x00, 0x01, 0x61 };
463         uint16_t len;
464         uint32_t bitlen;
465         int i;
466
467         if (!bit)
468                 return -1;
469
470         /* Skip initial header */
471         if (memcmp(bit, initial_header, sizeof(initial_header)))
472                 return -1;
473
474         bit += sizeof(initial_header);
475
476         /* Design name */
477         len = be16_to_cpu(*(uint16_t *)bit);
478         bit += sizeof(uint16_t);
479         debug("Design: %s\n", bit);
480         bit += len;
481
482         /* Device identifier */
483         if (*bit != 'b')
484                 return -1;
485         bit++;
486         len = be16_to_cpu(*(uint16_t *)bit);
487         bit += sizeof(uint16_t);
488         debug("Device: %s\n", bit);
489         bit += len;
490
491         /* Date */
492         if (*bit != 'c')
493                 return -1;
494         bit++;
495         len = be16_to_cpu(*(uint16_t *)bit);
496         bit += sizeof(uint16_t);
497         debug("Date: %s\n", bit);
498         bit += len;
499
500         /* Time */
501         if (*bit != 'd')
502                 return -1;
503         bit++;
504         len = be16_to_cpu(*(uint16_t *)bit);
505         bit += sizeof(uint16_t);
506         debug("Time: %s\n", bit);
507         bit += len;
508
509         /* Bitstream length */
510         if (*bit != 'e')
511                 return -1;
512         bit++;
513         bitlen = be32_to_cpu(*(uint32_t *)bit);
514         bit += sizeof(uint32_t);
515         bitbin = bit;
516
517         debug("Bitstream Length: 0x%x\n", bitlen);
518         for (i = 0; i < bitlen; i += sizeof(uint32_t)) {
519                 uint32_t *bitbin32 = (uint32_t *)&bitbin[i];
520                 *bitbin32 = __builtin_bswap32(*bitbin32);
521         }
522
523         if (!bf->dest_dev)
524                 bf->dest_dev = PART_ATTR_DEST_DEVICE_PL;
525
526         bf->load = 0xffffffff;
527         bf->entry = 0;
528
529         bf->flags |= 1ULL << BIF_FLAG_BIT_FILE;
530         return bif_add_part(bf, bit, bitlen);
531 }
532
533 /* Add .bin bitstream */
534 static int bif_add_bin(struct bif_entry *bf)
535 {
536         size_t size;
537         char *bin = read_full_file(bf->filename, &size);
538
539         if (!bf->dest_dev)
540                 bf->dest_dev = PART_ATTR_DEST_DEVICE_PS;
541
542         bf->flags |= 1ULL << BIF_FLAG_BIN_FILE;
543         return bif_add_part(bf, bin, size);
544 }
545
546 /* Add elf file */
547 static char *elf2flat64(char *elf, size_t *flat_size, size_t *load_addr)
548 {
549         Elf64_Ehdr *ehdr;
550         Elf64_Shdr *shdr;
551         size_t min_addr = -1, max_addr = 0;
552         char *flat;
553         int i;
554
555         ehdr = (void *)elf;
556         shdr = (void *)(elf + le64_to_cpu(ehdr->e_shoff));
557
558         /* Look for smallest / biggest address */
559         for (i = 0; i < le64_to_cpu(ehdr->e_shnum); i++, shdr++) {
560                 if (!shdr->sh_size || !shdr->sh_addr ||
561                     !(shdr->sh_flags & SHF_ALLOC) ||
562                     (shdr->sh_type == SHT_NOBITS))
563                         continue;
564
565                 if (le64_to_cpu(shdr->sh_addr) < min_addr)
566                         min_addr = le64_to_cpu(shdr->sh_addr);
567                 if ((le64_to_cpu(shdr->sh_addr) + le64_to_cpu(shdr->sh_size)) >
568                         max_addr)
569                         max_addr = le64_to_cpu(shdr->sh_addr) +
570                                    le64_to_cpu(shdr->sh_size);
571         }
572
573         *load_addr = min_addr;
574         *flat_size = max_addr - min_addr;
575         flat = calloc(1, *flat_size);
576         if (!flat)
577                 return NULL;
578
579         shdr = (void *)(elf + le64_to_cpu(ehdr->e_shoff));
580         for (i = 0; i < le64_to_cpu(ehdr->e_shnum); i++, shdr++) {
581                 char *dst = flat + le64_to_cpu(shdr->sh_addr) - min_addr;
582                 char *src = elf + le64_to_cpu(shdr->sh_offset);
583
584                 if (!shdr->sh_size || !shdr->sh_addr ||
585                     !(shdr->sh_flags & SHF_ALLOC))
586                         continue;
587
588                 if (shdr->sh_type != SHT_NOBITS)
589                         memcpy(dst, src, le64_to_cpu(shdr->sh_size));
590         }
591
592         return flat;
593 }
594
595 static char *elf2flat32(char *elf, size_t *flat_size, size_t *load_addr)
596 {
597         Elf32_Ehdr *ehdr;
598         Elf32_Shdr *shdr;
599         size_t min_addr = -1, max_addr = 0;
600         char *flat;
601         int i;
602
603         ehdr = (void *)elf;
604         shdr = (void *)(elf + le32_to_cpu(ehdr->e_shoff));
605
606         /* Look for smallest / biggest address */
607         for (i = 0; i < le32_to_cpu(ehdr->e_shnum); i++, shdr++) {
608                 if (!shdr->sh_size || !shdr->sh_addr ||
609                     !(shdr->sh_flags & SHF_ALLOC) ||
610                     (shdr->sh_type == SHT_NOBITS))
611                         continue;
612
613                 if (le32_to_cpu(shdr->sh_addr) < min_addr)
614                         min_addr = le32_to_cpu(shdr->sh_addr);
615                 if ((le32_to_cpu(shdr->sh_addr) + le32_to_cpu(shdr->sh_size)) >
616                         max_addr)
617                         max_addr = le32_to_cpu(shdr->sh_addr) +
618                                    le32_to_cpu(shdr->sh_size);
619         }
620
621         *load_addr = min_addr;
622         *flat_size = max_addr - min_addr;
623         flat = calloc(1, *flat_size);
624         if (!flat)
625                 return NULL;
626
627         shdr = (void *)(elf + le32_to_cpu(ehdr->e_shoff));
628         for (i = 0; i < le32_to_cpu(ehdr->e_shnum); i++, shdr++) {
629                 char *dst = flat + le32_to_cpu(shdr->sh_addr) - min_addr;
630                 char *src = elf + le32_to_cpu(shdr->sh_offset);
631
632                 if (!shdr->sh_size || !shdr->sh_addr ||
633                     !(shdr->sh_flags & SHF_ALLOC))
634                         continue;
635
636                 if (shdr->sh_type != SHT_NOBITS)
637                         memcpy(dst, src, le32_to_cpu(shdr->sh_size));
638         }
639
640         return flat;
641 }
642
643 static int bif_add_elf(struct bif_entry *bf)
644 {
645         size_t size;
646         size_t elf_size;
647         char *elf;
648         char *flat;
649         size_t load_addr;
650         Elf32_Ehdr *ehdr32;
651         Elf64_Ehdr *ehdr64;
652
653         elf = read_full_file(bf->filename, &elf_size);
654         if (!elf)
655                 return -1;
656
657         ehdr32 = (void *)elf;
658         ehdr64 = (void *)elf;
659
660         switch (ehdr32->e_ident[EI_CLASS]) {
661         case ELFCLASS32:
662                 flat = elf2flat32(elf, &size, &load_addr);
663                 bf->entry = le32_to_cpu(ehdr32->e_entry);
664                 bf->flags |= 1ULL << BIF_FLAG_AARCH32;
665                 break;
666         case ELFCLASS64:
667                 flat = elf2flat64(elf, &size, &load_addr);
668                 bf->entry = le64_to_cpu(ehdr64->e_entry);
669                 break;
670         default:
671                 printf("Unknown ELF class: %d\n", ehdr32->e_ident[EI_CLASS]);
672                 return -1;
673         }
674
675         if (!flat)
676                 return -1;
677
678         bf->load = load_addr;
679         if (!bf->dest_dev)
680                 bf->dest_dev = PART_ATTR_DEST_DEVICE_PS;
681
682         bf->flags |= 1ULL << BIF_FLAG_ELF_FILE;
683         return bif_add_part(bf, flat, size);
684 }
685
686 static const struct bif_file_type bif_file_types[] = {
687         {
688                 .name = "bitstream (.bit)",
689                 .header = 0x00090ff0,
690                 .add = bif_add_bit,
691         },
692
693         {
694                 .name = "ELF",
695                 .header = 0x7f454c46,
696                 .add = bif_add_elf,
697         },
698
699         /* Anything else is a .bin file */
700         {
701                 .name = ".bin",
702                 .add = bif_add_bin,
703         },
704 };
705
706 static int bif_fsbl_config(struct bif_entry *fsbl_config,
707                            struct bif_entry *entries, int nr_entries)
708 {
709         int i;
710         int config_set = 0;
711         struct {
712                 const char *name;
713                 uint64_t flags;
714                 uint64_t dest_cpu;
715         } configs[] = {
716                 { .name = "a5x_x64", .dest_cpu = PART_ATTR_DEST_CPU_A53_0 },
717                 { .name = "a53_x64", .dest_cpu = PART_ATTR_DEST_CPU_A53_0 },
718                 { .name = "a5x_x32", .dest_cpu = PART_ATTR_DEST_CPU_A53_0,
719                                      .flags = 1ULL << BIF_FLAG_AARCH32 },
720                 { .name = "a53_x32", .dest_cpu = PART_ATTR_DEST_CPU_A53_0,
721                                      .flags = 1ULL << BIF_FLAG_AARCH32 },
722                 { .name = "r5_single", .dest_cpu = PART_ATTR_DEST_CPU_R5_0 },
723                 { .name = "r5_dual", .dest_cpu = PART_ATTR_DEST_CPU_R5_L },
724         };
725
726         /* Set target CPU of bootloader entry */
727         for (i = 0; i < nr_entries; i++) {
728                 struct bif_entry *b = &entries[i];
729                 const char *config_attr = fsbl_config->filename;
730                 int j;
731
732                 if (!(b->flags & (1ULL << BIF_FLAG_BOOTLOADER)))
733                         continue;
734
735                 for (j = 0; j < ARRAY_SIZE(configs); j++) {
736                         if (!strncmp(config_attr, configs[j].name,
737                                      strlen(configs[j].name))) {
738                                 b->dest_cpu = configs[j].dest_cpu;
739                                 b->flags |= configs[j].flags;
740                                 config_set = 1;
741                         }
742                 }
743
744                 if (!config_set) {
745                         printf("ERROR: Unsupported fsbl_config: %s\n",
746                                config_attr);
747                         return -1;
748                 }
749         }
750
751         if (!config_set) {
752                 printf("ERROR: fsbl_config w/o bootloader\n");
753                 return -1;
754         }
755
756         return 0;
757 }
758
759 static const struct bif_flags *find_flag(char *str)
760 {
761         const struct bif_flags *bf;
762         int i;
763
764         for (i = 0; i < ARRAY_SIZE(bif_flags); i++) {
765                 bf = &bif_flags[i];
766                 if (!strncmp(bf->name, str, strlen(bf->name)))
767                         return bf;
768         }
769
770         printf("ERROR: Flag '%s' not found\n", str);
771
772         return NULL;
773 }
774
775 static int bif_open_file(struct bif_entry *entry)
776 {
777         int fd = open(entry->filename, O_RDONLY);
778
779         if (fd < 0)
780                 printf("Error opening file %s\n", entry->filename);
781
782         return fd;
783 }
784
785 static const struct bif_file_type *get_file_type(struct bif_entry *entry)
786 {
787         int fd = bif_open_file(entry);
788         uint32_t header;
789         int i;
790
791         if (fd < 0)
792                 return NULL;
793
794         if (read(fd, &header, sizeof(header)) != sizeof(header)) {
795                 printf("Error reading file %s", entry->filename);
796                 return NULL;
797         }
798
799         close(fd);
800
801         for (i = 0; i < ARRAY_SIZE(bif_file_types); i++) {
802                 const struct bif_file_type *type = &bif_file_types[i];
803
804                 if (!type->header)
805                         return type;
806                 if (type->header == be32_to_cpu(header))
807                         return type;
808         }
809
810         return NULL;
811 }
812
813 #define NEXT_CHAR(str, chr) ({          \
814         char *_n = strchr(str, chr);    \
815         if (!_n)                        \
816                 goto err;               \
817         _n;                             \
818 })
819
820 static char *skip_whitespace(char *str)
821 {
822         while (*str == ' ' || *str == '\t')
823                 str++;
824
825         return str;
826 }
827
828 int zynqmpbif_copy_image(int outfd, struct image_tool_params *mparams)
829 {
830         char *bif, *bifp, *bifpn;
831         char *line;
832         struct bif_entry entries[32] = { { 0 } };
833         int nr_entries = 0;
834         struct bif_entry *entry = entries;
835         size_t len;
836         int i;
837         uint32_t csum;
838         int bldr = -1;
839
840         bif_init();
841
842         /* Read .bif input file */
843         bif = read_full_file(mparams->datafile, NULL);
844         if (!bif)
845                 goto err;
846
847         /* Interpret .bif file */
848         bifp = bif;
849
850         /* A bif description starts with a { section */
851         bifp = NEXT_CHAR(bifp, '{') + 1;
852
853         /* Read every line */
854         while (1) {
855                 bifpn = NEXT_CHAR(bifp, '\n');
856
857                 if (bifpn[-1] == '\r')
858                         bifpn[-1] = '\0';
859
860                 *bifpn = '\0';
861                 bifpn++;
862                 line = bifp;
863
864                 line = skip_whitespace(line);
865
866                 /* Attributes? */
867                 if (*line == '[') {
868                         line++;
869                         while (1) {
870                                 const struct bif_flags *bf;
871
872                                 line = skip_whitespace(line);
873                                 bf = find_flag(line);
874                                 if (!bf)
875                                         goto err;
876
877                                 line += strlen(bf->name);
878                                 if (bf->parse)
879                                         line = bf->parse(line, entry);
880                                 else
881                                         entry->flags |= 1ULL << bf->flag;
882
883                                 if (!line)
884                                         goto err;
885
886                                 /* Go to next attribute or quit */
887                                 if (*line == ']') {
888                                         line++;
889                                         break;
890                                 }
891                                 if (*line == ',')
892                                         line++;
893                         }
894                 }
895
896                 /* End of image description */
897                 if (*line == '}')
898                         break;
899
900                 if (*line) {
901                         line = skip_whitespace(line);
902                         entry->filename = line;
903                         nr_entries++;
904                         entry++;
905                 }
906
907                 /* Use next line */
908                 bifp = bifpn;
909         }
910
911         for (i = 0; i < nr_entries; i++) {
912                 debug("Entry flags=%#lx name=%s\n", entries[i].flags,
913                       entries[i].filename);
914         }
915
916         /*
917          * Some entries are actually configuration option for other ones,
918          * let's apply them in an intermediate step.
919          */
920         for (i = 0; i < nr_entries; i++) {
921                 struct bif_entry *entry = &entries[i];
922
923                 if (entry->flags & (1ULL << BIF_FLAG_FSBL_CONFIG))
924                         if (bif_fsbl_config(entry, entries, nr_entries))
925                                 goto err;
926         }
927
928         /* Make sure PMUFW comes before bootloader */
929         for (i = 0; i < nr_entries; i++) {
930                 struct bif_entry *entry = &entries[i];
931
932                 if (entry->flags & (1ULL << BIF_FLAG_BOOTLOADER))
933                         bldr = i;
934                 if (entry->flags & (1ULL << BIF_FLAG_PMUFW_IMAGE)) {
935                         if (bldr >= 0) {
936                                 struct bif_entry tmp = *entry;
937
938                                 *entry = entries[bldr];
939                                 entries[bldr] = tmp;
940                         }
941                 }
942         }
943
944         for (i = 0; i < nr_entries; i++) {
945                 struct bif_entry *entry = &entries[i];
946                 const struct bif_file_type *type;
947                 int r;
948
949                 if (entry->flags & (1ULL << BIF_FLAG_FSBL_CONFIG))
950                         continue;
951
952                 type = get_file_type(entry);
953                 if (!type)
954                         goto err;
955
956                 debug("type=%s file=%s\n", type->name, entry->filename);
957                 r = type->add(entry);
958                 if (r)
959                         goto err;
960         }
961
962         /* Calculate checksums */
963         csum = zynqmp_csum(&bif_output.header->width_detection,
964                            &bif_output.header->checksum);
965         bif_output.header->checksum = cpu_to_le32(csum);
966
967         if (bif_output.imgheader) {
968                 csum = zynqmp_csum(bif_output.imgheader,
969                                    &bif_output.imgheader->checksum);
970                 bif_output.imgheader->checksum = cpu_to_le32(csum);
971         }
972
973         /* Write headers and components */
974         if (lseek(outfd, 0, SEEK_SET) != 0)
975                 goto err;
976
977         len = bif_output.data_len;
978         bifp = bif_output.data;
979         while (len) {
980                 int r;
981
982                 r = write(outfd, bifp, len);
983                 if (r < 0)
984                         goto err;
985                 len -= r;
986                 bifp += r;
987         }
988
989         return 0;
990
991 err:
992         fprintf(stderr, "Error: Failed to create image.\n");
993         return -1;
994 }
995
996 /* Needs to be stubbed out so we can print after creation */
997 static void zynqmpbif_set_header(void *ptr, struct stat *sbuf, int ifd,
998                                  struct image_tool_params *params)
999 {
1000 }
1001
1002 static struct zynqmp_header zynqmpimage_header;
1003
1004 U_BOOT_IMAGE_TYPE(
1005         zynqmpbif,
1006         "Xilinx ZynqMP Boot Image support (bif)",
1007         sizeof(struct zynqmp_header),
1008         (void *)&zynqmpimage_header,
1009         zynqmpbif_check_params,
1010         NULL,
1011         zynqmpimage_print_header,
1012         zynqmpbif_set_header,
1013         NULL,
1014         zynqmpbif_check_image_types,
1015         NULL,
1016         NULL
1017 );