mirror of
https://github.com/fail0verflow/switch-coreboot.git
synced 2025-05-04 01:39:18 -04:00
Signed-off-by: Uwe Hermann <uwe@hermann-uwe.de> Acked-by: Uwe Hermann <uwe@hermann-uwe.de> git-svn-id: svn://coreboot.org/repository/coreboot-v3@951 f3766cd6-281f-0410-b1cd-43a5c92072e9
309 lines
9.9 KiB
C
309 lines
9.9 KiB
C
/*
|
|
* This file is part of the coreboot project.
|
|
*
|
|
* Copyright (C) 2006-2007 coresystems GmbH
|
|
* (Written by Stefan Reinauer <stepan@coresystems.de> for coresystems GmbH)
|
|
* Copyright (C) 2007 Advanced Micro Devices, Inc.
|
|
*
|
|
* 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.
|
|
*
|
|
* 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., 51 Franklin St, Fifth Floor, Boston, MA, 02110-1301 USA
|
|
*/
|
|
|
|
#include <types.h>
|
|
#include <string.h>
|
|
#include <lar.h>
|
|
#include <console.h>
|
|
|
|
#ifndef CONFIG_BIG_ENDIAN
|
|
#define ntohl(x) ( ((x&0xff)<<24) | ((x&0xff00)<<8) | \
|
|
((x&0xff0000) >> 8) | ((x&0xff000000) >> 24) )
|
|
#else
|
|
#define ntohl(x) (x)
|
|
#endif
|
|
|
|
static const char * const algo_name[] = {
|
|
"none",
|
|
"lzma",
|
|
"nrv2b",
|
|
"zeroes",
|
|
/* invalid should always be the last entry. */
|
|
"invalid"
|
|
};
|
|
|
|
/**
|
|
* run_address is passed the address of a function taking no parameters and
|
|
* jumps to it, returning the result.
|
|
* @param f the address to call as a function.
|
|
* returns value returned by the function.
|
|
*/
|
|
|
|
int run_address(void *f)
|
|
{
|
|
int (*v) (void);
|
|
v = f;
|
|
return v();
|
|
}
|
|
|
|
/**
|
|
* Given a file name in the LAR , search for it, and fill out a return struct with the
|
|
* result.
|
|
* @param archive A descriptor for current archive. This is actually a mem_file type,
|
|
* which is a machine-dependent representation of the actual archive. In particular,
|
|
* things like u32 are void * in the mem_file struct.
|
|
* @param filename filename to find
|
|
* @param result pointer to mem_file struct which is filled in if the file is found
|
|
* returns 0 on success, -1 otherwise
|
|
*/
|
|
|
|
int find_file(const struct mem_file *archive, const char *filename, struct mem_file *result)
|
|
{
|
|
char *walk, *fullname;
|
|
struct lar_header *header;
|
|
|
|
printk(BIOS_INFO, "LAR: Attempting to open '%s'.\n", filename);
|
|
printk(BIOS_SPEW, "LAR: Start %p len 0x%x\n", archive->start,
|
|
archive->len);
|
|
|
|
/* Why check for sizeof(struct lar_header) + 1? The code below expects
|
|
* a filename to follow directly after the LAR header and will
|
|
* dereference the address directly after the header. However, if
|
|
* archive->len == sizeof(struct lar_header), printing the filename
|
|
* will dereference memory outside the archive. Without looking at the
|
|
* filename, the only thing we can check is that there is at least room
|
|
* for an empty filename (only the terminating \0).
|
|
*/
|
|
if (archive->len < sizeof(struct lar_header) + 1)
|
|
printk(BIOS_ERR, "Error: truncated archive (%d bytes); minimum"
|
|
" possible size is %d bytes\n",
|
|
archive->len, sizeof(struct lar_header) + 1);
|
|
|
|
/* Getting this for loop right is harder than it looks. All quantities
|
|
* are unsigned. The LAR stretches from (e.g.) 0xfff0000 for 0x100000
|
|
* bytes, i.e. to address ZERO.
|
|
* As a result, 'walk', can wrap to zero and keep going (this has been
|
|
* seen in practice). Recall that these are unsigned; walk can
|
|
* wrap to zero; so, question, when is walk less than any of these:
|
|
* archive->start
|
|
* Answer: once walk wraps to zero, it is < archive->start
|
|
* archive->start + archive->len
|
|
* archive->start + archive->len - 1
|
|
* Answer: In the case that archive->start + archive->len == 0, ALWAYS!
|
|
* A lot of expressions have been tried and all have been wrong.
|
|
* So what would work? Simple:
|
|
* test for walk < archive->start + archive->len - sizeof(lar_header)
|
|
* to cover the case that the archive does NOT occupy ALL of the
|
|
* top of memory and wrap to zero; RESIST the temptation to change
|
|
* that comparison to <= because if a header did terminate the
|
|
* archive, the filename (stored directly after the header) would
|
|
* be outside the archive and you'd get a nice NULL pointer for
|
|
* the filename
|
|
* and test for walk >= archive->start,
|
|
* to cover the case that you wrapped to zero.
|
|
* Unsigned pointer arithmetic that wraps to zero can be messy.
|
|
*/
|
|
for (walk = archive->start;
|
|
(walk < (char *)(archive->start + archive->len - sizeof(struct lar_header))) &&
|
|
(walk >= (char *)archive->start); walk += 16) {
|
|
if (strncmp(walk, MAGIC, sizeof(header->magic)) != 0)
|
|
continue;
|
|
|
|
header = (struct lar_header *)walk;
|
|
fullname = walk + sizeof(struct lar_header);
|
|
|
|
printk(BIOS_SPEW, "LAR: seen member %s@%p, size %d\n",
|
|
fullname, header, ntohl(header->len));
|
|
// FIXME: check checksum
|
|
|
|
if (strcmp(fullname, filename) == 0) {
|
|
printk(BIOS_SPEW, "LAR: CHECK %s @ %p\n", fullname, header);
|
|
result->start = walk + ntohl(header->offset);
|
|
result->len = ntohl(header->len);
|
|
result->reallen = ntohl(header->reallen);
|
|
result->compression = ntohl(header->compression);
|
|
result->entry = (void *)ntohl((u32)header->entry);
|
|
result->loadaddress = (void *)ntohl((u32)header->loadaddress);
|
|
printk(BIOS_SPEW,
|
|
"start %p len %d reallen %d compression %x entry %p loadaddress %p\n",
|
|
result->start, result->len, result->reallen,
|
|
result->compression, result->entry, result->loadaddress);
|
|
return 0;
|
|
}
|
|
/* skip file:
|
|
* The next iteration of the for loop will add 16 to walk, so
|
|
* we now add offset (from header start to data start) and len
|
|
* (member length), subtract 1 (to get the address of the last
|
|
* byte of the member) and round this down to the next 16 byte
|
|
* boundary.
|
|
* In the case of consecutive archive members with header-
|
|
* before-member structure, the next iteration of the loop will
|
|
* start exactly at the beginning of the next header.
|
|
*/
|
|
walk += (ntohl(header->offset) + ntohl(header->len) - 1)
|
|
& 0xfffffff0;
|
|
}
|
|
printk(BIOS_SPEW, "LAR: File not found!\n");
|
|
return 1;
|
|
}
|
|
|
|
int process_file(const struct mem_file *archive, void *where)
|
|
{
|
|
const char *algoname = algo_name[(archive->compression >= ALGO_INVALID)
|
|
? ALGO_INVALID : archive->compression];
|
|
printk(BIOS_SPEW, "LAR: Compression algorithm #%i (%s) used\n",
|
|
archive->compression, algoname);
|
|
switch (archive->compression) {
|
|
/* no compression */
|
|
case ALGO_NONE:
|
|
memcpy(where, archive->start, archive->len);
|
|
return 0;
|
|
#ifdef CONFIG_COMPRESSION_LZMA
|
|
/* lzma */
|
|
case ALGO_LZMA: {
|
|
unsigned long ulzma(unsigned char *src, unsigned char *dst);
|
|
ulzma(archive->start, where);
|
|
return 0;
|
|
}
|
|
#endif
|
|
#ifdef CONFIG_COMPRESSION_NRV2B
|
|
/* nrv2b */
|
|
case ALGO_NRV2B: {
|
|
unsigned long unrv2b(u8 *src, u8 *dst, unsigned long *ilen_p);
|
|
unsigned long tmp;
|
|
unrv2b(archive->start, where, &tmp);
|
|
return 0;
|
|
}
|
|
#endif
|
|
/* zeroes */
|
|
case ALGO_ZEROES:
|
|
memset(where, 0, archive->reallen);
|
|
return 0;
|
|
default:
|
|
printk(BIOS_INFO, "LAR: Compression algorithm #%i (%s) not"
|
|
" supported!\n", archive->compression, algoname);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Given a file name, search the LAR for all segments of it, and load them
|
|
* into memory, using the loadaddress pointer in the mem_file struct.
|
|
* @param archive A descriptor for current archive.
|
|
* @param filename filename to find
|
|
* returns entry on success, (void*)-1 otherwise
|
|
* FIXME: Look for a cmdline as well.
|
|
*/
|
|
void *load_file_segments(const struct mem_file *archive, const char *filename)
|
|
{
|
|
void *entry = NULL;
|
|
void *newentry;
|
|
int i;
|
|
char tmpname[64];
|
|
|
|
for(i = 0, entry = (void *)0; ;i++) {
|
|
sprintf(tmpname, "%s/segment%d", filename, i);
|
|
newentry = load_file(archive, tmpname);
|
|
if (newentry == (void *)-1)
|
|
break;
|
|
if (!entry)
|
|
entry = newentry;
|
|
}
|
|
if (!entry) {
|
|
printk(BIOS_INFO, "LAR: load_file_segments: Failed for %s\n", filename);
|
|
return (void *)-1;
|
|
}
|
|
printk(BIOS_SPEW, "LAR: load_file_segments: All loaded, entry %p\n", entry);
|
|
return entry;
|
|
}
|
|
|
|
/**
|
|
* Given a file name in the LAR , search for it, and load it into memory, using
|
|
* the loadaddress pointer in the mem_file struct.
|
|
* @param archive A descriptor for current archive.
|
|
* @param filename filename to find
|
|
* returns entry on success, (void*)-1 otherwise
|
|
*/
|
|
|
|
void *load_file(const struct mem_file *archive, const char *filename)
|
|
{
|
|
int ret;
|
|
struct mem_file result;
|
|
void *where;
|
|
void *entry;
|
|
|
|
ret = find_file(archive, filename, &result);
|
|
if (ret) {
|
|
printk(BIOS_INFO, "LAR: load_file: No such file '%s'\n",
|
|
filename);
|
|
return (void *)-1;
|
|
}
|
|
entry = result.entry;
|
|
where = result.loadaddress;
|
|
|
|
if (process_file(&result, where) == 0)
|
|
return entry;
|
|
|
|
return (void *)-1;
|
|
}
|
|
|
|
/**
|
|
* Given a file name in the LAR , search for it, and load it into memory,
|
|
* using the passed-in pointer as the address
|
|
* @param archive A descriptor for current archive.
|
|
* @param filename filename to find
|
|
* @param where pointer to where to load the data
|
|
* returns 0 on success, -1 otherwise
|
|
*/
|
|
int copy_file(const struct mem_file *archive, const char *filename, void *where)
|
|
{
|
|
int ret;
|
|
struct mem_file result;
|
|
|
|
ret = find_file(archive, filename, &result);
|
|
if (ret) {
|
|
printk(BIOS_INFO, "LAR: copy_file: No such file '%s'\n",
|
|
filename);
|
|
return 1;
|
|
}
|
|
|
|
return process_file(&result, where);
|
|
}
|
|
|
|
|
|
/**
|
|
* Given a file name in the LAR , search for it, and execute it in place.
|
|
* @param archive A descriptor for current archive.
|
|
* @param filename filename to find
|
|
* returns 0 on success, -1 otherwise
|
|
*/
|
|
int execute_in_place(const struct mem_file *archive, const char *filename)
|
|
{
|
|
struct mem_file result;
|
|
int ret;
|
|
void *where;
|
|
|
|
if (find_file(archive, filename, &result)) {
|
|
printk(BIOS_INFO, "LAR: Run file %s failed: No such file.\n",
|
|
filename);
|
|
return 1;
|
|
}
|
|
if (result.compression != 0) {
|
|
printk(BIOS_INFO, "LAR: Run file %s failed: Compressed file"
|
|
" not supported for in-place execution\n", filename);
|
|
return 1;
|
|
}
|
|
where = result.start + (u32)result.entry;
|
|
printk(BIOS_SPEW, "Entry point is %p\n", where);
|
|
ret = run_address(where);
|
|
printk(BIOS_SPEW, "run_file returns with %d\n", ret);
|
|
return ret;
|
|
}
|