/* * This file is part of the coreboot project. * * Copyright (C) 2006-2007 coresystems GmbH * (Written by Stefan Reinauer 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 #include #include #include #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; }