|  | /* | 
|  | * (C) Copyright 2008 | 
|  | * Marvell Semiconductor <www.marvell.com> | 
|  | * Written-by: Prafulla Wadaskar <prafulla@marvell.com> | 
|  | * | 
|  | * See file CREDITS for list of people who contributed to this | 
|  | * project. | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU General Public License as | 
|  | * published by the Free Software Foundation; either version 2 of | 
|  | * the License, or (at your option) any later version. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
|  | * GNU General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU General Public License | 
|  | * along with this program; if not, write to the Free Software | 
|  | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | 
|  | * MA 02111-1307 USA | 
|  | */ | 
|  |  | 
|  | /* Required to obtain the getline prototype from stdio.h */ | 
|  | #define _GNU_SOURCE | 
|  |  | 
|  | #include "mkimage.h" | 
|  | #include <image.h> | 
|  | #include "kwbimage.h" | 
|  |  | 
|  | /* | 
|  | * Supported commands for configuration file | 
|  | */ | 
|  | static table_entry_t kwbimage_cmds[] = { | 
|  | {CMD_BOOT_FROM,		"BOOT_FROM",		"boot command",	}, | 
|  | {CMD_NAND_ECC_MODE,	"NAND_ECC_MODE",	"NAND mode",	}, | 
|  | {CMD_NAND_PAGE_SIZE,	"NAND_PAGE_SIZE",	"NAND size",	}, | 
|  | {CMD_SATA_PIO_MODE,	"SATA_PIO_MODE",	"SATA mode",	}, | 
|  | {CMD_DDR_INIT_DELAY,	"DDR_INIT_DELAY",	"DDR init dly",	}, | 
|  | {CMD_DATA,		"DATA",			"Reg Write Data", }, | 
|  | {CMD_INVALID,		"",			"",	}, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Supported Boot options for configuration file | 
|  | */ | 
|  | static table_entry_t kwbimage_bootops[] = { | 
|  | {IBR_HDR_SPI_ID,	"spi",		"SPI Flash",	}, | 
|  | {IBR_HDR_NAND_ID,	"nand",		"NAND Flash",	}, | 
|  | {IBR_HDR_SATA_ID,	"sata",		"Sata port",	}, | 
|  | {IBR_HDR_PEX_ID,	"pex",		"PCIe port",	}, | 
|  | {IBR_HDR_UART_ID,	"uart",		"Serial port",	}, | 
|  | {-1,			"",		"Invalid",	}, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Supported NAND ecc options configuration file | 
|  | */ | 
|  | static table_entry_t kwbimage_eccmodes[] = { | 
|  | {IBR_HDR_ECC_DEFAULT,		"default",	"Default mode",	}, | 
|  | {IBR_HDR_ECC_FORCED_HAMMING,	"hamming",	"Hamming mode",	}, | 
|  | {IBR_HDR_ECC_FORCED_RS,		"rs",		"RS mode",	}, | 
|  | {IBR_HDR_ECC_DISABLED,		"disabled",	"ECC Disabled",	}, | 
|  | {-1,				"",		"",	}, | 
|  | }; | 
|  |  | 
|  | static struct kwb_header kwbimage_header; | 
|  | static int datacmd_cnt = 0; | 
|  | static char * fname = "Unknown"; | 
|  | static int lineno = -1; | 
|  |  | 
|  | /* | 
|  | * Report Error if xflag is set in addition to default | 
|  | */ | 
|  | static int kwbimage_check_params (struct mkimage_params *params) | 
|  | { | 
|  | if (!strlen (params->imagename)) { | 
|  | printf ("Error:%s - Configuration file not specified, " | 
|  | "it is needed for kwbimage generation\n", | 
|  | params->cmdname); | 
|  | return CFG_INVALID; | 
|  | } | 
|  | return	((params->dflag && (params->fflag || params->lflag)) || | 
|  | (params->fflag && (params->dflag || params->lflag)) || | 
|  | (params->lflag && (params->dflag || params->fflag)) || | 
|  | (params->xflag) || !(strlen (params->imagename))); | 
|  | } | 
|  |  | 
|  | static uint32_t check_get_hexval (char *token) | 
|  | { | 
|  | uint32_t hexval; | 
|  |  | 
|  | if (!sscanf (token, "%x", &hexval)) { | 
|  | printf ("Error:%s[%d] - Invalid hex data(%s)\n", fname, | 
|  | lineno, token); | 
|  | exit (EXIT_FAILURE); | 
|  | } | 
|  | return hexval; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Generates 8 bit checksum | 
|  | */ | 
|  | static uint8_t kwbimage_checksum8 (void *start, uint32_t len, uint8_t csum) | 
|  | { | 
|  | register uint8_t sum = csum; | 
|  | volatile uint8_t *p = (volatile uint8_t *)start; | 
|  |  | 
|  | /* check len and return zero checksum if invalid */ | 
|  | if (!len) | 
|  | return 0; | 
|  |  | 
|  | do { | 
|  | sum += *p; | 
|  | p++; | 
|  | } while (--len); | 
|  | return (sum); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Generates 32 bit checksum | 
|  | */ | 
|  | static uint32_t kwbimage_checksum32 (uint32_t *start, uint32_t len, uint32_t csum) | 
|  | { | 
|  | register uint32_t sum = csum; | 
|  | volatile uint32_t *p = start; | 
|  |  | 
|  | /* check len and return zero checksum if invalid */ | 
|  | if (!len) | 
|  | return 0; | 
|  |  | 
|  | if (len % sizeof(uint32_t)) { | 
|  | printf ("Error:%s[%d] - length is not in multiple of %zu\n", | 
|  | __FUNCTION__, len, sizeof(uint32_t)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | do { | 
|  | sum += *p; | 
|  | p++; | 
|  | len -= sizeof(uint32_t); | 
|  | } while (len > 0); | 
|  | return (sum); | 
|  | } | 
|  |  | 
|  | static void kwbimage_check_cfgdata (char *token, enum kwbimage_cmd cmdsw, | 
|  | struct kwb_header *kwbhdr) | 
|  | { | 
|  | bhr_t *mhdr = &kwbhdr->kwb_hdr; | 
|  | extbhr_t *exthdr = &kwbhdr->kwb_exthdr; | 
|  | int i; | 
|  |  | 
|  | switch (cmdsw) { | 
|  | case CMD_BOOT_FROM: | 
|  | i = get_table_entry_id (kwbimage_bootops, | 
|  | "Kwbimage boot option", token); | 
|  |  | 
|  | if (i < 0) | 
|  | goto INVL_DATA; | 
|  |  | 
|  | mhdr->blockid = i; | 
|  | printf ("Preparing kirkwood boot image to boot " | 
|  | "from %s\n", token); | 
|  | break; | 
|  | case CMD_NAND_ECC_MODE: | 
|  | i = get_table_entry_id (kwbimage_eccmodes, | 
|  | "NAND ecc mode", token); | 
|  |  | 
|  | if (i < 0) | 
|  | goto INVL_DATA; | 
|  |  | 
|  | mhdr->nandeccmode = i; | 
|  | printf ("Nand ECC mode = %s\n", token); | 
|  | break; | 
|  | case CMD_NAND_PAGE_SIZE: | 
|  | mhdr->nandpagesize = | 
|  | (uint16_t) check_get_hexval (token); | 
|  | printf ("Nand page size = 0x%x\n", mhdr->nandpagesize); | 
|  | break; | 
|  | case CMD_SATA_PIO_MODE: | 
|  | mhdr->satapiomode = | 
|  | (uint8_t) check_get_hexval (token); | 
|  | printf ("Sata PIO mode = 0x%x\n", | 
|  | mhdr->satapiomode); | 
|  | break; | 
|  | case CMD_DDR_INIT_DELAY: | 
|  | mhdr->ddrinitdelay = | 
|  | (uint16_t) check_get_hexval (token); | 
|  | printf ("DDR init delay = %d msec\n", mhdr->ddrinitdelay); | 
|  | break; | 
|  | case CMD_DATA: | 
|  | exthdr->rcfg[datacmd_cnt].raddr = | 
|  | check_get_hexval (token); | 
|  |  | 
|  | break; | 
|  | case CMD_INVALID: | 
|  | goto INVL_DATA; | 
|  | default: | 
|  | goto INVL_DATA; | 
|  | } | 
|  | return; | 
|  |  | 
|  | INVL_DATA: | 
|  | printf ("Error:%s[%d] - Invalid data\n", fname, lineno); | 
|  | exit (EXIT_FAILURE); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * this function sets the kwbimage header by- | 
|  | * 	1. Abstracting input command line arguments data | 
|  | *	2. parses the kwbimage configuration file and update extebded header data | 
|  | *	3. calculates header, extended header and image checksums | 
|  | */ | 
|  | static void kwdimage_set_ext_header (struct kwb_header *kwbhdr, char* name) { | 
|  | bhr_t *mhdr = &kwbhdr->kwb_hdr; | 
|  | extbhr_t *exthdr = &kwbhdr->kwb_exthdr; | 
|  | FILE *fd = NULL; | 
|  | int j; | 
|  | char *line = NULL; | 
|  | char * token, *saveptr1, *saveptr2; | 
|  | size_t len = 0; | 
|  | enum kwbimage_cmd cmd; | 
|  |  | 
|  | fname = name; | 
|  | /* set dram register offset */ | 
|  | exthdr->dramregsoffs = (intptr_t)&exthdr->rcfg - (intptr_t)mhdr; | 
|  |  | 
|  | if ((fd = fopen (name, "r")) == 0) { | 
|  | printf ("Error:%s - Can't open\n", fname); | 
|  | exit (EXIT_FAILURE); | 
|  | } | 
|  |  | 
|  | /* Simple kwimage.cfg file parser */ | 
|  | lineno=0; | 
|  | while ((getline (&line, &len, fd)) > 0) { | 
|  | lineno++; | 
|  | token = strtok_r (line, "\r\n", &saveptr1); | 
|  | /* drop all lines with zero tokens (= empty lines) */ | 
|  | if (token == NULL) | 
|  | continue; | 
|  |  | 
|  | for (j = 0, cmd = CMD_INVALID, line = token; ; line = NULL) { | 
|  | token = strtok_r (line, " \t", &saveptr2); | 
|  | if (token == NULL) | 
|  | break; | 
|  | /* Drop all text starting with '#' as comments */ | 
|  | if (token[0] == '#') | 
|  | break; | 
|  |  | 
|  | /* Process rest as valid config command line */ | 
|  | switch (j) { | 
|  | case CFG_COMMAND: | 
|  | cmd = get_table_entry_id (kwbimage_cmds, | 
|  | "Kwbimage command", token); | 
|  |  | 
|  | if (cmd == CMD_INVALID) | 
|  | goto INVL_CMD; | 
|  | break; | 
|  |  | 
|  | case CFG_DATA0: | 
|  | kwbimage_check_cfgdata (token, cmd, kwbhdr); | 
|  | break; | 
|  |  | 
|  | case CFG_DATA1: | 
|  | if (cmd != CMD_DATA) | 
|  | goto INVL_CMD; | 
|  |  | 
|  | exthdr->rcfg[datacmd_cnt].rdata = | 
|  | check_get_hexval (token); | 
|  |  | 
|  | if (datacmd_cnt > KWBIMAGE_MAX_CONFIG ) { | 
|  | printf ("Error:%s[%d] - Found more " | 
|  | "than max(%zd) allowed " | 
|  | "data configurations\n", | 
|  | fname, lineno, | 
|  | KWBIMAGE_MAX_CONFIG); | 
|  | exit (EXIT_FAILURE); | 
|  | } else | 
|  | datacmd_cnt++; | 
|  | break; | 
|  |  | 
|  | default: | 
|  | goto INVL_CMD; | 
|  | } | 
|  | j++; | 
|  | } | 
|  | } | 
|  | if (line) | 
|  | free (line); | 
|  |  | 
|  | fclose (fd); | 
|  | return; | 
|  |  | 
|  | /* | 
|  | * Invalid Command error reporring | 
|  | * | 
|  | * command CMD_DATA needs three strings on a line | 
|  | * whereas other commands need only two. | 
|  | * | 
|  | * if more than two/three (as per command type) are observed, | 
|  | * then error will be reported | 
|  | */ | 
|  | INVL_CMD: | 
|  | printf ("Error:%s[%d] - Invalid command\n", fname, lineno); | 
|  | exit (EXIT_FAILURE); | 
|  | } | 
|  |  | 
|  | static void kwbimage_set_header (void *ptr, struct stat *sbuf, int ifd, | 
|  | struct mkimage_params *params) | 
|  | { | 
|  | struct kwb_header *hdr = (struct kwb_header *)ptr; | 
|  | bhr_t *mhdr = &hdr->kwb_hdr; | 
|  | extbhr_t *exthdr = &hdr->kwb_exthdr; | 
|  | uint32_t checksum; | 
|  | int size; | 
|  |  | 
|  | /* Build and add image checksum header */ | 
|  | checksum = kwbimage_checksum32 ((uint32_t *)ptr, sbuf->st_size, 0); | 
|  |  | 
|  | size = write (ifd, &checksum, sizeof(uint32_t)); | 
|  | if (size != sizeof(uint32_t)) { | 
|  | printf ("Error:%s - Checksum write %d bytes %s\n", | 
|  | params->cmdname, size, params->imagefile); | 
|  | exit (EXIT_FAILURE); | 
|  | } | 
|  |  | 
|  | sbuf->st_size += sizeof(uint32_t); | 
|  |  | 
|  | mhdr->blocksize = sbuf->st_size - sizeof(struct kwb_header); | 
|  | mhdr->srcaddr = sizeof(struct kwb_header); | 
|  | mhdr->destaddr= params->addr; | 
|  | mhdr->execaddr =params->ep; | 
|  | mhdr->ext = 0x1; /* header extension appended */ | 
|  |  | 
|  | kwdimage_set_ext_header (hdr, params->imagename); | 
|  | /* calculate checksums */ | 
|  | mhdr->checkSum = kwbimage_checksum8 ((void *)mhdr, sizeof(bhr_t), 0); | 
|  | exthdr->checkSum = kwbimage_checksum8 ((void *)exthdr, | 
|  | sizeof(extbhr_t), 0); | 
|  | } | 
|  |  | 
|  | static int kwbimage_verify_header (unsigned char *ptr, int image_size, | 
|  | struct mkimage_params *params) | 
|  | { | 
|  | struct kwb_header *hdr = (struct kwb_header *)ptr; | 
|  | bhr_t *mhdr = &hdr->kwb_hdr; | 
|  | extbhr_t *exthdr = &hdr->kwb_exthdr; | 
|  | uint8_t calc_hdrcsum; | 
|  | uint8_t calc_exthdrcsum; | 
|  |  | 
|  | calc_hdrcsum = kwbimage_checksum8 ((void *)mhdr, | 
|  | sizeof(bhr_t) - sizeof(uint8_t), 0); | 
|  | if (calc_hdrcsum != mhdr->checkSum) | 
|  | return -FDT_ERR_BADSTRUCTURE;	/* mhdr csum not matched */ | 
|  |  | 
|  | calc_exthdrcsum = kwbimage_checksum8 ((void *)exthdr, | 
|  | sizeof(extbhr_t) - sizeof(uint8_t), 0); | 
|  | if (calc_exthdrcsum != exthdr->checkSum) | 
|  | return -FDT_ERR_BADSTRUCTURE; /* exthdr csum not matched */ | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void kwbimage_print_header (const void *ptr) | 
|  | { | 
|  | struct kwb_header *hdr = (struct kwb_header *) ptr; | 
|  | bhr_t *mhdr = &hdr->kwb_hdr; | 
|  | char *name = get_table_entry_name (kwbimage_bootops, | 
|  | "Kwbimage boot option", | 
|  | (int) mhdr->blockid); | 
|  |  | 
|  | printf ("Image Type:   Kirkwood Boot from %s Image\n", name); | 
|  | printf ("Data Size:    "); | 
|  | genimg_print_size (mhdr->blocksize - sizeof(uint32_t)); | 
|  | printf ("Load Address: %08x\n", mhdr->destaddr); | 
|  | printf ("Entry Point:  %08x\n", mhdr->execaddr); | 
|  | } | 
|  |  | 
|  | static int kwbimage_check_image_types (uint8_t type) | 
|  | { | 
|  | if (type == IH_TYPE_KWBIMAGE) | 
|  | return EXIT_SUCCESS; | 
|  | else | 
|  | return EXIT_FAILURE; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * kwbimage type parameters definition | 
|  | */ | 
|  | static struct image_type_params kwbimage_params = { | 
|  | .name = "Kirkwood Boot Image support", | 
|  | .header_size = sizeof(struct kwb_header), | 
|  | .hdr = (void*)&kwbimage_header, | 
|  | .check_image_type = kwbimage_check_image_types, | 
|  | .verify_header = kwbimage_verify_header, | 
|  | .print_header = kwbimage_print_header, | 
|  | .set_header = kwbimage_set_header, | 
|  | .check_params = kwbimage_check_params, | 
|  | }; | 
|  |  | 
|  | void init_kwb_image_type (void) | 
|  | { | 
|  | mkimage_register (&kwbimage_params); | 
|  | } |