438 lines
13 KiB
C
438 lines
13 KiB
C
/*
|
|
* QEMU Eyetech AmigaOne/Mai Logic Teron emulation
|
|
*
|
|
* Copyright (c) 2023 BALATON Zoltan
|
|
*
|
|
* This work is licensed under the GNU GPL license version 2 or later.
|
|
*
|
|
*/
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "qemu/units.h"
|
|
#include "qemu/datadir.h"
|
|
#include "qemu/log.h"
|
|
#include "qemu/error-report.h"
|
|
#include "qapi/error.h"
|
|
#include "hw/ppc/ppc.h"
|
|
#include "hw/boards.h"
|
|
#include "hw/loader.h"
|
|
#include "hw/pci-host/articia.h"
|
|
#include "hw/isa/vt82c686.h"
|
|
#include "hw/ide/pci.h"
|
|
#include "hw/i2c/smbus_eeprom.h"
|
|
#include "hw/ppc/ppc.h"
|
|
#include "system/block-backend.h"
|
|
#include "system/qtest.h"
|
|
#include "system/reset.h"
|
|
#include "kvm_ppc.h"
|
|
#include "elf.h"
|
|
|
|
#include <zlib.h> /* for crc32 */
|
|
|
|
#define BUS_FREQ_HZ 100000000
|
|
|
|
#define INITRD_MIN_ADDR 0x600000
|
|
#define INIT_RAM_ADDR 0x40000000
|
|
|
|
#define PCI_HIGH_ADDR 0x80000000
|
|
#define PCI_HIGH_SIZE 0x7d000000
|
|
#define PCI_LOW_ADDR 0xfd000000
|
|
#define PCI_LOW_SIZE 0xe0000
|
|
|
|
#define ARTICIA_ADDR 0xfe000000
|
|
|
|
/*
|
|
* Firmware binary available at
|
|
* https://www.hyperion-entertainment.com/index.php/downloads?view=files&parent=28
|
|
* then "tail -c 524288 updater.image >u-boot-amigaone.bin"
|
|
*
|
|
* BIOS emulator in firmware cannot run QEMU vgabios and hangs on it, use
|
|
* -device VGA,romfile=VGABIOS-lgpl-latest.bin
|
|
* from http://www.nongnu.org/vgabios/ instead.
|
|
*/
|
|
#define PROM_ADDR 0xfff00000
|
|
#define PROM_SIZE (512 * KiB)
|
|
|
|
/* AmigaOS calls this routine from ROM, use this if no firmware loaded */
|
|
static const char dummy_fw[] = {
|
|
0x54, 0x63, 0xc2, 0x3e, /* srwi r3,r3,8 */
|
|
0x7c, 0x63, 0x18, 0xf8, /* not r3,r3 */
|
|
0x4e, 0x80, 0x00, 0x20, /* blr */
|
|
};
|
|
|
|
#define NVRAM_ADDR 0xfd0e0000
|
|
#define NVRAM_SIZE (4 * KiB)
|
|
|
|
static const char default_env[] =
|
|
"baudrate=115200\0"
|
|
"stdout=vga\0"
|
|
"stdin=ps2kbd\0"
|
|
"bootcmd=boota; menu; run menuboot_cmd\0"
|
|
"boot1=ide\0"
|
|
"boot2=cdrom\0"
|
|
"boota_timeout=3\0"
|
|
"ide_doreset=on\0"
|
|
"pci_irqa=9\0"
|
|
"pci_irqa_select=level\0"
|
|
"pci_irqb=10\0"
|
|
"pci_irqb_select=level\0"
|
|
"pci_irqc=11\0"
|
|
"pci_irqc_select=level\0"
|
|
"pci_irqd=7\0"
|
|
"pci_irqd_select=level\0"
|
|
"a1ide_irq=1111\0"
|
|
"a1ide_xfer=FFFF\0";
|
|
#define CRC32_DEFAULT_ENV 0xb5548481
|
|
#define CRC32_ALL_ZEROS 0x603b0489
|
|
|
|
#define TYPE_A1_NVRAM "a1-nvram"
|
|
OBJECT_DECLARE_SIMPLE_TYPE(A1NVRAMState, A1_NVRAM)
|
|
|
|
struct A1NVRAMState {
|
|
SysBusDevice parent_obj;
|
|
|
|
MemoryRegion mr;
|
|
BlockBackend *blk;
|
|
};
|
|
|
|
static uint64_t nvram_read(void *opaque, hwaddr addr, unsigned int size)
|
|
{
|
|
/* read callback not used because of romd mode */
|
|
g_assert_not_reached();
|
|
}
|
|
|
|
static void nvram_write(void *opaque, hwaddr addr, uint64_t val,
|
|
unsigned int size)
|
|
{
|
|
A1NVRAMState *s = opaque;
|
|
uint8_t *p = memory_region_get_ram_ptr(&s->mr);
|
|
|
|
p[addr] = val;
|
|
if (s->blk && blk_pwrite(s->blk, addr, 1, &val, 0) < 0) {
|
|
error_report("%s: could not write %s", __func__, blk_name(s->blk));
|
|
}
|
|
}
|
|
|
|
static const MemoryRegionOps nvram_ops = {
|
|
.read = nvram_read,
|
|
.write = nvram_write,
|
|
.endianness = DEVICE_BIG_ENDIAN,
|
|
.impl = {
|
|
.min_access_size = 1,
|
|
.max_access_size = 1,
|
|
},
|
|
};
|
|
|
|
static void nvram_realize(DeviceState *dev, Error **errp)
|
|
{
|
|
A1NVRAMState *s = A1_NVRAM(dev);
|
|
void *p;
|
|
uint32_t crc, *c;
|
|
|
|
memory_region_init_rom_device(&s->mr, NULL, &nvram_ops, s, "nvram",
|
|
NVRAM_SIZE, &error_fatal);
|
|
sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->mr);
|
|
c = p = memory_region_get_ram_ptr(&s->mr);
|
|
if (s->blk) {
|
|
if (blk_getlength(s->blk) != NVRAM_SIZE) {
|
|
error_setg(errp, "NVRAM backing file size must be %" PRId64 "bytes",
|
|
NVRAM_SIZE);
|
|
return;
|
|
}
|
|
blk_set_perm(s->blk, BLK_PERM_CONSISTENT_READ | BLK_PERM_WRITE,
|
|
BLK_PERM_ALL, &error_fatal);
|
|
if (blk_pread(s->blk, 0, NVRAM_SIZE, p, 0) < 0) {
|
|
error_setg(errp, "Cannot read NVRAM contents from backing file");
|
|
return;
|
|
}
|
|
}
|
|
crc = crc32(0, p + 4, NVRAM_SIZE - 4);
|
|
if (crc == CRC32_ALL_ZEROS) { /* If env is uninitialized set default */
|
|
*c = cpu_to_be32(CRC32_DEFAULT_ENV);
|
|
/* Also copies terminating \0 as env is terminated by \0\0 */
|
|
memcpy(p + 4, default_env, sizeof(default_env));
|
|
if (s->blk &&
|
|
blk_pwrite(s->blk, 0, sizeof(crc) + sizeof(default_env), p, 0) < 0
|
|
) {
|
|
error_report("%s: could not write %s", __func__, blk_name(s->blk));
|
|
}
|
|
return;
|
|
}
|
|
if (*c == 0) {
|
|
*c = cpu_to_be32(crc32(0, p + 4, NVRAM_SIZE - 4));
|
|
if (s->blk && blk_pwrite(s->blk, 0, 4, p, 0) < 0) {
|
|
error_report("%s: could not write %s", __func__, blk_name(s->blk));
|
|
}
|
|
}
|
|
if (be32_to_cpu(*c) != crc) {
|
|
warn_report("NVRAM checksum mismatch");
|
|
}
|
|
}
|
|
|
|
static const Property nvram_properties[] = {
|
|
DEFINE_PROP_DRIVE("drive", A1NVRAMState, blk),
|
|
};
|
|
|
|
static void nvram_class_init(ObjectClass *oc, void *data)
|
|
{
|
|
DeviceClass *dc = DEVICE_CLASS(oc);
|
|
|
|
dc->realize = nvram_realize;
|
|
device_class_set_props(dc, nvram_properties);
|
|
}
|
|
|
|
static const TypeInfo nvram_types[] = {
|
|
{
|
|
.name = TYPE_A1_NVRAM,
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
.instance_size = sizeof(A1NVRAMState),
|
|
.class_init = nvram_class_init,
|
|
},
|
|
};
|
|
DEFINE_TYPES(nvram_types)
|
|
|
|
struct boot_info {
|
|
hwaddr entry;
|
|
hwaddr stack;
|
|
hwaddr bd_info;
|
|
hwaddr initrd_start;
|
|
hwaddr initrd_end;
|
|
hwaddr cmdline_start;
|
|
hwaddr cmdline_end;
|
|
};
|
|
|
|
/* Board info struct from U-Boot */
|
|
struct bd_info {
|
|
uint32_t bi_memstart;
|
|
uint32_t bi_memsize;
|
|
uint32_t bi_flashstart;
|
|
uint32_t bi_flashsize;
|
|
uint32_t bi_flashoffset;
|
|
uint32_t bi_sramstart;
|
|
uint32_t bi_sramsize;
|
|
uint32_t bi_bootflags;
|
|
uint32_t bi_ip_addr;
|
|
uint8_t bi_enetaddr[6];
|
|
uint16_t bi_ethspeed;
|
|
uint32_t bi_intfreq;
|
|
uint32_t bi_busfreq;
|
|
uint32_t bi_baudrate;
|
|
} QEMU_PACKED;
|
|
|
|
static void create_bd_info(hwaddr addr, ram_addr_t ram_size)
|
|
{
|
|
struct bd_info *bd = g_new0(struct bd_info, 1);
|
|
|
|
bd->bi_memsize = cpu_to_be32(ram_size);
|
|
bd->bi_flashstart = cpu_to_be32(PROM_ADDR);
|
|
bd->bi_flashsize = cpu_to_be32(1); /* match what U-Boot detects */
|
|
bd->bi_bootflags = cpu_to_be32(1);
|
|
bd->bi_intfreq = cpu_to_be32(11.5 * BUS_FREQ_HZ);
|
|
bd->bi_busfreq = cpu_to_be32(BUS_FREQ_HZ);
|
|
bd->bi_baudrate = cpu_to_be32(115200);
|
|
|
|
cpu_physical_memory_write(addr, bd, sizeof(*bd));
|
|
}
|
|
|
|
static void amigaone_cpu_reset(void *opaque)
|
|
{
|
|
PowerPCCPU *cpu = opaque;
|
|
CPUPPCState *env = &cpu->env;
|
|
|
|
cpu_reset(CPU(cpu));
|
|
if (env->load_info) {
|
|
struct boot_info *bi = env->load_info;
|
|
|
|
env->gpr[1] = bi->stack;
|
|
env->gpr[2] = 1024;
|
|
env->gpr[3] = bi->bd_info;
|
|
env->gpr[4] = bi->initrd_start;
|
|
env->gpr[5] = bi->initrd_end;
|
|
env->gpr[6] = bi->cmdline_start;
|
|
env->gpr[7] = bi->cmdline_end;
|
|
env->nip = bi->entry;
|
|
}
|
|
cpu_ppc_tb_reset(env);
|
|
}
|
|
|
|
static void fix_spd_data(uint8_t *spd)
|
|
{
|
|
uint32_t bank_size = 4 * MiB * spd[31];
|
|
uint32_t rows = bank_size / spd[13] / spd[17];
|
|
spd[3] = ctz32(rows) - spd[4];
|
|
}
|
|
|
|
static void amigaone_init(MachineState *machine)
|
|
{
|
|
PowerPCCPU *cpu;
|
|
CPUPPCState *env;
|
|
MemoryRegion *rom, *pci_mem, *mr;
|
|
ssize_t sz;
|
|
PCIBus *pci_bus;
|
|
Object *via;
|
|
DeviceState *dev;
|
|
I2CBus *i2c_bus;
|
|
uint8_t *spd_data;
|
|
DriveInfo *di;
|
|
hwaddr loadaddr;
|
|
struct boot_info *bi = NULL;
|
|
|
|
/* init CPU */
|
|
cpu = POWERPC_CPU(cpu_create(machine->cpu_type));
|
|
env = &cpu->env;
|
|
if (PPC_INPUT(env) != PPC_FLAGS_INPUT_6xx) {
|
|
error_report("Incompatible CPU, only 6xx bus supported");
|
|
exit(1);
|
|
}
|
|
cpu_ppc_tb_init(env, BUS_FREQ_HZ / 4);
|
|
qemu_register_reset(amigaone_cpu_reset, cpu);
|
|
|
|
/* RAM */
|
|
if (machine->ram_size > 2 * GiB) {
|
|
error_report("RAM size more than 2 GiB is not supported");
|
|
exit(1);
|
|
}
|
|
memory_region_add_subregion(get_system_memory(), 0, machine->ram);
|
|
if (machine->ram_size < 1 * GiB + 32 * KiB) {
|
|
/* Firmware uses this area for startup */
|
|
mr = g_new(MemoryRegion, 1);
|
|
memory_region_init_ram(mr, NULL, "init-cache", 32 * KiB, &error_fatal);
|
|
memory_region_add_subregion(get_system_memory(), INIT_RAM_ADDR, mr);
|
|
}
|
|
|
|
/* nvram */
|
|
dev = qdev_new(TYPE_A1_NVRAM);
|
|
di = drive_get(IF_MTD, 0, 0);
|
|
if (di) {
|
|
qdev_prop_set_drive(dev, "drive", blk_by_legacy_dinfo(di));
|
|
}
|
|
sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
|
|
memory_region_add_subregion(get_system_memory(), NVRAM_ADDR,
|
|
sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0));
|
|
|
|
/* allocate and load firmware */
|
|
rom = g_new(MemoryRegion, 1);
|
|
memory_region_init_rom(rom, NULL, "rom", PROM_SIZE, &error_fatal);
|
|
memory_region_add_subregion(get_system_memory(), PROM_ADDR, rom);
|
|
if (!machine->firmware) {
|
|
rom_add_blob_fixed("dummy-fw", dummy_fw, sizeof(dummy_fw),
|
|
PROM_ADDR + PROM_SIZE - 0x80);
|
|
} else {
|
|
g_autofree char *filename = qemu_find_file(QEMU_FILE_TYPE_BIOS,
|
|
machine->firmware);
|
|
if (!filename) {
|
|
error_report("Could not find firmware '%s'", machine->firmware);
|
|
exit(1);
|
|
}
|
|
sz = load_image_targphys(filename, PROM_ADDR, PROM_SIZE);
|
|
if (sz <= 0 || sz > PROM_SIZE) {
|
|
error_report("Could not load firmware '%s'", filename);
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
/* Articia S */
|
|
dev = sysbus_create_simple(TYPE_ARTICIA, ARTICIA_ADDR, NULL);
|
|
|
|
i2c_bus = I2C_BUS(qdev_get_child_bus(dev, "smbus"));
|
|
if (machine->ram_size > 512 * MiB) {
|
|
spd_data = spd_data_generate(SDR, machine->ram_size / 2);
|
|
} else {
|
|
spd_data = spd_data_generate(SDR, machine->ram_size);
|
|
}
|
|
fix_spd_data(spd_data);
|
|
smbus_eeprom_init_one(i2c_bus, 0x51, spd_data);
|
|
if (machine->ram_size > 512 * MiB) {
|
|
smbus_eeprom_init_one(i2c_bus, 0x52, spd_data);
|
|
}
|
|
|
|
pci_mem = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 1);
|
|
mr = g_new(MemoryRegion, 1);
|
|
memory_region_init_alias(mr, OBJECT(dev), "pci-mem-low", pci_mem,
|
|
0, PCI_LOW_SIZE);
|
|
memory_region_add_subregion(get_system_memory(), PCI_LOW_ADDR, mr);
|
|
mr = g_new(MemoryRegion, 1);
|
|
memory_region_init_alias(mr, OBJECT(dev), "pci-mem-high", pci_mem,
|
|
PCI_HIGH_ADDR, PCI_HIGH_SIZE);
|
|
memory_region_add_subregion(get_system_memory(), PCI_HIGH_ADDR, mr);
|
|
pci_bus = PCI_BUS(qdev_get_child_bus(dev, "pci.0"));
|
|
|
|
/* VIA VT82c686B South Bridge (multifunction PCI device) */
|
|
via = OBJECT(pci_create_simple_multifunction(pci_bus, PCI_DEVFN(7, 0),
|
|
TYPE_VT82C686B_ISA));
|
|
object_property_add_alias(OBJECT(machine), "rtc-time",
|
|
object_resolve_path_component(via, "rtc"),
|
|
"date");
|
|
qdev_connect_gpio_out_named(DEVICE(via), "intr", 0,
|
|
qdev_get_gpio_in(DEVICE(cpu),
|
|
PPC6xx_INPUT_INT));
|
|
for (int i = 0; i < PCI_NUM_PINS; i++) {
|
|
qdev_connect_gpio_out(dev, i, qdev_get_gpio_in_named(DEVICE(via),
|
|
"pirq", i));
|
|
}
|
|
pci_ide_create_devs(PCI_DEVICE(object_resolve_path_component(via, "ide")));
|
|
pci_vga_init(pci_bus);
|
|
|
|
if (!machine->kernel_filename) {
|
|
return;
|
|
}
|
|
|
|
/* handle -kernel, -initrd, -append options and emulate U-Boot */
|
|
bi = g_new0(struct boot_info, 1);
|
|
cpu->env.load_info = bi;
|
|
|
|
loadaddr = MIN(machine->ram_size, 256 * MiB);
|
|
bi->bd_info = loadaddr - 8 * MiB;
|
|
create_bd_info(bi->bd_info, machine->ram_size);
|
|
bi->stack = bi->bd_info - 64 * KiB - 8;
|
|
|
|
if (machine->kernel_cmdline && machine->kernel_cmdline[0]) {
|
|
size_t len = strlen(machine->kernel_cmdline);
|
|
|
|
loadaddr = bi->bd_info + 1 * MiB;
|
|
cpu_physical_memory_write(loadaddr, machine->kernel_cmdline, len + 1);
|
|
bi->cmdline_start = loadaddr;
|
|
bi->cmdline_end = loadaddr + len + 1; /* including terminating '\0' */
|
|
}
|
|
|
|
sz = load_elf(machine->kernel_filename, NULL, NULL, NULL,
|
|
&bi->entry, &loadaddr, NULL, NULL,
|
|
ELFDATA2MSB, PPC_ELF_MACHINE, 0, 0);
|
|
if (sz <= 0) {
|
|
sz = load_uimage(machine->kernel_filename, &bi->entry, &loadaddr,
|
|
NULL, NULL, NULL);
|
|
}
|
|
if (sz <= 0) {
|
|
error_report("Could not load kernel '%s'",
|
|
machine->kernel_filename);
|
|
exit(1);
|
|
}
|
|
loadaddr += sz;
|
|
|
|
if (machine->initrd_filename) {
|
|
loadaddr = ROUND_UP(loadaddr + 4 * MiB, 4 * KiB);
|
|
loadaddr = MAX(loadaddr, INITRD_MIN_ADDR);
|
|
sz = load_image_targphys(machine->initrd_filename, loadaddr,
|
|
bi->bd_info - loadaddr);
|
|
if (sz <= 0) {
|
|
error_report("Could not load initrd '%s'",
|
|
machine->initrd_filename);
|
|
exit(1);
|
|
}
|
|
bi->initrd_start = loadaddr;
|
|
bi->initrd_end = loadaddr + sz;
|
|
}
|
|
}
|
|
|
|
static void amigaone_machine_init(MachineClass *mc)
|
|
{
|
|
mc->desc = "Eyetech AmigaOne/Mai Logic Teron";
|
|
mc->init = amigaone_init;
|
|
mc->block_default_type = IF_IDE;
|
|
mc->default_cpu_type = POWERPC_CPU_TYPE_NAME("7457_v1.2");
|
|
mc->default_display = "std";
|
|
mc->default_ram_id = "ram";
|
|
mc->default_ram_size = 512 * MiB;
|
|
}
|
|
|
|
DEFINE_MACHINE("amigaone", amigaone_machine_init)
|