135 lines
2.8 KiB
C
135 lines
2.8 KiB
C
/* This file is part of the coreboot project. */
|
|
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
static void usage(const char *s)
|
|
{
|
|
printf("insert firmware blobs:\n\t"
|
|
"%s <rom file> <fw1> <fw2> <fw1 offset> <fw2 offset>\n\n",
|
|
s);
|
|
printf("set addresses of firmware blobs only:\n\t"
|
|
"%s <rom file> <fw1 offset> <fw2 offset>\n\n",
|
|
s);
|
|
printf(
|
|
"offset can be (example is put it to 0x7ffa00 when ROM is 8MB):\n"
|
|
"- file offset: 0x7ffa00\n"
|
|
"- distance to the end of file: -0x600\n"
|
|
"- the address when ROM is mapped to the end of memory: "
|
|
"0xfffffa00\n");
|
|
exit(1);
|
|
}
|
|
|
|
static void FseekEnd(FILE *fp, long o)
|
|
{
|
|
if (fseek(fp, o, SEEK_END) != 0) {
|
|
puts("fseek() error!\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
static long negoffset(long a, long romsz)
|
|
{
|
|
if (a > 0) {
|
|
if (a & 0x80000000) /* the address in memory, and sizeof(long)
|
|
is 8 */
|
|
return a - 0x100000000;
|
|
else /* the file offset */
|
|
return a - romsz;
|
|
} else {
|
|
return a;
|
|
}
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
FILE *fp, *fw1, *fw2;
|
|
long offset1, offset2;
|
|
|
|
if (argc != 4 && argc != 6)
|
|
usage(argv[0]);
|
|
|
|
fp = fopen(argv[1], "rb+");
|
|
if (fp == NULL) {
|
|
puts("Error opening firmware image!");
|
|
exit(1);
|
|
}
|
|
|
|
if (argc == 6) {
|
|
fw1 = fopen(argv[2], "rb");
|
|
fw2 = fopen(argv[3], "rb");
|
|
offset1 = strtoul(argv[4], NULL, 0);
|
|
offset2 = strtoul(argv[5], NULL, 0);
|
|
|
|
if (fw1 == NULL || fw2 == NULL) {
|
|
puts("Error opening file!");
|
|
exit(1);
|
|
}
|
|
} else {
|
|
fw1 = NULL;
|
|
fw2 = NULL;
|
|
offset1 = strtoul(argv[2], NULL, 0);
|
|
offset2 = strtoul(argv[3], NULL, 0);
|
|
}
|
|
|
|
if ((offset1 & 0xff) || (offset2 & 0xff)) {
|
|
puts("The offsets must be aligned to 0x100");
|
|
exit(1);
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
offset1 = negoffset(offset1, romsz);
|
|
offset2 = negoffset(offset2, romsz);
|
|
|
|
/* write two offsets to $s-0x100 */
|
|
char offs[8];
|
|
long os;
|
|
os = 0x1000000 + offset1;
|
|
offs[0] = os >> 16;
|
|
offs[1] = os >> 8;
|
|
offs[2] = 0xff - offs[0];
|
|
offs[3] = 0xff - offs[1];
|
|
os = 0x1000000 + offset2;
|
|
offs[4] = os >> 16;
|
|
offs[5] = os >> 8;
|
|
offs[6] = 0xff - offs[4];
|
|
offs[7] = 0xff - offs[5];
|
|
for (size_t i = 0; i < 8; i++)
|
|
printf("%02hhx ", offs[i]);
|
|
|
|
puts("");
|
|
FseekEnd(fp, -0x100);
|
|
printf("writing to 0x%lx\n", ftell(fp));
|
|
fwrite(offs, 1, 8, fp);
|
|
|
|
if (argc == 6) {
|
|
/* write fw1 and fw2 */
|
|
char c;
|
|
FseekEnd(fp, offset1);
|
|
printf("writing to 0x%lx\n", ftell(fp));
|
|
while (fread(&c, 1, 1, fw1) == 1)
|
|
fwrite(&c, 1, 1, fp);
|
|
|
|
FseekEnd(fp, offset2);
|
|
printf("writing to 0x%lx\n", ftell(fp));
|
|
while (fread(&c, 1, 1, fw2) == 1)
|
|
fwrite(&c, 1, 1, fp);
|
|
|
|
fclose(fw1);
|
|
fclose(fw2);
|
|
}
|
|
|
|
fclose(fp);
|
|
return 0;
|
|
}
|