mirror of
https://gitlab.com/qemu-project/openbios.git
synced 2024-02-13 08:34:06 +08:00
Signed-off-by: Mark Cave-Ayland <mark.cave-ayland@siriusit.co.uk> git-svn-id: svn://coreboot.org/openbios/trunk/openbios-devel@806 f158a5a8-5612-0410-a976-696ce0be7e32
241 lines
4.9 KiB
C
241 lines
4.9 KiB
C
/*
|
|
*
|
|
*/
|
|
#undef BOOTSTRAP
|
|
#include "config.h"
|
|
#include "libopenbios/bindings.h"
|
|
#include "arch/common/nvram.h"
|
|
#include "drivers/drivers.h"
|
|
#include "libc/diskio.h"
|
|
#include "libc/vsprintf.h"
|
|
#include "libopenbios/sys_info.h"
|
|
#include "libopenbios/elf_load.h"
|
|
#include "libopenbios/aout_load.h"
|
|
#include "libopenbios/fcode_load.h"
|
|
#include "libopenbios/forth_load.h"
|
|
#include "openprom.h"
|
|
#include "boot.h"
|
|
|
|
struct sys_info sys_info;
|
|
uint32_t kernel_image;
|
|
uint32_t kernel_size;
|
|
uint32_t qemu_cmdline;
|
|
uint32_t cmdline_size;
|
|
char boot_device;
|
|
static const void *romvec;
|
|
static int (*entry)(const void *romvec_ptr, int p2, int p3, int p4, int p5);
|
|
|
|
|
|
static int try_path(const char *path, char *param)
|
|
{
|
|
void *boot_notes = NULL;
|
|
ucell valid;
|
|
ihandle_t dev;
|
|
|
|
push_str(path);
|
|
fword("pathres-resolve-aliases");
|
|
bootpath = pop_fstr_copy();
|
|
printk("Trying %s (%s)\n", path, bootpath);
|
|
|
|
/* Open device used by this path */
|
|
dev = open_dev(path);
|
|
|
|
#ifdef CONFIG_LOADER_ELF
|
|
/* ELF Boot loader */
|
|
elf_load(&sys_info, path, param, &boot_notes);
|
|
feval("state-valid @");
|
|
valid = POP();
|
|
if (valid)
|
|
goto start_image;
|
|
#endif
|
|
|
|
/* Linux loader (not using Forth) */
|
|
linux_load(&sys_info, path, param);
|
|
|
|
#ifdef CONFIG_LOADER_AOUT
|
|
/* a.out loader */
|
|
aout_load(&sys_info, dev);
|
|
feval("state-valid @");
|
|
valid = POP();
|
|
if (valid)
|
|
goto start_image;
|
|
#endif
|
|
|
|
#ifdef CONFIG_LOADER_FCODE
|
|
/* Fcode loader */
|
|
fcode_load(dev);
|
|
feval("state-valid @");
|
|
valid = POP();
|
|
if (valid)
|
|
goto start_image;
|
|
#endif
|
|
|
|
#ifdef CONFIG_LOADER_FORTH
|
|
/* Forth loader */
|
|
forth_load(dev);
|
|
feval("state-valid @");
|
|
valid = POP();
|
|
if (valid)
|
|
goto start_image;
|
|
#endif
|
|
|
|
close_dev(dev);
|
|
|
|
return 0;
|
|
|
|
|
|
start_image:
|
|
go();
|
|
return -1;
|
|
}
|
|
|
|
|
|
void go(void)
|
|
{
|
|
ucell address, type, size;
|
|
int image_retval = 0;
|
|
|
|
/* Get the entry point and the type (see forth/debugging/client.fs) */
|
|
feval("saved-program-state >sps.entry @");
|
|
address = POP();
|
|
feval("saved-program-state >sps.file-type @");
|
|
type = POP();
|
|
feval("saved-program-state >sps.file-size @");
|
|
size = POP();
|
|
|
|
printk("\nJumping to entry point " FMT_ucellx " for type " FMT_ucellx "...\n", address, type);
|
|
|
|
switch (type) {
|
|
case 0x0:
|
|
/* Start ELF boot image */
|
|
entry = (void *) address;
|
|
image_retval = entry(romvec, 0, 0, 0, 0);
|
|
|
|
break;
|
|
|
|
case 0x1:
|
|
/* Start ELF image */
|
|
entry = (void *) address;
|
|
image_retval = entry(romvec, 0, 0, 0, 0);
|
|
|
|
break;
|
|
|
|
case 0x5:
|
|
/* Start a.out image */
|
|
entry = (void *) address;
|
|
image_retval = entry(romvec, 0, 0, 0, 0);
|
|
|
|
break;
|
|
|
|
case 0x10:
|
|
/* Start Fcode image */
|
|
printk("Evaluating FCode...\n");
|
|
PUSH(address);
|
|
PUSH(1);
|
|
fword("byte-load");
|
|
image_retval = 0;
|
|
break;
|
|
|
|
case 0x11:
|
|
/* Start Forth image */
|
|
PUSH(address);
|
|
PUSH(size);
|
|
fword("eval2");
|
|
image_retval = 0;
|
|
break;
|
|
}
|
|
|
|
printk("Image returned with return value %#x\n", image_retval);
|
|
}
|
|
|
|
|
|
void boot(void)
|
|
{
|
|
char *path = pop_fstr_copy(), *param, altpath[256];
|
|
const char *oldpath = path;
|
|
int unit = 0;
|
|
int result;
|
|
|
|
if(!path) {
|
|
/* No path specified, so grab defaults from firmware */
|
|
switch (boot_device) {
|
|
case 'a':
|
|
path = strdup("/obio/SUNW,fdtwo");
|
|
oldpath = "fd()";
|
|
break;
|
|
case 'c':
|
|
path = strdup("disk");
|
|
oldpath = "sd(0,0,0):d";
|
|
break;
|
|
default:
|
|
case 'd':
|
|
path = strdup("cdrom");
|
|
// FIXME: hardcoding this looks almost definitely wrong.
|
|
// With sd(0,2,0):b we get to see the solaris kernel though
|
|
//oldpath = "sd(0,2,0):d";
|
|
oldpath = "sd(0,2,0):b";
|
|
unit = 2;
|
|
break;
|
|
case 'n':
|
|
path = strdup("net");
|
|
oldpath = "le()";
|
|
break;
|
|
}
|
|
}
|
|
|
|
obp_arg.boot_dev_ctrl = 0;
|
|
obp_arg.boot_dev_unit = unit;
|
|
obp_arg.dev_partition = 0;
|
|
obp_arg.boot_dev[0] = oldpath[0];
|
|
obp_arg.boot_dev[1] = oldpath[1];
|
|
obp_arg.argv[0] = oldpath;
|
|
obp_arg.argv[1] = (void *)(long)qemu_cmdline;
|
|
|
|
param = strchr(path, ' ');
|
|
if(param) {
|
|
*param = '\0';
|
|
param++;
|
|
} else if (cmdline_size) {
|
|
param = (char *)qemu_cmdline;
|
|
} else {
|
|
push_str("boot-args");
|
|
push_str("/options");
|
|
fword("(find-dev)");
|
|
POP();
|
|
fword("get-package-property");
|
|
POP();
|
|
param = pop_fstr_copy();
|
|
obp_arg.argv[1] = param;
|
|
}
|
|
|
|
romvec = init_openprom();
|
|
|
|
if (kernel_size) {
|
|
printk("[sparc] Kernel already loaded\n");
|
|
entry = (void *) kernel_image;
|
|
entry(romvec, 0, 0, 0, 0);
|
|
}
|
|
|
|
printk("[sparc] Booting file '%s' ", path);
|
|
if (param)
|
|
printk("with parameters '%s'\n", param);
|
|
else
|
|
printk("without parameters.\n");
|
|
|
|
result = try_path(path, param);
|
|
if (!result) {
|
|
push_str(path);
|
|
PUSH(':');
|
|
fword("left-split");
|
|
snprintf(altpath, sizeof(altpath), "%s:d", pop_fstr_copy());
|
|
POP();
|
|
POP();
|
|
|
|
try_path(altpath, param);
|
|
}
|
|
|
|
printk("Unsupported image format\n");
|
|
|
|
free(path);
|
|
}
|