xref: /openbmc/qemu/hw/arm/fsl-imx8mp.c (revision 0c105b261551a9b9fed086a7b9ecd0b6d6063bc4)
1 /*
2  * i.MX 8M Plus SoC Implementation
3  *
4  * Based on hw/arm/fsl-imx6.c
5  *
6  * Copyright (c) 2024, Bernhard Beschow <shentey@gmail.com>
7  *
8  * SPDX-License-Identifier: GPL-2.0-or-later
9  */
10 
11 #include "qemu/osdep.h"
12 #include "exec/address-spaces.h"
13 #include "hw/arm/bsa.h"
14 #include "hw/arm/fsl-imx8mp.h"
15 #include "hw/intc/arm_gicv3.h"
16 #include "hw/misc/unimp.h"
17 #include "hw/boards.h"
18 #include "system/system.h"
19 #include "target/arm/cpu-qom.h"
20 #include "qapi/error.h"
21 #include "qobject/qlist.h"
22 
23 static const struct {
24     hwaddr addr;
25     size_t size;
26     const char *name;
27 } fsl_imx8mp_memmap[] = {
28     [FSL_IMX8MP_RAM] = { FSL_IMX8MP_RAM_START, FSL_IMX8MP_RAM_SIZE_MAX, "ram" },
29     [FSL_IMX8MP_DDR_PHY_BROADCAST] = { 0x3dc00000, 4 * MiB, "ddr_phy_broadcast" },
30     [FSL_IMX8MP_DDR_PERF_MON] = { 0x3d800000, 4 * MiB, "ddr_perf_mon" },
31     [FSL_IMX8MP_DDR_CTL] = { 0x3d400000, 4 * MiB, "ddr_ctl" },
32     [FSL_IMX8MP_DDR_BLK_CTRL] = { 0x3d000000, 1 * MiB, "ddr_blk_ctrl" },
33     [FSL_IMX8MP_DDR_PHY] = { 0x3c000000, 16 * MiB, "ddr_phy" },
34     [FSL_IMX8MP_AUDIO_DSP] = { 0x3b000000, 16 * MiB, "audio_dsp" },
35     [FSL_IMX8MP_GIC_DIST] = { 0x38800000, 512 * KiB, "gic_dist" },
36     [FSL_IMX8MP_GIC_REDIST] = { 0x38880000, 512 * KiB, "gic_redist" },
37     [FSL_IMX8MP_NPU] = { 0x38500000, 2 * MiB, "npu" },
38     [FSL_IMX8MP_VPU] = { 0x38340000, 2 * MiB, "vpu" },
39     [FSL_IMX8MP_VPU_BLK_CTRL] = { 0x38330000, 2 * MiB, "vpu_blk_ctrl" },
40     [FSL_IMX8MP_VPU_VC8000E_ENCODER] = { 0x38320000, 2 * MiB, "vpu_vc8000e_encoder" },
41     [FSL_IMX8MP_VPU_G2_DECODER] = { 0x38310000, 2 * MiB, "vpu_g2_decoder" },
42     [FSL_IMX8MP_VPU_G1_DECODER] = { 0x38300000, 2 * MiB, "vpu_g1_decoder" },
43     [FSL_IMX8MP_USB2] = { 0x38200000, 1 * MiB, "usb2" },
44     [FSL_IMX8MP_USB1] = { 0x38100000, 1 * MiB, "usb1" },
45     [FSL_IMX8MP_GPU2D] = { 0x38008000, 32 * KiB, "gpu2d" },
46     [FSL_IMX8MP_GPU3D] = { 0x38000000, 32 * KiB, "gpu3d" },
47     [FSL_IMX8MP_QSPI1_RX_BUFFER] = { 0x34000000, 32 * MiB, "qspi1_rx_buffer" },
48     [FSL_IMX8MP_PCIE1] = { 0x33800000, 4 * MiB, "pcie1" },
49     [FSL_IMX8MP_QSPI1_TX_BUFFER] = { 0x33008000, 32 * KiB, "qspi1_tx_buffer" },
50     [FSL_IMX8MP_APBH_DMA] = { 0x33000000, 32 * KiB, "apbh_dma" },
51 
52     /* AIPS-5 Begin */
53     [FSL_IMX8MP_MU_3_B] = { 0x30e90000, 64 * KiB, "mu_3_b" },
54     [FSL_IMX8MP_MU_3_A] = { 0x30e80000, 64 * KiB, "mu_3_a" },
55     [FSL_IMX8MP_MU_2_B] = { 0x30e70000, 64 * KiB, "mu_2_b" },
56     [FSL_IMX8MP_MU_2_A] = { 0x30e60000, 64 * KiB, "mu_2_a" },
57     [FSL_IMX8MP_EDMA_CHANNELS] = { 0x30e40000, 128 * KiB, "edma_channels" },
58     [FSL_IMX8MP_EDMA_MANAGEMENT_PAGE] = { 0x30e30000, 64 * KiB, "edma_management_page" },
59     [FSL_IMX8MP_AUDIO_BLK_CTRL] = { 0x30e20000, 64 * KiB, "audio_blk_ctrl" },
60     [FSL_IMX8MP_SDMA2] = { 0x30e10000, 64 * KiB, "sdma2" },
61     [FSL_IMX8MP_SDMA3] = { 0x30e00000, 64 * KiB, "sdma3" },
62     [FSL_IMX8MP_AIPS5_CONFIGURATION] = { 0x30df0000, 64 * KiB, "aips5_configuration" },
63     [FSL_IMX8MP_SPBA2] = { 0x30cf0000, 64 * KiB, "spba2" },
64     [FSL_IMX8MP_AUDIO_XCVR_RX] = { 0x30cc0000, 64 * KiB, "audio_xcvr_rx" },
65     [FSL_IMX8MP_HDMI_TX_AUDLNK_MSTR] = { 0x30cb0000, 64 * KiB, "hdmi_tx_audlnk_mstr" },
66     [FSL_IMX8MP_PDM] = { 0x30ca0000, 64 * KiB, "pdm" },
67     [FSL_IMX8MP_ASRC] = { 0x30c90000, 64 * KiB, "asrc" },
68     [FSL_IMX8MP_SAI7] = { 0x30c80000, 64 * KiB, "sai7" },
69     [FSL_IMX8MP_SAI6] = { 0x30c60000, 64 * KiB, "sai6" },
70     [FSL_IMX8MP_SAI5] = { 0x30c50000, 64 * KiB, "sai5" },
71     [FSL_IMX8MP_SAI3] = { 0x30c30000, 64 * KiB, "sai3" },
72     [FSL_IMX8MP_SAI2] = { 0x30c20000, 64 * KiB, "sai2" },
73     [FSL_IMX8MP_SAI1] = { 0x30c10000, 64 * KiB, "sai1" },
74     /* AIPS-5 End */
75 
76     /* AIPS-4 Begin */
77     [FSL_IMX8MP_HDMI_TX] = { 0x32fc0000, 128 * KiB, "hdmi_tx" },
78     [FSL_IMX8MP_TZASC] = { 0x32f80000, 64 * KiB, "tzasc" },
79     [FSL_IMX8MP_HSIO_BLK_CTL] = { 0x32f10000, 64 * KiB, "hsio_blk_ctl" },
80     [FSL_IMX8MP_PCIE_PHY1] = { 0x32f00000, 64 * KiB, "pcie_phy1" },
81     [FSL_IMX8MP_MEDIA_BLK_CTL] = { 0x32ec0000, 64 * KiB, "media_blk_ctl" },
82     [FSL_IMX8MP_LCDIF2] = { 0x32e90000, 64 * KiB, "lcdif2" },
83     [FSL_IMX8MP_LCDIF1] = { 0x32e80000, 64 * KiB, "lcdif1" },
84     [FSL_IMX8MP_MIPI_DSI1] = { 0x32e60000, 64 * KiB, "mipi_dsi1" },
85     [FSL_IMX8MP_MIPI_CSI2] = { 0x32e50000, 64 * KiB, "mipi_csi2" },
86     [FSL_IMX8MP_MIPI_CSI1] = { 0x32e40000, 64 * KiB, "mipi_csi1" },
87     [FSL_IMX8MP_IPS_DEWARP] = { 0x32e30000, 64 * KiB, "ips_dewarp" },
88     [FSL_IMX8MP_ISP2] = { 0x32e20000, 64 * KiB, "isp2" },
89     [FSL_IMX8MP_ISP1] = { 0x32e10000, 64 * KiB, "isp1" },
90     [FSL_IMX8MP_ISI] = { 0x32e00000, 64 * KiB, "isi" },
91     [FSL_IMX8MP_AIPS4_CONFIGURATION] = { 0x32df0000, 64 * KiB, "aips4_configuration" },
92     /* AIPS-4 End */
93 
94     [FSL_IMX8MP_INTERCONNECT] = { 0x32700000, 1 * MiB, "interconnect" },
95 
96     /* AIPS-3 Begin */
97     [FSL_IMX8MP_ENET2_TSN] = { 0x30bf0000, 64 * KiB, "enet2_tsn" },
98     [FSL_IMX8MP_ENET1] = { 0x30be0000, 64 * KiB, "enet1" },
99     [FSL_IMX8MP_SDMA1] = { 0x30bd0000, 64 * KiB, "sdma1" },
100     [FSL_IMX8MP_QSPI] = { 0x30bb0000, 64 * KiB, "qspi" },
101     [FSL_IMX8MP_USDHC3] = { 0x30b60000, 64 * KiB, "usdhc3" },
102     [FSL_IMX8MP_USDHC2] = { 0x30b50000, 64 * KiB, "usdhc2" },
103     [FSL_IMX8MP_USDHC1] = { 0x30b40000, 64 * KiB, "usdhc1" },
104     [FSL_IMX8MP_I2C6] = { 0x30ae0000, 64 * KiB, "i2c6" },
105     [FSL_IMX8MP_I2C5] = { 0x30ad0000, 64 * KiB, "i2c5" },
106     [FSL_IMX8MP_SEMAPHORE_HS] = { 0x30ac0000, 64 * KiB, "semaphore_hs" },
107     [FSL_IMX8MP_MU_1_B] = { 0x30ab0000, 64 * KiB, "mu_1_b" },
108     [FSL_IMX8MP_MU_1_A] = { 0x30aa0000, 64 * KiB, "mu_1_a" },
109     [FSL_IMX8MP_AUD_IRQ_STEER] = { 0x30a80000, 64 * KiB, "aud_irq_steer" },
110     [FSL_IMX8MP_UART4] = { 0x30a60000, 64 * KiB, "uart4" },
111     [FSL_IMX8MP_I2C4] = { 0x30a50000, 64 * KiB, "i2c4" },
112     [FSL_IMX8MP_I2C3] = { 0x30a40000, 64 * KiB, "i2c3" },
113     [FSL_IMX8MP_I2C2] = { 0x30a30000, 64 * KiB, "i2c2" },
114     [FSL_IMX8MP_I2C1] = { 0x30a20000, 64 * KiB, "i2c1" },
115     [FSL_IMX8MP_AIPS3_CONFIGURATION] = { 0x309f0000, 64 * KiB, "aips3_configuration" },
116     [FSL_IMX8MP_CAAM] = { 0x30900000, 256 * KiB, "caam" },
117     [FSL_IMX8MP_SPBA1] = { 0x308f0000, 64 * KiB, "spba1" },
118     [FSL_IMX8MP_FLEXCAN2] = { 0x308d0000, 64 * KiB, "flexcan2" },
119     [FSL_IMX8MP_FLEXCAN1] = { 0x308c0000, 64 * KiB, "flexcan1" },
120     [FSL_IMX8MP_UART2] = { 0x30890000, 64 * KiB, "uart2" },
121     [FSL_IMX8MP_UART3] = { 0x30880000, 64 * KiB, "uart3" },
122     [FSL_IMX8MP_UART1] = { 0x30860000, 64 * KiB, "uart1" },
123     [FSL_IMX8MP_ECSPI3] = { 0x30840000, 64 * KiB, "ecspi3" },
124     [FSL_IMX8MP_ECSPI2] = { 0x30830000, 64 * KiB, "ecspi2" },
125     [FSL_IMX8MP_ECSPI1] = { 0x30820000, 64 * KiB, "ecspi1" },
126     /* AIPS-3 End */
127 
128     /* AIPS-2 Begin */
129     [FSL_IMX8MP_QOSC] = { 0x307f0000, 64 * KiB, "qosc" },
130     [FSL_IMX8MP_PERFMON2] = { 0x307d0000, 64 * KiB, "perfmon2" },
131     [FSL_IMX8MP_PERFMON1] = { 0x307c0000, 64 * KiB, "perfmon1" },
132     [FSL_IMX8MP_GPT4] = { 0x30700000, 64 * KiB, "gpt4" },
133     [FSL_IMX8MP_GPT5] = { 0x306f0000, 64 * KiB, "gpt5" },
134     [FSL_IMX8MP_GPT6] = { 0x306e0000, 64 * KiB, "gpt6" },
135     [FSL_IMX8MP_SYSCNT_CTRL] = { 0x306c0000, 64 * KiB, "syscnt_ctrl" },
136     [FSL_IMX8MP_SYSCNT_CMP] = { 0x306b0000, 64 * KiB, "syscnt_cmp" },
137     [FSL_IMX8MP_SYSCNT_RD] = { 0x306a0000, 64 * KiB, "syscnt_rd" },
138     [FSL_IMX8MP_PWM4] = { 0x30690000, 64 * KiB, "pwm4" },
139     [FSL_IMX8MP_PWM3] = { 0x30680000, 64 * KiB, "pwm3" },
140     [FSL_IMX8MP_PWM2] = { 0x30670000, 64 * KiB, "pwm2" },
141     [FSL_IMX8MP_PWM1] = { 0x30660000, 64 * KiB, "pwm1" },
142     [FSL_IMX8MP_AIPS2_CONFIGURATION] = { 0x305f0000, 64 * KiB, "aips2_configuration" },
143     /* AIPS-2 End */
144 
145     /* AIPS-1 Begin */
146     [FSL_IMX8MP_CSU] = { 0x303e0000, 64 * KiB, "csu" },
147     [FSL_IMX8MP_RDC] = { 0x303d0000, 64 * KiB, "rdc" },
148     [FSL_IMX8MP_SEMAPHORE2] = { 0x303c0000, 64 * KiB, "semaphore2" },
149     [FSL_IMX8MP_SEMAPHORE1] = { 0x303b0000, 64 * KiB, "semaphore1" },
150     [FSL_IMX8MP_GPC] = { 0x303a0000, 64 * KiB, "gpc" },
151     [FSL_IMX8MP_SRC] = { 0x30390000, 64 * KiB, "src" },
152     [FSL_IMX8MP_CCM] = { 0x30380000, 64 * KiB, "ccm" },
153     [FSL_IMX8MP_SNVS_HP] = { 0x30370000, 64 * KiB, "snvs_hp" },
154     [FSL_IMX8MP_ANA_PLL] = { 0x30360000, 64 * KiB, "ana_pll" },
155     [FSL_IMX8MP_OCOTP_CTRL] = { 0x30350000, 64 * KiB, "ocotp_ctrl" },
156     [FSL_IMX8MP_IOMUXC_GPR] = { 0x30340000, 64 * KiB, "iomuxc_gpr" },
157     [FSL_IMX8MP_IOMUXC] = { 0x30330000, 64 * KiB, "iomuxc" },
158     [FSL_IMX8MP_GPT3] = { 0x302f0000, 64 * KiB, "gpt3" },
159     [FSL_IMX8MP_GPT2] = { 0x302e0000, 64 * KiB, "gpt2" },
160     [FSL_IMX8MP_GPT1] = { 0x302d0000, 64 * KiB, "gpt1" },
161     [FSL_IMX8MP_WDOG3] = { 0x302a0000, 64 * KiB, "wdog3" },
162     [FSL_IMX8MP_WDOG2] = { 0x30290000, 64 * KiB, "wdog2" },
163     [FSL_IMX8MP_WDOG1] = { 0x30280000, 64 * KiB, "wdog1" },
164     [FSL_IMX8MP_ANA_OSC] = { 0x30270000, 64 * KiB, "ana_osc" },
165     [FSL_IMX8MP_ANA_TSENSOR] = { 0x30260000, 64 * KiB, "ana_tsensor" },
166     [FSL_IMX8MP_GPIO5] = { 0x30240000, 64 * KiB, "gpio5" },
167     [FSL_IMX8MP_GPIO4] = { 0x30230000, 64 * KiB, "gpio4" },
168     [FSL_IMX8MP_GPIO3] = { 0x30220000, 64 * KiB, "gpio3" },
169     [FSL_IMX8MP_GPIO2] = { 0x30210000, 64 * KiB, "gpio2" },
170     [FSL_IMX8MP_GPIO1] = { 0x30200000, 64 * KiB, "gpio1" },
171     [FSL_IMX8MP_AIPS1_CONFIGURATION] = { 0x301f0000, 64 * KiB, "aips1_configuration" },
172     /* AIPS-1 End */
173 
174     [FSL_IMX8MP_A53_DAP] = { 0x28000000, 16 * MiB, "a53_dap" },
175     [FSL_IMX8MP_PCIE1_MEM] = { 0x18000000, 128 * MiB, "pcie1_mem" },
176     [FSL_IMX8MP_QSPI_MEM] = { 0x08000000, 256 * MiB, "qspi_mem" },
177     [FSL_IMX8MP_OCRAM] = { 0x00900000, 576 * KiB, "ocram" },
178     [FSL_IMX8MP_TCM_DTCM] = { 0x00800000, 128 * KiB, "tcm_dtcm" },
179     [FSL_IMX8MP_TCM_ITCM] = { 0x007e0000, 128 * KiB, "tcm_itcm" },
180     [FSL_IMX8MP_OCRAM_S] = { 0x00180000, 36 * KiB, "ocram_s" },
181     [FSL_IMX8MP_CAAM_MEM] = { 0x00100000, 32 * KiB, "caam_mem" },
182     [FSL_IMX8MP_BOOT_ROM_PROTECTED] = { 0x0003f000, 4 * KiB, "boot_rom_protected" },
183     [FSL_IMX8MP_BOOT_ROM] = { 0x00000000, 252 * KiB, "boot_rom" },
184 };
185 
186 static void fsl_imx8mp_init(Object *obj)
187 {
188     MachineState *ms = MACHINE(qdev_get_machine());
189     FslImx8mpState *s = FSL_IMX8MP(obj);
190     int i;
191 
192     for (i = 0; i < MIN(ms->smp.cpus, FSL_IMX8MP_NUM_CPUS); i++) {
193         g_autofree char *name = g_strdup_printf("cpu%d", i);
194         object_initialize_child(obj, name, &s->cpu[i],
195                                 ARM_CPU_TYPE_NAME("cortex-a53"));
196     }
197 
198     object_initialize_child(obj, "gic", &s->gic, TYPE_ARM_GICV3);
199 
200     object_initialize_child(obj, "ccm", &s->ccm, TYPE_IMX8MP_CCM);
201 
202     object_initialize_child(obj, "analog", &s->analog, TYPE_IMX8MP_ANALOG);
203 
204     object_initialize_child(obj, "snvs", &s->snvs, TYPE_IMX7_SNVS);
205 
206     for (i = 0; i < FSL_IMX8MP_NUM_UARTS; i++) {
207         g_autofree char *name = g_strdup_printf("uart%d", i + 1);
208         object_initialize_child(obj, name, &s->uart[i], TYPE_IMX_SERIAL);
209     }
210 
211     for (i = 0; i < FSL_IMX8MP_NUM_GPTS; i++) {
212         g_autofree char *name = g_strdup_printf("gpt%d", i + 1);
213         object_initialize_child(obj, name, &s->gpt[i], TYPE_IMX8MP_GPT);
214     }
215     object_initialize_child(obj, "gpt5-gpt6-irq", &s->gpt5_gpt6_irq,
216                             TYPE_OR_IRQ);
217 
218     for (i = 0; i < FSL_IMX8MP_NUM_I2CS; i++) {
219         g_autofree char *name = g_strdup_printf("i2c%d", i + 1);
220         object_initialize_child(obj, name, &s->i2c[i], TYPE_IMX_I2C);
221     }
222 
223     for (i = 0; i < FSL_IMX8MP_NUM_GPIOS; i++) {
224         g_autofree char *name = g_strdup_printf("gpio%d", i + 1);
225         object_initialize_child(obj, name, &s->gpio[i], TYPE_IMX_GPIO);
226     }
227 
228     for (i = 0; i < FSL_IMX8MP_NUM_USDHCS; i++) {
229         g_autofree char *name = g_strdup_printf("usdhc%d", i + 1);
230         object_initialize_child(obj, name, &s->usdhc[i], TYPE_IMX_USDHC);
231     }
232 
233     for (i = 0; i < FSL_IMX8MP_NUM_ECSPIS; i++) {
234         g_autofree char *name = g_strdup_printf("spi%d", i + 1);
235         object_initialize_child(obj, name, &s->spi[i], TYPE_IMX_SPI);
236     }
237 
238     for (i = 0; i < FSL_IMX8MP_NUM_WDTS; i++) {
239         g_autofree char *name = g_strdup_printf("wdt%d", i);
240         object_initialize_child(obj, name, &s->wdt[i], TYPE_IMX2_WDT);
241     }
242 
243     object_initialize_child(obj, "eth0", &s->enet, TYPE_IMX_ENET);
244 
245     object_initialize_child(obj, "pcie", &s->pcie, TYPE_DESIGNWARE_PCIE_HOST);
246     object_initialize_child(obj, "pcie_phy", &s->pcie_phy,
247                             TYPE_FSL_IMX8M_PCIE_PHY);
248 }
249 
250 static void fsl_imx8mp_realize(DeviceState *dev, Error **errp)
251 {
252     MachineState *ms = MACHINE(qdev_get_machine());
253     FslImx8mpState *s = FSL_IMX8MP(dev);
254     DeviceState *gicdev = DEVICE(&s->gic);
255     int i;
256 
257     if (ms->smp.cpus > FSL_IMX8MP_NUM_CPUS) {
258         error_setg(errp, "%s: Only %d CPUs are supported (%d requested)",
259                    TYPE_FSL_IMX8MP, FSL_IMX8MP_NUM_CPUS, ms->smp.cpus);
260         return;
261     }
262 
263     /* CPUs */
264     for (i = 0; i < ms->smp.cpus; i++) {
265         /* On uniprocessor, the CBAR is set to 0 */
266         if (ms->smp.cpus > 1) {
267             object_property_set_int(OBJECT(&s->cpu[i]), "reset-cbar",
268                                     fsl_imx8mp_memmap[FSL_IMX8MP_GIC_DIST].addr,
269                                     &error_abort);
270         }
271 
272         /*
273          * CNTFID0 base frequency in Hz of system counter
274          */
275         object_property_set_int(OBJECT(&s->cpu[i]), "cntfrq", 8000000,
276                                 &error_abort);
277 
278         if (i) {
279             /*
280              * Secondary CPUs start in powered-down state (and can be
281              * powered up via the SRC system reset controller)
282              */
283             object_property_set_bool(OBJECT(&s->cpu[i]), "start-powered-off",
284                                      true, &error_abort);
285         }
286 
287         if (!qdev_realize(DEVICE(&s->cpu[i]), NULL, errp)) {
288             return;
289         }
290     }
291 
292     /* GIC */
293     {
294         SysBusDevice *gicsbd = SYS_BUS_DEVICE(&s->gic);
295         QList *redist_region_count;
296 
297         qdev_prop_set_uint32(gicdev, "num-cpu", ms->smp.cpus);
298         qdev_prop_set_uint32(gicdev, "num-irq",
299                              FSL_IMX8MP_NUM_IRQS + GIC_INTERNAL);
300         redist_region_count = qlist_new();
301         qlist_append_int(redist_region_count, ms->smp.cpus);
302         qdev_prop_set_array(gicdev, "redist-region-count", redist_region_count);
303         object_property_set_link(OBJECT(&s->gic), "sysmem",
304                                  OBJECT(get_system_memory()), &error_fatal);
305         if (!sysbus_realize(gicsbd, errp)) {
306             return;
307         }
308         sysbus_mmio_map(gicsbd, 0, fsl_imx8mp_memmap[FSL_IMX8MP_GIC_DIST].addr);
309         sysbus_mmio_map(gicsbd, 1, fsl_imx8mp_memmap[FSL_IMX8MP_GIC_REDIST].addr);
310 
311         /*
312          * Wire the outputs from each CPU's generic timer and the GICv3
313          * maintenance interrupt signal to the appropriate GIC PPI inputs, and
314          * the GIC's IRQ/FIQ interrupt outputs to the CPU's inputs.
315          */
316         for (i = 0; i < ms->smp.cpus; i++) {
317             DeviceState *cpudev = DEVICE(&s->cpu[i]);
318             int intidbase = FSL_IMX8MP_NUM_IRQS + i * GIC_INTERNAL;
319             qemu_irq irq;
320 
321             /*
322              * Mapping from the output timer irq lines from the CPU to the
323              * GIC PPI inputs.
324              */
325             static const int timer_irqs[] = {
326                 [GTIMER_PHYS] = ARCH_TIMER_NS_EL1_IRQ,
327                 [GTIMER_VIRT] = ARCH_TIMER_VIRT_IRQ,
328                 [GTIMER_HYP]  = ARCH_TIMER_NS_EL2_IRQ,
329                 [GTIMER_SEC]  = ARCH_TIMER_S_EL1_IRQ,
330             };
331 
332             for (int j = 0; j < ARRAY_SIZE(timer_irqs); j++) {
333                 irq = qdev_get_gpio_in(gicdev, intidbase + timer_irqs[j]);
334                 qdev_connect_gpio_out(cpudev, j, irq);
335             }
336 
337             irq = qdev_get_gpio_in(gicdev, intidbase + ARCH_GIC_MAINT_IRQ);
338             qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt",
339                                         0, irq);
340 
341             irq = qdev_get_gpio_in(gicdev, intidbase + VIRTUAL_PMU_IRQ);
342             qdev_connect_gpio_out_named(cpudev, "pmu-interrupt", 0, irq);
343 
344             sysbus_connect_irq(gicsbd, i,
345                                qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));
346             sysbus_connect_irq(gicsbd, i + ms->smp.cpus,
347                                qdev_get_gpio_in(cpudev, ARM_CPU_FIQ));
348         }
349     }
350 
351     /* CCM */
352     if (!sysbus_realize(SYS_BUS_DEVICE(&s->ccm), errp)) {
353         return;
354     }
355     sysbus_mmio_map(SYS_BUS_DEVICE(&s->ccm), 0,
356                     fsl_imx8mp_memmap[FSL_IMX8MP_CCM].addr);
357 
358     /* Analog */
359     if (!sysbus_realize(SYS_BUS_DEVICE(&s->analog), errp)) {
360         return;
361     }
362     sysbus_mmio_map(SYS_BUS_DEVICE(&s->analog), 0,
363                     fsl_imx8mp_memmap[FSL_IMX8MP_ANA_PLL].addr);
364 
365     /* UARTs */
366     for (i = 0; i < FSL_IMX8MP_NUM_UARTS; i++) {
367         struct {
368             hwaddr addr;
369             unsigned int irq;
370         } serial_table[FSL_IMX8MP_NUM_UARTS] = {
371             { fsl_imx8mp_memmap[FSL_IMX8MP_UART1].addr, FSL_IMX8MP_UART1_IRQ },
372             { fsl_imx8mp_memmap[FSL_IMX8MP_UART2].addr, FSL_IMX8MP_UART2_IRQ },
373             { fsl_imx8mp_memmap[FSL_IMX8MP_UART3].addr, FSL_IMX8MP_UART3_IRQ },
374             { fsl_imx8mp_memmap[FSL_IMX8MP_UART4].addr, FSL_IMX8MP_UART4_IRQ },
375         };
376 
377         qdev_prop_set_chr(DEVICE(&s->uart[i]), "chardev", serial_hd(i));
378         if (!sysbus_realize(SYS_BUS_DEVICE(&s->uart[i]), errp)) {
379             return;
380         }
381 
382         sysbus_mmio_map(SYS_BUS_DEVICE(&s->uart[i]), 0, serial_table[i].addr);
383         sysbus_connect_irq(SYS_BUS_DEVICE(&s->uart[i]), 0,
384                            qdev_get_gpio_in(gicdev, serial_table[i].irq));
385     }
386 
387     /* GPTs */
388     object_property_set_int(OBJECT(&s->gpt5_gpt6_irq), "num-lines", 2,
389                             &error_abort);
390     if (!qdev_realize(DEVICE(&s->gpt5_gpt6_irq), NULL, errp)) {
391         return;
392     }
393 
394     qdev_connect_gpio_out(DEVICE(&s->gpt5_gpt6_irq), 0,
395                           qdev_get_gpio_in(gicdev, FSL_IMX8MP_GPT5_GPT6_IRQ));
396 
397     for (i = 0; i < FSL_IMX8MP_NUM_GPTS; i++) {
398         hwaddr gpt_addrs[FSL_IMX8MP_NUM_GPTS] = {
399             fsl_imx8mp_memmap[FSL_IMX8MP_GPT1].addr,
400             fsl_imx8mp_memmap[FSL_IMX8MP_GPT2].addr,
401             fsl_imx8mp_memmap[FSL_IMX8MP_GPT3].addr,
402             fsl_imx8mp_memmap[FSL_IMX8MP_GPT4].addr,
403             fsl_imx8mp_memmap[FSL_IMX8MP_GPT5].addr,
404             fsl_imx8mp_memmap[FSL_IMX8MP_GPT6].addr,
405         };
406 
407         s->gpt[i].ccm = IMX_CCM(&s->ccm);
408 
409         if (!sysbus_realize(SYS_BUS_DEVICE(&s->gpt[i]), errp)) {
410             return;
411         }
412 
413         sysbus_mmio_map(SYS_BUS_DEVICE(&s->gpt[i]), 0, gpt_addrs[i]);
414 
415         if (i < FSL_IMX8MP_NUM_GPTS - 2) {
416             static const unsigned int gpt_irqs[FSL_IMX8MP_NUM_GPTS - 2] = {
417                 FSL_IMX8MP_GPT1_IRQ,
418                 FSL_IMX8MP_GPT2_IRQ,
419                 FSL_IMX8MP_GPT3_IRQ,
420                 FSL_IMX8MP_GPT4_IRQ,
421             };
422 
423             sysbus_connect_irq(SYS_BUS_DEVICE(&s->gpt[i]), 0,
424                                qdev_get_gpio_in(gicdev, gpt_irqs[i]));
425         } else {
426             int irq = i - FSL_IMX8MP_NUM_GPTS + 2;
427 
428             sysbus_connect_irq(SYS_BUS_DEVICE(&s->gpt[i]), 0,
429                                qdev_get_gpio_in(DEVICE(&s->gpt5_gpt6_irq), irq));
430         }
431     }
432 
433     /* I2Cs */
434     for (i = 0; i < FSL_IMX8MP_NUM_I2CS; i++) {
435         struct {
436             hwaddr addr;
437             unsigned int irq;
438         } i2c_table[FSL_IMX8MP_NUM_I2CS] = {
439             { fsl_imx8mp_memmap[FSL_IMX8MP_I2C1].addr, FSL_IMX8MP_I2C1_IRQ },
440             { fsl_imx8mp_memmap[FSL_IMX8MP_I2C2].addr, FSL_IMX8MP_I2C2_IRQ },
441             { fsl_imx8mp_memmap[FSL_IMX8MP_I2C3].addr, FSL_IMX8MP_I2C3_IRQ },
442             { fsl_imx8mp_memmap[FSL_IMX8MP_I2C4].addr, FSL_IMX8MP_I2C4_IRQ },
443             { fsl_imx8mp_memmap[FSL_IMX8MP_I2C5].addr, FSL_IMX8MP_I2C5_IRQ },
444             { fsl_imx8mp_memmap[FSL_IMX8MP_I2C6].addr, FSL_IMX8MP_I2C6_IRQ },
445         };
446 
447         if (!sysbus_realize(SYS_BUS_DEVICE(&s->i2c[i]), errp)) {
448             return;
449         }
450 
451         sysbus_mmio_map(SYS_BUS_DEVICE(&s->i2c[i]), 0, i2c_table[i].addr);
452         sysbus_connect_irq(SYS_BUS_DEVICE(&s->i2c[i]), 0,
453                            qdev_get_gpio_in(gicdev, i2c_table[i].irq));
454     }
455 
456     /* GPIOs */
457     for (i = 0; i < FSL_IMX8MP_NUM_GPIOS; i++) {
458         struct {
459             hwaddr addr;
460             unsigned int irq_low;
461             unsigned int irq_high;
462         } gpio_table[FSL_IMX8MP_NUM_GPIOS] = {
463             {
464                 fsl_imx8mp_memmap[FSL_IMX8MP_GPIO1].addr,
465                 FSL_IMX8MP_GPIO1_LOW_IRQ,
466                 FSL_IMX8MP_GPIO1_HIGH_IRQ
467             },
468             {
469                 fsl_imx8mp_memmap[FSL_IMX8MP_GPIO2].addr,
470                 FSL_IMX8MP_GPIO2_LOW_IRQ,
471                 FSL_IMX8MP_GPIO2_HIGH_IRQ
472             },
473             {
474                 fsl_imx8mp_memmap[FSL_IMX8MP_GPIO3].addr,
475                 FSL_IMX8MP_GPIO3_LOW_IRQ,
476                 FSL_IMX8MP_GPIO3_HIGH_IRQ
477             },
478             {
479                 fsl_imx8mp_memmap[FSL_IMX8MP_GPIO4].addr,
480                 FSL_IMX8MP_GPIO4_LOW_IRQ,
481                 FSL_IMX8MP_GPIO4_HIGH_IRQ
482             },
483             {
484                 fsl_imx8mp_memmap[FSL_IMX8MP_GPIO5].addr,
485                 FSL_IMX8MP_GPIO5_LOW_IRQ,
486                 FSL_IMX8MP_GPIO5_HIGH_IRQ
487             },
488         };
489 
490         object_property_set_bool(OBJECT(&s->gpio[i]), "has-edge-sel", true,
491                                  &error_abort);
492         object_property_set_bool(OBJECT(&s->gpio[i]), "has-upper-pin-irq",
493                                  true, &error_abort);
494         if (!sysbus_realize(SYS_BUS_DEVICE(&s->gpio[i]), errp)) {
495             return;
496         }
497 
498         sysbus_mmio_map(SYS_BUS_DEVICE(&s->gpio[i]), 0, gpio_table[i].addr);
499         sysbus_connect_irq(SYS_BUS_DEVICE(&s->gpio[i]), 0,
500                            qdev_get_gpio_in(gicdev, gpio_table[i].irq_low));
501         sysbus_connect_irq(SYS_BUS_DEVICE(&s->gpio[i]), 1,
502                            qdev_get_gpio_in(gicdev, gpio_table[i].irq_high));
503     }
504 
505     /* USDHCs */
506     for (i = 0; i < FSL_IMX8MP_NUM_USDHCS; i++) {
507         struct {
508             hwaddr addr;
509             unsigned int irq;
510         } usdhc_table[FSL_IMX8MP_NUM_USDHCS] = {
511             { fsl_imx8mp_memmap[FSL_IMX8MP_USDHC1].addr, FSL_IMX8MP_USDHC1_IRQ },
512             { fsl_imx8mp_memmap[FSL_IMX8MP_USDHC2].addr, FSL_IMX8MP_USDHC2_IRQ },
513             { fsl_imx8mp_memmap[FSL_IMX8MP_USDHC3].addr, FSL_IMX8MP_USDHC3_IRQ },
514         };
515 
516         object_property_set_uint(OBJECT(&s->usdhc[i]), "vendor",
517                                  SDHCI_VENDOR_IMX, &error_abort);
518         if (!sysbus_realize(SYS_BUS_DEVICE(&s->usdhc[i]), errp)) {
519             return;
520         }
521 
522         sysbus_mmio_map(SYS_BUS_DEVICE(&s->usdhc[i]), 0, usdhc_table[i].addr);
523         sysbus_connect_irq(SYS_BUS_DEVICE(&s->usdhc[i]), 0,
524                            qdev_get_gpio_in(gicdev, usdhc_table[i].irq));
525     }
526 
527     /* ECSPIs */
528     for (i = 0; i < FSL_IMX8MP_NUM_ECSPIS; i++) {
529         struct {
530             hwaddr addr;
531             unsigned int irq;
532         } spi_table[FSL_IMX8MP_NUM_ECSPIS] = {
533             { fsl_imx8mp_memmap[FSL_IMX8MP_ECSPI1].addr, FSL_IMX8MP_ECSPI1_IRQ },
534             { fsl_imx8mp_memmap[FSL_IMX8MP_ECSPI2].addr, FSL_IMX8MP_ECSPI2_IRQ },
535             { fsl_imx8mp_memmap[FSL_IMX8MP_ECSPI3].addr, FSL_IMX8MP_ECSPI3_IRQ },
536         };
537 
538         if (!sysbus_realize(SYS_BUS_DEVICE(&s->spi[i]), errp)) {
539             return;
540         }
541 
542         sysbus_mmio_map(SYS_BUS_DEVICE(&s->spi[i]), 0, spi_table[i].addr);
543         sysbus_connect_irq(SYS_BUS_DEVICE(&s->spi[i]), 0,
544                            qdev_get_gpio_in(gicdev, spi_table[i].irq));
545     }
546 
547     /* ENET1 */
548     object_property_set_uint(OBJECT(&s->enet), "phy-num", s->phy_num,
549                              &error_abort);
550     object_property_set_uint(OBJECT(&s->enet), "tx-ring-num", 3, &error_abort);
551     qemu_configure_nic_device(DEVICE(&s->enet), true, NULL);
552     if (!sysbus_realize(SYS_BUS_DEVICE(&s->enet), errp)) {
553         return;
554     }
555     sysbus_mmio_map(SYS_BUS_DEVICE(&s->enet), 0,
556                     fsl_imx8mp_memmap[FSL_IMX8MP_ENET1].addr);
557     sysbus_connect_irq(SYS_BUS_DEVICE(&s->enet), 0,
558                        qdev_get_gpio_in(gicdev, FSL_IMX8MP_ENET1_MAC_IRQ));
559     sysbus_connect_irq(SYS_BUS_DEVICE(&s->enet), 1,
560                        qdev_get_gpio_in(gicdev, FSL_IMX6_ENET1_MAC_1588_IRQ));
561 
562     /* SNVS */
563     if (!sysbus_realize(SYS_BUS_DEVICE(&s->snvs), errp)) {
564         return;
565     }
566     sysbus_mmio_map(SYS_BUS_DEVICE(&s->snvs), 0,
567                     fsl_imx8mp_memmap[FSL_IMX8MP_SNVS_HP].addr);
568 
569     /* Watchdogs */
570     for (i = 0; i < FSL_IMX8MP_NUM_WDTS; i++) {
571         struct {
572             hwaddr addr;
573             unsigned int irq;
574         } wdog_table[FSL_IMX8MP_NUM_WDTS] = {
575             { fsl_imx8mp_memmap[FSL_IMX8MP_WDOG1].addr, FSL_IMX8MP_WDOG1_IRQ },
576             { fsl_imx8mp_memmap[FSL_IMX8MP_WDOG2].addr, FSL_IMX8MP_WDOG2_IRQ },
577             { fsl_imx8mp_memmap[FSL_IMX8MP_WDOG3].addr, FSL_IMX8MP_WDOG3_IRQ },
578         };
579 
580         object_property_set_bool(OBJECT(&s->wdt[i]), "pretimeout-support",
581                                  true, &error_abort);
582         if (!sysbus_realize(SYS_BUS_DEVICE(&s->wdt[i]), errp)) {
583             return;
584         }
585 
586         sysbus_mmio_map(SYS_BUS_DEVICE(&s->wdt[i]), 0, wdog_table[i].addr);
587         sysbus_connect_irq(SYS_BUS_DEVICE(&s->wdt[i]), 0,
588                            qdev_get_gpio_in(gicdev, wdog_table[i].irq));
589     }
590 
591     /* PCIe */
592     if (!sysbus_realize(SYS_BUS_DEVICE(&s->pcie), errp)) {
593         return;
594     }
595     sysbus_mmio_map(SYS_BUS_DEVICE(&s->pcie), 0,
596                     fsl_imx8mp_memmap[FSL_IMX8MP_PCIE1].addr);
597 
598     sysbus_connect_irq(SYS_BUS_DEVICE(&s->pcie), 0,
599                        qdev_get_gpio_in(gicdev, FSL_IMX8MP_PCI_INTA_IRQ));
600     sysbus_connect_irq(SYS_BUS_DEVICE(&s->pcie), 1,
601                        qdev_get_gpio_in(gicdev, FSL_IMX8MP_PCI_INTB_IRQ));
602     sysbus_connect_irq(SYS_BUS_DEVICE(&s->pcie), 2,
603                        qdev_get_gpio_in(gicdev, FSL_IMX8MP_PCI_INTC_IRQ));
604     sysbus_connect_irq(SYS_BUS_DEVICE(&s->pcie), 3,
605                        qdev_get_gpio_in(gicdev, FSL_IMX8MP_PCI_INTD_IRQ));
606     sysbus_connect_irq(SYS_BUS_DEVICE(&s->pcie), 4,
607                        qdev_get_gpio_in(gicdev, FSL_IMX8MP_PCI_MSI_IRQ));
608 
609     if (!sysbus_realize(SYS_BUS_DEVICE(&s->pcie_phy), errp)) {
610         return;
611     }
612     sysbus_mmio_map(SYS_BUS_DEVICE(&s->pcie_phy), 0,
613                     fsl_imx8mp_memmap[FSL_IMX8MP_PCIE_PHY1].addr);
614 
615     /* Unimplemented devices */
616     for (i = 0; i < ARRAY_SIZE(fsl_imx8mp_memmap); i++) {
617         switch (i) {
618         case FSL_IMX8MP_ANA_PLL:
619         case FSL_IMX8MP_CCM:
620         case FSL_IMX8MP_GIC_DIST:
621         case FSL_IMX8MP_GIC_REDIST:
622         case FSL_IMX8MP_GPIO1 ... FSL_IMX8MP_GPIO5:
623         case FSL_IMX8MP_ECSPI1 ... FSL_IMX8MP_ECSPI3:
624         case FSL_IMX8MP_ENET1:
625         case FSL_IMX8MP_I2C1 ... FSL_IMX8MP_I2C6:
626         case FSL_IMX8MP_PCIE1:
627         case FSL_IMX8MP_PCIE_PHY1:
628         case FSL_IMX8MP_RAM:
629         case FSL_IMX8MP_SNVS_HP:
630         case FSL_IMX8MP_UART1 ... FSL_IMX8MP_UART4:
631         case FSL_IMX8MP_USDHC1 ... FSL_IMX8MP_USDHC3:
632         case FSL_IMX8MP_WDOG1 ... FSL_IMX8MP_WDOG3:
633             /* device implemented and treated above */
634             break;
635 
636         default:
637             create_unimplemented_device(fsl_imx8mp_memmap[i].name,
638                                         fsl_imx8mp_memmap[i].addr,
639                                         fsl_imx8mp_memmap[i].size);
640             break;
641         }
642     }
643 }
644 
645 static const Property fsl_imx8mp_properties[] = {
646     DEFINE_PROP_UINT32("fec1-phy-num", FslImx8mpState, phy_num, 0),
647     DEFINE_PROP_BOOL("fec1-phy-connected", FslImx8mpState, phy_connected, true),
648 };
649 
650 static void fsl_imx8mp_class_init(ObjectClass *oc, void *data)
651 {
652     DeviceClass *dc = DEVICE_CLASS(oc);
653 
654     device_class_set_props(dc, fsl_imx8mp_properties);
655     dc->realize = fsl_imx8mp_realize;
656 
657     dc->desc = "i.MX 8M Plus SoC";
658 }
659 
660 static const TypeInfo fsl_imx8mp_types[] = {
661     {
662         .name = TYPE_FSL_IMX8MP,
663         .parent = TYPE_DEVICE,
664         .instance_size = sizeof(FslImx8mpState),
665         .instance_init = fsl_imx8mp_init,
666         .class_init = fsl_imx8mp_class_init,
667     },
668 };
669 
670 DEFINE_TYPES(fsl_imx8mp_types)
671