switch-coreboot/lib/lar.c
Uwe Hermann 9b90a6f22b Fix a bunch of Doxygen warnings in v3 (trivial).
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
2008-10-23 18:55:01 +00:00

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;
}