xref: /openbmc/u-boot/tools/kwbimage.c (revision 98a48c5d)
1 /*
2  * (C) Copyright 2008
3  * Marvell Semiconductor <www.marvell.com>
4  * Written-by: Prafulla Wadaskar <prafulla@marvell.com>
5  *
6  * See file CREDITS for list of people who contributed to this
7  * project.
8  *
9  * This program is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU General Public License as
11  * published by the Free Software Foundation; either version 2 of
12  * the License, or (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
22  * MA 02111-1307 USA
23  */
24 
25 /* Required to obtain the getline prototype from stdio.h */
26 #define _GNU_SOURCE
27 
28 #include "mkimage.h"
29 #include <image.h>
30 #include "kwbimage.h"
31 
32 /*
33  * Supported commands for configuration file
34  */
35 static table_entry_t kwbimage_cmds[] = {
36 	{CMD_BOOT_FROM,		"BOOT_FROM",		"boot command",	},
37 	{CMD_NAND_ECC_MODE,	"NAND_ECC_MODE",	"NAND mode",	},
38 	{CMD_NAND_PAGE_SIZE,	"NAND_PAGE_SIZE",	"NAND size",	},
39 	{CMD_SATA_PIO_MODE,	"SATA_PIO_MODE",	"SATA mode",	},
40 	{CMD_DDR_INIT_DELAY,	"DDR_INIT_DELAY",	"DDR init dly",	},
41 	{CMD_DATA,		"DATA",			"Reg Write Data", },
42 	{CMD_INVALID,		"",			"",	},
43 };
44 
45 /*
46  * Supported Boot options for configuration file
47  */
48 static table_entry_t kwbimage_bootops[] = {
49 	{IBR_HDR_SPI_ID,	"spi",		"SPI Flash",	},
50 	{IBR_HDR_NAND_ID,	"nand",		"NAND Flash",	},
51 	{IBR_HDR_SATA_ID,	"sata",		"Sata port",	},
52 	{IBR_HDR_PEX_ID,	"pex",		"PCIe port",	},
53 	{IBR_HDR_UART_ID,	"uart",		"Serial port",	},
54 	{-1,			"",		"Invalid",	},
55 };
56 
57 /*
58  * Supported NAND ecc options configuration file
59  */
60 static table_entry_t kwbimage_eccmodes[] = {
61 	{IBR_HDR_ECC_DEFAULT,		"default",	"Default mode",	},
62 	{IBR_HDR_ECC_FORCED_HAMMING,	"hamming",	"Hamming mode",	},
63 	{IBR_HDR_ECC_FORCED_RS,		"rs",		"RS mode",	},
64 	{IBR_HDR_ECC_DISABLED,		"disabled",	"ECC Disabled",	},
65 	{-1,				"",		"",	},
66 };
67 
68 static struct kwb_header kwbimage_header;
69 static int datacmd_cnt = 0;
70 static char * fname = "Unknown";
71 static int lineno = -1;
72 
73 /*
74  * Report Error if xflag is set in addition to default
75  */
76 static int kwbimage_check_params (struct mkimage_params *params)
77 {
78 	if (!strlen (params->imagename)) {
79 		printf ("Error:%s - Configuration file not specified, "
80 			"it is needed for kwbimage generation\n",
81 			params->cmdname);
82 		return CFG_INVALID;
83 	}
84 	return	((params->dflag && (params->fflag || params->lflag)) ||
85 		(params->fflag && (params->dflag || params->lflag)) ||
86 		(params->lflag && (params->dflag || params->fflag)) ||
87 		(params->xflag) || !(strlen (params->imagename)));
88 }
89 
90 static uint32_t check_get_hexval (char *token)
91 {
92 	uint32_t hexval;
93 
94 	if (!sscanf (token, "%x", &hexval)) {
95 		printf ("Error:%s[%d] - Invalid hex data(%s)\n", fname,
96 			lineno, token);
97 		exit (EXIT_FAILURE);
98 	}
99 	return hexval;
100 }
101 
102 /*
103  * Generates 8 bit checksum
104  */
105 static uint8_t kwbimage_checksum8 (void *start, uint32_t len, uint8_t csum)
106 {
107 	register uint8_t sum = csum;
108 	volatile uint8_t *p = (volatile uint8_t *)start;
109 
110 	/* check len and return zero checksum if invalid */
111 	if (!len)
112 		return 0;
113 
114 	do {
115 		sum += *p;
116 		p++;
117 	} while (--len);
118 	return (sum);
119 }
120 
121 /*
122  * Generates 32 bit checksum
123  */
124 static uint32_t kwbimage_checksum32 (uint32_t *start, uint32_t len, uint32_t csum)
125 {
126 	register uint32_t sum = csum;
127 	volatile uint32_t *p = start;
128 
129 	/* check len and return zero checksum if invalid */
130 	if (!len)
131 		return 0;
132 
133 	if (len % sizeof(uint32_t)) {
134 		printf ("Error:%s[%d] - length is not in multiple of %zu\n",
135 			__FUNCTION__, len, sizeof(uint32_t));
136 		return 0;
137 	}
138 
139 	do {
140 		sum += *p;
141 		p++;
142 		len -= sizeof(uint32_t);
143 	} while (len > 0);
144 	return (sum);
145 }
146 
147 static void kwbimage_check_cfgdata (char *token, enum kwbimage_cmd cmdsw,
148 					struct kwb_header *kwbhdr)
149 {
150 	bhr_t *mhdr = &kwbhdr->kwb_hdr;
151 	extbhr_t *exthdr = &kwbhdr->kwb_exthdr;
152 	int i;
153 
154 	switch (cmdsw) {
155 	case CMD_BOOT_FROM:
156 		i = get_table_entry_id (kwbimage_bootops,
157 				"Kwbimage boot option", token);
158 
159 		if (i < 0)
160 			goto INVL_DATA;
161 
162 		mhdr->blockid = i;
163 		printf ("Preparing kirkwood boot image to boot "
164 			"from %s\n", token);
165 		break;
166 	case CMD_NAND_ECC_MODE:
167 		i = get_table_entry_id (kwbimage_eccmodes,
168 			"NAND ecc mode", token);
169 
170 		if (i < 0)
171 			goto INVL_DATA;
172 
173 		mhdr->nandeccmode = i;
174 		printf ("Nand ECC mode = %s\n", token);
175 		break;
176 	case CMD_NAND_PAGE_SIZE:
177 		mhdr->nandpagesize =
178 			(uint16_t) check_get_hexval (token);
179 		printf ("Nand page size = 0x%x\n", mhdr->nandpagesize);
180 		break;
181 	case CMD_SATA_PIO_MODE:
182 		mhdr->satapiomode =
183 			(uint8_t) check_get_hexval (token);
184 		printf ("Sata PIO mode = 0x%x\n",
185 				mhdr->satapiomode);
186 		break;
187 	case CMD_DDR_INIT_DELAY:
188 		mhdr->ddrinitdelay =
189 			(uint16_t) check_get_hexval (token);
190 		printf ("DDR init delay = %d msec\n", mhdr->ddrinitdelay);
191 		break;
192 	case CMD_DATA:
193 		exthdr->rcfg[datacmd_cnt].raddr =
194 			check_get_hexval (token);
195 
196 		break;
197 	case CMD_INVALID:
198 		goto INVL_DATA;
199 	default:
200 		goto INVL_DATA;
201 	}
202 	return;
203 
204 INVL_DATA:
205 	printf ("Error:%s[%d] - Invalid data\n", fname, lineno);
206 	exit (EXIT_FAILURE);
207 }
208 
209 /*
210  * this function sets the kwbimage header by-
211  * 	1. Abstracting input command line arguments data
212  *	2. parses the kwbimage configuration file and update extebded header data
213  *	3. calculates header, extended header and image checksums
214  */
215 static void kwdimage_set_ext_header (struct kwb_header *kwbhdr, char* name) {
216 	bhr_t *mhdr = &kwbhdr->kwb_hdr;
217 	extbhr_t *exthdr = &kwbhdr->kwb_exthdr;
218 	FILE *fd = NULL;
219 	int j;
220 	char *line = NULL;
221 	char * token, *saveptr1, *saveptr2;
222 	size_t len = 0;
223 	enum kwbimage_cmd cmd;
224 
225 	fname = name;
226 	/* set dram register offset */
227 	exthdr->dramregsoffs = (intptr_t)&exthdr->rcfg - (intptr_t)mhdr;
228 
229 	if ((fd = fopen (name, "r")) == 0) {
230 		printf ("Error:%s - Can't open\n", fname);
231 		exit (EXIT_FAILURE);
232 	}
233 
234 	/* Simple kwimage.cfg file parser */
235 	lineno=0;
236 	while ((getline (&line, &len, fd)) > 0) {
237 		lineno++;
238 		token = strtok_r (line, "\r\n", &saveptr1);
239 		/* drop all lines with zero tokens (= empty lines) */
240 		if (token == NULL)
241 			continue;
242 
243 		for (j = 0, cmd = CMD_INVALID, line = token; ; line = NULL) {
244 			token = strtok_r (line, " \t", &saveptr2);
245 			if (token == NULL)
246 			break;
247 			/* Drop all text starting with '#' as comments */
248 			if (token[0] == '#')
249 				break;
250 
251 			/* Process rest as valid config command line */
252 			switch (j) {
253 			case CFG_COMMAND:
254 				cmd = get_table_entry_id (kwbimage_cmds,
255 						"Kwbimage command", token);
256 
257 				if (cmd == CMD_INVALID)
258 					goto INVL_CMD;
259 				break;
260 
261 			case CFG_DATA0:
262 				kwbimage_check_cfgdata (token, cmd, kwbhdr);
263 				break;
264 
265 			case CFG_DATA1:
266 				if (cmd != CMD_DATA)
267 					goto INVL_CMD;
268 
269 				exthdr->rcfg[datacmd_cnt].rdata =
270 						check_get_hexval (token);
271 
272 				if (datacmd_cnt > KWBIMAGE_MAX_CONFIG ) {
273 					printf ("Error:%s[%d] - Found more "
274 						"than max(%zd) allowed "
275 						"data configurations\n",
276 						fname, lineno,
277 						KWBIMAGE_MAX_CONFIG);
278 				exit (EXIT_FAILURE);
279 				} else
280 					datacmd_cnt++;
281 				break;
282 
283 			default:
284 				goto INVL_CMD;
285 			}
286 			j++;
287 		}
288 	}
289 	if (line)
290 		free (line);
291 
292 	fclose (fd);
293 	return;
294 
295 /*
296  * Invalid Command error reporring
297  *
298  * command CMD_DATA needs three strings on a line
299  * whereas other commands need only two.
300  *
301  * if more than two/three (as per command type) are observed,
302  * then error will be reported
303  */
304 INVL_CMD:
305 	printf ("Error:%s[%d] - Invalid command\n", fname, lineno);
306 	exit (EXIT_FAILURE);
307 }
308 
309 static void kwbimage_set_header (void *ptr, struct stat *sbuf, int ifd,
310 				struct mkimage_params *params)
311 {
312 	struct kwb_header *hdr = (struct kwb_header *)ptr;
313 	bhr_t *mhdr = &hdr->kwb_hdr;
314 	extbhr_t *exthdr = &hdr->kwb_exthdr;
315 	uint32_t checksum;
316 	int size;
317 
318 	/* Build and add image checksum header */
319 	checksum = kwbimage_checksum32 ((uint32_t *)ptr, sbuf->st_size, 0);
320 
321 	size = write (ifd, &checksum, sizeof(uint32_t));
322 	if (size != sizeof(uint32_t)) {
323 		printf ("Error:%s - Checksum write %d bytes %s\n",
324 			params->cmdname, size, params->imagefile);
325 		exit (EXIT_FAILURE);
326 	}
327 
328 	sbuf->st_size += sizeof(uint32_t);
329 
330 	mhdr->blocksize = sbuf->st_size - sizeof(struct kwb_header);
331 	mhdr->srcaddr = sizeof(struct kwb_header);
332 	mhdr->destaddr= params->addr;
333 	mhdr->execaddr =params->ep;
334 	mhdr->ext = 0x1; /* header extension appended */
335 
336 	kwdimage_set_ext_header (hdr, params->imagename);
337 	/* calculate checksums */
338 	mhdr->checkSum = kwbimage_checksum8 ((void *)mhdr, sizeof(bhr_t), 0);
339 	exthdr->checkSum = kwbimage_checksum8 ((void *)exthdr,
340 						sizeof(extbhr_t), 0);
341 }
342 
343 static int kwbimage_verify_header (unsigned char *ptr, int image_size,
344 			struct mkimage_params *params)
345 {
346 	struct kwb_header *hdr = (struct kwb_header *)ptr;
347 	bhr_t *mhdr = &hdr->kwb_hdr;
348 	extbhr_t *exthdr = &hdr->kwb_exthdr;
349 	uint8_t calc_hdrcsum;
350 	uint8_t calc_exthdrcsum;
351 
352 	calc_hdrcsum = kwbimage_checksum8 ((void *)mhdr,
353 			sizeof(bhr_t) - sizeof(uint8_t), 0);
354 	if (calc_hdrcsum != mhdr->checkSum)
355 		return -FDT_ERR_BADSTRUCTURE;	/* mhdr csum not matched */
356 
357 	calc_exthdrcsum = kwbimage_checksum8 ((void *)exthdr,
358 			sizeof(extbhr_t) - sizeof(uint8_t), 0);
359 	if (calc_exthdrcsum != exthdr->checkSum)
360 		return -FDT_ERR_BADSTRUCTURE; /* exthdr csum not matched */
361 
362 	return 0;
363 }
364 
365 static void kwbimage_print_header (const void *ptr)
366 {
367 	struct kwb_header *hdr = (struct kwb_header *) ptr;
368 	bhr_t *mhdr = &hdr->kwb_hdr;
369 	char *name = get_table_entry_name (kwbimage_bootops,
370 				"Kwbimage boot option",
371 				(int) mhdr->blockid);
372 
373 	printf ("Image Type:   Kirkwood Boot from %s Image\n", name);
374 	printf ("Data Size:    ");
375 	genimg_print_size (mhdr->blocksize - sizeof(uint32_t));
376 	printf ("Load Address: %08x\n", mhdr->destaddr);
377 	printf ("Entry Point:  %08x\n", mhdr->execaddr);
378 }
379 
380 static int kwbimage_check_image_types (uint8_t type)
381 {
382 	if (type == IH_TYPE_KWBIMAGE)
383 		return EXIT_SUCCESS;
384 	else
385 		return EXIT_FAILURE;
386 }
387 
388 /*
389  * kwbimage type parameters definition
390  */
391 static struct image_type_params kwbimage_params = {
392 	.name = "Kirkwood Boot Image support",
393 	.header_size = sizeof(struct kwb_header),
394 	.hdr = (void*)&kwbimage_header,
395 	.check_image_type = kwbimage_check_image_types,
396 	.verify_header = kwbimage_verify_header,
397 	.print_header = kwbimage_print_header,
398 	.set_header = kwbimage_set_header,
399 	.check_params = kwbimage_check_params,
400 };
401 
402 void init_kwb_image_type (void)
403 {
404 	mkimage_register (&kwbimage_params);
405 }
406