mirror of
https://github.com/fail0verflow/switch-coreboot.git
synced 2025-05-04 01:39:18 -04:00
entry. Signed-off-by: Alex Beregszaszi <alex@rtfs.hu> Acked-by: Stefan Reinauer <stepan@coresystems.de> git-svn-id: svn://coreboot.org/repository/LinuxBIOSv3@492 f3766cd6-281f-0410-b1cd-43a5c92072e9
279 lines
8.7 KiB
C
279 lines
8.7 KiB
C
/*
|
|
* This file is part of the LinuxBIOS 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
|
|
|
|
/**
|
|
* run_address is passed the address of a function taking no parameters and
|
|
* jumps to it, returning the result.
|
|
* @param v 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 hte 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(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);
|
|
|
|
if (archive->len < sizeof(struct lar_header))
|
|
printk(BIOS_ERR, "Error: truncated archive (%d bytes); minimum possible size is %d bytes\n",
|
|
archive->len, sizeof(struct lar_header));
|
|
|
|
/* 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 - 1 to cover the case
|
|
* that the archive does NOT occupy ALL of the top of memory and
|
|
* wrap to zero;
|
|
* 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 (strcmp(walk, MAGIC) != 0)
|
|
continue;
|
|
|
|
header = (struct lar_header *)walk;
|
|
fullname = walk + sizeof(struct lar_header);
|
|
|
|
printk(BIOS_SPEW, "LAR: search for %s\n", fullname);
|
|
// 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(header->entry);
|
|
result->loadaddress = (void *)ntohl(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
|
|
walk += (ntohl(header->len) + ntohl(header->offset) -
|
|
1) & 0xfffffff0;
|
|
}
|
|
printk(BIOS_SPEW, "NO FILE FOUND\n");
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* 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 0 on success, -1 otherwise
|
|
*/
|
|
|
|
void *load_file(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;
|
|
printk(BIOS_SPEW, "LAR: Compression algorithm #%i used\n", result.compression);
|
|
/* no compression */
|
|
if (result.compression == 0) {
|
|
memcpy(where, result.start, result.len);
|
|
return entry;
|
|
}
|
|
#ifdef CONFIG_COMPRESSION_LZMA
|
|
/* lzma */
|
|
unsigned long ulzma(unsigned char *src, unsigned char *dst);
|
|
if (result.compression == 1) {
|
|
ulzma(result.start, where);
|
|
return entry;
|
|
}
|
|
#endif
|
|
#ifdef CONFIG_COMPRESSION_NRV2B
|
|
/* nrv2b */
|
|
unsigned long unrv2b(u8 *src, u8 *dst, unsigned long *ilen_p);
|
|
if (result.compression == 2) {
|
|
int tmp;
|
|
unrv2b(result.start, where, &tmp);
|
|
return entry;
|
|
}
|
|
#endif
|
|
printk(BIOS_INFO, "LAR: Compression algorithm #%i not supported!\n", result.compression);
|
|
return (void *)-1;
|
|
}
|
|
|
|
/* FIXME -- most of copy_file should be replaced by load_file */
|
|
/**
|
|
* 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(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;
|
|
}
|
|
|
|
printk(BIOS_SPEW, "LAR: Compression algorithm #%i used\n", result.compression);
|
|
/* no compression */
|
|
if (result.compression == 0) {
|
|
memcpy(where, result.start, result.len);
|
|
return 0;
|
|
}
|
|
#ifdef CONFIG_COMPRESSION_LZMA
|
|
/* lzma */
|
|
unsigned long ulzma(unsigned char *src, unsigned char *dst);
|
|
if (result.compression == 1) {
|
|
ulzma(result.start, where);
|
|
return 0;
|
|
}
|
|
#endif
|
|
#ifdef CONFIG_COMPRESSION_NRV2B
|
|
/* nrv2b */
|
|
unsigned long unrv2b(u8 *src, u8 *dst, unsigned long *ilen_p);
|
|
if (result.compression == 2) {
|
|
int tmp;
|
|
unrv2b(result.start, where, &tmp);
|
|
return 0;
|
|
}
|
|
#endif
|
|
printk(BIOS_INFO, "LAR: Compression algorithm #%i not supported!\n", result.compression);
|
|
return 1;
|
|
}
|
|
|
|
|
|
/**
|
|
* Given a file name in the LAR , search for it, and load it into memory,
|
|
* using the passed-in pointer as the address; jump to the file.
|
|
* If the passed-in pointer is (void *)-1, then execute the file in place.
|
|
* @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 run_file(struct mem_file *archive, const char *filename, void *where)
|
|
{
|
|
int (*v) (void);
|
|
struct mem_file result;
|
|
int ret;
|
|
|
|
if ((u32) where != 0xFFFFFFFF) {
|
|
if (copy_file(archive, filename, where)) {
|
|
printk(BIOS_INFO,
|
|
"LAR: Run file %s failed: No such file.\n",
|
|
filename);
|
|
return 1;
|
|
}
|
|
} else { /* XIP */
|
|
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;
|
|
}
|
|
printk(BIOS_SPEW, "where is %p\n", where);
|
|
v = where;
|
|
ret = v();
|
|
printk(BIOS_SPEW, "run_file returns with %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* 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(struct mem_file *archive, const char *filename)
|
|
{
|
|
return run_file(archive, filename, (void *) 0xFFFFFFFF);
|
|
}
|