Headers in include/sysemu/ are not only related to system *emulation*, they are also used by virtualization. Rename as system/ which is clearer. Files renamed manually then mechanical change using sed tool. Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org> Reviewed-by: Richard Henderson <richard.henderson@linaro.org> Tested-by: Lei Yang <leiyang@redhat.com> Message-Id: <20241203172445.28576-1-philmd@linaro.org>
		
			
				
	
	
		
			641 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			641 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Arm MPS3 board emulation for Cortex-R-based FPGA images.
 | 
						|
 * (For M-profile images see mps2.c and mps2tz.c.)
 | 
						|
 *
 | 
						|
 * Copyright (c) 2017 Linaro Limited
 | 
						|
 * Written by Peter Maydell
 | 
						|
 *
 | 
						|
 *  This program is free software; you can redistribute it and/or modify
 | 
						|
 *  it under the terms of the GNU General Public License version 2 or
 | 
						|
 *  (at your option) any later version.
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * The MPS3 is an FPGA based dev board. This file handles FPGA images
 | 
						|
 * which use the Cortex-R CPUs. We model these separately from the
 | 
						|
 * M-profile images, because on M-profile the FPGA image is based on
 | 
						|
 * a "Subsystem for Embedded" which is similar to an SoC, whereas
 | 
						|
 * the R-profile FPGA images don't have that abstraction layer.
 | 
						|
 *
 | 
						|
 * We model the following FPGA images here:
 | 
						|
 *  "mps3-an536" -- dual Cortex-R52 as documented in Arm Application Note AN536
 | 
						|
 *
 | 
						|
 * Application Note AN536:
 | 
						|
 * https://developer.arm.com/documentation/dai0536/latest/
 | 
						|
 */
 | 
						|
 | 
						|
#include "qemu/osdep.h"
 | 
						|
#include "qemu/units.h"
 | 
						|
#include "qapi/error.h"
 | 
						|
#include "qapi/qmp/qlist.h"
 | 
						|
#include "exec/address-spaces.h"
 | 
						|
#include "cpu.h"
 | 
						|
#include "system/system.h"
 | 
						|
#include "hw/boards.h"
 | 
						|
#include "hw/or-irq.h"
 | 
						|
#include "hw/qdev-clock.h"
 | 
						|
#include "hw/qdev-properties.h"
 | 
						|
#include "hw/arm/boot.h"
 | 
						|
#include "hw/arm/bsa.h"
 | 
						|
#include "hw/char/cmsdk-apb-uart.h"
 | 
						|
#include "hw/i2c/arm_sbcon_i2c.h"
 | 
						|
#include "hw/intc/arm_gicv3.h"
 | 
						|
#include "hw/misc/mps2-scc.h"
 | 
						|
#include "hw/misc/mps2-fpgaio.h"
 | 
						|
#include "hw/misc/unimp.h"
 | 
						|
#include "hw/net/lan9118.h"
 | 
						|
#include "hw/rtc/pl031.h"
 | 
						|
#include "hw/ssi/pl022.h"
 | 
						|
#include "hw/timer/cmsdk-apb-dualtimer.h"
 | 
						|
#include "hw/watchdog/cmsdk-apb-watchdog.h"
 | 
						|
 | 
						|
/* Define the layout of RAM and ROM in a board */
 | 
						|
typedef struct RAMInfo {
 | 
						|
    const char *name;
 | 
						|
    hwaddr base;
 | 
						|
    hwaddr size;
 | 
						|
    int mrindex; /* index into rams[]; -1 for the system RAM block */
 | 
						|
    int flags;
 | 
						|
} RAMInfo;
 | 
						|
 | 
						|
/*
 | 
						|
 * The MPS3 DDR is 3GiB, but on a 32-bit host QEMU doesn't permit
 | 
						|
 * emulation of that much guest RAM, so artificially make it smaller.
 | 
						|
 */
 | 
						|
#if HOST_LONG_BITS == 32
 | 
						|
#define MPS3_DDR_SIZE (1 * GiB)
 | 
						|
#else
 | 
						|
#define MPS3_DDR_SIZE (3 * GiB)
 | 
						|
#endif
 | 
						|
 | 
						|
/*
 | 
						|
 * Flag values:
 | 
						|
 * IS_MAIN: this is the main machine RAM
 | 
						|
 * IS_ROM: this area is read-only
 | 
						|
 */
 | 
						|
#define IS_MAIN 1
 | 
						|
#define IS_ROM 2
 | 
						|
 | 
						|
#define MPS3R_RAM_MAX 9
 | 
						|
#define MPS3R_CPU_MAX 2
 | 
						|
#define MPS3R_UART_MAX 4 /* shared UART count */
 | 
						|
 | 
						|
#define PERIPHBASE 0xf0000000
 | 
						|
#define NUM_SPIS 96
 | 
						|
 | 
						|
typedef enum MPS3RFPGAType {
 | 
						|
    FPGA_AN536,
 | 
						|
} MPS3RFPGAType;
 | 
						|
 | 
						|
struct MPS3RMachineClass {
 | 
						|
    MachineClass parent;
 | 
						|
    MPS3RFPGAType fpga_type;
 | 
						|
    const RAMInfo *raminfo;
 | 
						|
    hwaddr loader_start;
 | 
						|
};
 | 
						|
 | 
						|
struct MPS3RMachineState {
 | 
						|
    MachineState parent;
 | 
						|
    struct arm_boot_info bootinfo;
 | 
						|
    MemoryRegion ram[MPS3R_RAM_MAX];
 | 
						|
    Object *cpu[MPS3R_CPU_MAX];
 | 
						|
    MemoryRegion cpu_sysmem[MPS3R_CPU_MAX];
 | 
						|
    MemoryRegion sysmem_alias[MPS3R_CPU_MAX];
 | 
						|
    MemoryRegion cpu_ram[MPS3R_CPU_MAX];
 | 
						|
    GICv3State gic;
 | 
						|
    /* per-CPU UARTs followed by the shared UARTs */
 | 
						|
    CMSDKAPBUART uart[MPS3R_CPU_MAX + MPS3R_UART_MAX];
 | 
						|
    OrIRQState cpu_uart_oflow[MPS3R_CPU_MAX];
 | 
						|
    OrIRQState uart_oflow;
 | 
						|
    CMSDKAPBWatchdog watchdog;
 | 
						|
    CMSDKAPBDualTimer dualtimer;
 | 
						|
    ArmSbconI2CState i2c[5];
 | 
						|
    PL022State spi[3];
 | 
						|
    MPS2SCC scc;
 | 
						|
    MPS2FPGAIO fpgaio;
 | 
						|
    UnimplementedDeviceState i2s_audio;
 | 
						|
    PL031State rtc;
 | 
						|
    Clock *clk;
 | 
						|
};
 | 
						|
 | 
						|
#define TYPE_MPS3R_MACHINE "mps3r"
 | 
						|
#define TYPE_MPS3R_AN536_MACHINE MACHINE_TYPE_NAME("mps3-an536")
 | 
						|
 | 
						|
OBJECT_DECLARE_TYPE(MPS3RMachineState, MPS3RMachineClass, MPS3R_MACHINE)
 | 
						|
 | 
						|
/*
 | 
						|
 * Main clock frequency CLK in Hz (50MHz). In the image there are also
 | 
						|
 * ACLK, MCLK, GPUCLK and PERIPHCLK at the same frequency; for our
 | 
						|
 * model we just roll them all into one.
 | 
						|
 */
 | 
						|
#define CLK_FRQ 50000000
 | 
						|
 | 
						|
static const RAMInfo an536_raminfo[] = {
 | 
						|
    {
 | 
						|
        .name = "ATCM",
 | 
						|
        .base = 0x00000000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 0,
 | 
						|
    }, {
 | 
						|
        /* We model the QSPI flash as simple ROM for now */
 | 
						|
        .name = "QSPI",
 | 
						|
        .base = 0x08000000,
 | 
						|
        .size = 0x00800000,
 | 
						|
        .flags = IS_ROM,
 | 
						|
        .mrindex = 1,
 | 
						|
    }, {
 | 
						|
        .name = "BRAM",
 | 
						|
        .base = 0x10000000,
 | 
						|
        .size = 0x00080000,
 | 
						|
        .mrindex = 2,
 | 
						|
    }, {
 | 
						|
        .name = "DDR",
 | 
						|
        .base = 0x20000000,
 | 
						|
        .size = MPS3_DDR_SIZE,
 | 
						|
        .mrindex = -1,
 | 
						|
    }, {
 | 
						|
        .name = "ATCM0",
 | 
						|
        .base = 0xee000000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 3,
 | 
						|
    }, {
 | 
						|
        .name = "BTCM0",
 | 
						|
        .base = 0xee100000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 4,
 | 
						|
    }, {
 | 
						|
        .name = "CTCM0",
 | 
						|
        .base = 0xee200000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 5,
 | 
						|
    }, {
 | 
						|
        .name = "ATCM1",
 | 
						|
        .base = 0xee400000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 6,
 | 
						|
    }, {
 | 
						|
        .name = "BTCM1",
 | 
						|
        .base = 0xee500000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 7,
 | 
						|
    }, {
 | 
						|
        .name = "CTCM1",
 | 
						|
        .base = 0xee600000,
 | 
						|
        .size = 0x00008000,
 | 
						|
        .mrindex = 8,
 | 
						|
    }, {
 | 
						|
        .name = NULL,
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
static const int an536_oscclk[] = {
 | 
						|
    24000000, /* 24MHz reference for RTC and timers */
 | 
						|
    50000000, /* 50MHz ACLK */
 | 
						|
    50000000, /* 50MHz MCLK */
 | 
						|
    50000000, /* 50MHz GPUCLK */
 | 
						|
    24576000, /* 24.576MHz AUDCLK */
 | 
						|
    23750000, /* 23.75MHz HDLCDCLK */
 | 
						|
    100000000, /* 100MHz DDR4_REF_CLK */
 | 
						|
};
 | 
						|
 | 
						|
static MemoryRegion *mr_for_raminfo(MPS3RMachineState *mms,
 | 
						|
                                    const RAMInfo *raminfo)
 | 
						|
{
 | 
						|
    /* Return an initialized MemoryRegion for the RAMInfo. */
 | 
						|
    MemoryRegion *ram;
 | 
						|
 | 
						|
    if (raminfo->mrindex < 0) {
 | 
						|
        /* Means this RAMInfo is for QEMU's "system memory" */
 | 
						|
        MachineState *machine = MACHINE(mms);
 | 
						|
        assert(!(raminfo->flags & IS_ROM));
 | 
						|
        return machine->ram;
 | 
						|
    }
 | 
						|
 | 
						|
    assert(raminfo->mrindex < MPS3R_RAM_MAX);
 | 
						|
    ram = &mms->ram[raminfo->mrindex];
 | 
						|
 | 
						|
    memory_region_init_ram(ram, NULL, raminfo->name,
 | 
						|
                           raminfo->size, &error_fatal);
 | 
						|
    if (raminfo->flags & IS_ROM) {
 | 
						|
        memory_region_set_readonly(ram, true);
 | 
						|
    }
 | 
						|
    return ram;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * There is no defined secondary boot protocol for Linux for the AN536,
 | 
						|
 * because real hardware has a restriction that atomic operations between
 | 
						|
 * the two CPUs do not function correctly, and so true SMP is not
 | 
						|
 * possible. Therefore for cases where the user is directly booting
 | 
						|
 * a kernel, we treat the system as essentially uniprocessor, and
 | 
						|
 * put the secondary CPU into power-off state (as if the user on the
 | 
						|
 * real hardware had configured the secondary to be halted via the
 | 
						|
 * SCC config registers).
 | 
						|
 *
 | 
						|
 * Note that the default secondary boot code would not work here anyway
 | 
						|
 * as it assumes a GICv2, and we have a GICv3.
 | 
						|
 */
 | 
						|
static void mps3r_write_secondary_boot(ARMCPU *cpu,
 | 
						|
                                       const struct arm_boot_info *info)
 | 
						|
{
 | 
						|
    /*
 | 
						|
     * Power the secondary CPU off. This means we don't need to write any
 | 
						|
     * boot code into guest memory. Note that the 'cpu' argument to this
 | 
						|
     * function is the primary CPU we passed to arm_load_kernel(), not
 | 
						|
     * the secondary. Loop around all the other CPUs, as the boot.c
 | 
						|
     * code does for the "disable secondaries if PSCI is enabled" case.
 | 
						|
     */
 | 
						|
    for (CPUState *cs = first_cpu; cs; cs = CPU_NEXT(cs)) {
 | 
						|
        if (cs != first_cpu) {
 | 
						|
            object_property_set_bool(OBJECT(cs), "start-powered-off", true,
 | 
						|
                                     &error_abort);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void mps3r_secondary_cpu_reset(ARMCPU *cpu,
 | 
						|
                                      const struct arm_boot_info *info)
 | 
						|
{
 | 
						|
    /* We don't need to do anything here because the CPU will be off */
 | 
						|
}
 | 
						|
 | 
						|
static void create_gic(MPS3RMachineState *mms, MemoryRegion *sysmem)
 | 
						|
{
 | 
						|
    MachineState *machine = MACHINE(mms);
 | 
						|
    DeviceState *gicdev;
 | 
						|
    QList *redist_region_count;
 | 
						|
 | 
						|
    object_initialize_child(OBJECT(mms), "gic", &mms->gic, TYPE_ARM_GICV3);
 | 
						|
    gicdev = DEVICE(&mms->gic);
 | 
						|
    qdev_prop_set_uint32(gicdev, "num-cpu", machine->smp.cpus);
 | 
						|
    qdev_prop_set_uint32(gicdev, "num-irq", NUM_SPIS + GIC_INTERNAL);
 | 
						|
    redist_region_count = qlist_new();
 | 
						|
    qlist_append_int(redist_region_count, machine->smp.cpus);
 | 
						|
    qdev_prop_set_array(gicdev, "redist-region-count", redist_region_count);
 | 
						|
    object_property_set_link(OBJECT(&mms->gic), "sysmem",
 | 
						|
                             OBJECT(sysmem), &error_fatal);
 | 
						|
    sysbus_realize(SYS_BUS_DEVICE(&mms->gic), &error_fatal);
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->gic), 0, PERIPHBASE);
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->gic), 1, PERIPHBASE + 0x100000);
 | 
						|
    /*
 | 
						|
     * Wire the outputs from each CPU's generic timer and the GICv3
 | 
						|
     * maintenance interrupt signal to the appropriate GIC PPI inputs,
 | 
						|
     * and the GIC's IRQ/FIQ/VIRQ/VFIQ interrupt outputs to the CPU's inputs.
 | 
						|
     */
 | 
						|
    for (int i = 0; i < machine->smp.cpus; i++) {
 | 
						|
        DeviceState *cpudev = DEVICE(mms->cpu[i]);
 | 
						|
        SysBusDevice *gicsbd = SYS_BUS_DEVICE(&mms->gic);
 | 
						|
        int intidbase = NUM_SPIS + i * GIC_INTERNAL;
 | 
						|
        int irq;
 | 
						|
        /*
 | 
						|
         * Mapping from the output timer irq lines from the CPU to the
 | 
						|
         * GIC PPI inputs used for this board. This isn't a BSA board,
 | 
						|
         * but it uses the standard convention for the PPI numbers.
 | 
						|
         */
 | 
						|
        const int timer_irq[] = {
 | 
						|
            [GTIMER_PHYS] = ARCH_TIMER_NS_EL1_IRQ,
 | 
						|
            [GTIMER_VIRT] = ARCH_TIMER_VIRT_IRQ,
 | 
						|
            [GTIMER_HYP]  = ARCH_TIMER_NS_EL2_IRQ,
 | 
						|
        };
 | 
						|
 | 
						|
        for (irq = 0; irq < ARRAY_SIZE(timer_irq); irq++) {
 | 
						|
            qdev_connect_gpio_out(cpudev, irq,
 | 
						|
                                  qdev_get_gpio_in(gicdev,
 | 
						|
                                                   intidbase + timer_irq[irq]));
 | 
						|
        }
 | 
						|
 | 
						|
        qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt", 0,
 | 
						|
                                    qdev_get_gpio_in(gicdev,
 | 
						|
                                                     intidbase + ARCH_GIC_MAINT_IRQ));
 | 
						|
 | 
						|
        qdev_connect_gpio_out_named(cpudev, "pmu-interrupt", 0,
 | 
						|
                                    qdev_get_gpio_in(gicdev,
 | 
						|
                                                     intidbase + VIRTUAL_PMU_IRQ));
 | 
						|
 | 
						|
        sysbus_connect_irq(gicsbd, i,
 | 
						|
                           qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));
 | 
						|
        sysbus_connect_irq(gicsbd, i + machine->smp.cpus,
 | 
						|
                           qdev_get_gpio_in(cpudev, ARM_CPU_FIQ));
 | 
						|
        sysbus_connect_irq(gicsbd, i + 2 * machine->smp.cpus,
 | 
						|
                           qdev_get_gpio_in(cpudev, ARM_CPU_VIRQ));
 | 
						|
        sysbus_connect_irq(gicsbd, i + 3 * machine->smp.cpus,
 | 
						|
                           qdev_get_gpio_in(cpudev, ARM_CPU_VFIQ));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Create UART uartno, and map it into the MemoryRegion mem at address baseaddr.
 | 
						|
 * The qemu_irq arguments are where we connect the various IRQs from the UART.
 | 
						|
 */
 | 
						|
static void create_uart(MPS3RMachineState *mms, int uartno, MemoryRegion *mem,
 | 
						|
                        hwaddr baseaddr, qemu_irq txirq, qemu_irq rxirq,
 | 
						|
                        qemu_irq txoverirq, qemu_irq rxoverirq,
 | 
						|
                        qemu_irq combirq)
 | 
						|
{
 | 
						|
    g_autofree char *s = g_strdup_printf("uart%d", uartno);
 | 
						|
    SysBusDevice *sbd;
 | 
						|
 | 
						|
    assert(uartno < ARRAY_SIZE(mms->uart));
 | 
						|
    object_initialize_child(OBJECT(mms), s, &mms->uart[uartno],
 | 
						|
                            TYPE_CMSDK_APB_UART);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->uart[uartno]), "pclk-frq", CLK_FRQ);
 | 
						|
    qdev_prop_set_chr(DEVICE(&mms->uart[uartno]), "chardev", serial_hd(uartno));
 | 
						|
    sbd = SYS_BUS_DEVICE(&mms->uart[uartno]);
 | 
						|
    sysbus_realize(sbd, &error_fatal);
 | 
						|
    memory_region_add_subregion(mem, baseaddr,
 | 
						|
                                sysbus_mmio_get_region(sbd, 0));
 | 
						|
    sysbus_connect_irq(sbd, 0, txirq);
 | 
						|
    sysbus_connect_irq(sbd, 1, rxirq);
 | 
						|
    sysbus_connect_irq(sbd, 2, txoverirq);
 | 
						|
    sysbus_connect_irq(sbd, 3, rxoverirq);
 | 
						|
    sysbus_connect_irq(sbd, 4, combirq);
 | 
						|
}
 | 
						|
 | 
						|
static void mps3r_common_init(MachineState *machine)
 | 
						|
{
 | 
						|
    MPS3RMachineState *mms = MPS3R_MACHINE(machine);
 | 
						|
    MPS3RMachineClass *mmc = MPS3R_MACHINE_GET_CLASS(mms);
 | 
						|
    MemoryRegion *sysmem = get_system_memory();
 | 
						|
    DeviceState *gicdev;
 | 
						|
    QList *oscclk;
 | 
						|
 | 
						|
    mms->clk = clock_new(OBJECT(machine), "CLK");
 | 
						|
    clock_set_hz(mms->clk, CLK_FRQ);
 | 
						|
 | 
						|
    for (const RAMInfo *ri = mmc->raminfo; ri->name; ri++) {
 | 
						|
        MemoryRegion *mr = mr_for_raminfo(mms, ri);
 | 
						|
        memory_region_add_subregion(sysmem, ri->base, mr);
 | 
						|
    }
 | 
						|
 | 
						|
    assert(machine->smp.cpus <= MPS3R_CPU_MAX);
 | 
						|
    for (int i = 0; i < machine->smp.cpus; i++) {
 | 
						|
        g_autofree char *sysmem_name = g_strdup_printf("cpu-%d-memory", i);
 | 
						|
        g_autofree char *ramname = g_strdup_printf("cpu-%d-memory", i);
 | 
						|
        g_autofree char *alias_name = g_strdup_printf("sysmem-alias-%d", i);
 | 
						|
 | 
						|
        /*
 | 
						|
         * Each CPU has some private RAM/peripherals, so create the container
 | 
						|
         * which will house those, with the whole-machine system memory being
 | 
						|
         * used where there's no CPU-specific device. Note that we need the
 | 
						|
         * sysmem_alias aliases because we can't put one MR (the original
 | 
						|
         * 'sysmem') into more than one other MR.
 | 
						|
         */
 | 
						|
        memory_region_init(&mms->cpu_sysmem[i], OBJECT(machine),
 | 
						|
                           sysmem_name, UINT64_MAX);
 | 
						|
        memory_region_init_alias(&mms->sysmem_alias[i], OBJECT(machine),
 | 
						|
                                 alias_name, sysmem, 0, UINT64_MAX);
 | 
						|
        memory_region_add_subregion_overlap(&mms->cpu_sysmem[i], 0,
 | 
						|
                                            &mms->sysmem_alias[i], -1);
 | 
						|
 | 
						|
        mms->cpu[i] = object_new(machine->cpu_type);
 | 
						|
        object_property_set_link(mms->cpu[i], "memory",
 | 
						|
                                 OBJECT(&mms->cpu_sysmem[i]), &error_abort);
 | 
						|
        object_property_set_int(mms->cpu[i], "reset-cbar",
 | 
						|
                                PERIPHBASE, &error_abort);
 | 
						|
        qdev_realize(DEVICE(mms->cpu[i]), NULL, &error_fatal);
 | 
						|
        object_unref(mms->cpu[i]);
 | 
						|
 | 
						|
        /* Per-CPU RAM */
 | 
						|
        memory_region_init_ram(&mms->cpu_ram[i], NULL, ramname,
 | 
						|
                               0x1000, &error_fatal);
 | 
						|
        memory_region_add_subregion(&mms->cpu_sysmem[i], 0xe7c01000,
 | 
						|
                                    &mms->cpu_ram[i]);
 | 
						|
    }
 | 
						|
 | 
						|
    create_gic(mms, sysmem);
 | 
						|
    gicdev = DEVICE(&mms->gic);
 | 
						|
 | 
						|
    /*
 | 
						|
     * UARTs 0 and 1 are per-CPU; their interrupts are wired to
 | 
						|
     * the relevant CPU's PPI 0..3, aka INTID 16..19
 | 
						|
     */
 | 
						|
    for (int i = 0; i < machine->smp.cpus; i++) {
 | 
						|
        int intidbase = NUM_SPIS + i * GIC_INTERNAL;
 | 
						|
        g_autofree char *s = g_strdup_printf("cpu-uart-oflow-orgate%d", i);
 | 
						|
        DeviceState *orgate;
 | 
						|
 | 
						|
        /* The two overflow IRQs from the UART are ORed together into PPI 3 */
 | 
						|
        object_initialize_child(OBJECT(mms), s, &mms->cpu_uart_oflow[i],
 | 
						|
                                TYPE_OR_IRQ);
 | 
						|
        orgate = DEVICE(&mms->cpu_uart_oflow[i]);
 | 
						|
        qdev_prop_set_uint32(orgate, "num-lines", 2);
 | 
						|
        qdev_realize(orgate, NULL, &error_fatal);
 | 
						|
        qdev_connect_gpio_out(orgate, 0,
 | 
						|
                              qdev_get_gpio_in(gicdev, intidbase + 19));
 | 
						|
 | 
						|
        create_uart(mms, i, &mms->cpu_sysmem[i], 0xe7c00000,
 | 
						|
                    qdev_get_gpio_in(gicdev, intidbase + 17), /* tx */
 | 
						|
                    qdev_get_gpio_in(gicdev, intidbase + 16), /* rx */
 | 
						|
                    qdev_get_gpio_in(orgate, 0), /* txover */
 | 
						|
                    qdev_get_gpio_in(orgate, 1), /* rxover */
 | 
						|
                    qdev_get_gpio_in(gicdev, intidbase + 18) /* combined */);
 | 
						|
    }
 | 
						|
    /*
 | 
						|
     * UARTs 2 to 5 are whole-system; all overflow IRQs are ORed
 | 
						|
     * together into IRQ 17
 | 
						|
     */
 | 
						|
    object_initialize_child(OBJECT(mms), "uart-oflow-orgate",
 | 
						|
                            &mms->uart_oflow, TYPE_OR_IRQ);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->uart_oflow), "num-lines",
 | 
						|
                         MPS3R_UART_MAX * 2);
 | 
						|
    qdev_realize(DEVICE(&mms->uart_oflow), NULL, &error_fatal);
 | 
						|
    qdev_connect_gpio_out(DEVICE(&mms->uart_oflow), 0,
 | 
						|
                          qdev_get_gpio_in(gicdev, 17));
 | 
						|
 | 
						|
    for (int i = 0; i < MPS3R_UART_MAX; i++) {
 | 
						|
        hwaddr baseaddr = 0xe0205000 + i * 0x1000;
 | 
						|
        int rxirq = 5 + i * 2, txirq = 6 + i * 2, combirq = 13 + i;
 | 
						|
 | 
						|
        create_uart(mms, i + MPS3R_CPU_MAX, sysmem, baseaddr,
 | 
						|
                    qdev_get_gpio_in(gicdev, txirq),
 | 
						|
                    qdev_get_gpio_in(gicdev, rxirq),
 | 
						|
                    qdev_get_gpio_in(DEVICE(&mms->uart_oflow), i * 2),
 | 
						|
                    qdev_get_gpio_in(DEVICE(&mms->uart_oflow), i * 2 + 1),
 | 
						|
                    qdev_get_gpio_in(gicdev, combirq));
 | 
						|
    }
 | 
						|
 | 
						|
    for (int i = 0; i < 4; i++) {
 | 
						|
        /* CMSDK GPIO controllers */
 | 
						|
        g_autofree char *s = g_strdup_printf("gpio%d", i);
 | 
						|
        create_unimplemented_device(s, 0xe0000000 + i * 0x1000, 0x1000);
 | 
						|
    }
 | 
						|
 | 
						|
    object_initialize_child(OBJECT(mms), "watchdog", &mms->watchdog,
 | 
						|
                            TYPE_CMSDK_APB_WATCHDOG);
 | 
						|
    qdev_connect_clock_in(DEVICE(&mms->watchdog), "WDOGCLK", mms->clk);
 | 
						|
    sysbus_realize(SYS_BUS_DEVICE(&mms->watchdog), &error_fatal);
 | 
						|
    sysbus_connect_irq(SYS_BUS_DEVICE(&mms->watchdog), 0,
 | 
						|
                       qdev_get_gpio_in(gicdev, 0));
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->watchdog), 0, 0xe0100000);
 | 
						|
 | 
						|
    object_initialize_child(OBJECT(mms), "dualtimer", &mms->dualtimer,
 | 
						|
                            TYPE_CMSDK_APB_DUALTIMER);
 | 
						|
    qdev_connect_clock_in(DEVICE(&mms->dualtimer), "TIMCLK", mms->clk);
 | 
						|
    sysbus_realize(SYS_BUS_DEVICE(&mms->dualtimer), &error_fatal);
 | 
						|
    sysbus_connect_irq(SYS_BUS_DEVICE(&mms->dualtimer), 0,
 | 
						|
                       qdev_get_gpio_in(gicdev, 3));
 | 
						|
    sysbus_connect_irq(SYS_BUS_DEVICE(&mms->dualtimer), 1,
 | 
						|
                       qdev_get_gpio_in(gicdev, 1));
 | 
						|
    sysbus_connect_irq(SYS_BUS_DEVICE(&mms->dualtimer), 2,
 | 
						|
                       qdev_get_gpio_in(gicdev, 2));
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->dualtimer), 0, 0xe0101000);
 | 
						|
 | 
						|
    for (int i = 0; i < ARRAY_SIZE(mms->i2c); i++) {
 | 
						|
        static const hwaddr i2cbase[] = {0xe0102000,    /* Touch */
 | 
						|
                                         0xe0103000,    /* Audio */
 | 
						|
                                         0xe0107000,    /* Shield0 */
 | 
						|
                                         0xe0108000,    /* Shield1 */
 | 
						|
                                         0xe0109000};   /* DDR4 EEPROM */
 | 
						|
        g_autofree char *s = g_strdup_printf("i2c%d", i);
 | 
						|
 | 
						|
        object_initialize_child(OBJECT(mms), s, &mms->i2c[i],
 | 
						|
                                TYPE_ARM_SBCON_I2C);
 | 
						|
        sysbus_realize(SYS_BUS_DEVICE(&mms->i2c[i]), &error_fatal);
 | 
						|
        sysbus_mmio_map(SYS_BUS_DEVICE(&mms->i2c[i]), 0, i2cbase[i]);
 | 
						|
        if (i != 2 && i != 3) {
 | 
						|
            /*
 | 
						|
             * internal-only bus: mark it full to avoid user-created
 | 
						|
             * i2c devices being plugged into it.
 | 
						|
             */
 | 
						|
            qbus_mark_full(qdev_get_child_bus(DEVICE(&mms->i2c[i]), "i2c"));
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    for (int i = 0; i < ARRAY_SIZE(mms->spi); i++) {
 | 
						|
        g_autofree char *s = g_strdup_printf("spi%d", i);
 | 
						|
        hwaddr baseaddr = 0xe0104000 + i * 0x1000;
 | 
						|
 | 
						|
        object_initialize_child(OBJECT(mms), s, &mms->spi[i], TYPE_PL022);
 | 
						|
        sysbus_realize(SYS_BUS_DEVICE(&mms->spi[i]), &error_fatal);
 | 
						|
        sysbus_mmio_map(SYS_BUS_DEVICE(&mms->spi[i]), 0, baseaddr);
 | 
						|
        sysbus_connect_irq(SYS_BUS_DEVICE(&mms->spi[i]), 0,
 | 
						|
                           qdev_get_gpio_in(gicdev, 22 + i));
 | 
						|
    }
 | 
						|
 | 
						|
    object_initialize_child(OBJECT(mms), "scc", &mms->scc, TYPE_MPS2_SCC);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->scc), "scc-cfg0", 0);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->scc), "scc-cfg4", 0x2);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->scc), "scc-aid", 0x00200008);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->scc), "scc-id", 0x41055360);
 | 
						|
    oscclk = qlist_new();
 | 
						|
    for (int i = 0; i < ARRAY_SIZE(an536_oscclk); i++) {
 | 
						|
        qlist_append_int(oscclk, an536_oscclk[i]);
 | 
						|
    }
 | 
						|
    qdev_prop_set_array(DEVICE(&mms->scc), "oscclk", oscclk);
 | 
						|
    sysbus_realize(SYS_BUS_DEVICE(&mms->scc), &error_fatal);
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->scc), 0, 0xe0200000);
 | 
						|
 | 
						|
    create_unimplemented_device("i2s-audio", 0xe0201000, 0x1000);
 | 
						|
 | 
						|
    object_initialize_child(OBJECT(mms), "fpgaio", &mms->fpgaio,
 | 
						|
                            TYPE_MPS2_FPGAIO);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->fpgaio), "prescale-clk", an536_oscclk[1]);
 | 
						|
    qdev_prop_set_uint32(DEVICE(&mms->fpgaio), "num-leds", 10);
 | 
						|
    qdev_prop_set_bit(DEVICE(&mms->fpgaio), "has-switches", true);
 | 
						|
    qdev_prop_set_bit(DEVICE(&mms->fpgaio), "has-dbgctrl", false);
 | 
						|
    sysbus_realize(SYS_BUS_DEVICE(&mms->fpgaio), &error_fatal);
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->fpgaio), 0, 0xe0202000);
 | 
						|
 | 
						|
    create_unimplemented_device("clcd", 0xe0209000, 0x1000);
 | 
						|
 | 
						|
    object_initialize_child(OBJECT(mms), "rtc", &mms->rtc, TYPE_PL031);
 | 
						|
    sysbus_realize(SYS_BUS_DEVICE(&mms->rtc), &error_fatal);
 | 
						|
    sysbus_mmio_map(SYS_BUS_DEVICE(&mms->rtc), 0, 0xe020a000);
 | 
						|
    sysbus_connect_irq(SYS_BUS_DEVICE(&mms->rtc), 0,
 | 
						|
                       qdev_get_gpio_in(gicdev, 4));
 | 
						|
 | 
						|
    /*
 | 
						|
     * In hardware this is a LAN9220; the LAN9118 is software compatible
 | 
						|
     * except that it doesn't support the checksum-offload feature.
 | 
						|
     */
 | 
						|
    lan9118_init(0xe0300000,
 | 
						|
                 qdev_get_gpio_in(gicdev, 18));
 | 
						|
 | 
						|
    create_unimplemented_device("usb", 0xe0301000, 0x1000);
 | 
						|
    create_unimplemented_device("qspi-write-config", 0xe0600000, 0x1000);
 | 
						|
 | 
						|
    mms->bootinfo.ram_size = machine->ram_size;
 | 
						|
    mms->bootinfo.board_id = -1;
 | 
						|
    mms->bootinfo.loader_start = mmc->loader_start;
 | 
						|
    mms->bootinfo.write_secondary_boot = mps3r_write_secondary_boot;
 | 
						|
    mms->bootinfo.secondary_cpu_reset_hook = mps3r_secondary_cpu_reset;
 | 
						|
    arm_load_kernel(ARM_CPU(mms->cpu[0]), machine, &mms->bootinfo);
 | 
						|
}
 | 
						|
 | 
						|
static void mps3r_set_default_ram_info(MPS3RMachineClass *mmc)
 | 
						|
{
 | 
						|
    /*
 | 
						|
     * Set mc->default_ram_size and default_ram_id from the
 | 
						|
     * information in mmc->raminfo.
 | 
						|
     */
 | 
						|
    MachineClass *mc = MACHINE_CLASS(mmc);
 | 
						|
    const RAMInfo *p;
 | 
						|
 | 
						|
    for (p = mmc->raminfo; p->name; p++) {
 | 
						|
        if (p->mrindex < 0) {
 | 
						|
            /* Found the entry for "system memory" */
 | 
						|
            mc->default_ram_size = p->size;
 | 
						|
            mc->default_ram_id = p->name;
 | 
						|
            mmc->loader_start = p->base;
 | 
						|
            return;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    g_assert_not_reached();
 | 
						|
}
 | 
						|
 | 
						|
static void mps3r_class_init(ObjectClass *oc, void *data)
 | 
						|
{
 | 
						|
    MachineClass *mc = MACHINE_CLASS(oc);
 | 
						|
 | 
						|
    mc->init = mps3r_common_init;
 | 
						|
}
 | 
						|
 | 
						|
static void mps3r_an536_class_init(ObjectClass *oc, void *data)
 | 
						|
{
 | 
						|
    MachineClass *mc = MACHINE_CLASS(oc);
 | 
						|
    MPS3RMachineClass *mmc = MPS3R_MACHINE_CLASS(oc);
 | 
						|
    static const char * const valid_cpu_types[] = {
 | 
						|
        ARM_CPU_TYPE_NAME("cortex-r52"),
 | 
						|
        NULL
 | 
						|
    };
 | 
						|
 | 
						|
    mc->desc = "ARM MPS3 with AN536 FPGA image for Cortex-R52";
 | 
						|
    /*
 | 
						|
     * In the real FPGA image there are always two cores, but the standard
 | 
						|
     * initial setting for the SCC SYSCON 0x000 register is 0x21, meaning
 | 
						|
     * that the second core is held in reset and halted. Many images built for
 | 
						|
     * the board do not expect the second core to run at startup (especially
 | 
						|
     * since on the real FPGA image it is not possible to use LDREX/STREX
 | 
						|
     * in RAM between the two cores, so a true SMP setup isn't supported).
 | 
						|
     *
 | 
						|
     * As QEMU's equivalent of this, we support both -smp 1 and -smp 2,
 | 
						|
     * with the default being -smp 1. This seems a more intuitive UI for
 | 
						|
     * QEMU users than, for instance, having a machine property to allow
 | 
						|
     * the user to set the initial value of the SYSCON 0x000 register.
 | 
						|
     */
 | 
						|
    mc->default_cpus = 1;
 | 
						|
    mc->min_cpus = 1;
 | 
						|
    mc->max_cpus = 2;
 | 
						|
    mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-r52");
 | 
						|
    mc->valid_cpu_types = valid_cpu_types;
 | 
						|
    mmc->raminfo = an536_raminfo;
 | 
						|
    mps3r_set_default_ram_info(mmc);
 | 
						|
}
 | 
						|
 | 
						|
static const TypeInfo mps3r_machine_types[] = {
 | 
						|
    {
 | 
						|
        .name = TYPE_MPS3R_MACHINE,
 | 
						|
        .parent = TYPE_MACHINE,
 | 
						|
        .abstract = true,
 | 
						|
        .instance_size = sizeof(MPS3RMachineState),
 | 
						|
        .class_size = sizeof(MPS3RMachineClass),
 | 
						|
        .class_init = mps3r_class_init,
 | 
						|
    }, {
 | 
						|
        .name = TYPE_MPS3R_AN536_MACHINE,
 | 
						|
        .parent = TYPE_MPS3R_MACHINE,
 | 
						|
        .class_init = mps3r_an536_class_init,
 | 
						|
    },
 | 
						|
};
 | 
						|
 | 
						|
DEFINE_TYPES(mps3r_machine_types);
 |