blob: e40385a70b9c3af4e12473608cbb18339f65b613 [file] [log] [blame]
/*
* Creation Date: <2004/08/28 18:38:22 greg>
* Time-stamp: <2004/08/28 18:38:22 greg>
*
* <init.c>
*
* Initialization for qemu
*
* Copyright (C) 2004 Greg Watson
* Copyright (C) 2005 Stefan Reinauer
*
* based on mol/init.c:
*
* Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004 Samuel & David Rydh
* (samuel@ibrium.se, dary@lindesign.se)
*
* 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
*
*/
#include "config.h"
#include "libopenbios/openbios.h"
#include "libopenbios/bindings.h"
#include "libopenbios/console.h"
#include "drivers/pci.h"
#include "arch/common/nvram.h"
#include "drivers/drivers.h"
#include "qemu/qemu.h"
#include "libopenbios/ofmem.h"
#include "openbios-version.h"
#include "libc/byteorder.h"
#include "libc/vsprintf.h"
#define NO_QEMU_PROTOS
#include "arch/common/fw_cfg.h"
#include "arch/ppc/processor.h"
#include "context.h"
#define UUID_FMT "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x"
struct cpudef {
unsigned int iu_version;
const char *name;
int icache_size, dcache_size;
int icache_sets, dcache_sets;
int icache_block_size, dcache_block_size;
int tlb_sets, tlb_size;
void (*initfn)(const struct cpudef *cpu);
};
static uint16_t machine_id = 0;
extern void unexpected_excep(int vector);
void
unexpected_excep(int vector)
{
printk("openbios panic: Unexpected exception %x\n", vector);
for (;;) {
}
}
extern void __divide_error(void);
void
__divide_error(void)
{
return;
}
enum {
ARCH_PREP = 0,
ARCH_MAC99,
ARCH_HEATHROW,
ARCH_MAC99_U3,
};
int is_apple(void)
{
return is_oldworld() || is_newworld();
}
int is_oldworld(void)
{
return machine_id == ARCH_HEATHROW;
}
int is_newworld(void)
{
return (machine_id == ARCH_MAC99) ||
(machine_id == ARCH_MAC99_U3);
}
#define CORE99_VIA_CONFIG_CUDA 0x0
#define CORE99_VIA_CONFIG_PMU 0x1
#define CORE99_VIA_CONFIG_PMU_ADB 0x2
int has_pmu(void)
{
uint32_t via_config = fw_cfg_read_i32(FW_CFG_PPC_VIACONFIG);
return (via_config != CORE99_VIA_CONFIG_CUDA);
}
int has_adb(void)
{
uint32_t via_config = fw_cfg_read_i32(FW_CFG_PPC_VIACONFIG);
return (via_config == CORE99_VIA_CONFIG_CUDA ||
via_config == CORE99_VIA_CONFIG_PMU_ADB);
}
static const pci_arch_t known_arch[] = {
[ARCH_PREP] = {
.name = "PREP",
.vendor_id = PCI_VENDOR_ID_MOTOROLA,
.device_id = PCI_DEVICE_ID_MOTOROLA_RAVEN,
.cfg_addr = 0x80000cf8,
.cfg_data = 0x80000cfc,
.cfg_base = 0x80000000,
.cfg_len = 0x00100000,
.host_pci_base = 0xc0000000,
.pci_mem_base = 0x100000, /* avoid VGA at 0xa0000 */
.mem_len = 0x10000000,
.io_base = 0x80000000,
.io_len = 0x00010000,
.host_ranges = {
{ .type = IO_SPACE, .parentaddr = 0, .childaddr = 0x80000000, .len = 0x00010000 },
{ .type = MEMORY_SPACE_32, .parentaddr = 0, .childaddr = 0xc0100000, .len = 0x10000000 },
{ .type = 0, .parentaddr = 0, .childaddr = 0, .len = 0 }
},
.irqs = { 15, 15, 15, 15 }
},
[ARCH_MAC99] = {
.name = "MAC99",
.vendor_id = PCI_VENDOR_ID_APPLE,
.device_id = PCI_DEVICE_ID_APPLE_UNI_N_PCI,
.cfg_addr = 0xf2800000,
.cfg_data = 0xf2c00000,
.cfg_base = 0xf2000000,
.cfg_len = 0x02000000,
.host_pci_base = 0x0,
.pci_mem_base = 0x80000000,
.mem_len = 0x10000000,
.io_base = 0xf2000000,
.io_len = 0x00800000,
.host_ranges = {
{ .type = IO_SPACE, .parentaddr = 0, .childaddr = 0xf2000000, .len = 0x00800000 },
{ .type = MEMORY_SPACE_32, .parentaddr = 0x80000000, .childaddr = 0x80000000, .len = 0x10000000 },
{ .type = 0, .parentaddr = 0, .childaddr = 0, .len = 0 }
},
.irqs = { 0x1b, 0x1c, 0x1d, 0x1e }
},
[ARCH_MAC99_U3] = {
.name = "MAC99_U3",
.vendor_id = PCI_VENDOR_ID_APPLE,
.device_id = PCI_DEVICE_ID_APPLE_U3_AGP,
.cfg_addr = 0xf0800000,
.cfg_data = 0xf0c00000,
.cfg_base = 0xf0000000,
.cfg_len = 0x02000000,
.host_pci_base = 0x0,
.pci_mem_base = 0x80000000,
.mem_len = 0x10000000,
.io_base = 0xf2000000,
.io_len = 0x00800000,
.host_ranges = {
{ .type = IO_SPACE, .parentaddr = 0, .childaddr = 0xf2000000, .len = 0x00800000 },
{ .type = MEMORY_SPACE_32, .parentaddr = 0x80000000, .childaddr = 0x80000000, .len = 0x10000000 },
{ .type = 0, .parentaddr = 0, .childaddr = 0, .len = 0 }
},
.irqs = { 0x1b, 0x1c, 0x1d, 0x1e }
},
[ARCH_HEATHROW] = {
.name = "HEATHROW",
.vendor_id = PCI_VENDOR_ID_MOTOROLA,
.device_id = PCI_DEVICE_ID_MOTOROLA_MPC106,
.cfg_addr = 0xfec00000,
.cfg_data = 0xfee00000,
.cfg_base = 0x80000000,
.cfg_len = 0x7f000000,
.host_pci_base = 0x0,
.pci_mem_base = 0x80000000,
.mem_len = 0x10000000,
.io_base = 0xfe000000,
.io_len = 0x00800000,
.host_ranges = {
{ .type = IO_SPACE, .parentaddr = 0, .childaddr = 0xfe000000, .len = 0x00800000 },
{ .type = MEMORY_SPACE_32, .parentaddr = 0, .childaddr = 0xfd000000, .len = 0x01000000 },
{ .type = MEMORY_SPACE_32, .parentaddr = 0x80000000, .childaddr = 0x80000000, .len = 0x10000000 },
{ .type = 0, .parentaddr = 0, .childaddr = 0, .len = 0 }
},
.irqs = { 21, 22, 23, 24 }
},
};
unsigned long isa_io_base;
extern struct _console_ops mac_console_ops, prep_console_ops;
void
entry(void)
{
uint32_t temp = 0;
char buf[5];
arch = &known_arch[ARCH_HEATHROW];
fw_cfg_init();
fw_cfg_read(FW_CFG_SIGNATURE, buf, 4);
buf[4] = '\0';
if (strncmp(buf, "QEMU", 4) == 0) {
temp = fw_cfg_read_i32(FW_CFG_ID);
if (temp == 1) {
machine_id = fw_cfg_read_i16(FW_CFG_MACHINE_ID);
arch = &known_arch[machine_id];
}
}
isa_io_base = arch->io_base;
#ifdef CONFIG_DEBUG_CONSOLE
if (is_apple()) {
init_console(mac_console_ops);
} else {
init_console(prep_console_ops);
}
#endif
if (temp != 1) {
printk("Incompatible configuration device version, freezing\n");
for (;;) {
}
}
ofmem_init();
initialize_forth();
/* won't return */
printk("of_startup returned!\n");
for (;;) {
}
}
/* -- phys.lo ... phys.hi */
static void
push_physaddr(phys_addr_t value)
{
PUSH(value);
#ifdef CONFIG_PPC64
PUSH(value >> 32);
#endif
}
/* From drivers/timer.c */
extern unsigned long timer_freq;
static void
cpu_generic_init(const struct cpudef *cpu)
{
push_str("/cpus");
fword("find-device");
fword("new-device");
push_str(cpu->name);
fword("device-name");
push_str("cpu");
fword("device-type");
PUSH(mfpvr());
fword("encode-int");
push_str("cpu-version");
fword("property");
PUSH(cpu->dcache_size);
fword("encode-int");
push_str("d-cache-size");
fword("property");
PUSH(cpu->icache_size);
fword("encode-int");
push_str("i-cache-size");
fword("property");
PUSH(cpu->dcache_sets);
fword("encode-int");
push_str("d-cache-sets");
fword("property");
PUSH(cpu->icache_sets);
fword("encode-int");
push_str("i-cache-sets");
fword("property");
PUSH(cpu->dcache_block_size);
fword("encode-int");
push_str("d-cache-block-size");
fword("property");
PUSH(cpu->icache_block_size);
fword("encode-int");
push_str("i-cache-block-size");
fword("property");
PUSH(cpu->tlb_sets);
fword("encode-int");
push_str("tlb-sets");
fword("property");
PUSH(cpu->tlb_size);
fword("encode-int");
push_str("tlb-size");
fword("property");
timer_freq = fw_cfg_read_i32(FW_CFG_PPC_TBFREQ);
PUSH(timer_freq);
fword("encode-int");
push_str("timebase-frequency");
fword("property");
PUSH(fw_cfg_read_i32(FW_CFG_PPC_CLOCKFREQ));
fword("encode-int");
push_str("clock-frequency");
fword("property");
PUSH(fw_cfg_read_i32(FW_CFG_PPC_BUSFREQ));
fword("encode-int");
push_str("bus-frequency");
fword("property");
push_str("running");
fword("encode-string");
push_str("state");
fword("property");
PUSH(0x20);
fword("encode-int");
push_str("reservation-granule-size");
fword("property");
}
static void
cpu_add_pir_property(void)
{
unsigned long pir;
asm("mfspr %0, 1023\n"
: "=r"(pir) :);
PUSH(pir);
fword("encode-int");
push_str("reg");
fword("property");
}
static void
cpu_604_init(const struct cpudef *cpu)
{
cpu_generic_init(cpu);
cpu_add_pir_property();
fword("finish-device");
}
static void
cpu_750_init(const struct cpudef *cpu)
{
cpu_generic_init(cpu);
PUSH(0);
fword("encode-int");
push_str("reg");
fword("property");
fword("finish-device");
}
static void
cpu_g4_init(const struct cpudef *cpu)
{
cpu_generic_init(cpu);
cpu_add_pir_property();
fword("finish-device");
}
#ifdef CONFIG_PPC_64BITSUPPORT
/* In order to get 64 bit aware handlers that rescue all our
GPRs from getting truncated to 32 bits, we need to patch the
existing handlers so they jump to our 64 bit aware ones. */
static void
ppc64_patch_handlers(void)
{
uint32_t *dsi = (uint32_t *)0x300UL;
uint32_t *isi = (uint32_t *)0x400UL;
// Patch the first DSI handler instruction to: ba 0x2000
*dsi = 0x48002002;
// Patch the first ISI handler instruction to: ba 0x2200
*isi = 0x48002202;
// Invalidate the cache lines
asm ("icbi 0, %0" : : "r"(dsi));
asm ("icbi 0, %0" : : "r"(isi));
}
#endif
static void
cpu_970_init(const struct cpudef *cpu)
{
cpu_generic_init(cpu);
PUSH(0);
fword("encode-int");
push_str("reg");
fword("property");
PUSH(0);
PUSH(0);
fword("encode-bytes");
push_str("64-bit");
fword("property");
fword("finish-device");
#ifdef CONFIG_PPC_64BITSUPPORT
/* The 970 is a PPC64 CPU, so we need to activate
* 64bit aware interrupt handlers */
ppc64_patch_handlers();
#endif
/* The 970 also implements the HIOR which we need to set to 0 */
mtspr(S_HIOR, 0);
}
static const struct cpudef ppc_defs[] = {
{
.iu_version = 0x00040000,
.name = "PowerPC,604",
.icache_size = 0x4000,
.dcache_size = 0x4000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_604_init,
},
{ // XXX find out real values
.iu_version = 0x00090000,
.name = "PowerPC,604e",
.icache_size = 0x4000,
.dcache_size = 0x4000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_604_init,
},
{ // XXX find out real values
.iu_version = 0x000a0000,
.name = "PowerPC,604r",
.icache_size = 0x4000,
.dcache_size = 0x4000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_604_init,
},
{ // XXX find out real values
.iu_version = 0x80040000,
.name = "PowerPC,MPC86xx",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_750_init,
},
{
.iu_version = 0x000080000,
.name = "PowerPC,750",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_750_init,
},
{ // XXX find out real values
.iu_version = 0x10080000,
.name = "PowerPC,750",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_750_init,
},
{ // XXX find out real values
.iu_version = 0x70000000,
.name = "PowerPC,750",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_750_init,
},
{ // XXX find out real values
.iu_version = 0x70020000,
.name = "PowerPC,750",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_750_init,
},
{ // XXX find out real values
.iu_version = 0x800c0000,
.name = "PowerPC,74xx",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_750_init,
},
{
.iu_version = 0x0000c0000,
.name = "PowerPC,G4",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_g4_init,
},
{
.iu_version = 0x80000000,
.name = "PowerPC,G4",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_g4_init,
},
{
.iu_version = 0x80010000,
.name = "PowerPC,G4",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_g4_init,
},
{
.iu_version = 0x80020000,
.name = "PowerPC,G4",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_g4_init,
},
{
.iu_version = 0x80030000,
.name = "PowerPC,G4",
.icache_size = 0x8000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x20,
.dcache_block_size = 0x20,
.tlb_sets = 0x40,
.tlb_size = 0x80,
.initfn = cpu_g4_init,
},
{
.iu_version = 0x00390000,
.name = "PowerPC,970",
.icache_size = 0x10000,
.dcache_size = 0x8000,
.icache_sets = 0x200,
.dcache_sets = 0x80,
.icache_block_size = 0x80,
.dcache_block_size = 0x80,
.tlb_sets = 0x100,
.tlb_size = 0x1000,
.initfn = cpu_970_init,
},
{ // XXX find out real values
.iu_version = 0x003C0000,
.name = "PowerPC,970FX",
.icache_size = 0x10000,
.dcache_size = 0x8000,
.icache_sets = 0x80,
.dcache_sets = 0x80,
.icache_block_size = 0x80,
.dcache_block_size = 0x80,
.tlb_sets = 0x100,
.tlb_size = 0x1000,
.initfn = cpu_970_init,
},
{
.iu_version = 0x00350000,
.name = "PowerPC,POWER4",
.icache_size = 0x10000,
.dcache_size = 0x8000,
.icache_sets = 0x100,
.dcache_sets = 0x40,
.icache_block_size = 0x80,
.dcache_block_size = 0x80,
.tlb_sets = 0x100,
.tlb_size = 0x1000,
.initfn = cpu_970_init,
},
};
static const struct cpudef *
id_cpu(void)
{
unsigned int iu_version;
unsigned int i;
iu_version = mfpvr() & 0xffff0000;
for (i = 0; i < sizeof(ppc_defs) / sizeof(struct cpudef); i++) {
if (iu_version == ppc_defs[i].iu_version)
return &ppc_defs[i];
}
printk("Unknown cpu (pvr %x), freezing!\n", iu_version);
for (;;) {
}
}
static void arch_go(void);
static void
arch_go(void)
{
phandle_t ph;
xt_t xt;
/* Insert copyright property for MacOS 9 and below */
if (find_dev("/rom/macos")) {
fword("insert-copyright-property");
}
/* PReP machines expect a standard VGA console, so disable
VBE extensions just before we transfer control */
if (!is_apple()) {
ph = dt_iterate_type(find_dev("/"), "display");
if (ph != 0) {
xt = find_package_method("vbe-deinit", ph);
if (xt != 0) {
PUSH(xt);
fword("execute");
}
}
}
}
static void kvm_of_init(void)
{
char hypercall[4 * 4];
uint32_t *hc32;
/* Don't expose /hypervisor when not in KVM */
if (!fw_cfg_read_i32(FW_CFG_PPC_IS_KVM))
return;
push_str("/");
fword("find-device");
fword("new-device");
push_str("hypervisor");
fword("device-name");
push_str("hypervisor");
fword("device-type");
/* compatible */
push_str("linux,kvm");
fword("encode-string");
push_str("epapr,hypervisor-0.2");
fword("encode-string");
fword("encode+");
push_str("compatible");
fword("property");
/* Tell the guest about the hypercall instructions */
fw_cfg_read(FW_CFG_PPC_KVM_HC, hypercall, 4 * 4);
hc32 = (uint32_t*)hypercall;
PUSH(hc32[0]);
fword("encode-int");
PUSH(hc32[1]);
fword("encode-int");
fword("encode+");
PUSH(hc32[2]);
fword("encode-int");
fword("encode+");
PUSH(hc32[3]);
fword("encode-int");
fword("encode+");
push_str("hcall-instructions");
fword("property");
/* ePAPR requires us to provide a unique guest id */
PUSH(fw_cfg_read_i32(FW_CFG_PPC_KVM_PID));
fword("encode-int");
push_str("guest-id");
fword("property");
/* ePAPR requires us to provide a guest name */
push_str("KVM guest");
fword("encode-string");
push_str("guest-name");
fword("property");
fword("finish-device");
}
/*
* filll ( addr bytes quad -- )
*/
static void ffilll(void)
{
const u32 longval = POP();
u32 bytes = POP();
u32 *laddr = (u32 *)cell2pointer(POP());
u32 len;
for (len = 0; len < bytes / sizeof(u32); len++) {
*laddr++ = longval;
}
}
/*
* adler32 ( adler buf len -- checksum )
*
* Adapted from Mark Adler's original implementation (zlib license)
*
* Both OS 9 and BootX require this word for payload validation.
*/
#define DO1(buf,i) {s1 += buf[i]; s2 += s1;}
#define DO2(buf,i) DO1(buf,i); DO1(buf,i+1);
#define DO4(buf,i) DO2(buf,i); DO2(buf,i+2);
#define DO8(buf,i) DO4(buf,i); DO4(buf,i+4);
#define DO16(buf) DO8(buf,0); DO8(buf,8);
static void adler32(void)
{
uint32_t len = (uint32_t)POP();
char *buf = (char *)POP();
uint32_t adler = (uint32_t)POP();
if (buf == NULL) {
RET(-1);
}
uint32_t base = 65521;
uint32_t nmax = 5552;
uint32_t s1 = adler & 0xffff;
uint32_t s2 = (adler >> 16) & 0xffff;
uint32_t k;
while (len > 0) {
k = (len < nmax ? len : nmax);
len -= k;
while (k >= 16) {
DO16(buf);
buf += 16;
k -= 16;
}
if (k != 0) {
do {
s1 += *buf++;
s2 += s1;
} while (--k);
}
s1 %= base;
s2 %= base;
}
RET(s2 << 16 | s1);
}
/* ( size -- virt ) */
static void
dma_alloc(void)
{
ucell size = POP();
ucell addr;
int ret;
ret = ofmem_posix_memalign((void *)&addr, size, PAGE_SIZE);
if (ret) {
PUSH(0);
} else {
PUSH(addr);
}
}
/* ( virt size cacheable? -- devaddr ) */
static void
dma_map_in(void)
{
POP();
POP();
ucell va = POP();
if (is_apple()) {
PUSH(va);
} else {
/* PReP */
PUSH(va + 0x80000000);
}
}
/* ( virt devaddr size -- ) */
static void
dma_sync(void)
{
ucell size = POP();
POP();
ucell virt = POP();
flush_dcache_range(cell2pointer(virt), cell2pointer(virt + size));
flush_icache_range(cell2pointer(virt), cell2pointer(virt + size));
}
void
arch_of_init(void)
{
#ifdef CONFIG_RTAS
phandle_t ph;
#endif
uint64_t ram_size;
const struct cpudef *cpu;
char buf[256], qemu_uuid[16];
const char *stdin_path, *stdout_path, *boot_path;
uint32_t temp = 0;
char *boot_device, *bootorder_file;
uint32_t bootorder_sz, sz;
ofmem_t *ofmem = ofmem_arch_get_private();
ucell load_base;
openbios_init();
modules_init();
setup_timers();
bind_func("ppc-dma-alloc", dma_alloc);
feval("['] ppc-dma-alloc to (dma-alloc)");
bind_func("ppc-dma-map-in", dma_map_in);
feval("['] ppc-dma-map-in to (dma-map-in)");
bind_func("ppc-dma-sync", dma_sync);
feval("['] ppc-dma-sync to (dma-sync)");
#ifdef CONFIG_DRIVER_PCI
push_str("/");
fword("find-device");
feval("\" /\" open-dev to my-self");
switch (machine_id) {
case ARCH_MAC99:
case ARCH_MAC99_U3:
/* The NewWorld NVRAM is not located in the MacIO device */
macio_nvram_init("/", 0);
ob_pci_init();
ob_unin_init();
break;
default:
ob_pci_init();
}
feval("0 to my-self");
#endif
printk("\n");
printk("=============================================================\n");
printk(PROGRAM_NAME " " OPENBIOS_VERSION_STR " [%s]\n",
OPENBIOS_BUILD_DATE);
fw_cfg_read(FW_CFG_SIGNATURE, buf, 4);
buf[4] = '\0';
printk("Configuration device id %s", buf);
temp = fw_cfg_read_i32(FW_CFG_ID);
printk(" version %d machine id %d\n", temp, machine_id);
temp = fw_cfg_read_i32(FW_CFG_NB_CPUS);
printk("CPUs: %x\n", temp);
ram_size = ofmem->ramsize;
printk("Memory: %lldM\n", ram_size / 1024 / 1024);
fw_cfg_read(FW_CFG_UUID, qemu_uuid, 16);
printk("UUID: " UUID_FMT "\n", qemu_uuid[0], qemu_uuid[1], qemu_uuid[2],
qemu_uuid[3], qemu_uuid[4], qemu_uuid[5], qemu_uuid[6],
qemu_uuid[7], qemu_uuid[8], qemu_uuid[9], qemu_uuid[10],
qemu_uuid[11], qemu_uuid[12], qemu_uuid[13], qemu_uuid[14],
qemu_uuid[15]);
/* set device tree root info */
push_str("/");
fword("find-device");
switch(machine_id) {
case ARCH_HEATHROW: /* OldWorld */
/* model */
push_str("Power Macintosh");
fword("model");
/* compatible */
push_str("AAPL,PowerMac G3");
fword("encode-string");
push_str("MacRISC");
fword("encode-string");
fword("encode+");
push_str("compatible");
fword("property");
/* misc */
push_str("device-tree");
fword("encode-string");
push_str("AAPL,original-name");
fword("property");
PUSH(0);
fword("encode-int");
push_str("AAPL,cpu-id");
fword("property");
PUSH(fw_cfg_read_i32(FW_CFG_PPC_BUSFREQ));
fword("encode-int");
push_str("clock-frequency");
fword("property");
break;
case ARCH_MAC99:
case ARCH_MAC99_U3:
case ARCH_PREP:
default:
/* model */
push_str("PowerMac3,1");
fword("model");
/* compatible */
push_str("PowerMac3,1");
fword("encode-string");
push_str("MacRISC");
fword("encode-string");
fword("encode+");
push_str("MacRISC2");
fword("encode-string");
fword("encode+");
push_str("Power Macintosh");
fword("encode-string");
fword("encode+");
push_str("compatible");
fword("property");
/* misc */
push_str("bootrom");
fword("device-type");
PUSH(fw_cfg_read_i32(FW_CFG_PPC_BUSFREQ));
fword("encode-int");
push_str("clock-frequency");
fword("property");
break;
}
/* Perhaps we can store UUID here ? */
push_str("0000000000000");
fword("encode-string");
push_str("system-id");
fword("property");
/* memory info */
push_str("/memory");
fword("find-device");
/* all memory */
push_physaddr(0);
fword("encode-phys");
/* This needs adjusting if #size-cells gets increased.
Alternatively use multiple (address, size) tuples. */
PUSH(ram_size & 0xffffffff);
fword("encode-int");
fword("encode+");
push_str("reg");
fword("property");
cpu = id_cpu();
cpu->initfn(cpu);
printk("CPU type %s\n", cpu->name);
snprintf(buf, sizeof(buf), "/cpus/%s", cpu->name);
ofmem_register(find_dev("/memory"), find_dev(buf));
node_methods_init(buf);
#ifdef CONFIG_RTAS
/* OldWorld Macs don't have an /rtas node. */
switch (machine_id) {
case ARCH_MAC99:
case ARCH_MAC99_U3:
if (!(ph = find_dev("/rtas"))) {
printk("Warning: No /rtas node\n");
} else {
unsigned long size = 0x1000;
while (size < (unsigned long)of_rtas_end - (unsigned long)of_rtas_start)
size *= 2;
set_property(ph, "rtas-size", (char*)&size, sizeof(size));
set_int_property(ph, "rtas-version", is_apple() ? 0x41 : 1);
}
break;
}
#endif
if (fw_cfg_read_i16(FW_CFG_NOGRAPHIC)) {
if (is_apple()) {
if (CONFIG_SERIAL_PORT) {
stdin_path = "sccb";
stdout_path = "sccb";
} else {
stdin_path = "scca";
stdout_path = "scca";
}
} else {
stdin_path = "ttya";
stdout_path = "ttya";
}
/* Some bootloaders force the output to the screen device, so
let's create a screen alias for the serial device too */
push_str("/aliases");
fword("find-device");
push_str(stdout_path);
fword("pathres-resolve-aliases");
fword("encode-string");
push_str("screen");
fword("property");
} else {
stdin_path = "keyboard";
stdout_path = "screen";
}
kvm_of_init();
/* Setup nvram variables */
push_str("/options");
fword("find-device");
/* Boot order */
bootorder_file = fw_cfg_read_file("bootorder", &bootorder_sz);
if (bootorder_file == NULL) {
/* No bootorder present, use fw_cfg device if no custom
boot-device specified */
fword("boot-device");
boot_device = pop_fstr_copy();
if (boot_device && strcmp(boot_device, "disk") == 0) {
switch (fw_cfg_read_i16(FW_CFG_BOOT_DEVICE)) {
case 'c':
boot_path = "hd";
break;
default:
case 'd':
boot_path = "cd";
break;
}
snprintf(buf, sizeof(buf),
"%s:,\\\\:tbxi "
"%s:,\\ppc\\bootinfo.txt "
"%s:,%%BOOT",
boot_path, boot_path, boot_path);
push_str(buf);
fword("encode-string");
push_str("boot-device");
fword("property");
}
free(boot_device);
} else {
sz = bootorder_sz * (3 * 2);
boot_device = malloc(sz);
memset(boot_device, 0, sz);
while ((boot_path = strsep(&bootorder_file, "\n")) != NULL) {
snprintf(buf, sizeof(buf),
"%s:,\\\\:tbxi "
"%s:,\\ppc\\bootinfo.txt "
"%s:,%%BOOT ",
boot_path, boot_path, boot_path);
strncat(boot_device, buf, sz);
}
push_str(boot_device);
fword("encode-string");
push_str("boot-device");
fword("property");
free(boot_device);
}
/* Set up other properties */
push_str("/chosen");
fword("find-device");
push_str(stdin_path);
fword("pathres-resolve-aliases");
push_str("input-device");
fword("$setenv");
push_str(stdout_path);
fword("pathres-resolve-aliases");
push_str("output-device");
fword("$setenv");
#if 0
if(getbool("tty-interface?") == 1)
#endif
fword("activate-tty-interface");
device_end();
/* Implementation of filll word (required by BootX) */
bind_func("filll", ffilll);
/* Implementation of adler32 word (required by OS 9, BootX) */
bind_func("(adler32)", adler32);
bind_func("platform-boot", boot);
bind_func("(arch-go)", arch_go);
/* Allocate 8MB memory at load-base */
fword("load-base");
load_base = POP();
ofmem_claim_phys(load_base, 0x800000, 0);
ofmem_claim_virt(load_base, 0x800000, 0);
ofmem_map(load_base, load_base, 0x800000, 0);
}