summaryrefslogtreecommitdiff
path: root/util/kbc1126/kbc1126_ec_dump.c
blob: a96464a76f63ebb4a834ff69489da2db01fc4b98 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
/* This file is part of the coreboot 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; version 2 of the License.
 *
 * 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.
 */

#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

static void usage(const char *s)
{
	printf("%s <rom file>\n", s);
	exit(1);
}

static void FseekEnd(FILE *fp, long o)
{
	if (fseek(fp, o, SEEK_END) != 0) {
		puts("fseek() error!\n");
		exit(1);
	}
}

void dump_fw(FILE *dst, FILE *src, long offset)
{
	static unsigned char buf[65536];

	if (offset > 0)
		offset -= 0x1000000;

	printf("Dumping firmware at -0x%lx...", -offset);

	FseekEnd(src, offset);
	unsigned short len;
	unsigned short cksum;
	unsigned short _cksum = 0;
	fread(&len, 2, 1, src);
	fread(&cksum, 2, 1, src);
	fread(buf, len, 1, src);

	for (size_t i = 0; i < len; i++)
		_cksum += buf[i];

	if (_cksum == cksum) {
		puts("checksum ok");
	} else {
		puts("checksum fail");
		exit(1);
	}

	fwrite(&len, 2, 1, dst);
	fwrite(&cksum, 2, 1, dst);
	fwrite(buf, len, 1, dst);
}

int main(int argc, char *argv[])
{
	if (argc != 2)
		usage(argv[0]);

	FILE *fp = fopen(argv[1], "rb");

	if (fp == NULL) {
		puts("Error opening file!");
		exit(1);
	}

	char *basename = strrchr(argv[1], '/');
	if (basename == NULL)
		basename = argv[1];
	else
		basename = basename + 1;

	int len = strlen(basename);
	char fn1[len + 5], fn2[len + 5];
	strcpy(fn1, basename);
	strcpy(fn2, basename);
	strcat(fn1, ".fw1");
	strcat(fn2, ".fw2");

	FILE *fw1 = fopen(fn1, "wb");
	FILE *fw2 = fopen(fn2, "wb");

	long romsz;
	FseekEnd(fp, -1);
	romsz = ftell(fp) + 1;
	printf("size of %s: 0x%lx\n", argv[1], romsz);

	if (romsz & 0xff) {
		puts("The ROM size must be multiple of 0x100");
		exit(1);
	}

	/* read offset of fw1 and fw2 */
	unsigned char offs[8];
	FseekEnd(fp, -0x100);
	fread(offs, 8, 1, fp);

	assert(offs[0] + offs[2] == 0xff);
	assert(offs[1] + offs[3] == 0xff);
	assert(offs[4] + offs[6] == 0xff);
	assert(offs[5] + offs[7] == 0xff);
	long offw1 = (offs[0] << 16) | (offs[1] << 8);
	long offw2 = (offs[4] << 16) | (offs[5] << 8);

	dump_fw(fw1, fp, offw1);
	dump_fw(fw2, fp, offw2);

	fclose(fp);
	fclose(fw1);
	fclose(fw2);
	return 0;
}