xref: /openbmc/u-boot/cmd/armflash.c (revision e8f80a5a)
1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
22e192b24SSimon Glass /*
32e192b24SSimon Glass  * (C) Copyright 2015
42e192b24SSimon Glass  * Linus Walleij, Linaro
52e192b24SSimon Glass  *
62e192b24SSimon Glass  * Support for ARM Flash Partitions
72e192b24SSimon Glass  */
82e192b24SSimon Glass #include <common.h>
92e192b24SSimon Glass #include <command.h>
102e192b24SSimon Glass #include <console.h>
112e192b24SSimon Glass #include <asm/io.h>
122e192b24SSimon Glass 
132e192b24SSimon Glass #define MAX_REGIONS 4
142e192b24SSimon Glass #define MAX_IMAGES 32
152e192b24SSimon Glass 
162e192b24SSimon Glass struct afs_region {
172e192b24SSimon Glass 	u32 load_address;
182e192b24SSimon Glass 	u32 size;
192e192b24SSimon Glass 	u32 offset;
202e192b24SSimon Glass };
212e192b24SSimon Glass 
222e192b24SSimon Glass struct afs_image {
232e192b24SSimon Glass 	flash_info_t *flinfo;
242e192b24SSimon Glass 	const char *name;
252e192b24SSimon Glass 	u32 version;
262e192b24SSimon Glass 	u32 entrypoint;
272e192b24SSimon Glass 	u32 attributes;
282e192b24SSimon Glass 	u32 region_count;
292e192b24SSimon Glass 	struct afs_region regions[MAX_REGIONS];
302e192b24SSimon Glass 	ulong flash_mem_start;
312e192b24SSimon Glass 	ulong flash_mem_end;
322e192b24SSimon Glass };
332e192b24SSimon Glass 
342e192b24SSimon Glass static struct afs_image afs_images[MAX_IMAGES];
352e192b24SSimon Glass static int num_afs_images;
362e192b24SSimon Glass 
compute_crc(ulong start,u32 len)372e192b24SSimon Glass static u32 compute_crc(ulong start, u32 len)
382e192b24SSimon Glass {
392e192b24SSimon Glass 	u32 sum = 0;
402e192b24SSimon Glass 	int i;
412e192b24SSimon Glass 
422e192b24SSimon Glass 	if (len % 4 != 0) {
432e192b24SSimon Glass 		printf("bad checksumming\n");
442e192b24SSimon Glass 		return 0;
452e192b24SSimon Glass 	}
462e192b24SSimon Glass 
472e192b24SSimon Glass 	for (i = 0; i < len; i += 4) {
482e192b24SSimon Glass 		u32 val;
492e192b24SSimon Glass 
502e192b24SSimon Glass 		val = readl((void *)start + i);
512e192b24SSimon Glass 		if (val > ~sum)
522e192b24SSimon Glass 			sum++;
532e192b24SSimon Glass 		sum += val;
542e192b24SSimon Glass 	}
552e192b24SSimon Glass 	return ~sum;
562e192b24SSimon Glass }
572e192b24SSimon Glass 
parse_bank(ulong bank)582e192b24SSimon Glass static void parse_bank(ulong bank)
592e192b24SSimon Glass {
602e192b24SSimon Glass 	int i;
612e192b24SSimon Glass 	ulong flstart, flend;
622e192b24SSimon Glass 	flash_info_t *info;
632e192b24SSimon Glass 
642e192b24SSimon Glass 	info = &flash_info[bank];
652e192b24SSimon Glass 	if (info->flash_id != FLASH_MAN_CFI) {
662e192b24SSimon Glass 		printf("Bank %lu: missing or unknown FLASH type\n", bank);
672e192b24SSimon Glass 		return;
682e192b24SSimon Glass 	}
692e192b24SSimon Glass 	if (!info->sector_count) {
702e192b24SSimon Glass 		printf("Bank %lu: no FLASH sectors\n", bank);
712e192b24SSimon Glass 		return;
722e192b24SSimon Glass 	}
732e192b24SSimon Glass 
742e192b24SSimon Glass 	flstart = info->start[0];
752e192b24SSimon Glass 	flend = flstart + info->size;
762e192b24SSimon Glass 
772e192b24SSimon Glass 	for (i = 0; i < info->sector_count; ++i) {
782e192b24SSimon Glass 		ulong secend;
792e192b24SSimon Glass 		u32 foot1, foot2;
802e192b24SSimon Glass 
812e192b24SSimon Glass 		if (ctrlc())
822e192b24SSimon Glass 			break;
832e192b24SSimon Glass 
842e192b24SSimon Glass 		if (i == info->sector_count-1)
852e192b24SSimon Glass 			secend = flend;
862e192b24SSimon Glass 		else
872e192b24SSimon Glass 			secend = info->start[i+1];
882e192b24SSimon Glass 
892e192b24SSimon Glass 		/* Check for v1 header */
902e192b24SSimon Glass 		foot1 = readl((void *)secend - 0x0c);
912e192b24SSimon Glass 		if (foot1 == 0xA0FFFF9FU) {
922e192b24SSimon Glass 			struct afs_image *afi = &afs_images[num_afs_images];
932e192b24SSimon Glass 			ulong imginfo;
942e192b24SSimon Glass 
952e192b24SSimon Glass 			afi->flinfo = info;
962e192b24SSimon Glass 			afi->version = 1;
972e192b24SSimon Glass 			afi->flash_mem_start = readl((void *)secend - 0x10);
982e192b24SSimon Glass 			afi->flash_mem_end = readl((void *)secend - 0x14);
992e192b24SSimon Glass 			afi->attributes = readl((void *)secend - 0x08);
1002e192b24SSimon Glass 			/* Adjust to even address */
1012e192b24SSimon Glass 			imginfo = afi->flash_mem_end + afi->flash_mem_end % 4;
1022e192b24SSimon Glass 			/* Record as a single region */
1032e192b24SSimon Glass 			afi->region_count = 1;
1042e192b24SSimon Glass 			afi->regions[0].offset = readl((void *)imginfo + 0x04);
1052e192b24SSimon Glass 			afi->regions[0].load_address =
1062e192b24SSimon Glass 				readl((void *)imginfo + 0x08);
1072e192b24SSimon Glass 			afi->regions[0].size = readl((void *)imginfo + 0x0C);
1082e192b24SSimon Glass 			afi->entrypoint = readl((void *)imginfo + 0x10);
1092e192b24SSimon Glass 			afi->name = (const char *)imginfo + 0x14;
1102e192b24SSimon Glass 			num_afs_images++;
1112e192b24SSimon Glass 		}
1122e192b24SSimon Glass 
1132e192b24SSimon Glass 		/* Check for v2 header */
1142e192b24SSimon Glass 		foot1 = readl((void *)secend - 0x04);
1152e192b24SSimon Glass 		foot2 = readl((void *)secend - 0x08);
1162e192b24SSimon Glass 		/* This makes up the string "HSLFTOOF" flash footer */
1172e192b24SSimon Glass 		if (foot1 == 0x464F4F54U && foot2 == 0x464C5348U) {
1182e192b24SSimon Glass 			struct afs_image *afi = &afs_images[num_afs_images];
1192e192b24SSimon Glass 			ulong imginfo;
1202e192b24SSimon Glass 			u32 block_start, block_end;
1212e192b24SSimon Glass 			int j;
1222e192b24SSimon Glass 
1232e192b24SSimon Glass 			afi->flinfo = info;
1242e192b24SSimon Glass 			afi->version = readl((void *)secend - 0x0c);
1252e192b24SSimon Glass 			imginfo = secend - 0x30 - readl((void *)secend - 0x10);
1262e192b24SSimon Glass 			afi->name = (const char *)secend - 0x30;
1272e192b24SSimon Glass 
1282e192b24SSimon Glass 			afi->entrypoint = readl((void *)imginfo+0x08);
1292e192b24SSimon Glass 			afi->attributes = readl((void *)imginfo+0x0c);
1302e192b24SSimon Glass 			afi->region_count = readl((void *)imginfo+0x10);
1312e192b24SSimon Glass 			block_start = readl((void *)imginfo+0x54);
1322e192b24SSimon Glass 			block_end = readl((void *)imginfo+0x58);
1332e192b24SSimon Glass 			afi->flash_mem_start = afi->flinfo->start[block_start];
1342e192b24SSimon Glass 			afi->flash_mem_end = afi->flinfo->start[block_end];
1352e192b24SSimon Glass 
1362e192b24SSimon Glass 			/*
1372e192b24SSimon Glass 			 * Check footer CRC, the algorithm saves the inverse
1382e192b24SSimon Glass 			 * checksum as part of the summed words, and thus
1392e192b24SSimon Glass 			 * the result should be zero.
1402e192b24SSimon Glass 			 */
1412e192b24SSimon Glass 			if (compute_crc(imginfo + 8, 0x88) != 0) {
1422e192b24SSimon Glass 				printf("BAD CRC on ARM image info\n");
1432e192b24SSimon Glass 				printf("(continuing anyway)\n");
1442e192b24SSimon Glass 			}
1452e192b24SSimon Glass 
1462e192b24SSimon Glass 			/* Parse regions */
1472e192b24SSimon Glass 			for (j = 0; j < afi->region_count; j++) {
1482e192b24SSimon Glass 				afi->regions[j].load_address =
1492e192b24SSimon Glass 					readl((void *)imginfo+0x14 + j*0x10);
1502e192b24SSimon Glass 				afi->regions[j].size =
1512e192b24SSimon Glass 					readl((void *)imginfo+0x18 + j*0x10);
1522e192b24SSimon Glass 				afi->regions[j].offset =
1532e192b24SSimon Glass 					readl((void *)imginfo+0x1c + j*0x10);
1542e192b24SSimon Glass 				/*
1552e192b24SSimon Glass 				 * At offset 0x20 + j*0x10 there is a region
1562e192b24SSimon Glass 				 * checksum which seems to be the running
1572e192b24SSimon Glass 				 * sum + 3, however since we anyway checksum
1582e192b24SSimon Glass 				 * the entire footer this is skipped over for
1592e192b24SSimon Glass 				 * checking here.
1602e192b24SSimon Glass 				 */
1612e192b24SSimon Glass 			}
1622e192b24SSimon Glass 			num_afs_images++;
1632e192b24SSimon Glass 		}
1642e192b24SSimon Glass 	}
1652e192b24SSimon Glass }
1662e192b24SSimon Glass 
parse_flash(void)1672e192b24SSimon Glass static void parse_flash(void)
1682e192b24SSimon Glass {
1692e192b24SSimon Glass 	ulong bank;
1702e192b24SSimon Glass 
1712e192b24SSimon Glass 	/* We have already parsed the images in flash */
1722e192b24SSimon Glass 	if (num_afs_images > 0)
1732e192b24SSimon Glass 		return;
1742e192b24SSimon Glass 	for (bank = 0; bank < CONFIG_SYS_MAX_FLASH_BANKS; ++bank)
1752e192b24SSimon Glass 		parse_bank(bank);
1762e192b24SSimon Glass }
1772e192b24SSimon Glass 
load_image(const char * const name,const ulong address)1782e192b24SSimon Glass static int load_image(const char * const name, const ulong address)
1792e192b24SSimon Glass {
1802e192b24SSimon Glass 	struct afs_image *afi = NULL;
1812e192b24SSimon Glass 	int i;
1822e192b24SSimon Glass 
1832e192b24SSimon Glass 	parse_flash();
1842e192b24SSimon Glass 	for (i = 0; i < num_afs_images; i++) {
1852e192b24SSimon Glass 		struct afs_image *tmp = &afs_images[i];
1862e192b24SSimon Glass 
1872e192b24SSimon Glass 		if (!strcmp(tmp->name, name)) {
1882e192b24SSimon Glass 			afi = tmp;
1892e192b24SSimon Glass 			break;
1902e192b24SSimon Glass 		}
1912e192b24SSimon Glass 	}
1922e192b24SSimon Glass 	if (!afi) {
1932e192b24SSimon Glass 		printf("image \"%s\" not found in flash\n", name);
1942e192b24SSimon Glass 		return CMD_RET_FAILURE;
1952e192b24SSimon Glass 	}
1962e192b24SSimon Glass 
1972e192b24SSimon Glass 	for (i = 0; i < afi->region_count; i++) {
1982e192b24SSimon Glass 		ulong from, to;
1992e192b24SSimon Glass 
2002e192b24SSimon Glass 		from = afi->flash_mem_start + afi->regions[i].offset;
2012e192b24SSimon Glass 		if (address) {
2022e192b24SSimon Glass 			to = address;
2032e192b24SSimon Glass 		} else if (afi->regions[i].load_address) {
2042e192b24SSimon Glass 			to = afi->regions[i].load_address;
2052e192b24SSimon Glass 		} else {
2062e192b24SSimon Glass 			printf("no valid load address\n");
2072e192b24SSimon Glass 			return CMD_RET_FAILURE;
2082e192b24SSimon Glass 		}
2092e192b24SSimon Glass 
2102e192b24SSimon Glass 		memcpy((void *)to, (void *)from, afi->regions[i].size);
2112e192b24SSimon Glass 
2122e192b24SSimon Glass 		printf("loaded region %d from %08lX to %08lX, %08X bytes\n",
2132e192b24SSimon Glass 		       i,
2142e192b24SSimon Glass 		       from,
2152e192b24SSimon Glass 		       to,
2162e192b24SSimon Glass 		       afi->regions[i].size);
2172e192b24SSimon Glass 	}
2182e192b24SSimon Glass 	return CMD_RET_SUCCESS;
2192e192b24SSimon Glass }
2202e192b24SSimon Glass 
print_images(void)2212e192b24SSimon Glass static void print_images(void)
2222e192b24SSimon Glass {
2232e192b24SSimon Glass 	int i;
2242e192b24SSimon Glass 
2252e192b24SSimon Glass 	parse_flash();
2262e192b24SSimon Glass 	for (i = 0; i < num_afs_images; i++) {
2272e192b24SSimon Glass 		struct afs_image *afi = &afs_images[i];
2282e192b24SSimon Glass 		int j;
2292e192b24SSimon Glass 
2302e192b24SSimon Glass 		printf("Image: \"%s\" (v%d):\n", afi->name, afi->version);
2312e192b24SSimon Glass 		printf("    Entry point: 0x%08X\n", afi->entrypoint);
2322e192b24SSimon Glass 		printf("    Attributes: 0x%08X: ", afi->attributes);
2332e192b24SSimon Glass 		if (afi->attributes == 0x01)
2342e192b24SSimon Glass 			printf("ARM executable");
2352e192b24SSimon Glass 		if (afi->attributes == 0x08)
2362e192b24SSimon Glass 			printf("ARM backup");
2372e192b24SSimon Glass 		printf("\n");
2382e192b24SSimon Glass 		printf("    Flash mem start: 0x%08lX\n",
2392e192b24SSimon Glass 		       afi->flash_mem_start);
2402e192b24SSimon Glass 		printf("    Flash mem end: 0x%08lX\n",
2412e192b24SSimon Glass 		       afi->flash_mem_end);
2422e192b24SSimon Glass 		for (j = 0; j < afi->region_count; j++) {
2432e192b24SSimon Glass 			printf("    region %d\n"
2442e192b24SSimon Glass 			       "        load address: %08X\n"
2452e192b24SSimon Glass 			       "        size: %08X\n"
2462e192b24SSimon Glass 			       "        offset: %08X\n",
2472e192b24SSimon Glass 			       j,
2482e192b24SSimon Glass 			       afi->regions[j].load_address,
2492e192b24SSimon Glass 			       afi->regions[j].size,
2502e192b24SSimon Glass 			       afi->regions[j].offset);
2512e192b24SSimon Glass 		}
2522e192b24SSimon Glass 	}
2532e192b24SSimon Glass }
2542e192b24SSimon Glass 
exists(const char * const name)2552e192b24SSimon Glass static int exists(const char * const name)
2562e192b24SSimon Glass {
2572e192b24SSimon Glass 	int i;
2582e192b24SSimon Glass 
2592e192b24SSimon Glass 	parse_flash();
2602e192b24SSimon Glass 	for (i = 0; i < num_afs_images; i++) {
2612e192b24SSimon Glass 		struct afs_image *afi = &afs_images[i];
2622e192b24SSimon Glass 
2632e192b24SSimon Glass 		if (strcmp(afi->name, name) == 0)
2642e192b24SSimon Glass 			return CMD_RET_SUCCESS;
2652e192b24SSimon Glass 	}
2662e192b24SSimon Glass 	return CMD_RET_FAILURE;
2672e192b24SSimon Glass }
2682e192b24SSimon Glass 
do_afs(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])2692e192b24SSimon Glass static int do_afs(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
2702e192b24SSimon Glass {
2712e192b24SSimon Glass 	int ret = CMD_RET_SUCCESS;
2722e192b24SSimon Glass 
2732e192b24SSimon Glass 	if (argc == 1) {
2742e192b24SSimon Glass 		print_images();
2752e192b24SSimon Glass 	} else if (argc == 3 && !strcmp(argv[1], "exists")) {
2762e192b24SSimon Glass 		ret = exists(argv[2]);
2772e192b24SSimon Glass 	} else if (argc == 3 && !strcmp(argv[1], "load")) {
2782e192b24SSimon Glass 		ret = load_image(argv[2], 0x0);
2792e192b24SSimon Glass 	} else if (argc == 4 && !strcmp(argv[1], "load")) {
2802e192b24SSimon Glass 		ulong load_addr;
2812e192b24SSimon Glass 
2822e192b24SSimon Glass 		load_addr = simple_strtoul(argv[3], NULL, 16);
2832e192b24SSimon Glass 		ret = load_image(argv[2], load_addr);
2842e192b24SSimon Glass 	} else {
2852e192b24SSimon Glass 		return CMD_RET_USAGE;
2862e192b24SSimon Glass 	}
2872e192b24SSimon Glass 
2882e192b24SSimon Glass 	return ret;
2892e192b24SSimon Glass }
2902e192b24SSimon Glass 
2912e192b24SSimon Glass U_BOOT_CMD(afs, 4, 0, do_afs, "show AFS partitions",
2922e192b24SSimon Glass 	   "no arguments\n"
2932e192b24SSimon Glass 	   "    - list images in flash\n"
2942e192b24SSimon Glass 	   "exists <image>\n"
2952e192b24SSimon Glass 	   "    - returns 1 if an image exists, else 0\n"
2962e192b24SSimon Glass 	   "load <image>\n"
2972e192b24SSimon Glass 	   "    - load an image to the location indicated in the header\n"
2982e192b24SSimon Glass 	   "load <image> 0x<address>\n"
2992e192b24SSimon Glass 	   "    - load an image to the location specified\n");
300