xref: /qemu/hw/ppc/amigaone.c (revision 06b40d250ecfa1633209c2e431a7a38acfd03a98)
1d9656f86SBALATON Zoltan /*
2d9656f86SBALATON Zoltan  * QEMU Eyetech AmigaOne/Mai Logic Teron emulation
3d9656f86SBALATON Zoltan  *
4d9656f86SBALATON Zoltan  * Copyright (c) 2023 BALATON Zoltan
5d9656f86SBALATON Zoltan  *
6d9656f86SBALATON Zoltan  * This work is licensed under the GNU GPL license version 2 or later.
7d9656f86SBALATON Zoltan  *
8d9656f86SBALATON Zoltan  */
9d9656f86SBALATON Zoltan 
10d9656f86SBALATON Zoltan #include "qemu/osdep.h"
11d9656f86SBALATON Zoltan #include "qemu/units.h"
12d9656f86SBALATON Zoltan #include "qemu/datadir.h"
13d9656f86SBALATON Zoltan #include "qemu/log.h"
14d9656f86SBALATON Zoltan #include "qemu/error-report.h"
15d9656f86SBALATON Zoltan #include "qapi/error.h"
16d9656f86SBALATON Zoltan #include "hw/ppc/ppc.h"
17d9656f86SBALATON Zoltan #include "hw/boards.h"
18d9656f86SBALATON Zoltan #include "hw/loader.h"
19d9656f86SBALATON Zoltan #include "hw/pci-host/articia.h"
20d9656f86SBALATON Zoltan #include "hw/isa/vt82c686.h"
21d9656f86SBALATON Zoltan #include "hw/ide/pci.h"
22d9656f86SBALATON Zoltan #include "hw/i2c/smbus_eeprom.h"
23d9656f86SBALATON Zoltan #include "hw/ppc/ppc.h"
24addff513SBALATON Zoltan #include "system/block-backend.h"
2532cad1ffSPhilippe Mathieu-Daudé #include "system/qtest.h"
2632cad1ffSPhilippe Mathieu-Daudé #include "system/reset.h"
27d9656f86SBALATON Zoltan #include "kvm_ppc.h"
2834f053d8SBALATON Zoltan #include "elf.h"
29d9656f86SBALATON Zoltan 
30addff513SBALATON Zoltan #include <zlib.h> /* for crc32 */
31addff513SBALATON Zoltan 
32d9656f86SBALATON Zoltan #define BUS_FREQ_HZ 100000000
33d9656f86SBALATON Zoltan 
3434f053d8SBALATON Zoltan #define INITRD_MIN_ADDR 0x600000
35e6521e41SBALATON Zoltan #define INIT_RAM_ADDR 0x40000000
36e6521e41SBALATON Zoltan 
37e6521e41SBALATON Zoltan #define PCI_HIGH_ADDR 0x80000000
38e6521e41SBALATON Zoltan #define PCI_HIGH_SIZE 0x7d000000
39e6521e41SBALATON Zoltan #define PCI_LOW_ADDR  0xfd000000
40e6521e41SBALATON Zoltan #define PCI_LOW_SIZE  0xe0000
41e6521e41SBALATON Zoltan 
42e6521e41SBALATON Zoltan #define ARTICIA_ADDR 0xfe000000
4334f053d8SBALATON Zoltan 
44d9656f86SBALATON Zoltan /*
45d9656f86SBALATON Zoltan  * Firmware binary available at
46d9656f86SBALATON Zoltan  * https://www.hyperion-entertainment.com/index.php/downloads?view=files&parent=28
47d9656f86SBALATON Zoltan  * then "tail -c 524288 updater.image >u-boot-amigaone.bin"
48d9656f86SBALATON Zoltan  *
49d9656f86SBALATON Zoltan  * BIOS emulator in firmware cannot run QEMU vgabios and hangs on it, use
50d9656f86SBALATON Zoltan  * -device VGA,romfile=VGABIOS-lgpl-latest.bin
51d9656f86SBALATON Zoltan  * from http://www.nongnu.org/vgabios/ instead.
52d9656f86SBALATON Zoltan  */
53d9656f86SBALATON Zoltan #define PROM_ADDR 0xfff00000
54d9656f86SBALATON Zoltan #define PROM_SIZE (512 * KiB)
55d9656f86SBALATON Zoltan 
56e25acd61SBALATON Zoltan /* AmigaOS calls this routine from ROM, use this if no firmware loaded */
57e25acd61SBALATON Zoltan static const char dummy_fw[] = {
58222d37d3SBALATON Zoltan     0x54, 0x63, 0xc2, 0x3e, /* srwi    r3,r3,8 */
59e25acd61SBALATON Zoltan     0x7c, 0x63, 0x18, 0xf8, /* not     r3,r3 */
60e25acd61SBALATON Zoltan     0x4e, 0x80, 0x00, 0x20, /* blr */
61e25acd61SBALATON Zoltan };
62e25acd61SBALATON Zoltan 
63addff513SBALATON Zoltan #define NVRAM_ADDR 0xfd0e0000
64addff513SBALATON Zoltan #define NVRAM_SIZE (4 * KiB)
65addff513SBALATON Zoltan 
66667413f5SBALATON Zoltan static const char default_env[] =
67c2bed995SBALATON Zoltan     "baudrate=115200\0"
68c2bed995SBALATON Zoltan     "stdout=vga\0"
69c2bed995SBALATON Zoltan     "stdin=ps2kbd\0"
70c2bed995SBALATON Zoltan     "bootcmd=boota; menu; run menuboot_cmd\0"
71c2bed995SBALATON Zoltan     "boot1=ide\0"
72c2bed995SBALATON Zoltan     "boot2=cdrom\0"
73c2bed995SBALATON Zoltan     "boota_timeout=3\0"
74c2bed995SBALATON Zoltan     "ide_doreset=on\0"
75c2bed995SBALATON Zoltan     "pci_irqa=9\0"
76c2bed995SBALATON Zoltan     "pci_irqa_select=level\0"
77c2bed995SBALATON Zoltan     "pci_irqb=10\0"
78c2bed995SBALATON Zoltan     "pci_irqb_select=level\0"
79c2bed995SBALATON Zoltan     "pci_irqc=11\0"
80c2bed995SBALATON Zoltan     "pci_irqc_select=level\0"
81c2bed995SBALATON Zoltan     "pci_irqd=7\0"
82c2bed995SBALATON Zoltan     "pci_irqd_select=level\0"
83c2bed995SBALATON Zoltan     "a1ide_irq=1111\0"
84c2bed995SBALATON Zoltan     "a1ide_xfer=FFFF\0";
85c2bed995SBALATON Zoltan #define CRC32_DEFAULT_ENV 0xb5548481
86c2bed995SBALATON Zoltan #define CRC32_ALL_ZEROS   0x603b0489
87c2bed995SBALATON Zoltan 
88addff513SBALATON Zoltan #define TYPE_A1_NVRAM "a1-nvram"
89addff513SBALATON Zoltan OBJECT_DECLARE_SIMPLE_TYPE(A1NVRAMState, A1_NVRAM)
90addff513SBALATON Zoltan 
91addff513SBALATON Zoltan struct A1NVRAMState {
92addff513SBALATON Zoltan     SysBusDevice parent_obj;
93addff513SBALATON Zoltan 
94addff513SBALATON Zoltan     MemoryRegion mr;
95addff513SBALATON Zoltan     BlockBackend *blk;
96addff513SBALATON Zoltan };
97addff513SBALATON Zoltan 
nvram_read(void * opaque,hwaddr addr,unsigned int size)98addff513SBALATON Zoltan static uint64_t nvram_read(void *opaque, hwaddr addr, unsigned int size)
99addff513SBALATON Zoltan {
100addff513SBALATON Zoltan     /* read callback not used because of romd mode */
101addff513SBALATON Zoltan     g_assert_not_reached();
102addff513SBALATON Zoltan }
103addff513SBALATON Zoltan 
nvram_write(void * opaque,hwaddr addr,uint64_t val,unsigned int size)104addff513SBALATON Zoltan static void nvram_write(void *opaque, hwaddr addr, uint64_t val,
105addff513SBALATON Zoltan                         unsigned int size)
106addff513SBALATON Zoltan {
107addff513SBALATON Zoltan     A1NVRAMState *s = opaque;
108addff513SBALATON Zoltan     uint8_t *p = memory_region_get_ram_ptr(&s->mr);
109addff513SBALATON Zoltan 
110addff513SBALATON Zoltan     p[addr] = val;
1110cb6498bSBALATON Zoltan     if (s->blk && blk_pwrite(s->blk, addr, 1, &val, 0) < 0) {
1120cb6498bSBALATON Zoltan         error_report("%s: could not write %s", __func__, blk_name(s->blk));
113addff513SBALATON Zoltan     }
114addff513SBALATON Zoltan }
115addff513SBALATON Zoltan 
116addff513SBALATON Zoltan static const MemoryRegionOps nvram_ops = {
117addff513SBALATON Zoltan     .read = nvram_read,
118addff513SBALATON Zoltan     .write = nvram_write,
119addff513SBALATON Zoltan     .endianness = DEVICE_BIG_ENDIAN,
120addff513SBALATON Zoltan     .impl = {
121addff513SBALATON Zoltan         .min_access_size = 1,
122addff513SBALATON Zoltan         .max_access_size = 1,
123addff513SBALATON Zoltan     },
124addff513SBALATON Zoltan };
125addff513SBALATON Zoltan 
nvram_realize(DeviceState * dev,Error ** errp)126addff513SBALATON Zoltan static void nvram_realize(DeviceState *dev, Error **errp)
127addff513SBALATON Zoltan {
128addff513SBALATON Zoltan     A1NVRAMState *s = A1_NVRAM(dev);
129addff513SBALATON Zoltan     void *p;
130c2bed995SBALATON Zoltan     uint32_t crc, *c;
131addff513SBALATON Zoltan 
132addff513SBALATON Zoltan     memory_region_init_rom_device(&s->mr, NULL, &nvram_ops, s, "nvram",
133addff513SBALATON Zoltan                                   NVRAM_SIZE, &error_fatal);
134addff513SBALATON Zoltan     sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->mr);
135addff513SBALATON Zoltan     c = p = memory_region_get_ram_ptr(&s->mr);
136addff513SBALATON Zoltan     if (s->blk) {
137addff513SBALATON Zoltan         if (blk_getlength(s->blk) != NVRAM_SIZE) {
138addff513SBALATON Zoltan             error_setg(errp, "NVRAM backing file size must be %" PRId64 "bytes",
139addff513SBALATON Zoltan                        NVRAM_SIZE);
140addff513SBALATON Zoltan             return;
141addff513SBALATON Zoltan         }
142addff513SBALATON Zoltan         blk_set_perm(s->blk, BLK_PERM_CONSISTENT_READ | BLK_PERM_WRITE,
143addff513SBALATON Zoltan                      BLK_PERM_ALL, &error_fatal);
144addff513SBALATON Zoltan         if (blk_pread(s->blk, 0, NVRAM_SIZE, p, 0) < 0) {
145addff513SBALATON Zoltan             error_setg(errp, "Cannot read NVRAM contents from backing file");
146addff513SBALATON Zoltan             return;
147addff513SBALATON Zoltan         }
148addff513SBALATON Zoltan     }
149c2bed995SBALATON Zoltan     crc = crc32(0, p + 4, NVRAM_SIZE - 4);
150c2bed995SBALATON Zoltan     if (crc == CRC32_ALL_ZEROS) { /* If env is uninitialized set default */
151c2bed995SBALATON Zoltan         *c = cpu_to_be32(CRC32_DEFAULT_ENV);
152c2bed995SBALATON Zoltan         /* Also copies terminating \0 as env is terminated by \0\0 */
153c2bed995SBALATON Zoltan         memcpy(p + 4, default_env, sizeof(default_env));
1540cb6498bSBALATON Zoltan         if (s->blk &&
1550cb6498bSBALATON Zoltan             blk_pwrite(s->blk, 0, sizeof(crc) + sizeof(default_env), p, 0) < 0
1560cb6498bSBALATON Zoltan            ) {
1570cb6498bSBALATON Zoltan             error_report("%s: could not write %s", __func__, blk_name(s->blk));
158c2bed995SBALATON Zoltan         }
159c2bed995SBALATON Zoltan         return;
160c2bed995SBALATON Zoltan     }
161addff513SBALATON Zoltan     if (*c == 0) {
162addff513SBALATON Zoltan         *c = cpu_to_be32(crc32(0, p + 4, NVRAM_SIZE - 4));
1630cb6498bSBALATON Zoltan         if (s->blk && blk_pwrite(s->blk, 0, 4, p, 0) < 0) {
1640cb6498bSBALATON Zoltan             error_report("%s: could not write %s", __func__, blk_name(s->blk));
165addff513SBALATON Zoltan         }
166addff513SBALATON Zoltan     }
167c2bed995SBALATON Zoltan     if (be32_to_cpu(*c) != crc) {
168c2bed995SBALATON Zoltan         warn_report("NVRAM checksum mismatch");
169c2bed995SBALATON Zoltan     }
170addff513SBALATON Zoltan }
171addff513SBALATON Zoltan 
172addff513SBALATON Zoltan static const Property nvram_properties[] = {
173addff513SBALATON Zoltan     DEFINE_PROP_DRIVE("drive", A1NVRAMState, blk),
174addff513SBALATON Zoltan };
175addff513SBALATON Zoltan 
nvram_class_init(ObjectClass * oc,const void * data)176*12d1a768SPhilippe Mathieu-Daudé static void nvram_class_init(ObjectClass *oc, const void *data)
177addff513SBALATON Zoltan {
178addff513SBALATON Zoltan     DeviceClass *dc = DEVICE_CLASS(oc);
179addff513SBALATON Zoltan 
180addff513SBALATON Zoltan     dc->realize = nvram_realize;
181addff513SBALATON Zoltan     device_class_set_props(dc, nvram_properties);
182addff513SBALATON Zoltan }
183addff513SBALATON Zoltan 
184addff513SBALATON Zoltan static const TypeInfo nvram_types[] = {
185addff513SBALATON Zoltan     {
186addff513SBALATON Zoltan         .name = TYPE_A1_NVRAM,
187addff513SBALATON Zoltan         .parent = TYPE_SYS_BUS_DEVICE,
188addff513SBALATON Zoltan         .instance_size = sizeof(A1NVRAMState),
189addff513SBALATON Zoltan         .class_init = nvram_class_init,
190addff513SBALATON Zoltan     },
191addff513SBALATON Zoltan };
192addff513SBALATON Zoltan DEFINE_TYPES(nvram_types)
193addff513SBALATON Zoltan 
19434f053d8SBALATON Zoltan struct boot_info {
19534f053d8SBALATON Zoltan     hwaddr entry;
19634f053d8SBALATON Zoltan     hwaddr stack;
19734f053d8SBALATON Zoltan     hwaddr bd_info;
19834f053d8SBALATON Zoltan     hwaddr initrd_start;
19934f053d8SBALATON Zoltan     hwaddr initrd_end;
20034f053d8SBALATON Zoltan     hwaddr cmdline_start;
20134f053d8SBALATON Zoltan     hwaddr cmdline_end;
20234f053d8SBALATON Zoltan };
20334f053d8SBALATON Zoltan 
20434f053d8SBALATON Zoltan /* Board info struct from U-Boot */
20534f053d8SBALATON Zoltan struct bd_info {
20634f053d8SBALATON Zoltan     uint32_t bi_memstart;
20734f053d8SBALATON Zoltan     uint32_t bi_memsize;
20834f053d8SBALATON Zoltan     uint32_t bi_flashstart;
20934f053d8SBALATON Zoltan     uint32_t bi_flashsize;
21034f053d8SBALATON Zoltan     uint32_t bi_flashoffset;
21134f053d8SBALATON Zoltan     uint32_t bi_sramstart;
21234f053d8SBALATON Zoltan     uint32_t bi_sramsize;
21334f053d8SBALATON Zoltan     uint32_t bi_bootflags;
21434f053d8SBALATON Zoltan     uint32_t bi_ip_addr;
21534f053d8SBALATON Zoltan     uint8_t  bi_enetaddr[6];
21634f053d8SBALATON Zoltan     uint16_t bi_ethspeed;
21734f053d8SBALATON Zoltan     uint32_t bi_intfreq;
21834f053d8SBALATON Zoltan     uint32_t bi_busfreq;
21934f053d8SBALATON Zoltan     uint32_t bi_baudrate;
22034f053d8SBALATON Zoltan } QEMU_PACKED;
22134f053d8SBALATON Zoltan 
create_bd_info(hwaddr addr,ram_addr_t ram_size)22234f053d8SBALATON Zoltan static void create_bd_info(hwaddr addr, ram_addr_t ram_size)
22334f053d8SBALATON Zoltan {
22434f053d8SBALATON Zoltan     struct bd_info *bd = g_new0(struct bd_info, 1);
22534f053d8SBALATON Zoltan 
22634f053d8SBALATON Zoltan     bd->bi_memsize =    cpu_to_be32(ram_size);
22734f053d8SBALATON Zoltan     bd->bi_flashstart = cpu_to_be32(PROM_ADDR);
22834f053d8SBALATON Zoltan     bd->bi_flashsize =  cpu_to_be32(1); /* match what U-Boot detects */
22934f053d8SBALATON Zoltan     bd->bi_bootflags =  cpu_to_be32(1);
23034f053d8SBALATON Zoltan     bd->bi_intfreq =    cpu_to_be32(11.5 * BUS_FREQ_HZ);
23134f053d8SBALATON Zoltan     bd->bi_busfreq =    cpu_to_be32(BUS_FREQ_HZ);
23234f053d8SBALATON Zoltan     bd->bi_baudrate =   cpu_to_be32(115200);
23334f053d8SBALATON Zoltan 
23434f053d8SBALATON Zoltan     cpu_physical_memory_write(addr, bd, sizeof(*bd));
23534f053d8SBALATON Zoltan }
23634f053d8SBALATON Zoltan 
amigaone_cpu_reset(void * opaque)237d9656f86SBALATON Zoltan static void amigaone_cpu_reset(void *opaque)
238d9656f86SBALATON Zoltan {
239d9656f86SBALATON Zoltan     PowerPCCPU *cpu = opaque;
24034f053d8SBALATON Zoltan     CPUPPCState *env = &cpu->env;
241d9656f86SBALATON Zoltan 
242d9656f86SBALATON Zoltan     cpu_reset(CPU(cpu));
24334f053d8SBALATON Zoltan     if (env->load_info) {
24434f053d8SBALATON Zoltan         struct boot_info *bi = env->load_info;
24534f053d8SBALATON Zoltan 
24634f053d8SBALATON Zoltan         env->gpr[1] = bi->stack;
24734f053d8SBALATON Zoltan         env->gpr[2] = 1024;
24834f053d8SBALATON Zoltan         env->gpr[3] = bi->bd_info;
24934f053d8SBALATON Zoltan         env->gpr[4] = bi->initrd_start;
25034f053d8SBALATON Zoltan         env->gpr[5] = bi->initrd_end;
25134f053d8SBALATON Zoltan         env->gpr[6] = bi->cmdline_start;
25234f053d8SBALATON Zoltan         env->gpr[7] = bi->cmdline_end;
25334f053d8SBALATON Zoltan         env->nip = bi->entry;
25434f053d8SBALATON Zoltan     }
25534f053d8SBALATON Zoltan     cpu_ppc_tb_reset(env);
256d9656f86SBALATON Zoltan }
257d9656f86SBALATON Zoltan 
fix_spd_data(uint8_t * spd)258d9656f86SBALATON Zoltan static void fix_spd_data(uint8_t *spd)
259d9656f86SBALATON Zoltan {
260d9656f86SBALATON Zoltan     uint32_t bank_size = 4 * MiB * spd[31];
261d9656f86SBALATON Zoltan     uint32_t rows = bank_size / spd[13] / spd[17];
262d9656f86SBALATON Zoltan     spd[3] = ctz32(rows) - spd[4];
263d9656f86SBALATON Zoltan }
264d9656f86SBALATON Zoltan 
amigaone_init(MachineState * machine)265d9656f86SBALATON Zoltan static void amigaone_init(MachineState *machine)
266d9656f86SBALATON Zoltan {
267d9656f86SBALATON Zoltan     PowerPCCPU *cpu;
268d9656f86SBALATON Zoltan     CPUPPCState *env;
269d9656f86SBALATON Zoltan     MemoryRegion *rom, *pci_mem, *mr;
270d9656f86SBALATON Zoltan     ssize_t sz;
271d9656f86SBALATON Zoltan     PCIBus *pci_bus;
272d9656f86SBALATON Zoltan     Object *via;
273d9656f86SBALATON Zoltan     DeviceState *dev;
274d9656f86SBALATON Zoltan     I2CBus *i2c_bus;
275d9656f86SBALATON Zoltan     uint8_t *spd_data;
276addff513SBALATON Zoltan     DriveInfo *di;
27734f053d8SBALATON Zoltan     hwaddr loadaddr;
27834f053d8SBALATON Zoltan     struct boot_info *bi = NULL;
279d9656f86SBALATON Zoltan 
280d9656f86SBALATON Zoltan     /* init CPU */
281d9656f86SBALATON Zoltan     cpu = POWERPC_CPU(cpu_create(machine->cpu_type));
282d9656f86SBALATON Zoltan     env = &cpu->env;
283d9656f86SBALATON Zoltan     if (PPC_INPUT(env) != PPC_FLAGS_INPUT_6xx) {
284d9656f86SBALATON Zoltan         error_report("Incompatible CPU, only 6xx bus supported");
285d9656f86SBALATON Zoltan         exit(1);
286d9656f86SBALATON Zoltan     }
287d9656f86SBALATON Zoltan     cpu_ppc_tb_init(env, BUS_FREQ_HZ / 4);
288d9656f86SBALATON Zoltan     qemu_register_reset(amigaone_cpu_reset, cpu);
289d9656f86SBALATON Zoltan 
290d9656f86SBALATON Zoltan     /* RAM */
291d9656f86SBALATON Zoltan     if (machine->ram_size > 2 * GiB) {
292d9656f86SBALATON Zoltan         error_report("RAM size more than 2 GiB is not supported");
293d9656f86SBALATON Zoltan         exit(1);
294d9656f86SBALATON Zoltan     }
295d9656f86SBALATON Zoltan     memory_region_add_subregion(get_system_memory(), 0, machine->ram);
296d9656f86SBALATON Zoltan     if (machine->ram_size < 1 * GiB + 32 * KiB) {
297d9656f86SBALATON Zoltan         /* Firmware uses this area for startup */
298d9656f86SBALATON Zoltan         mr = g_new(MemoryRegion, 1);
299d9656f86SBALATON Zoltan         memory_region_init_ram(mr, NULL, "init-cache", 32 * KiB, &error_fatal);
300e6521e41SBALATON Zoltan         memory_region_add_subregion(get_system_memory(), INIT_RAM_ADDR, mr);
301d9656f86SBALATON Zoltan     }
302d9656f86SBALATON Zoltan 
303addff513SBALATON Zoltan     /* nvram */
304addff513SBALATON Zoltan     dev = qdev_new(TYPE_A1_NVRAM);
305addff513SBALATON Zoltan     di = drive_get(IF_MTD, 0, 0);
306addff513SBALATON Zoltan     if (di) {
307addff513SBALATON Zoltan         qdev_prop_set_drive(dev, "drive", blk_by_legacy_dinfo(di));
308addff513SBALATON Zoltan     }
309addff513SBALATON Zoltan     sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
310addff513SBALATON Zoltan     memory_region_add_subregion(get_system_memory(), NVRAM_ADDR,
311addff513SBALATON Zoltan                                 sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0));
312addff513SBALATON Zoltan 
313d9656f86SBALATON Zoltan     /* allocate and load firmware */
314d9656f86SBALATON Zoltan     rom = g_new(MemoryRegion, 1);
315d9656f86SBALATON Zoltan     memory_region_init_rom(rom, NULL, "rom", PROM_SIZE, &error_fatal);
316d9656f86SBALATON Zoltan     memory_region_add_subregion(get_system_memory(), PROM_ADDR, rom);
317e25acd61SBALATON Zoltan     if (!machine->firmware) {
318e25acd61SBALATON Zoltan         rom_add_blob_fixed("dummy-fw", dummy_fw, sizeof(dummy_fw),
319e25acd61SBALATON Zoltan                            PROM_ADDR + PROM_SIZE - 0x80);
320e25acd61SBALATON Zoltan     } else {
321e25acd61SBALATON Zoltan         g_autofree char *filename = qemu_find_file(QEMU_FILE_TYPE_BIOS,
322e25acd61SBALATON Zoltan                                                    machine->firmware);
323e25acd61SBALATON Zoltan         if (!filename) {
324e25acd61SBALATON Zoltan             error_report("Could not find firmware '%s'", machine->firmware);
325e25acd61SBALATON Zoltan             exit(1);
326e25acd61SBALATON Zoltan         }
327d9656f86SBALATON Zoltan         sz = load_image_targphys(filename, PROM_ADDR, PROM_SIZE);
328d9656f86SBALATON Zoltan         if (sz <= 0 || sz > PROM_SIZE) {
329d9656f86SBALATON Zoltan             error_report("Could not load firmware '%s'", filename);
330d9656f86SBALATON Zoltan             exit(1);
331d9656f86SBALATON Zoltan         }
332d9656f86SBALATON Zoltan     }
333d9656f86SBALATON Zoltan 
334d9656f86SBALATON Zoltan     /* Articia S */
335e6521e41SBALATON Zoltan     dev = sysbus_create_simple(TYPE_ARTICIA, ARTICIA_ADDR, NULL);
336d9656f86SBALATON Zoltan 
337d9656f86SBALATON Zoltan     i2c_bus = I2C_BUS(qdev_get_child_bus(dev, "smbus"));
338d9656f86SBALATON Zoltan     if (machine->ram_size > 512 * MiB) {
339d9656f86SBALATON Zoltan         spd_data = spd_data_generate(SDR, machine->ram_size / 2);
340d9656f86SBALATON Zoltan     } else {
341d9656f86SBALATON Zoltan         spd_data = spd_data_generate(SDR, machine->ram_size);
342d9656f86SBALATON Zoltan     }
343d9656f86SBALATON Zoltan     fix_spd_data(spd_data);
344d9656f86SBALATON Zoltan     smbus_eeprom_init_one(i2c_bus, 0x51, spd_data);
345d9656f86SBALATON Zoltan     if (machine->ram_size > 512 * MiB) {
346d9656f86SBALATON Zoltan         smbus_eeprom_init_one(i2c_bus, 0x52, spd_data);
347d9656f86SBALATON Zoltan     }
348d9656f86SBALATON Zoltan 
349d9656f86SBALATON Zoltan     pci_mem = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 1);
350d9656f86SBALATON Zoltan     mr = g_new(MemoryRegion, 1);
351d9656f86SBALATON Zoltan     memory_region_init_alias(mr, OBJECT(dev), "pci-mem-low", pci_mem,
352e6521e41SBALATON Zoltan                              0, PCI_LOW_SIZE);
353e6521e41SBALATON Zoltan     memory_region_add_subregion(get_system_memory(), PCI_LOW_ADDR, mr);
354d9656f86SBALATON Zoltan     mr = g_new(MemoryRegion, 1);
355d9656f86SBALATON Zoltan     memory_region_init_alias(mr, OBJECT(dev), "pci-mem-high", pci_mem,
356e6521e41SBALATON Zoltan                              PCI_HIGH_ADDR, PCI_HIGH_SIZE);
357e6521e41SBALATON Zoltan     memory_region_add_subregion(get_system_memory(), PCI_HIGH_ADDR, mr);
358d9656f86SBALATON Zoltan     pci_bus = PCI_BUS(qdev_get_child_bus(dev, "pci.0"));
359d9656f86SBALATON Zoltan 
360d9656f86SBALATON Zoltan     /* VIA VT82c686B South Bridge (multifunction PCI device) */
361d9656f86SBALATON Zoltan     via = OBJECT(pci_create_simple_multifunction(pci_bus, PCI_DEVFN(7, 0),
362d9656f86SBALATON Zoltan                                                  TYPE_VT82C686B_ISA));
363d9656f86SBALATON Zoltan     object_property_add_alias(OBJECT(machine), "rtc-time",
364d9656f86SBALATON Zoltan                               object_resolve_path_component(via, "rtc"),
365d9656f86SBALATON Zoltan                               "date");
3669a365c25SBernhard Beschow     qdev_connect_gpio_out_named(DEVICE(via), "intr", 0,
3679a365c25SBernhard Beschow                                 qdev_get_gpio_in(DEVICE(cpu),
3689a365c25SBernhard Beschow                                 PPC6xx_INPUT_INT));
369addff513SBALATON Zoltan     for (int i = 0; i < PCI_NUM_PINS; i++) {
370d9656f86SBALATON Zoltan         qdev_connect_gpio_out(dev, i, qdev_get_gpio_in_named(DEVICE(via),
371d9656f86SBALATON Zoltan                                                              "pirq", i));
372d9656f86SBALATON Zoltan     }
373d9656f86SBALATON Zoltan     pci_ide_create_devs(PCI_DEVICE(object_resolve_path_component(via, "ide")));
374d9656f86SBALATON Zoltan     pci_vga_init(pci_bus);
37534f053d8SBALATON Zoltan 
37634f053d8SBALATON Zoltan     if (!machine->kernel_filename) {
37734f053d8SBALATON Zoltan         return;
37834f053d8SBALATON Zoltan     }
37934f053d8SBALATON Zoltan 
38034f053d8SBALATON Zoltan     /* handle -kernel, -initrd, -append options and emulate U-Boot */
38134f053d8SBALATON Zoltan     bi = g_new0(struct boot_info, 1);
38234f053d8SBALATON Zoltan     cpu->env.load_info = bi;
38334f053d8SBALATON Zoltan 
38434f053d8SBALATON Zoltan     loadaddr = MIN(machine->ram_size, 256 * MiB);
38534f053d8SBALATON Zoltan     bi->bd_info = loadaddr - 8 * MiB;
38634f053d8SBALATON Zoltan     create_bd_info(bi->bd_info, machine->ram_size);
38734f053d8SBALATON Zoltan     bi->stack = bi->bd_info - 64 * KiB - 8;
38834f053d8SBALATON Zoltan 
38934f053d8SBALATON Zoltan     if (machine->kernel_cmdline && machine->kernel_cmdline[0]) {
39034f053d8SBALATON Zoltan         size_t len = strlen(machine->kernel_cmdline);
39134f053d8SBALATON Zoltan 
39234f053d8SBALATON Zoltan         loadaddr = bi->bd_info + 1 * MiB;
39334f053d8SBALATON Zoltan         cpu_physical_memory_write(loadaddr, machine->kernel_cmdline, len + 1);
39434f053d8SBALATON Zoltan         bi->cmdline_start = loadaddr;
39534f053d8SBALATON Zoltan         bi->cmdline_end = loadaddr + len + 1; /* including terminating '\0' */
39634f053d8SBALATON Zoltan     }
39734f053d8SBALATON Zoltan 
39834f053d8SBALATON Zoltan     sz = load_elf(machine->kernel_filename, NULL, NULL, NULL,
39934f053d8SBALATON Zoltan                   &bi->entry, &loadaddr, NULL, NULL,
40034f053d8SBALATON Zoltan                   ELFDATA2MSB, PPC_ELF_MACHINE, 0, 0);
40134f053d8SBALATON Zoltan     if (sz <= 0) {
40234f053d8SBALATON Zoltan         sz = load_uimage(machine->kernel_filename, &bi->entry, &loadaddr,
40334f053d8SBALATON Zoltan                          NULL, NULL, NULL);
40434f053d8SBALATON Zoltan     }
40534f053d8SBALATON Zoltan     if (sz <= 0) {
40634f053d8SBALATON Zoltan         error_report("Could not load kernel '%s'",
40734f053d8SBALATON Zoltan                      machine->kernel_filename);
40834f053d8SBALATON Zoltan         exit(1);
40934f053d8SBALATON Zoltan     }
41034f053d8SBALATON Zoltan     loadaddr += sz;
41134f053d8SBALATON Zoltan 
41234f053d8SBALATON Zoltan     if (machine->initrd_filename) {
41334f053d8SBALATON Zoltan         loadaddr = ROUND_UP(loadaddr + 4 * MiB, 4 * KiB);
41434f053d8SBALATON Zoltan         loadaddr = MAX(loadaddr, INITRD_MIN_ADDR);
41534f053d8SBALATON Zoltan         sz = load_image_targphys(machine->initrd_filename, loadaddr,
41634f053d8SBALATON Zoltan                                  bi->bd_info - loadaddr);
41734f053d8SBALATON Zoltan         if (sz <= 0) {
41834f053d8SBALATON Zoltan             error_report("Could not load initrd '%s'",
41934f053d8SBALATON Zoltan                          machine->initrd_filename);
42034f053d8SBALATON Zoltan             exit(1);
42134f053d8SBALATON Zoltan         }
42234f053d8SBALATON Zoltan         bi->initrd_start = loadaddr;
42334f053d8SBALATON Zoltan         bi->initrd_end = loadaddr + sz;
42434f053d8SBALATON Zoltan     }
425d9656f86SBALATON Zoltan }
426d9656f86SBALATON Zoltan 
amigaone_machine_init(MachineClass * mc)427d9656f86SBALATON Zoltan static void amigaone_machine_init(MachineClass *mc)
428d9656f86SBALATON Zoltan {
429d9656f86SBALATON Zoltan     mc->desc = "Eyetech AmigaOne/Mai Logic Teron";
430d9656f86SBALATON Zoltan     mc->init = amigaone_init;
431d9656f86SBALATON Zoltan     mc->block_default_type = IF_IDE;
432d9656f86SBALATON Zoltan     mc->default_cpu_type = POWERPC_CPU_TYPE_NAME("7457_v1.2");
433d9656f86SBALATON Zoltan     mc->default_display = "std";
434d9656f86SBALATON Zoltan     mc->default_ram_id = "ram";
435d9656f86SBALATON Zoltan     mc->default_ram_size = 512 * MiB;
436d9656f86SBALATON Zoltan }
437d9656f86SBALATON Zoltan 
438d9656f86SBALATON Zoltan DEFINE_MACHINE("amigaone", amigaone_machine_init)
439