* use relative paths in ldscript.ld and crt0_includes.h * avoid use of dd(1) in xcompile * build libregex for kconfig, if necessary * work around missing utsname on win32 * unlink targets before rename on win32 * implement (crude) mkstemp for win32 * avoid open/read/close, use fopen/fread/fclose instead * don't free certain data structures in romcc on win32 to avoid crashes (likely use-after-free()) * handle "\CRLF" and win32 style absolute paths (X:/ or X:\) in romcc * make lzma (part of cbfstool) build on XP * implement ntohl/htonl on win32 * handle CRLF in awk script * set larger stack for romcc on win32 Signed-off-by: Patrick Georgi <patrick.georgi@coresystems.de> Acked-by: Peter Stuge <peter@stuge.se> git-svn-id: svn://svn.coreboot.org/coreboot/trunk@4952 2b7e53f0-3cfb-0310-b3e9-8179ed1497e1
		
			
				
	
	
		
			613 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			613 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <unistd.h>
 | |
| #include <ctype.h>
 | |
| #include <errno.h>
 | |
| #include <libgen.h>
 | |
| #include "../../src/include/pc80/mc146818rtc.h"
 | |
| #include "../../src/include/boot/coreboot_tables.h"
 | |
| 
 | |
| #define CMOS_IMAGE_BUFFER_SIZE 256
 | |
| #define INPUT_LINE_MAX 256
 | |
| #define MAX_VALUE_BYTE_LENGTH 64
 | |
| 
 | |
| #define TMPFILE_LEN 256
 | |
| #define TMPFILE_TEMPLATE "/build_opt_tbl_XXXXXX"
 | |
| 
 | |
| static unsigned char cmos_table[4096];
 | |
| void test_for_entry_overlaps(void *entry_start, void *entry_end);
 | |
| 
 | |
| /* This array is used to isolate bits that are to be changed in a byte */
 | |
| static unsigned char clip[9]={0,1,3,7,0x0f,0x1f,0x3f,0x7f,0xff};
 | |
| 
 | |
| #ifdef WIN32
 | |
| #include <fcntl.h>
 | |
| char* mkstemp(char* name) {
 | |
| 	static char val='0';
 | |
| 	char *c=name;
 | |
| 	while (*c!='X') c++;
 | |
| 	*c=val++;
 | |
| 	return open(name,O_CREAT | O_RDWR);
 | |
| }
 | |
| #define UNLINK_IF_NECESSARY(x) unlink(x)
 | |
| #else
 | |
| #define UNLINK_IF_NECESSARY(x)
 | |
| #endif
 | |
| 
 | |
| /* This routine loops through the entried and tests if any of the fields overlap
 | |
| 	input entry_start = the memory pointer to the start of the entries.
 | |
| 	      entry_end = the byte past the entries.
 | |
| 	output  none
 | |
| 		if there is an overlap, the routine exits, other wise it returns.
 | |
| */
 | |
| void test_for_entry_overlaps(void *entry_start, void *entry_end)
 | |
| {
 | |
| 	int ptr;
 | |
| 	char *cptr;
 | |
| 	int buffer_bit_size;
 | |
| 	int offset;
 | |
| 	int byte;
 | |
| 	int byte_length;
 | |
|         struct cmos_entries *ce;
 | |
| 	unsigned char test[CMOS_IMAGE_BUFFER_SIZE];
 | |
| 	unsigned char set;
 | |
| 
 | |
| 	/* calculate the size of the cmos buffer in bits */
 | |
| 	buffer_bit_size=(CMOS_IMAGE_BUFFER_SIZE*8);
 | |
| 	/* clear the temporary test buffer */
 | |
| 	for(ptr=0; ptr < CMOS_IMAGE_BUFFER_SIZE; ptr++)
 | |
| 		test[ptr]=0;
 | |
| 
 | |
| 	/* loop through each entry in the table testing for errors */
 | |
| 	for(cptr = entry_start; cptr < (char *)entry_end; cptr += ce->size) {
 | |
| 		ce=(struct cmos_entries *)cptr;
 | |
| 		/* test if entry goes past the end of the buffer */
 | |
| 		if((ce->bit+ce->length)>buffer_bit_size) {
 | |
| 			printf("Error - Entry %s start bit + length must be less than %d\n",
 | |
| 				ce->name,buffer_bit_size);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		byte=ce->bit/8;
 | |
| 		offset=ce->bit%8;
 | |
| 		byte_length=ce->length/8;
 | |
| 		if(byte_length) {	/* entry is 8 bits long or more */
 | |
| 			if(offset) { /* if 8 bits or more long, it must be byte aligned */
 | |
| 				printf("Error - Entry %s length over 8 must be byte aligned\n",
 | |
| 					ce->name);
 | |
| 				exit(1);
 | |
| 			}
 | |
| 			/* test if entries 8 or more in length are even bytes */ 
 | |
| 			if(ce->length%8){
 | |
|                                 printf("Error - Entry %s length over 8 must be a multiple of 8\n",
 | |
|                                         ce->name);
 | |
|                                 exit(1);
 | |
|                         }
 | |
| 			/* test if any of the bits have been previously used */
 | |
| 			for(;byte_length;byte_length--,byte++) {
 | |
| 				if(test[byte]) {
 | |
| 					printf("Error - Entry %s uses same bits previously used\n",
 | |
| 						ce->name);
 | |
| 					exit(1);
 | |
| 	                        }
 | |
| 				test[byte]=clip[8]; /* set the bits defined in test */
 | |
| 			}
 | |
| 		} else {
 | |
| 			/* test if bits overlap byte boundaries */
 | |
| 			if(ce->length>(8-offset)) {
 | |
|                                 printf("Error - Entry %s length overlaps a byte boundry\n",
 | |
| 					ce->name);
 | |
|                                 exit(1);
 | |
|                         }
 | |
| 			/* test for bits previously used */
 | |
| 			set=(clip[ce->length]<<offset);
 | |
| 			if(test[byte]&set) {
 | |
|                         	printf("Error - Entry %s uses same bits previously used\n",
 | |
|                                                 ce->name);
 | |
|                                 exit(1);
 | |
|                         }
 | |
|                         test[byte]|=set;  /* set the bits defined in test */
 | |
| 		}
 | |
| 	}
 | |
| 	return;
 | |
| }
 | |
| 
 | |
| /* This routine displays the usage options */
 | |
| void display_usage(char *name)
 | |
| {
 | |
| 	printf("Usage: %s [--config filename]\n", name);
 | |
| 	printf("                       [--option filename]\n");
 | |
| 	printf("                       [--header filename]\n\n");
 | |
| 	printf("--config = Build the definitions table from the given file.\n");
 | |
| 	printf("--option = Output a C source file with the definitions.\n");
 | |
| 	printf("--header = Ouput a C header file with the definitions.\n");
 | |
| 	exit(1);
 | |
| }
 | |
| 
 | |
| 
 | |
| static void skip_spaces(char *line, char **ptr)
 | |
| {
 | |
| 	if (!isspace(**ptr)) {
 | |
| 		printf("Error missing whitespace in line\n%s\n", line);
 | |
| 		exit(1);
 | |
| 	}
 | |
| 	while(isspace(**ptr)) {
 | |
| 		(*ptr)++;
 | |
| 	}
 | |
| 	return;
 | |
| }
 | |
| static unsigned long get_number(char *line, char **ptr, int base)
 | |
| {
 | |
| 	unsigned long value;
 | |
| 	char *ptr2;
 | |
| 	value = strtoul(*ptr, &ptr2, base);
 | |
| 	if (ptr2 == *ptr) {
 | |
| 		printf("Error missing digits at: \n%s\n in line:\n%s\n", 
 | |
| 			*ptr, line);
 | |
| 		exit(1);
 | |
| 	}
 | |
| 	*ptr = ptr2;
 | |
| 	return value;
 | |
| }
 | |
| 
 | |
| static int is_ident_digit(int c)
 | |
| {
 | |
| 	int result;
 | |
| 	switch(c) {
 | |
| 	case '0':	case '1':	case '2':	case '3':
 | |
| 	case '4':	case '5':	case '6':	case '7':
 | |
| 	case '8':	case '9':
 | |
| 		result = 1;
 | |
| 		break;
 | |
| 	default:
 | |
| 		result = 0;
 | |
| 		break;
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| static int is_ident_nondigit(int c)
 | |
| {
 | |
| 	int result;
 | |
| 	switch(c) {
 | |
| 	case 'A':	case 'B':	case 'C':	case 'D':
 | |
| 	case 'E':	case 'F':	case 'G':	case 'H':
 | |
| 	case 'I':	case 'J':	case 'K':	case 'L':
 | |
| 	case 'M':	case 'N':	case 'O':	case 'P':
 | |
| 	case 'Q':	case 'R':	case 'S':	case 'T':
 | |
| 	case 'U':	case 'V':	case 'W':	case 'X':
 | |
| 	case 'Y':	case 'Z':
 | |
| 	case 'a':	case 'b':	case 'c':	case 'd':
 | |
| 	case 'e':	case 'f':	case 'g':	case 'h':
 | |
| 	case 'i':	case 'j':	case 'k':	case 'l':
 | |
| 	case 'm':	case 'n':	case 'o':	case 'p':
 | |
| 	case 'q':	case 'r':	case 's':	case 't':
 | |
| 	case 'u':	case 'v':	case 'w':	case 'x':
 | |
| 	case 'y':	case 'z':
 | |
| 	case '_':
 | |
| 		result = 1;
 | |
| 		break;
 | |
| 	default:
 | |
| 		result = 0;
 | |
| 		break;
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| static int is_ident(char *str)
 | |
| {
 | |
| 	int result;
 | |
| 	int ch;
 | |
| 	ch = *str;
 | |
| 	result = 0;
 | |
| 	if (is_ident_nondigit(ch)) {
 | |
| 		do {
 | |
| 			str++;
 | |
| 			ch = *str;
 | |
| 		} while(ch && (is_ident_nondigit(ch) || (is_ident_digit(ch))));
 | |
| 		result = (ch == '\0');
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* This routine builds the cmos definition table from the cmos layout file
 | |
| 	input The input comes from the configuration file which contains two parts
 | |
| 		entries and enumerations. Each section is started with the key words
 | |
| 		entries and enumerations.  Records then follow in their respective 
 | |
| 		formats.
 | |
| 	output The output of this program is the cmos definitions table.  It is stored
 | |
| 		in the cmos_table array. If this module is called, and the global 
 | |
| 		table_file has been implimented by the user, the table is also written
 | |
| 		to the specified file.
 | |
| 		This program exits on and error.  It returns a 1 on successful 
 | |
| 		completion
 | |
| */
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
| 	int i;
 | |
| 	char *config=0;
 | |
| 	char *option=0;
 | |
| 	char *header=0;
 | |
| 	FILE *fp;
 | |
| 	int tmpfile;
 | |
| 	char tmpfilename[TMPFILE_LEN];
 | |
| 	struct cmos_option_table *ct;
 | |
| 	struct cmos_entries *ce;
 | |
| 	struct cmos_enums *c_enums, *c_enums_start;
 | |
| 	struct cmos_checksum *cs;
 | |
| 	char line[INPUT_LINE_MAX];
 | |
| 	unsigned char uc;
 | |
| 	int entry_mode=0;
 | |
| 	int enum_mode=0;
 | |
| 	int checksum_mode=0;
 | |
| 	long ptr;
 | |
| 	int cnt;
 | |
| 	char *cptr;
 | |
| 	void *entry_start, *entry_end;
 | |
| 	int entries_length;
 | |
| 	int enum_length;
 | |
| 	int len;
 | |
| 	char buf[16];
 | |
| 
 | |
|         for(i=1;i<argc;i++) {
 | |
|                 if(argv[i][0]!='-') {
 | |
|                         display_usage(argv[0]);
 | |
|                 }
 | |
|                 switch(argv[i][1]) {
 | |
|                         case '-':       /* data is requested from a file */
 | |
|                                 switch(argv[i][2]) {
 | |
|                                         case 'c':  /* use a configuration file */
 | |
|                                                 if(strcmp(&argv[i][2],"config")) {
 | |
|                                                         display_usage(argv[0]);
 | |
|                                                 }
 | |
|                                                 config=argv[++i];
 | |
|                                                 break;
 | |
|                                         case 'o':  /* use a cmos definitions table file */
 | |
|                                                 if(strcmp(&argv[i][2],"option")) {
 | |
|                                                         display_usage(argv[0]);
 | |
|                                                 }
 | |
|                                                 option=argv[++i];
 | |
|                                                 break;
 | |
| 					case 'h': /* Output a header file */
 | |
| 						if (strcmp(&argv[i][2], "header") != 0) {
 | |
| 							display_usage(argv[0]);
 | |
| 						}
 | |
| 						header=argv[++i];
 | |
| 						break;
 | |
|                                         default:
 | |
|                                                 display_usage(argv[0]);
 | |
|                                                 break;
 | |
|                                 }
 | |
|                                 break;
 | |
| 
 | |
|                         default:
 | |
|                                 display_usage(argv[0]);
 | |
|                                 break;
 | |
|                 }
 | |
|         }
 | |
| 
 | |
| 
 | |
| 	/* Has the user specified a configuration file */
 | |
| 	if(config) {	/* if yes, open it */
 | |
| 		if((fp=fopen(config,"r"))==NULL){
 | |
| 			fprintf(stderr, "Error - Can not open config file %s\n",config);
 | |
| 			exit(1);  /* exit if it can not be opened */
 | |
| 		}
 | |
| 	}
 | |
| 	else {  /* no configuration file specified, so try the default */
 | |
| 		if((fp=fopen("cmos.layout","r"))==NULL){
 | |
| 			fprintf(stderr, "Error - Can not open cmos.layout\n");
 | |
| 			exit(1);  /* end of no configuration file is found */
 | |
| 		}
 | |
| 	}
 | |
| 	/* type cast a pointer, so we can us the structure */
 | |
| 	ct=(struct cmos_option_table*)cmos_table;
 | |
| 	/* start the table with the type signature */
 | |
| 	ct->tag = LB_TAG_CMOS_OPTION_TABLE;
 | |
| 	/* put in the header length */
 | |
| 	ct->header_length=sizeof(*ct);
 | |
| 
 | |
| 	/* Get the entry records */
 | |
| 	ce=(struct cmos_entries*)(cmos_table+(ct->header_length));
 | |
| 	cptr = (char*)ce;
 | |
| 	for(;;){  /* this section loops through the entry records */
 | |
| 		if(fgets(line,INPUT_LINE_MAX,fp)==NULL) 
 | |
| 			break; /* end if no more input */
 | |
| 		if(!entry_mode) {  /* skip input until the entries key word */
 | |
| 			if (strstr(line,"entries") != 0) {
 | |
| 				entry_mode=1;
 | |
| 				continue;
 | |
| 			}
 | |
| 		}
 | |
| 		else{  /* Test if we are done with entries and starting enumerations */
 | |
| 			if (strstr(line,"enumerations") != 0){
 | |
| 				entry_mode=0;
 | |
| 				enum_mode=1;
 | |
| 				break;
 | |
| 			}
 | |
| 			if (strstr(line, "checksums") != 0) {
 | |
| 				enum_mode=0;
 | |
| 				checksum_mode=1;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		/* skip commented and blank lines */
 | |
| 		if(line[0]=='#') continue;
 | |
| 		if(line[strspn(line," ")]=='\n') continue;
 | |
| 		/* scan in the input data */
 | |
| 		sscanf(line,"%d %d %c %d %s",
 | |
| 			&ce->bit,&ce->length,&uc,&ce->config_id,&ce->name[0]);
 | |
| 		ce->config=(int)uc;
 | |
| 		/* check bit and length ranges */
 | |
| 		if(ce->bit>(CMOS_IMAGE_BUFFER_SIZE*8)) {
 | |
|                         fprintf(stderr, "Error - bit is to big in line \n%s\n",line);
 | |
|                         exit(1);
 | |
|                 }
 | |
| 		if((ce->length>(MAX_VALUE_BYTE_LENGTH*8))&&(uc!='r')) {
 | |
| 			fprintf(stderr, "Error - Length is to long in line \n%s\n",line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if (!is_ident((char *)ce->name)) {
 | |
| 			fprintf(stderr, 
 | |
| 				"Error - Name %s is an invalid identifier in line\n %s\n", 
 | |
| 				ce->name, line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		/* put in the record type */
 | |
| 		ce->tag=LB_TAG_OPTION;
 | |
| 		/* calculate and save the record length */
 | |
| 		len=strlen((char *)ce->name)+1;
 | |
| 		/* make the record int aligned */
 | |
| 		if(len%4)
 | |
| 			len+=(4-(len%4));
 | |
| 		ce->size=sizeof(struct cmos_entries)-32+len;
 | |
| 		cptr = (char*)ce;
 | |
| 		cptr += ce->size;  /* increment to the next table position */
 | |
| 		ce = (struct cmos_entries*) cptr;
 | |
| 	}
 | |
| 
 | |
| 	/* put the length of the entries into the header section */
 | |
| 	entries_length = (cptr - (char *)&cmos_table) - ct->header_length;
 | |
| 
 | |
| 	/* compute the start of the enumerations section */
 | |
| 	entry_start = ((char*)&cmos_table) + ct->header_length;
 | |
| 	entry_end   = ((char *)entry_start) + entries_length;
 | |
| 	c_enums_start = c_enums = (struct cmos_enums*)entry_end;
 | |
|   	/* test for overlaps in the entry records */
 | |
| 	test_for_entry_overlaps(entry_start, entry_end);
 | |
| 
 | |
| 	for(;enum_mode;){ /* loop to build the enumerations section */
 | |
| 		if(fgets(line,INPUT_LINE_MAX,fp)==NULL) 
 | |
| 			break; /* go till end of input */
 | |
| 
 | |
| 		if (strstr(line, "checksums") != 0) {
 | |
| 			enum_mode=0;
 | |
| 			checksum_mode=1;
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		/* skip commented and blank lines */
 | |
| 		if(line[0]=='#') continue;
 | |
| 		if(line[strspn(line," ")]=='\n') continue;
 | |
| 
 | |
| 		/* scan in the data */
 | |
| 		for(ptr=0;(line[ptr]==' ')||(line[ptr]=='\t');ptr++);
 | |
| 		c_enums->config_id=strtol(&line[ptr],(char**)NULL,10);
 | |
| 		for(;(line[ptr]!=' ')&&(line[ptr]!='\t');ptr++);
 | |
| 		for(;(line[ptr]==' ')||(line[ptr]=='\t');ptr++);
 | |
| 		c_enums->value=strtol(&line[ptr],(char**)NULL,10);
 | |
| 		for(;(line[ptr]!=' ')&&(line[ptr]!='\t');ptr++);
 | |
| 		for(;(line[ptr]==' ')||(line[ptr]=='\t');ptr++);
 | |
| 		for(cnt=0;(line[ptr]!='\n')&&(cnt<31);ptr++,cnt++)
 | |
| 			c_enums->text[cnt]=line[ptr];
 | |
| 		c_enums->text[cnt]=0;
 | |
| 	
 | |
| 		/* make the record int aligned */
 | |
| 		cnt++;
 | |
| 		if(cnt%4)
 | |
| 			cnt+=4-(cnt%4);
 | |
| 		/* store the record length */
 | |
| 		c_enums->size=((char *)&c_enums->text[cnt]) - (char *)c_enums;
 | |
| 		/* store the record type */
 | |
| 		c_enums->tag=LB_TAG_OPTION_ENUM;
 | |
| 		/* increment to the next record */
 | |
| 		c_enums=(struct cmos_enums*)&c_enums->text[cnt];
 | |
| 	}
 | |
| 	/* save the enumerations length */
 | |
| 	enum_length= (char *)c_enums - (char *)c_enums_start;
 | |
| 	ct->size=ct->header_length+enum_length+entries_length;
 | |
| 
 | |
| 	/* Get the checksum records */
 | |
| 	cs=(struct cmos_checksum *)(cmos_table+(ct->size));
 | |
| 	cptr = (char*)cs;
 | |
| 	for(;checksum_mode;) { /* This section finds the checksums */
 | |
| 		char *ptr;
 | |
| 		if(fgets(line, INPUT_LINE_MAX,fp)==NULL)
 | |
| 			break; /* end if no more input */
 | |
| 
 | |
| 		/* skip commented and blank lines */
 | |
| 		if (line[0]=='#') continue;
 | |
| 		if (line[strspn(line, " ")]=='\n') continue;
 | |
| 		if (memcmp(line, "checksum", 8) != 0) continue;
 | |
| 
 | |
| 		/* get the information */
 | |
| 		ptr = line + 8;
 | |
| 		skip_spaces(line, &ptr);
 | |
| 		cs->range_start = get_number(line, &ptr, 10);
 | |
| 
 | |
| 		skip_spaces(line, &ptr);
 | |
| 		cs->range_end = get_number(line, &ptr, 10);
 | |
| 
 | |
| 		skip_spaces(line, &ptr);
 | |
| 		cs->location = get_number(line, &ptr, 10);
 | |
| 		
 | |
| 		/* Make certain there are spaces until the end of the line */
 | |
| 		skip_spaces(line, &ptr);
 | |
| 
 | |
| 		if ((cs->range_start%8) != 0) {
 | |
| 			fprintf(stderr, "Error - range start is not byte aligned in line\n%s\n", line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if (cs->range_start >= (CMOS_IMAGE_BUFFER_SIZE*8)) {
 | |
| 			fprintf(stderr, "Error - range start is to big in line\n%s\n", line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if ((cs->range_end%8) != 7) {
 | |
| 			fprintf(stderr, "Error - range end is not byte aligned in line\n%s\n", line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if ((cs->range_end) >= (CMOS_IMAGE_BUFFER_SIZE*8)) {
 | |
| 			fprintf(stderr, "Error - range end is to long in line\n%s\n", line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if ((cs->location%8) != 0) {
 | |
| 			fprintf(stderr, "Error - location is not byte aligned in line\n%s\n", line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if ((cs->location >= (CMOS_IMAGE_BUFFER_SIZE*8)) ||
 | |
| 			((cs->location + 16) > (CMOS_IMAGE_BUFFER_SIZE*8))) 
 | |
| 		{
 | |
| 			fprintf(stderr, "Error - location is to big in line\n%s\n", line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		/* And since we are not ready to be fully general purpose yet.. */
 | |
| 		if ((cs->range_start/8) != CONFIG_LB_CKS_RANGE_START) {
 | |
| 			fprintf(stderr, "Error - Range start((value in file #%d), which is  byte #%d)"
 | |
| 				"does not match the value of the config variable LB_CKS_RANGE_START)(%d) in line\n%s\n", 
 | |
| 				cs->range_start, cs->range_start/8, CONFIG_LB_CKS_RANGE_START, line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if ((cs->range_end/8) != CONFIG_LB_CKS_RANGE_END) {
 | |
| 			fprintf(stderr, "Error - Range end ((value in file #%d), which is  byte #%d)"
 | |
| 					"does not match the value of the config variable LB_CKS_RANGE_END (%d) in line\n%s\n", 
 | |
| 					cs->range_end, cs->range_end/8, 
 | |
| 					CONFIG_LB_CKS_RANGE_END, line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 		if ((cs->location/8) != CONFIG_LB_CKS_LOC) {
 | |
| 			fprintf(stderr, "Error - Location ((value in file #%d), which is  byte #%d) does not match LB_CKS_LOC (%d) in line\n%s\n", 
 | |
| 				cs->location, cs->location/8, CONFIG_LB_CKS_LOC, line);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 
 | |
| 		cs->tag = LB_TAG_OPTION_CHECKSUM;
 | |
| 		cs->size = sizeof(*cs);
 | |
| 		cs->type = CHECKSUM_PCBIOS;
 | |
| 		cptr = (char *)cs;
 | |
| 		cptr += cs->size;
 | |
| 		cs = (struct cmos_checksum *)cptr;
 | |
| 
 | |
| 	}
 | |
| 	ct->size += (cptr - (char *)(cmos_table + ct->size));
 | |
| 	fclose(fp);
 | |
| 
 | |
| 	/* See if we want to output a C source file */
 | |
| 	if(option) {
 | |
| 		int err=0;
 | |
| 		strncpy(tmpfilename, dirname(strdup(option)), TMPFILE_LEN);
 | |
| 	        strncat(tmpfilename, TMPFILE_TEMPLATE, TMPFILE_LEN);
 | |
| 		tmpfile = mkstemp(tmpfilename);
 | |
| 		if(tmpfile == -1) {
 | |
|                         perror("Error - Could not create temporary file");
 | |
|                         exit(1);
 | |
| 		}
 | |
| 
 | |
| 		if((fp=fdopen(tmpfile,"w"))==NULL){
 | |
| 			perror("Error - Could not open temporary file");
 | |
| 			unlink(tmpfilename);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 
 | |
| 		/* write the header */
 | |
|         	if(!fwrite("unsigned char option_table[] = {",1,32,fp)) {
 | |
|         	        perror("Error - Could not write image file");
 | |
|         	        fclose(fp);
 | |
| 			unlink(tmpfilename);
 | |
|         	        exit(1);
 | |
|         	}
 | |
| 		/* write the array values */
 | |
| 		for(i=0;i<(ct->size-1);i++) {
 | |
| 			if(!(i%10) && !err) err=!fwrite("\n\t",1,2,fp);
 | |
| 			sprintf(buf,"0x%02x,",cmos_table[i]);
 | |
| 			if(!err) err=!fwrite(buf,1,5,fp);
 | |
| 		}
 | |
| 		/* write the end */
 | |
| 		sprintf(buf,"0x%02x\n",cmos_table[i]);
 | |
| 		if(!err) err=!fwrite(buf,1,4,fp);
 | |
|         	if(!fwrite("};\n",1,3,fp)) {
 | |
|         	        perror("Error - Could not write image file");
 | |
|         	        fclose(fp);
 | |
| 			unlink(tmpfilename);
 | |
|         	        exit(1);
 | |
|         	}
 | |
| 
 | |
|         	fclose(fp);
 | |
| 		UNLINK_IF_NECESSARY(option);
 | |
| 		if (rename(tmpfilename, option)) {
 | |
| 			fprintf(stderr, "Error - Could not write %s: ", option);
 | |
| 			perror(NULL);
 | |
| 			unlink(tmpfilename);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* See if we also want to output a C header file */
 | |
| 	if (header) {
 | |
| 		struct cmos_option_table *hdr;
 | |
| 		struct lb_record *ptr, *end;
 | |
| 
 | |
| 		strncpy(tmpfilename, dirname(strdup(option)), TMPFILE_LEN);
 | |
| 	        strncat(tmpfilename, TMPFILE_TEMPLATE, TMPFILE_LEN);
 | |
| 		tmpfile = mkstemp(tmpfilename);
 | |
| 		if(tmpfile == -1) {
 | |
| 			perror("Error - Could not create temporary file");
 | |
| 			exit(1);
 | |
| 		}
 | |
| 
 | |
| 		fp = fdopen(tmpfile, "w");
 | |
| 		if (!fp) {
 | |
| 			perror("Error - Could not open temporary file");
 | |
| 			unlink(tmpfilename);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 
 | |
| 		/* Get the cmos table header */
 | |
| 		hdr = (struct cmos_option_table *)cmos_table;
 | |
| 		/* Walk through the entry records */
 | |
| 		ptr = (struct lb_record *)(cmos_table + hdr->header_length);
 | |
| 		end = (struct lb_record *)(cmos_table + hdr->size);
 | |
| 		for(;ptr < end; ptr = (struct lb_record *)(((char *)ptr) + ptr->size)) {
 | |
| 			if (ptr->tag != LB_TAG_OPTION) {
 | |
| 				continue;
 | |
| 			}
 | |
| 			ce = (struct cmos_entries *)ptr;
 | |
| 
 | |
| 			if (!is_ident((char *)ce->name)) {
 | |
| 				fprintf(stderr, "Invalid identifier: %s\n",
 | |
| 					ce->name);
 | |
| 				fclose(fp);
 | |
| 				unlink(tmpfilename);
 | |
| 				exit(1);
 | |
| 			}
 | |
| 			fprintf(fp, "#define CMOS_VSTART_%s %d\n",
 | |
| 				ce->name, ce->bit);
 | |
| 			fprintf(fp, "#define CMOS_VLEN_%s %d\n",
 | |
| 				ce->name, ce->length);
 | |
| 		}
 | |
| 		fclose(fp);
 | |
| 
 | |
| 		UNLINK_IF_NECESSARY(header);
 | |
| 		if (rename(tmpfilename, header)) {
 | |
| 			fprintf(stderr, "Error - Could not write %s: ", header);
 | |
| 			perror(NULL);
 | |
| 			unlink(tmpfilename);
 | |
| 			exit(1);
 | |
| 		}
 | |
| 	}
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| 
 |