xref: /rk3399_rockchip-uboot/cmd/armflash.c (revision 2e192b245ed36a63bab0ef576999a95e23f60ecd)
1*2e192b24SSimon Glass /*
2*2e192b24SSimon Glass  * (C) Copyright 2015
3*2e192b24SSimon Glass  * Linus Walleij, Linaro
4*2e192b24SSimon Glass  *
5*2e192b24SSimon Glass  * Support for ARM Flash Partitions
6*2e192b24SSimon Glass  *
7*2e192b24SSimon Glass  * SPDX-License-Identifier:     GPL-2.0+
8*2e192b24SSimon Glass  */
9*2e192b24SSimon Glass #include <common.h>
10*2e192b24SSimon Glass #include <command.h>
11*2e192b24SSimon Glass #include <console.h>
12*2e192b24SSimon Glass #include <asm/io.h>
13*2e192b24SSimon Glass 
14*2e192b24SSimon Glass #define MAX_REGIONS 4
15*2e192b24SSimon Glass #define MAX_IMAGES 32
16*2e192b24SSimon Glass 
17*2e192b24SSimon Glass struct afs_region {
18*2e192b24SSimon Glass 	u32 load_address;
19*2e192b24SSimon Glass 	u32 size;
20*2e192b24SSimon Glass 	u32 offset;
21*2e192b24SSimon Glass };
22*2e192b24SSimon Glass 
23*2e192b24SSimon Glass struct afs_image {
24*2e192b24SSimon Glass 	flash_info_t *flinfo;
25*2e192b24SSimon Glass 	const char *name;
26*2e192b24SSimon Glass 	u32 version;
27*2e192b24SSimon Glass 	u32 entrypoint;
28*2e192b24SSimon Glass 	u32 attributes;
29*2e192b24SSimon Glass 	u32 region_count;
30*2e192b24SSimon Glass 	struct afs_region regions[MAX_REGIONS];
31*2e192b24SSimon Glass 	ulong flash_mem_start;
32*2e192b24SSimon Glass 	ulong flash_mem_end;
33*2e192b24SSimon Glass };
34*2e192b24SSimon Glass 
35*2e192b24SSimon Glass static struct afs_image afs_images[MAX_IMAGES];
36*2e192b24SSimon Glass static int num_afs_images;
37*2e192b24SSimon Glass 
compute_crc(ulong start,u32 len)38*2e192b24SSimon Glass static u32 compute_crc(ulong start, u32 len)
39*2e192b24SSimon Glass {
40*2e192b24SSimon Glass 	u32 sum = 0;
41*2e192b24SSimon Glass 	int i;
42*2e192b24SSimon Glass 
43*2e192b24SSimon Glass 	if (len % 4 != 0) {
44*2e192b24SSimon Glass 		printf("bad checksumming\n");
45*2e192b24SSimon Glass 		return 0;
46*2e192b24SSimon Glass 	}
47*2e192b24SSimon Glass 
48*2e192b24SSimon Glass 	for (i = 0; i < len; i += 4) {
49*2e192b24SSimon Glass 		u32 val;
50*2e192b24SSimon Glass 
51*2e192b24SSimon Glass 		val = readl((void *)start + i);
52*2e192b24SSimon Glass 		if (val > ~sum)
53*2e192b24SSimon Glass 			sum++;
54*2e192b24SSimon Glass 		sum += val;
55*2e192b24SSimon Glass 	}
56*2e192b24SSimon Glass 	return ~sum;
57*2e192b24SSimon Glass }
58*2e192b24SSimon Glass 
parse_bank(ulong bank)59*2e192b24SSimon Glass static void parse_bank(ulong bank)
60*2e192b24SSimon Glass {
61*2e192b24SSimon Glass 	int i;
62*2e192b24SSimon Glass 	ulong flstart, flend;
63*2e192b24SSimon Glass 	flash_info_t *info;
64*2e192b24SSimon Glass 
65*2e192b24SSimon Glass 	info = &flash_info[bank];
66*2e192b24SSimon Glass 	if (info->flash_id != FLASH_MAN_CFI) {
67*2e192b24SSimon Glass 		printf("Bank %lu: missing or unknown FLASH type\n", bank);
68*2e192b24SSimon Glass 		return;
69*2e192b24SSimon Glass 	}
70*2e192b24SSimon Glass 	if (!info->sector_count) {
71*2e192b24SSimon Glass 		printf("Bank %lu: no FLASH sectors\n", bank);
72*2e192b24SSimon Glass 		return;
73*2e192b24SSimon Glass 	}
74*2e192b24SSimon Glass 
75*2e192b24SSimon Glass 	flstart = info->start[0];
76*2e192b24SSimon Glass 	flend = flstart + info->size;
77*2e192b24SSimon Glass 
78*2e192b24SSimon Glass 	for (i = 0; i < info->sector_count; ++i) {
79*2e192b24SSimon Glass 		ulong secend;
80*2e192b24SSimon Glass 		u32 foot1, foot2;
81*2e192b24SSimon Glass 
82*2e192b24SSimon Glass 		if (ctrlc())
83*2e192b24SSimon Glass 			break;
84*2e192b24SSimon Glass 
85*2e192b24SSimon Glass 		if (i == info->sector_count-1)
86*2e192b24SSimon Glass 			secend = flend;
87*2e192b24SSimon Glass 		else
88*2e192b24SSimon Glass 			secend = info->start[i+1];
89*2e192b24SSimon Glass 
90*2e192b24SSimon Glass 		/* Check for v1 header */
91*2e192b24SSimon Glass 		foot1 = readl((void *)secend - 0x0c);
92*2e192b24SSimon Glass 		if (foot1 == 0xA0FFFF9FU) {
93*2e192b24SSimon Glass 			struct afs_image *afi = &afs_images[num_afs_images];
94*2e192b24SSimon Glass 			ulong imginfo;
95*2e192b24SSimon Glass 
96*2e192b24SSimon Glass 			afi->flinfo = info;
97*2e192b24SSimon Glass 			afi->version = 1;
98*2e192b24SSimon Glass 			afi->flash_mem_start = readl((void *)secend - 0x10);
99*2e192b24SSimon Glass 			afi->flash_mem_end = readl((void *)secend - 0x14);
100*2e192b24SSimon Glass 			afi->attributes = readl((void *)secend - 0x08);
101*2e192b24SSimon Glass 			/* Adjust to even address */
102*2e192b24SSimon Glass 			imginfo = afi->flash_mem_end + afi->flash_mem_end % 4;
103*2e192b24SSimon Glass 			/* Record as a single region */
104*2e192b24SSimon Glass 			afi->region_count = 1;
105*2e192b24SSimon Glass 			afi->regions[0].offset = readl((void *)imginfo + 0x04);
106*2e192b24SSimon Glass 			afi->regions[0].load_address =
107*2e192b24SSimon Glass 				readl((void *)imginfo + 0x08);
108*2e192b24SSimon Glass 			afi->regions[0].size = readl((void *)imginfo + 0x0C);
109*2e192b24SSimon Glass 			afi->entrypoint = readl((void *)imginfo + 0x10);
110*2e192b24SSimon Glass 			afi->name = (const char *)imginfo + 0x14;
111*2e192b24SSimon Glass 			num_afs_images++;
112*2e192b24SSimon Glass 		}
113*2e192b24SSimon Glass 
114*2e192b24SSimon Glass 		/* Check for v2 header */
115*2e192b24SSimon Glass 		foot1 = readl((void *)secend - 0x04);
116*2e192b24SSimon Glass 		foot2 = readl((void *)secend - 0x08);
117*2e192b24SSimon Glass 		/* This makes up the string "HSLFTOOF" flash footer */
118*2e192b24SSimon Glass 		if (foot1 == 0x464F4F54U && foot2 == 0x464C5348U) {
119*2e192b24SSimon Glass 			struct afs_image *afi = &afs_images[num_afs_images];
120*2e192b24SSimon Glass 			ulong imginfo;
121*2e192b24SSimon Glass 			u32 block_start, block_end;
122*2e192b24SSimon Glass 			int j;
123*2e192b24SSimon Glass 
124*2e192b24SSimon Glass 			afi->flinfo = info;
125*2e192b24SSimon Glass 			afi->version = readl((void *)secend - 0x0c);
126*2e192b24SSimon Glass 			imginfo = secend - 0x30 - readl((void *)secend - 0x10);
127*2e192b24SSimon Glass 			afi->name = (const char *)secend - 0x30;
128*2e192b24SSimon Glass 
129*2e192b24SSimon Glass 			afi->entrypoint = readl((void *)imginfo+0x08);
130*2e192b24SSimon Glass 			afi->attributes = readl((void *)imginfo+0x0c);
131*2e192b24SSimon Glass 			afi->region_count = readl((void *)imginfo+0x10);
132*2e192b24SSimon Glass 			block_start = readl((void *)imginfo+0x54);
133*2e192b24SSimon Glass 			block_end = readl((void *)imginfo+0x58);
134*2e192b24SSimon Glass 			afi->flash_mem_start = afi->flinfo->start[block_start];
135*2e192b24SSimon Glass 			afi->flash_mem_end = afi->flinfo->start[block_end];
136*2e192b24SSimon Glass 
137*2e192b24SSimon Glass 			/*
138*2e192b24SSimon Glass 			 * Check footer CRC, the algorithm saves the inverse
139*2e192b24SSimon Glass 			 * checksum as part of the summed words, and thus
140*2e192b24SSimon Glass 			 * the result should be zero.
141*2e192b24SSimon Glass 			 */
142*2e192b24SSimon Glass 			if (compute_crc(imginfo + 8, 0x88) != 0) {
143*2e192b24SSimon Glass 				printf("BAD CRC on ARM image info\n");
144*2e192b24SSimon Glass 				printf("(continuing anyway)\n");
145*2e192b24SSimon Glass 			}
146*2e192b24SSimon Glass 
147*2e192b24SSimon Glass 			/* Parse regions */
148*2e192b24SSimon Glass 			for (j = 0; j < afi->region_count; j++) {
149*2e192b24SSimon Glass 				afi->regions[j].load_address =
150*2e192b24SSimon Glass 					readl((void *)imginfo+0x14 + j*0x10);
151*2e192b24SSimon Glass 				afi->regions[j].size =
152*2e192b24SSimon Glass 					readl((void *)imginfo+0x18 + j*0x10);
153*2e192b24SSimon Glass 				afi->regions[j].offset =
154*2e192b24SSimon Glass 					readl((void *)imginfo+0x1c + j*0x10);
155*2e192b24SSimon Glass 				/*
156*2e192b24SSimon Glass 				 * At offset 0x20 + j*0x10 there is a region
157*2e192b24SSimon Glass 				 * checksum which seems to be the running
158*2e192b24SSimon Glass 				 * sum + 3, however since we anyway checksum
159*2e192b24SSimon Glass 				 * the entire footer this is skipped over for
160*2e192b24SSimon Glass 				 * checking here.
161*2e192b24SSimon Glass 				 */
162*2e192b24SSimon Glass 			}
163*2e192b24SSimon Glass 			num_afs_images++;
164*2e192b24SSimon Glass 		}
165*2e192b24SSimon Glass 	}
166*2e192b24SSimon Glass }
167*2e192b24SSimon Glass 
parse_flash(void)168*2e192b24SSimon Glass static void parse_flash(void)
169*2e192b24SSimon Glass {
170*2e192b24SSimon Glass 	ulong bank;
171*2e192b24SSimon Glass 
172*2e192b24SSimon Glass 	/* We have already parsed the images in flash */
173*2e192b24SSimon Glass 	if (num_afs_images > 0)
174*2e192b24SSimon Glass 		return;
175*2e192b24SSimon Glass 	for (bank = 0; bank < CONFIG_SYS_MAX_FLASH_BANKS; ++bank)
176*2e192b24SSimon Glass 		parse_bank(bank);
177*2e192b24SSimon Glass }
178*2e192b24SSimon Glass 
load_image(const char * const name,const ulong address)179*2e192b24SSimon Glass static int load_image(const char * const name, const ulong address)
180*2e192b24SSimon Glass {
181*2e192b24SSimon Glass 	struct afs_image *afi = NULL;
182*2e192b24SSimon Glass 	int i;
183*2e192b24SSimon Glass 
184*2e192b24SSimon Glass 	parse_flash();
185*2e192b24SSimon Glass 	for (i = 0; i < num_afs_images; i++) {
186*2e192b24SSimon Glass 		struct afs_image *tmp = &afs_images[i];
187*2e192b24SSimon Glass 
188*2e192b24SSimon Glass 		if (!strcmp(tmp->name, name)) {
189*2e192b24SSimon Glass 			afi = tmp;
190*2e192b24SSimon Glass 			break;
191*2e192b24SSimon Glass 		}
192*2e192b24SSimon Glass 	}
193*2e192b24SSimon Glass 	if (!afi) {
194*2e192b24SSimon Glass 		printf("image \"%s\" not found in flash\n", name);
195*2e192b24SSimon Glass 		return CMD_RET_FAILURE;
196*2e192b24SSimon Glass 	}
197*2e192b24SSimon Glass 
198*2e192b24SSimon Glass 	for (i = 0; i < afi->region_count; i++) {
199*2e192b24SSimon Glass 		ulong from, to;
200*2e192b24SSimon Glass 
201*2e192b24SSimon Glass 		from = afi->flash_mem_start + afi->regions[i].offset;
202*2e192b24SSimon Glass 		if (address) {
203*2e192b24SSimon Glass 			to = address;
204*2e192b24SSimon Glass 		} else if (afi->regions[i].load_address) {
205*2e192b24SSimon Glass 			to = afi->regions[i].load_address;
206*2e192b24SSimon Glass 		} else {
207*2e192b24SSimon Glass 			printf("no valid load address\n");
208*2e192b24SSimon Glass 			return CMD_RET_FAILURE;
209*2e192b24SSimon Glass 		}
210*2e192b24SSimon Glass 
211*2e192b24SSimon Glass 		memcpy((void *)to, (void *)from, afi->regions[i].size);
212*2e192b24SSimon Glass 
213*2e192b24SSimon Glass 		printf("loaded region %d from %08lX to %08lX, %08X bytes\n",
214*2e192b24SSimon Glass 		       i,
215*2e192b24SSimon Glass 		       from,
216*2e192b24SSimon Glass 		       to,
217*2e192b24SSimon Glass 		       afi->regions[i].size);
218*2e192b24SSimon Glass 	}
219*2e192b24SSimon Glass 	return CMD_RET_SUCCESS;
220*2e192b24SSimon Glass }
221*2e192b24SSimon Glass 
print_images(void)222*2e192b24SSimon Glass static void print_images(void)
223*2e192b24SSimon Glass {
224*2e192b24SSimon Glass 	int i;
225*2e192b24SSimon Glass 
226*2e192b24SSimon Glass 	parse_flash();
227*2e192b24SSimon Glass 	for (i = 0; i < num_afs_images; i++) {
228*2e192b24SSimon Glass 		struct afs_image *afi = &afs_images[i];
229*2e192b24SSimon Glass 		int j;
230*2e192b24SSimon Glass 
231*2e192b24SSimon Glass 		printf("Image: \"%s\" (v%d):\n", afi->name, afi->version);
232*2e192b24SSimon Glass 		printf("    Entry point: 0x%08X\n", afi->entrypoint);
233*2e192b24SSimon Glass 		printf("    Attributes: 0x%08X: ", afi->attributes);
234*2e192b24SSimon Glass 		if (afi->attributes == 0x01)
235*2e192b24SSimon Glass 			printf("ARM executable");
236*2e192b24SSimon Glass 		if (afi->attributes == 0x08)
237*2e192b24SSimon Glass 			printf("ARM backup");
238*2e192b24SSimon Glass 		printf("\n");
239*2e192b24SSimon Glass 		printf("    Flash mem start: 0x%08lX\n",
240*2e192b24SSimon Glass 		       afi->flash_mem_start);
241*2e192b24SSimon Glass 		printf("    Flash mem end: 0x%08lX\n",
242*2e192b24SSimon Glass 		       afi->flash_mem_end);
243*2e192b24SSimon Glass 		for (j = 0; j < afi->region_count; j++) {
244*2e192b24SSimon Glass 			printf("    region %d\n"
245*2e192b24SSimon Glass 			       "        load address: %08X\n"
246*2e192b24SSimon Glass 			       "        size: %08X\n"
247*2e192b24SSimon Glass 			       "        offset: %08X\n",
248*2e192b24SSimon Glass 			       j,
249*2e192b24SSimon Glass 			       afi->regions[j].load_address,
250*2e192b24SSimon Glass 			       afi->regions[j].size,
251*2e192b24SSimon Glass 			       afi->regions[j].offset);
252*2e192b24SSimon Glass 		}
253*2e192b24SSimon Glass 	}
254*2e192b24SSimon Glass }
255*2e192b24SSimon Glass 
exists(const char * const name)256*2e192b24SSimon Glass static int exists(const char * const name)
257*2e192b24SSimon Glass {
258*2e192b24SSimon Glass 	int i;
259*2e192b24SSimon Glass 
260*2e192b24SSimon Glass 	parse_flash();
261*2e192b24SSimon Glass 	for (i = 0; i < num_afs_images; i++) {
262*2e192b24SSimon Glass 		struct afs_image *afi = &afs_images[i];
263*2e192b24SSimon Glass 
264*2e192b24SSimon Glass 		if (strcmp(afi->name, name) == 0)
265*2e192b24SSimon Glass 			return CMD_RET_SUCCESS;
266*2e192b24SSimon Glass 	}
267*2e192b24SSimon Glass 	return CMD_RET_FAILURE;
268*2e192b24SSimon Glass }
269*2e192b24SSimon Glass 
do_afs(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])270*2e192b24SSimon Glass static int do_afs(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
271*2e192b24SSimon Glass {
272*2e192b24SSimon Glass 	int ret = CMD_RET_SUCCESS;
273*2e192b24SSimon Glass 
274*2e192b24SSimon Glass 	if (argc == 1) {
275*2e192b24SSimon Glass 		print_images();
276*2e192b24SSimon Glass 	} else if (argc == 3 && !strcmp(argv[1], "exists")) {
277*2e192b24SSimon Glass 		ret = exists(argv[2]);
278*2e192b24SSimon Glass 	} else if (argc == 3 && !strcmp(argv[1], "load")) {
279*2e192b24SSimon Glass 		ret = load_image(argv[2], 0x0);
280*2e192b24SSimon Glass 	} else if (argc == 4 && !strcmp(argv[1], "load")) {
281*2e192b24SSimon Glass 		ulong load_addr;
282*2e192b24SSimon Glass 
283*2e192b24SSimon Glass 		load_addr = simple_strtoul(argv[3], NULL, 16);
284*2e192b24SSimon Glass 		ret = load_image(argv[2], load_addr);
285*2e192b24SSimon Glass 	} else {
286*2e192b24SSimon Glass 		return CMD_RET_USAGE;
287*2e192b24SSimon Glass 	}
288*2e192b24SSimon Glass 
289*2e192b24SSimon Glass 	return ret;
290*2e192b24SSimon Glass }
291*2e192b24SSimon Glass 
292*2e192b24SSimon Glass U_BOOT_CMD(afs, 4, 0, do_afs, "show AFS partitions",
293*2e192b24SSimon Glass 	   "no arguments\n"
294*2e192b24SSimon Glass 	   "    - list images in flash\n"
295*2e192b24SSimon Glass 	   "exists <image>\n"
296*2e192b24SSimon Glass 	   "    - returns 1 if an image exists, else 0\n"
297*2e192b24SSimon Glass 	   "load <image>\n"
298*2e192b24SSimon Glass 	   "    - load an image to the location indicated in the header\n"
299*2e192b24SSimon Glass 	   "load <image> 0x<address>\n"
300*2e192b24SSimon Glass 	   "    - load an image to the location specified\n");
301