xref: /openbmc/qemu/hw/vmapple/vmapple.c (revision f5e6e13124440797308d2c044f44d9e655fcb74d)
1*59f4d655SAlexander Graf /*
2*59f4d655SAlexander Graf  * VMApple machine emulation
3*59f4d655SAlexander Graf  *
4*59f4d655SAlexander Graf  * Copyright © 2023 Amazon.com, Inc. or its affiliates. All Rights Reserved.
5*59f4d655SAlexander Graf  *
6*59f4d655SAlexander Graf  * This work is licensed under the terms of the GNU GPL, version 2 or later.
7*59f4d655SAlexander Graf  * See the COPYING file in the top-level directory.
8*59f4d655SAlexander Graf  *
9*59f4d655SAlexander Graf  * SPDX-License-Identifier: GPL-2.0-or-later
10*59f4d655SAlexander Graf  *
11*59f4d655SAlexander Graf  * VMApple is the device model that the macOS built-in hypervisor called
12*59f4d655SAlexander Graf  * "Virtualization.framework" exposes to Apple Silicon macOS guests. The
13*59f4d655SAlexander Graf  * machine model in this file implements the same device model in QEMU, but
14*59f4d655SAlexander Graf  * does not use any code from Virtualization.Framework.
15*59f4d655SAlexander Graf  */
16*59f4d655SAlexander Graf 
17*59f4d655SAlexander Graf #include "qemu/osdep.h"
18*59f4d655SAlexander Graf #include "qemu/bitops.h"
19*59f4d655SAlexander Graf #include "qemu/datadir.h"
20*59f4d655SAlexander Graf #include "qemu/error-report.h"
21*59f4d655SAlexander Graf #include "qemu/guest-random.h"
22*59f4d655SAlexander Graf #include "qemu/help-texts.h"
23*59f4d655SAlexander Graf #include "qemu/log.h"
24*59f4d655SAlexander Graf #include "qemu/module.h"
25*59f4d655SAlexander Graf #include "qemu/option.h"
26*59f4d655SAlexander Graf #include "qemu/units.h"
27*59f4d655SAlexander Graf #include "monitor/qdev.h"
28*59f4d655SAlexander Graf #include "hw/boards.h"
29*59f4d655SAlexander Graf #include "hw/irq.h"
30*59f4d655SAlexander Graf #include "hw/loader.h"
31*59f4d655SAlexander Graf #include "hw/qdev-properties.h"
32*59f4d655SAlexander Graf #include "hw/sysbus.h"
33*59f4d655SAlexander Graf #include "hw/usb.h"
34*59f4d655SAlexander Graf #include "hw/arm/boot.h"
35*59f4d655SAlexander Graf #include "hw/arm/primecell.h"
36*59f4d655SAlexander Graf #include "hw/char/pl011.h"
37*59f4d655SAlexander Graf #include "hw/intc/arm_gic.h"
38*59f4d655SAlexander Graf #include "hw/intc/arm_gicv3_common.h"
39*59f4d655SAlexander Graf #include "hw/misc/pvpanic.h"
40*59f4d655SAlexander Graf #include "hw/pci-host/gpex.h"
41*59f4d655SAlexander Graf #include "hw/usb/hcd-xhci-pci.h"
42*59f4d655SAlexander Graf #include "hw/virtio/virtio-pci.h"
43*59f4d655SAlexander Graf #include "hw/vmapple/vmapple.h"
44*59f4d655SAlexander Graf #include "net/net.h"
45*59f4d655SAlexander Graf #include "qapi/error.h"
46*59f4d655SAlexander Graf #include "qapi/visitor.h"
47*59f4d655SAlexander Graf #include "qapi/qapi-visit-common.h"
48*59f4d655SAlexander Graf #include "qobject/qlist.h"
49*59f4d655SAlexander Graf #include "standard-headers/linux/input.h"
50*59f4d655SAlexander Graf #include "system/hvf.h"
51*59f4d655SAlexander Graf #include "system/reset.h"
52*59f4d655SAlexander Graf #include "system/runstate.h"
53*59f4d655SAlexander Graf #include "system/system.h"
54*59f4d655SAlexander Graf 
55*59f4d655SAlexander Graf struct VMAppleMachineState {
56*59f4d655SAlexander Graf     MachineState parent;
57*59f4d655SAlexander Graf 
58*59f4d655SAlexander Graf     Notifier machine_done;
59*59f4d655SAlexander Graf     struct arm_boot_info bootinfo;
60*59f4d655SAlexander Graf     const MemMapEntry *memmap;
61*59f4d655SAlexander Graf     const int *irqmap;
62*59f4d655SAlexander Graf     DeviceState *gic;
63*59f4d655SAlexander Graf     DeviceState *cfg;
64*59f4d655SAlexander Graf     DeviceState *pvpanic;
65*59f4d655SAlexander Graf     Notifier powerdown_notifier;
66*59f4d655SAlexander Graf     PCIBus *bus;
67*59f4d655SAlexander Graf     MemoryRegion fw_mr;
68*59f4d655SAlexander Graf     MemoryRegion ecam_alias;
69*59f4d655SAlexander Graf     uint64_t uuid;
70*59f4d655SAlexander Graf };
71*59f4d655SAlexander Graf 
72*59f4d655SAlexander Graf #define TYPE_VMAPPLE_MACHINE   MACHINE_TYPE_NAME("vmapple")
73*59f4d655SAlexander Graf OBJECT_DECLARE_SIMPLE_TYPE(VMAppleMachineState, VMAPPLE_MACHINE)
74*59f4d655SAlexander Graf 
75*59f4d655SAlexander Graf /* Number of external interrupt lines to configure the GIC with */
76*59f4d655SAlexander Graf #define NUM_IRQS 256
77*59f4d655SAlexander Graf 
78*59f4d655SAlexander Graf enum {
79*59f4d655SAlexander Graf     VMAPPLE_FIRMWARE,
80*59f4d655SAlexander Graf     VMAPPLE_CONFIG,
81*59f4d655SAlexander Graf     VMAPPLE_MEM,
82*59f4d655SAlexander Graf     VMAPPLE_GIC_DIST,
83*59f4d655SAlexander Graf     VMAPPLE_GIC_REDIST,
84*59f4d655SAlexander Graf     VMAPPLE_UART,
85*59f4d655SAlexander Graf     VMAPPLE_RTC,
86*59f4d655SAlexander Graf     VMAPPLE_PCIE,
87*59f4d655SAlexander Graf     VMAPPLE_PCIE_MMIO,
88*59f4d655SAlexander Graf     VMAPPLE_PCIE_ECAM,
89*59f4d655SAlexander Graf     VMAPPLE_GPIO,
90*59f4d655SAlexander Graf     VMAPPLE_PVPANIC,
91*59f4d655SAlexander Graf     VMAPPLE_APV_GFX,
92*59f4d655SAlexander Graf     VMAPPLE_APV_IOSFC,
93*59f4d655SAlexander Graf     VMAPPLE_AES_1,
94*59f4d655SAlexander Graf     VMAPPLE_AES_2,
95*59f4d655SAlexander Graf     VMAPPLE_BDOOR,
96*59f4d655SAlexander Graf     VMAPPLE_MEMMAP_LAST,
97*59f4d655SAlexander Graf };
98*59f4d655SAlexander Graf 
99*59f4d655SAlexander Graf static const MemMapEntry memmap[] = {
100*59f4d655SAlexander Graf     [VMAPPLE_FIRMWARE] =           { 0x00100000, 0x00100000 },
101*59f4d655SAlexander Graf     [VMAPPLE_CONFIG] =             { 0x00400000, 0x00010000 },
102*59f4d655SAlexander Graf 
103*59f4d655SAlexander Graf     [VMAPPLE_GIC_DIST] =           { 0x10000000, 0x00010000 },
104*59f4d655SAlexander Graf     [VMAPPLE_GIC_REDIST] =         { 0x10010000, 0x00400000 },
105*59f4d655SAlexander Graf 
106*59f4d655SAlexander Graf     [VMAPPLE_UART] =               { 0x20010000, 0x00010000 },
107*59f4d655SAlexander Graf     [VMAPPLE_RTC] =                { 0x20050000, 0x00001000 },
108*59f4d655SAlexander Graf     [VMAPPLE_GPIO] =               { 0x20060000, 0x00001000 },
109*59f4d655SAlexander Graf     [VMAPPLE_PVPANIC] =            { 0x20070000, 0x00000002 },
110*59f4d655SAlexander Graf     [VMAPPLE_BDOOR] =              { 0x30000000, 0x00200000 },
111*59f4d655SAlexander Graf     [VMAPPLE_APV_GFX] =            { 0x30200000, 0x00010000 },
112*59f4d655SAlexander Graf     [VMAPPLE_APV_IOSFC] =          { 0x30210000, 0x00010000 },
113*59f4d655SAlexander Graf     [VMAPPLE_AES_1] =              { 0x30220000, 0x00004000 },
114*59f4d655SAlexander Graf     [VMAPPLE_AES_2] =              { 0x30230000, 0x00004000 },
115*59f4d655SAlexander Graf     [VMAPPLE_PCIE_ECAM] =          { 0x40000000, 0x10000000 },
116*59f4d655SAlexander Graf     [VMAPPLE_PCIE_MMIO] =          { 0x50000000, 0x1fff0000 },
117*59f4d655SAlexander Graf 
118*59f4d655SAlexander Graf     /* Actual RAM size depends on configuration */
119*59f4d655SAlexander Graf     [VMAPPLE_MEM] =                { 0x70000000ULL, GiB},
120*59f4d655SAlexander Graf };
121*59f4d655SAlexander Graf 
122*59f4d655SAlexander Graf static const int irqmap[] = {
123*59f4d655SAlexander Graf     [VMAPPLE_UART] = 1,
124*59f4d655SAlexander Graf     [VMAPPLE_RTC] = 2,
125*59f4d655SAlexander Graf     [VMAPPLE_GPIO] = 0x5,
126*59f4d655SAlexander Graf     [VMAPPLE_APV_IOSFC] = 0x10,
127*59f4d655SAlexander Graf     [VMAPPLE_APV_GFX] = 0x11,
128*59f4d655SAlexander Graf     [VMAPPLE_AES_1] = 0x12,
129*59f4d655SAlexander Graf     [VMAPPLE_PCIE] = 0x20,
130*59f4d655SAlexander Graf };
131*59f4d655SAlexander Graf 
132*59f4d655SAlexander Graf #define GPEX_NUM_IRQS 16
133*59f4d655SAlexander Graf 
create_bdif(VMAppleMachineState * vms,MemoryRegion * mem)134*59f4d655SAlexander Graf static void create_bdif(VMAppleMachineState *vms, MemoryRegion *mem)
135*59f4d655SAlexander Graf {
136*59f4d655SAlexander Graf     DeviceState *bdif;
137*59f4d655SAlexander Graf     SysBusDevice *bdif_sb;
138*59f4d655SAlexander Graf     DriveInfo *di_aux = drive_get(IF_PFLASH, 0, 0);
139*59f4d655SAlexander Graf     DriveInfo *di_root = drive_get(IF_PFLASH, 0, 1);
140*59f4d655SAlexander Graf 
141*59f4d655SAlexander Graf     if (!di_aux) {
142*59f4d655SAlexander Graf         error_report("No AUX device. Please specify one as pflash drive.");
143*59f4d655SAlexander Graf         exit(1);
144*59f4d655SAlexander Graf     }
145*59f4d655SAlexander Graf 
146*59f4d655SAlexander Graf     if (!di_root) {
147*59f4d655SAlexander Graf         /* Fall back to the first IF_VIRTIO device as root device */
148*59f4d655SAlexander Graf         di_root = drive_get(IF_VIRTIO, 0, 0);
149*59f4d655SAlexander Graf     }
150*59f4d655SAlexander Graf 
151*59f4d655SAlexander Graf     if (!di_root) {
152*59f4d655SAlexander Graf         error_report("No root device. Please specify one as virtio drive.");
153*59f4d655SAlexander Graf         exit(1);
154*59f4d655SAlexander Graf     }
155*59f4d655SAlexander Graf 
156*59f4d655SAlexander Graf     /* PV backdoor device */
157*59f4d655SAlexander Graf     bdif = qdev_new(TYPE_VMAPPLE_BDIF);
158*59f4d655SAlexander Graf     bdif_sb = SYS_BUS_DEVICE(bdif);
159*59f4d655SAlexander Graf     sysbus_mmio_map(bdif_sb, 0, vms->memmap[VMAPPLE_BDOOR].base);
160*59f4d655SAlexander Graf 
161*59f4d655SAlexander Graf     qdev_prop_set_drive(DEVICE(bdif), "aux", blk_by_legacy_dinfo(di_aux));
162*59f4d655SAlexander Graf     qdev_prop_set_drive(DEVICE(bdif), "root", blk_by_legacy_dinfo(di_root));
163*59f4d655SAlexander Graf 
164*59f4d655SAlexander Graf     sysbus_realize_and_unref(bdif_sb, &error_fatal);
165*59f4d655SAlexander Graf }
166*59f4d655SAlexander Graf 
create_pvpanic(VMAppleMachineState * vms,MemoryRegion * mem)167*59f4d655SAlexander Graf static void create_pvpanic(VMAppleMachineState *vms, MemoryRegion *mem)
168*59f4d655SAlexander Graf {
169*59f4d655SAlexander Graf     SysBusDevice *pvpanic;
170*59f4d655SAlexander Graf 
171*59f4d655SAlexander Graf     vms->pvpanic = qdev_new(TYPE_PVPANIC_MMIO_DEVICE);
172*59f4d655SAlexander Graf     pvpanic = SYS_BUS_DEVICE(vms->pvpanic);
173*59f4d655SAlexander Graf     sysbus_mmio_map(pvpanic, 0, vms->memmap[VMAPPLE_PVPANIC].base);
174*59f4d655SAlexander Graf 
175*59f4d655SAlexander Graf     sysbus_realize_and_unref(pvpanic, &error_fatal);
176*59f4d655SAlexander Graf }
177*59f4d655SAlexander Graf 
create_cfg(VMAppleMachineState * vms,MemoryRegion * mem,Error ** errp)178*59f4d655SAlexander Graf static bool create_cfg(VMAppleMachineState *vms, MemoryRegion *mem,
179*59f4d655SAlexander Graf                        Error **errp)
180*59f4d655SAlexander Graf {
181*59f4d655SAlexander Graf     ERRP_GUARD();
182*59f4d655SAlexander Graf     SysBusDevice *cfg;
183*59f4d655SAlexander Graf     MachineState *machine = MACHINE(vms);
184*59f4d655SAlexander Graf     uint32_t rnd = 1;
185*59f4d655SAlexander Graf 
186*59f4d655SAlexander Graf     vms->cfg = qdev_new(TYPE_VMAPPLE_CFG);
187*59f4d655SAlexander Graf     cfg = SYS_BUS_DEVICE(vms->cfg);
188*59f4d655SAlexander Graf     sysbus_mmio_map(cfg, 0, vms->memmap[VMAPPLE_CONFIG].base);
189*59f4d655SAlexander Graf 
190*59f4d655SAlexander Graf     qemu_guest_getrandom_nofail(&rnd, sizeof(rnd));
191*59f4d655SAlexander Graf 
192*59f4d655SAlexander Graf     qdev_prop_set_uint32(vms->cfg, "nr-cpus", machine->smp.cpus);
193*59f4d655SAlexander Graf     qdev_prop_set_uint64(vms->cfg, "ecid", vms->uuid);
194*59f4d655SAlexander Graf     qdev_prop_set_uint64(vms->cfg, "ram-size", machine->ram_size);
195*59f4d655SAlexander Graf     qdev_prop_set_uint32(vms->cfg, "rnd", rnd);
196*59f4d655SAlexander Graf 
197*59f4d655SAlexander Graf     if (!sysbus_realize_and_unref(cfg, errp)) {
198*59f4d655SAlexander Graf         error_prepend(errp, "Error creating vmapple cfg device: ");
199*59f4d655SAlexander Graf         return false;
200*59f4d655SAlexander Graf     }
201*59f4d655SAlexander Graf 
202*59f4d655SAlexander Graf     return true;
203*59f4d655SAlexander Graf }
204*59f4d655SAlexander Graf 
create_gfx(VMAppleMachineState * vms,MemoryRegion * mem)205*59f4d655SAlexander Graf static void create_gfx(VMAppleMachineState *vms, MemoryRegion *mem)
206*59f4d655SAlexander Graf {
207*59f4d655SAlexander Graf     int irq_gfx = vms->irqmap[VMAPPLE_APV_GFX];
208*59f4d655SAlexander Graf     int irq_iosfc = vms->irqmap[VMAPPLE_APV_IOSFC];
209*59f4d655SAlexander Graf     SysBusDevice *gfx;
210*59f4d655SAlexander Graf 
211*59f4d655SAlexander Graf     gfx = SYS_BUS_DEVICE(qdev_new("apple-gfx-mmio"));
212*59f4d655SAlexander Graf     sysbus_mmio_map(gfx, 0, vms->memmap[VMAPPLE_APV_GFX].base);
213*59f4d655SAlexander Graf     sysbus_mmio_map(gfx, 1, vms->memmap[VMAPPLE_APV_IOSFC].base);
214*59f4d655SAlexander Graf     sysbus_connect_irq(gfx, 0, qdev_get_gpio_in(vms->gic, irq_gfx));
215*59f4d655SAlexander Graf     sysbus_connect_irq(gfx, 1, qdev_get_gpio_in(vms->gic, irq_iosfc));
216*59f4d655SAlexander Graf     sysbus_realize_and_unref(gfx, &error_fatal);
217*59f4d655SAlexander Graf }
218*59f4d655SAlexander Graf 
create_aes(VMAppleMachineState * vms,MemoryRegion * mem)219*59f4d655SAlexander Graf static void create_aes(VMAppleMachineState *vms, MemoryRegion *mem)
220*59f4d655SAlexander Graf {
221*59f4d655SAlexander Graf     int irq = vms->irqmap[VMAPPLE_AES_1];
222*59f4d655SAlexander Graf     SysBusDevice *aes;
223*59f4d655SAlexander Graf 
224*59f4d655SAlexander Graf     aes = SYS_BUS_DEVICE(qdev_new(TYPE_APPLE_AES));
225*59f4d655SAlexander Graf     sysbus_mmio_map(aes, 0, vms->memmap[VMAPPLE_AES_1].base);
226*59f4d655SAlexander Graf     sysbus_mmio_map(aes, 1, vms->memmap[VMAPPLE_AES_2].base);
227*59f4d655SAlexander Graf     sysbus_connect_irq(aes, 0, qdev_get_gpio_in(vms->gic, irq));
228*59f4d655SAlexander Graf     sysbus_realize_and_unref(aes, &error_fatal);
229*59f4d655SAlexander Graf }
230*59f4d655SAlexander Graf 
arm_gic_ppi_index(int cpu_nr,int ppi_index)231*59f4d655SAlexander Graf static int arm_gic_ppi_index(int cpu_nr, int ppi_index)
232*59f4d655SAlexander Graf {
233*59f4d655SAlexander Graf     return NUM_IRQS + cpu_nr * GIC_INTERNAL + ppi_index;
234*59f4d655SAlexander Graf }
235*59f4d655SAlexander Graf 
create_gic(VMAppleMachineState * vms,MemoryRegion * mem)236*59f4d655SAlexander Graf static void create_gic(VMAppleMachineState *vms, MemoryRegion *mem)
237*59f4d655SAlexander Graf {
238*59f4d655SAlexander Graf     MachineState *ms = MACHINE(vms);
239*59f4d655SAlexander Graf     /* We create a standalone GIC */
240*59f4d655SAlexander Graf     SysBusDevice *gicbusdev;
241*59f4d655SAlexander Graf     QList *redist_region_count;
242*59f4d655SAlexander Graf     int i;
243*59f4d655SAlexander Graf     unsigned int smp_cpus = ms->smp.cpus;
244*59f4d655SAlexander Graf 
245*59f4d655SAlexander Graf     vms->gic = qdev_new(gicv3_class_name());
246*59f4d655SAlexander Graf     qdev_prop_set_uint32(vms->gic, "revision", 3);
247*59f4d655SAlexander Graf     qdev_prop_set_uint32(vms->gic, "num-cpu", smp_cpus);
248*59f4d655SAlexander Graf     /*
249*59f4d655SAlexander Graf      * Note that the num-irq property counts both internal and external
250*59f4d655SAlexander Graf      * interrupts; there are always 32 of the former (mandated by GIC spec).
251*59f4d655SAlexander Graf      */
252*59f4d655SAlexander Graf     qdev_prop_set_uint32(vms->gic, "num-irq", NUM_IRQS + 32);
253*59f4d655SAlexander Graf 
254*59f4d655SAlexander Graf     uint32_t redist0_capacity =
255*59f4d655SAlexander Graf                 vms->memmap[VMAPPLE_GIC_REDIST].size / GICV3_REDIST_SIZE;
256*59f4d655SAlexander Graf     uint32_t redist0_count = MIN(smp_cpus, redist0_capacity);
257*59f4d655SAlexander Graf 
258*59f4d655SAlexander Graf     redist_region_count = qlist_new();
259*59f4d655SAlexander Graf     qlist_append_int(redist_region_count, redist0_count);
260*59f4d655SAlexander Graf     qdev_prop_set_array(vms->gic, "redist-region-count", redist_region_count);
261*59f4d655SAlexander Graf 
262*59f4d655SAlexander Graf     gicbusdev = SYS_BUS_DEVICE(vms->gic);
263*59f4d655SAlexander Graf     sysbus_realize_and_unref(gicbusdev, &error_fatal);
264*59f4d655SAlexander Graf     sysbus_mmio_map(gicbusdev, 0, vms->memmap[VMAPPLE_GIC_DIST].base);
265*59f4d655SAlexander Graf     sysbus_mmio_map(gicbusdev, 1, vms->memmap[VMAPPLE_GIC_REDIST].base);
266*59f4d655SAlexander Graf 
267*59f4d655SAlexander Graf     /*
268*59f4d655SAlexander Graf      * Wire the outputs from each CPU's generic timer and the GICv3
269*59f4d655SAlexander Graf      * maintenance interrupt signal to the appropriate GIC PPI inputs,
270*59f4d655SAlexander Graf      * and the GIC's IRQ/FIQ/VIRQ/VFIQ interrupt outputs to the CPU's inputs.
271*59f4d655SAlexander Graf      */
272*59f4d655SAlexander Graf     for (i = 0; i < smp_cpus; i++) {
273*59f4d655SAlexander Graf         DeviceState *cpudev = DEVICE(qemu_get_cpu(i));
274*59f4d655SAlexander Graf 
275*59f4d655SAlexander Graf         /* Map the virt timer to PPI 27 */
276*59f4d655SAlexander Graf         qdev_connect_gpio_out(cpudev, GTIMER_VIRT,
277*59f4d655SAlexander Graf                               qdev_get_gpio_in(vms->gic,
278*59f4d655SAlexander Graf                                                arm_gic_ppi_index(i, 27)));
279*59f4d655SAlexander Graf 
280*59f4d655SAlexander Graf         /* Map the GIC IRQ and FIQ lines to CPU */
281*59f4d655SAlexander Graf         sysbus_connect_irq(gicbusdev, i, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));
282*59f4d655SAlexander Graf         sysbus_connect_irq(gicbusdev, i + smp_cpus,
283*59f4d655SAlexander Graf                            qdev_get_gpio_in(cpudev, ARM_CPU_FIQ));
284*59f4d655SAlexander Graf     }
285*59f4d655SAlexander Graf }
286*59f4d655SAlexander Graf 
create_uart(const VMAppleMachineState * vms,int uart,MemoryRegion * mem,Chardev * chr)287*59f4d655SAlexander Graf static void create_uart(const VMAppleMachineState *vms, int uart,
288*59f4d655SAlexander Graf                         MemoryRegion *mem, Chardev *chr)
289*59f4d655SAlexander Graf {
290*59f4d655SAlexander Graf     hwaddr base = vms->memmap[uart].base;
291*59f4d655SAlexander Graf     int irq = vms->irqmap[uart];
292*59f4d655SAlexander Graf     DeviceState *dev = qdev_new(TYPE_PL011);
293*59f4d655SAlexander Graf     SysBusDevice *s = SYS_BUS_DEVICE(dev);
294*59f4d655SAlexander Graf 
295*59f4d655SAlexander Graf     qdev_prop_set_chr(dev, "chardev", chr);
296*59f4d655SAlexander Graf     sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
297*59f4d655SAlexander Graf     memory_region_add_subregion(mem, base,
298*59f4d655SAlexander Graf                                 sysbus_mmio_get_region(s, 0));
299*59f4d655SAlexander Graf     sysbus_connect_irq(s, 0, qdev_get_gpio_in(vms->gic, irq));
300*59f4d655SAlexander Graf }
301*59f4d655SAlexander Graf 
create_rtc(const VMAppleMachineState * vms)302*59f4d655SAlexander Graf static void create_rtc(const VMAppleMachineState *vms)
303*59f4d655SAlexander Graf {
304*59f4d655SAlexander Graf     hwaddr base = vms->memmap[VMAPPLE_RTC].base;
305*59f4d655SAlexander Graf     int irq = vms->irqmap[VMAPPLE_RTC];
306*59f4d655SAlexander Graf 
307*59f4d655SAlexander Graf     sysbus_create_simple("pl031", base, qdev_get_gpio_in(vms->gic, irq));
308*59f4d655SAlexander Graf }
309*59f4d655SAlexander Graf 
310*59f4d655SAlexander Graf static DeviceState *gpio_key_dev;
vmapple_powerdown_req(Notifier * n,void * opaque)311*59f4d655SAlexander Graf static void vmapple_powerdown_req(Notifier *n, void *opaque)
312*59f4d655SAlexander Graf {
313*59f4d655SAlexander Graf     /* use gpio Pin 3 for power button event */
314*59f4d655SAlexander Graf     qemu_set_irq(qdev_get_gpio_in(gpio_key_dev, 0), 1);
315*59f4d655SAlexander Graf }
316*59f4d655SAlexander Graf 
create_gpio_devices(const VMAppleMachineState * vms,int gpio,MemoryRegion * mem)317*59f4d655SAlexander Graf static void create_gpio_devices(const VMAppleMachineState *vms, int gpio,
318*59f4d655SAlexander Graf                                 MemoryRegion *mem)
319*59f4d655SAlexander Graf {
320*59f4d655SAlexander Graf     DeviceState *pl061_dev;
321*59f4d655SAlexander Graf     hwaddr base = vms->memmap[gpio].base;
322*59f4d655SAlexander Graf     int irq = vms->irqmap[gpio];
323*59f4d655SAlexander Graf     SysBusDevice *s;
324*59f4d655SAlexander Graf 
325*59f4d655SAlexander Graf     pl061_dev = qdev_new("pl061");
326*59f4d655SAlexander Graf     /* Pull lines down to 0 if not driven by the PL061 */
327*59f4d655SAlexander Graf     qdev_prop_set_uint32(pl061_dev, "pullups", 0);
328*59f4d655SAlexander Graf     qdev_prop_set_uint32(pl061_dev, "pulldowns", 0xff);
329*59f4d655SAlexander Graf     s = SYS_BUS_DEVICE(pl061_dev);
330*59f4d655SAlexander Graf     sysbus_realize_and_unref(s, &error_fatal);
331*59f4d655SAlexander Graf     memory_region_add_subregion(mem, base, sysbus_mmio_get_region(s, 0));
332*59f4d655SAlexander Graf     sysbus_connect_irq(s, 0, qdev_get_gpio_in(vms->gic, irq));
333*59f4d655SAlexander Graf     gpio_key_dev = sysbus_create_simple("gpio-key", -1,
334*59f4d655SAlexander Graf                                         qdev_get_gpio_in(pl061_dev, 3));
335*59f4d655SAlexander Graf }
336*59f4d655SAlexander Graf 
vmapple_firmware_init(VMAppleMachineState * vms,MemoryRegion * sysmem)337*59f4d655SAlexander Graf static void vmapple_firmware_init(VMAppleMachineState *vms,
338*59f4d655SAlexander Graf                                   MemoryRegion *sysmem)
339*59f4d655SAlexander Graf {
340*59f4d655SAlexander Graf     hwaddr size = vms->memmap[VMAPPLE_FIRMWARE].size;
341*59f4d655SAlexander Graf     hwaddr base = vms->memmap[VMAPPLE_FIRMWARE].base;
342*59f4d655SAlexander Graf     const char *bios_name;
343*59f4d655SAlexander Graf     int image_size;
344*59f4d655SAlexander Graf     char *fname;
345*59f4d655SAlexander Graf 
346*59f4d655SAlexander Graf     bios_name = MACHINE(vms)->firmware;
347*59f4d655SAlexander Graf     if (!bios_name) {
348*59f4d655SAlexander Graf         error_report("No firmware specified");
349*59f4d655SAlexander Graf         exit(1);
350*59f4d655SAlexander Graf     }
351*59f4d655SAlexander Graf 
352*59f4d655SAlexander Graf     fname = qemu_find_file(QEMU_FILE_TYPE_BIOS, bios_name);
353*59f4d655SAlexander Graf     if (!fname) {
354*59f4d655SAlexander Graf         error_report("Could not find ROM image '%s'", bios_name);
355*59f4d655SAlexander Graf         exit(1);
356*59f4d655SAlexander Graf     }
357*59f4d655SAlexander Graf 
358*59f4d655SAlexander Graf     memory_region_init_ram(&vms->fw_mr, NULL, "firmware", size, &error_fatal);
359*59f4d655SAlexander Graf     image_size = load_image_mr(fname, &vms->fw_mr);
360*59f4d655SAlexander Graf 
361*59f4d655SAlexander Graf     g_free(fname);
362*59f4d655SAlexander Graf     if (image_size < 0) {
363*59f4d655SAlexander Graf         error_report("Could not load ROM image '%s'", bios_name);
364*59f4d655SAlexander Graf         exit(1);
365*59f4d655SAlexander Graf     }
366*59f4d655SAlexander Graf 
367*59f4d655SAlexander Graf     memory_region_add_subregion(get_system_memory(), base, &vms->fw_mr);
368*59f4d655SAlexander Graf }
369*59f4d655SAlexander Graf 
create_pcie(VMAppleMachineState * vms)370*59f4d655SAlexander Graf static void create_pcie(VMAppleMachineState *vms)
371*59f4d655SAlexander Graf {
372*59f4d655SAlexander Graf     hwaddr base_mmio = vms->memmap[VMAPPLE_PCIE_MMIO].base;
373*59f4d655SAlexander Graf     hwaddr size_mmio = vms->memmap[VMAPPLE_PCIE_MMIO].size;
374*59f4d655SAlexander Graf     hwaddr base_ecam = vms->memmap[VMAPPLE_PCIE_ECAM].base;
375*59f4d655SAlexander Graf     hwaddr size_ecam = vms->memmap[VMAPPLE_PCIE_ECAM].size;
376*59f4d655SAlexander Graf     int irq = vms->irqmap[VMAPPLE_PCIE];
377*59f4d655SAlexander Graf     MemoryRegion *mmio_alias;
378*59f4d655SAlexander Graf     MemoryRegion *mmio_reg;
379*59f4d655SAlexander Graf     MemoryRegion *ecam_reg;
380*59f4d655SAlexander Graf     DeviceState *dev;
381*59f4d655SAlexander Graf     int i;
382*59f4d655SAlexander Graf     PCIHostState *pci;
383*59f4d655SAlexander Graf     DeviceState *usb_controller;
384*59f4d655SAlexander Graf     USBBus *usb_bus;
385*59f4d655SAlexander Graf 
386*59f4d655SAlexander Graf     dev = qdev_new(TYPE_GPEX_HOST);
387*59f4d655SAlexander Graf     qdev_prop_set_uint32(dev, "num-irqs", GPEX_NUM_IRQS);
388*59f4d655SAlexander Graf     sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
389*59f4d655SAlexander Graf 
390*59f4d655SAlexander Graf     /* Map only the first size_ecam bytes of ECAM space */
391*59f4d655SAlexander Graf     ecam_reg = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
392*59f4d655SAlexander Graf     memory_region_init_alias(&vms->ecam_alias, OBJECT(dev), "pcie-ecam",
393*59f4d655SAlexander Graf                              ecam_reg, 0, size_ecam);
394*59f4d655SAlexander Graf     memory_region_add_subregion(get_system_memory(), base_ecam,
395*59f4d655SAlexander Graf                                 &vms->ecam_alias);
396*59f4d655SAlexander Graf 
397*59f4d655SAlexander Graf     /*
398*59f4d655SAlexander Graf      * Map the MMIO window from [0x50000000-0x7fff0000] in PCI space into
399*59f4d655SAlexander Graf      * system address space at [0x50000000-0x7fff0000].
400*59f4d655SAlexander Graf      */
401*59f4d655SAlexander Graf     mmio_alias = g_new0(MemoryRegion, 1);
402*59f4d655SAlexander Graf     mmio_reg = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 1);
403*59f4d655SAlexander Graf     memory_region_init_alias(mmio_alias, OBJECT(dev), "pcie-mmio",
404*59f4d655SAlexander Graf                              mmio_reg, base_mmio, size_mmio);
405*59f4d655SAlexander Graf     memory_region_add_subregion(get_system_memory(), base_mmio, mmio_alias);
406*59f4d655SAlexander Graf 
407*59f4d655SAlexander Graf     for (i = 0; i < GPEX_NUM_IRQS; i++) {
408*59f4d655SAlexander Graf         sysbus_connect_irq(SYS_BUS_DEVICE(dev), i,
409*59f4d655SAlexander Graf                            qdev_get_gpio_in(vms->gic, irq + i));
410*59f4d655SAlexander Graf         gpex_set_irq_num(GPEX_HOST(dev), i, irq + i);
411*59f4d655SAlexander Graf     }
412*59f4d655SAlexander Graf 
413*59f4d655SAlexander Graf     pci = PCI_HOST_BRIDGE(dev);
414*59f4d655SAlexander Graf     vms->bus = pci->bus;
415*59f4d655SAlexander Graf     g_assert(vms->bus);
416*59f4d655SAlexander Graf 
417*59f4d655SAlexander Graf     while ((dev = qemu_create_nic_device("virtio-net-pci", true, NULL))) {
418*59f4d655SAlexander Graf         qdev_realize_and_unref(dev, BUS(vms->bus), &error_fatal);
419*59f4d655SAlexander Graf     }
420*59f4d655SAlexander Graf 
421*59f4d655SAlexander Graf     if (defaults_enabled()) {
422*59f4d655SAlexander Graf         usb_controller = qdev_new(TYPE_QEMU_XHCI);
423*59f4d655SAlexander Graf         qdev_realize_and_unref(usb_controller, BUS(pci->bus), &error_fatal);
424*59f4d655SAlexander Graf 
425*59f4d655SAlexander Graf         usb_bus = USB_BUS(object_resolve_type_unambiguous(TYPE_USB_BUS,
426*59f4d655SAlexander Graf                                                           &error_fatal));
427*59f4d655SAlexander Graf         usb_create_simple(usb_bus, "usb-kbd");
428*59f4d655SAlexander Graf         usb_create_simple(usb_bus, "usb-tablet");
429*59f4d655SAlexander Graf     }
430*59f4d655SAlexander Graf }
431*59f4d655SAlexander Graf 
vmapple_reset(void * opaque)432*59f4d655SAlexander Graf static void vmapple_reset(void *opaque)
433*59f4d655SAlexander Graf {
434*59f4d655SAlexander Graf     VMAppleMachineState *vms = opaque;
435*59f4d655SAlexander Graf     hwaddr base = vms->memmap[VMAPPLE_FIRMWARE].base;
436*59f4d655SAlexander Graf 
437*59f4d655SAlexander Graf     cpu_set_pc(first_cpu, base);
438*59f4d655SAlexander Graf }
439*59f4d655SAlexander Graf 
mach_vmapple_init(MachineState * machine)440*59f4d655SAlexander Graf static void mach_vmapple_init(MachineState *machine)
441*59f4d655SAlexander Graf {
442*59f4d655SAlexander Graf     VMAppleMachineState *vms = VMAPPLE_MACHINE(machine);
443*59f4d655SAlexander Graf     MachineClass *mc = MACHINE_GET_CLASS(machine);
444*59f4d655SAlexander Graf     const CPUArchIdList *possible_cpus;
445*59f4d655SAlexander Graf     MemoryRegion *sysmem = get_system_memory();
446*59f4d655SAlexander Graf     int n;
447*59f4d655SAlexander Graf     unsigned int smp_cpus = machine->smp.cpus;
448*59f4d655SAlexander Graf     unsigned int max_cpus = machine->smp.max_cpus;
449*59f4d655SAlexander Graf 
450*59f4d655SAlexander Graf     vms->memmap = memmap;
451*59f4d655SAlexander Graf     machine->usb = true;
452*59f4d655SAlexander Graf 
453*59f4d655SAlexander Graf     possible_cpus = mc->possible_cpu_arch_ids(machine);
454*59f4d655SAlexander Graf     assert(possible_cpus->len == max_cpus);
455*59f4d655SAlexander Graf     for (n = 0; n < possible_cpus->len; n++) {
456*59f4d655SAlexander Graf         Object *cpu;
457*59f4d655SAlexander Graf         CPUState *cs;
458*59f4d655SAlexander Graf 
459*59f4d655SAlexander Graf         if (n >= smp_cpus) {
460*59f4d655SAlexander Graf             break;
461*59f4d655SAlexander Graf         }
462*59f4d655SAlexander Graf 
463*59f4d655SAlexander Graf         cpu = object_new(possible_cpus->cpus[n].type);
464*59f4d655SAlexander Graf         object_property_set_int(cpu, "mp-affinity",
465*59f4d655SAlexander Graf                                 possible_cpus->cpus[n].arch_id, &error_fatal);
466*59f4d655SAlexander Graf 
467*59f4d655SAlexander Graf         cs = CPU(cpu);
468*59f4d655SAlexander Graf         cs->cpu_index = n;
469*59f4d655SAlexander Graf 
470*59f4d655SAlexander Graf         numa_cpu_pre_plug(&possible_cpus->cpus[cs->cpu_index], DEVICE(cpu),
471*59f4d655SAlexander Graf                           &error_fatal);
472*59f4d655SAlexander Graf 
473*59f4d655SAlexander Graf         if (object_property_find(cpu, "has_el3")) {
474*59f4d655SAlexander Graf             object_property_set_bool(cpu, "has_el3", false, &error_fatal);
475*59f4d655SAlexander Graf         }
476*59f4d655SAlexander Graf         if (object_property_find(cpu, "has_el2")) {
477*59f4d655SAlexander Graf             object_property_set_bool(cpu, "has_el2", false, &error_fatal);
478*59f4d655SAlexander Graf         }
479*59f4d655SAlexander Graf         object_property_set_int(cpu, "psci-conduit", QEMU_PSCI_CONDUIT_HVC,
480*59f4d655SAlexander Graf                                 &error_fatal);
481*59f4d655SAlexander Graf 
482*59f4d655SAlexander Graf         /* Secondary CPUs start in PSCI powered-down state */
483*59f4d655SAlexander Graf         if (n > 0) {
484*59f4d655SAlexander Graf             object_property_set_bool(cpu, "start-powered-off", true,
485*59f4d655SAlexander Graf                                      &error_fatal);
486*59f4d655SAlexander Graf         }
487*59f4d655SAlexander Graf 
488*59f4d655SAlexander Graf         object_property_set_link(cpu, "memory", OBJECT(sysmem), &error_abort);
489*59f4d655SAlexander Graf         qdev_realize(DEVICE(cpu), NULL, &error_fatal);
490*59f4d655SAlexander Graf         object_unref(cpu);
491*59f4d655SAlexander Graf     }
492*59f4d655SAlexander Graf 
493*59f4d655SAlexander Graf     memory_region_add_subregion(sysmem, vms->memmap[VMAPPLE_MEM].base,
494*59f4d655SAlexander Graf                                 machine->ram);
495*59f4d655SAlexander Graf 
496*59f4d655SAlexander Graf     create_gic(vms, sysmem);
497*59f4d655SAlexander Graf     create_bdif(vms, sysmem);
498*59f4d655SAlexander Graf     create_pvpanic(vms, sysmem);
499*59f4d655SAlexander Graf     create_aes(vms, sysmem);
500*59f4d655SAlexander Graf     create_gfx(vms, sysmem);
501*59f4d655SAlexander Graf     create_uart(vms, VMAPPLE_UART, sysmem, serial_hd(0));
502*59f4d655SAlexander Graf     create_rtc(vms);
503*59f4d655SAlexander Graf     create_pcie(vms);
504*59f4d655SAlexander Graf 
505*59f4d655SAlexander Graf     create_gpio_devices(vms, VMAPPLE_GPIO, sysmem);
506*59f4d655SAlexander Graf 
507*59f4d655SAlexander Graf     vmapple_firmware_init(vms, sysmem);
508*59f4d655SAlexander Graf     create_cfg(vms, sysmem, &error_fatal);
509*59f4d655SAlexander Graf 
510*59f4d655SAlexander Graf     /* connect powerdown request */
511*59f4d655SAlexander Graf     vms->powerdown_notifier.notify = vmapple_powerdown_req;
512*59f4d655SAlexander Graf     qemu_register_powerdown_notifier(&vms->powerdown_notifier);
513*59f4d655SAlexander Graf 
514*59f4d655SAlexander Graf     vms->bootinfo.ram_size = machine->ram_size;
515*59f4d655SAlexander Graf     vms->bootinfo.board_id = -1;
516*59f4d655SAlexander Graf     vms->bootinfo.loader_start = vms->memmap[VMAPPLE_MEM].base;
517*59f4d655SAlexander Graf     vms->bootinfo.skip_dtb_autoload = true;
518*59f4d655SAlexander Graf     vms->bootinfo.firmware_loaded = true;
519*59f4d655SAlexander Graf     arm_load_kernel(ARM_CPU(first_cpu), machine, &vms->bootinfo);
520*59f4d655SAlexander Graf 
521*59f4d655SAlexander Graf     qemu_register_reset(vmapple_reset, vms);
522*59f4d655SAlexander Graf }
523*59f4d655SAlexander Graf 
524*59f4d655SAlexander Graf static CpuInstanceProperties
vmapple_cpu_index_to_props(MachineState * ms,unsigned cpu_index)525*59f4d655SAlexander Graf vmapple_cpu_index_to_props(MachineState *ms, unsigned cpu_index)
526*59f4d655SAlexander Graf {
527*59f4d655SAlexander Graf     MachineClass *mc = MACHINE_GET_CLASS(ms);
528*59f4d655SAlexander Graf     const CPUArchIdList *possible_cpus = mc->possible_cpu_arch_ids(ms);
529*59f4d655SAlexander Graf 
530*59f4d655SAlexander Graf     assert(cpu_index < possible_cpus->len);
531*59f4d655SAlexander Graf     return possible_cpus->cpus[cpu_index].props;
532*59f4d655SAlexander Graf }
533*59f4d655SAlexander Graf 
534*59f4d655SAlexander Graf 
vmapple_get_default_cpu_node_id(const MachineState * ms,int idx)535*59f4d655SAlexander Graf static int64_t vmapple_get_default_cpu_node_id(const MachineState *ms, int idx)
536*59f4d655SAlexander Graf {
537*59f4d655SAlexander Graf     return idx % ms->numa_state->num_nodes;
538*59f4d655SAlexander Graf }
539*59f4d655SAlexander Graf 
vmapple_possible_cpu_arch_ids(MachineState * ms)540*59f4d655SAlexander Graf static const CPUArchIdList *vmapple_possible_cpu_arch_ids(MachineState *ms)
541*59f4d655SAlexander Graf {
542*59f4d655SAlexander Graf     int n;
543*59f4d655SAlexander Graf     unsigned int max_cpus = ms->smp.max_cpus;
544*59f4d655SAlexander Graf 
545*59f4d655SAlexander Graf     if (ms->possible_cpus) {
546*59f4d655SAlexander Graf         assert(ms->possible_cpus->len == max_cpus);
547*59f4d655SAlexander Graf         return ms->possible_cpus;
548*59f4d655SAlexander Graf     }
549*59f4d655SAlexander Graf 
550*59f4d655SAlexander Graf     ms->possible_cpus = g_malloc0(sizeof(CPUArchIdList) +
551*59f4d655SAlexander Graf                                   sizeof(CPUArchId) * max_cpus);
552*59f4d655SAlexander Graf     ms->possible_cpus->len = max_cpus;
553*59f4d655SAlexander Graf     for (n = 0; n < ms->possible_cpus->len; n++) {
554*59f4d655SAlexander Graf         ms->possible_cpus->cpus[n].type = ms->cpu_type;
555*59f4d655SAlexander Graf         ms->possible_cpus->cpus[n].arch_id =
556*59f4d655SAlexander Graf             arm_build_mp_affinity(n, GICV3_TARGETLIST_BITS);
557*59f4d655SAlexander Graf         ms->possible_cpus->cpus[n].props.has_thread_id = true;
558*59f4d655SAlexander Graf         ms->possible_cpus->cpus[n].props.thread_id = n;
559*59f4d655SAlexander Graf     }
560*59f4d655SAlexander Graf     return ms->possible_cpus;
561*59f4d655SAlexander Graf }
562*59f4d655SAlexander Graf 
563*59f4d655SAlexander Graf static GlobalProperty vmapple_compat_defaults[] = {
564*59f4d655SAlexander Graf     { TYPE_VIRTIO_PCI, "disable-legacy", "on" },
565*59f4d655SAlexander Graf     /*
566*59f4d655SAlexander Graf      * macOS XHCI driver attempts to schedule events onto even rings 1 & 2
567*59f4d655SAlexander Graf      * even when (as here) there is no MSI(-X) support. Disabling interrupter
568*59f4d655SAlexander Graf      * mapping in the XHCI controller works around the problem.
569*59f4d655SAlexander Graf      */
570*59f4d655SAlexander Graf     { TYPE_XHCI_PCI, "conditional-intr-mapping", "on" },
571*59f4d655SAlexander Graf };
572*59f4d655SAlexander Graf 
vmapple_machine_class_init(ObjectClass * oc,void * data)573*59f4d655SAlexander Graf static void vmapple_machine_class_init(ObjectClass *oc, void *data)
574*59f4d655SAlexander Graf {
575*59f4d655SAlexander Graf     MachineClass *mc = MACHINE_CLASS(oc);
576*59f4d655SAlexander Graf 
577*59f4d655SAlexander Graf     mc->init = mach_vmapple_init;
578*59f4d655SAlexander Graf     mc->max_cpus = 32;
579*59f4d655SAlexander Graf     mc->block_default_type = IF_VIRTIO;
580*59f4d655SAlexander Graf     mc->no_cdrom = 1;
581*59f4d655SAlexander Graf     mc->pci_allow_0_address = true;
582*59f4d655SAlexander Graf     mc->minimum_page_bits = 12;
583*59f4d655SAlexander Graf     mc->possible_cpu_arch_ids = vmapple_possible_cpu_arch_ids;
584*59f4d655SAlexander Graf     mc->cpu_index_to_instance_props = vmapple_cpu_index_to_props;
585*59f4d655SAlexander Graf     mc->default_cpu_type = ARM_CPU_TYPE_NAME("host");
586*59f4d655SAlexander Graf     mc->get_default_cpu_node_id = vmapple_get_default_cpu_node_id;
587*59f4d655SAlexander Graf     mc->default_ram_id = "mach-vmapple.ram";
588*59f4d655SAlexander Graf     mc->desc = "Apple aarch64 Virtual Machine";
589*59f4d655SAlexander Graf 
590*59f4d655SAlexander Graf     compat_props_add(mc->compat_props, vmapple_compat_defaults,
591*59f4d655SAlexander Graf                      G_N_ELEMENTS(vmapple_compat_defaults));
592*59f4d655SAlexander Graf }
593*59f4d655SAlexander Graf 
vmapple_instance_init(Object * obj)594*59f4d655SAlexander Graf static void vmapple_instance_init(Object *obj)
595*59f4d655SAlexander Graf {
596*59f4d655SAlexander Graf     VMAppleMachineState *vms = VMAPPLE_MACHINE(obj);
597*59f4d655SAlexander Graf 
598*59f4d655SAlexander Graf     vms->irqmap = irqmap;
599*59f4d655SAlexander Graf 
600*59f4d655SAlexander Graf     object_property_add_uint64_ptr(obj, "uuid", &vms->uuid,
601*59f4d655SAlexander Graf                                    OBJ_PROP_FLAG_READWRITE);
602*59f4d655SAlexander Graf     object_property_set_description(obj, "uuid", "Machine UUID (SDOM)");
603*59f4d655SAlexander Graf }
604*59f4d655SAlexander Graf 
605*59f4d655SAlexander Graf static const TypeInfo vmapple_machine_info = {
606*59f4d655SAlexander Graf     .name          = TYPE_VMAPPLE_MACHINE,
607*59f4d655SAlexander Graf     .parent        = TYPE_MACHINE,
608*59f4d655SAlexander Graf     .instance_size = sizeof(VMAppleMachineState),
609*59f4d655SAlexander Graf     .class_init    = vmapple_machine_class_init,
610*59f4d655SAlexander Graf     .instance_init = vmapple_instance_init,
611*59f4d655SAlexander Graf };
612*59f4d655SAlexander Graf 
machvmapple_machine_init(void)613*59f4d655SAlexander Graf static void machvmapple_machine_init(void)
614*59f4d655SAlexander Graf {
615*59f4d655SAlexander Graf     type_register_static(&vmapple_machine_info);
616*59f4d655SAlexander Graf }
617*59f4d655SAlexander Graf type_init(machvmapple_machine_init);
618*59f4d655SAlexander Graf 
619