1
Forgot to add system/index to docs/index.rst (one line tweak).
1
This is a respin of my pull request from earlier this week:
2
Rest of series same as v1.
2
* versal board compile failure fixed
3
* a few new patches:
4
- MAINTAINERS file fix
5
- use ARRAY_SIZE macro in xilinx_zynq
6
- avoid an array overrun in strongarm GPIO irq handling
7
- fix an assert running KVM on an aarch64-only host
3
8
4
The following changes since commit b7c359c748a2e3ccb97a184b9739feb2cd48de2f:
9
The following changes since commit 69e2d03843412b9c076515b3aa9a71db161b6a1a:
5
10
6
Merge remote-tracking branch 'remotes/vivier2/tags/linux-user-for-5.0-pull-request' into staging (2020-01-23 14:38:43 +0000)
11
Merge remote-tracking branch 'remotes/riscv/tags/riscv-for-master-3.1-sf1' into staging (2018-11-02 13:16:13 +0000)
7
12
8
are available in the Git repository at:
13
are available in the Git repository at:
9
14
10
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20200123-1
15
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20181102
11
16
12
for you to fetch changes up to 3efba2eac3f2ac1f84f75465597f361626a6d0d5:
17
for you to fetch changes up to 6f16da53ffe4567c0353f85055df04860eb4e6fc:
13
18
14
hw/arm/exynos4210: Connect serial port DMA busy signals with pl330 (2020-01-23 15:34:05 +0000)
19
hw/arm: versal: Add a virtual Xilinx Versal board (2018-11-02 14:11:31 +0000)
15
20
16
----------------------------------------------------------------
21
----------------------------------------------------------------
17
target-arm queue:
22
target-arm queue:
18
* fix bug in PAuth emulation
23
* microbit: Add the UART to our nRF51 SoC model
19
* add PMU to Cortex-R5, Cortex-R5F
24
* Add a virtual Xilinx Versal board "xlnx-versal-virt"
20
* qemu-nbd: Convert documentation to rST
25
* hw/arm/virt: Set VIRT_COMPAT_3_0 compat
21
* qemu-block-drivers: Convert documentation to rST
26
* MAINTAINERS: Remove bouncing email in ARM ACPI
22
* Fix Exynos4210 UART DMA support
27
* strongarm: mask off high[31:28] bits from dir and state registers
23
* Various minor code cleanups
28
* target/arm: Conditionalize some asserts on aarch32 support
29
* hw/arm/xilinx_zynq: Use the ARRAY_SIZE macro
24
30
25
----------------------------------------------------------------
31
----------------------------------------------------------------
26
Andrew Jones (1):
32
Edgar E. Iglesias (2):
27
target/arm/arch_dump: Add SVE notes
33
hw/arm: versal: Add a model of Xilinx Versal SoC
34
hw/arm: versal: Add a virtual Xilinx Versal board
28
35
29
Clement Deschamps (1):
36
Eric Auger (1):
30
target/arm: add PMU feature to cortex-r5 and cortex-r5f
37
hw/arm/virt: Set VIRT_COMPAT_3_0 compat
31
38
32
Guenter Roeck (8):
39
Julia Suvorova (3):
33
dma/pl330: Convert to support tracing
40
hw/char: Implement nRF51 SoC UART
34
hw/core/or-irq: Increase limit of or-lines to 48
41
hw/arm/nrf51_soc: Connect UART to nRF51 SoC
35
hw/arm/exynos4210: Fix DMA initialization
42
tests/boot-serial-test: Add microbit board testcase
36
hw/char/exynos4210_uart: Convert to support tracing
37
hw/char/exynos4210_uart: Implement post_load function
38
hw/char/exynos4210_uart: Implement Rx FIFO level triggers and timeouts
39
hw/char/exynos4210_uart: Add receive DMA support
40
hw/arm/exynos4210: Connect serial port DMA busy signals with pl330
41
43
42
Keqian Zhu (2):
44
Philippe Mathieu-Daudé (2):
43
hw/acpi: Remove extra indent in ACPI GED hotplug cb
45
MAINTAINERS: Remove bouncing email in ARM ACPI
44
hw/arm: Use helper function to trigger hotplug handler plug
46
hw/arm/xilinx_zynq: Use the ARRAY_SIZE macro
45
47
46
Peter Maydell (3):
48
Prasad J Pandit (1):
47
qemu-nbd: Convert invocation documentation to rST
49
strongarm: mask off high[31:28] bits from dir and state registers
48
docs: Create stub system manual
49
qemu-block-drivers: Convert to rST
50
50
51
Philippe Mathieu-Daudé (1):
51
Richard Henderson (1):
52
hw/misc/stm32f4xx_syscfg: Fix copy/paste error
52
target/arm: Conditionalize some asserts on aarch32 support
53
53
54
Richard Henderson (3):
54
hw/arm/Makefile.objs | 1 +
55
tests/tcg/aarch64: Fix compilation parameters for pauth-%
55
hw/char/Makefile.objs | 1 +
56
tests/tcg/aarch64: Add pauth-3
56
include/hw/arm/nrf51_soc.h | 3 +
57
tests/tcg/aarch64: Add pauth-4
57
include/hw/arm/xlnx-versal.h | 122 +++++++++
58
include/hw/char/nrf51_uart.h | 78 ++++++
59
target/arm/cpu.h | 5 +
60
hw/arm/microbit.c | 2 +
61
hw/arm/nrf51_soc.c | 20 ++
62
hw/arm/strongarm.c | 4 +-
63
hw/arm/virt.c | 4 +
64
hw/arm/xilinx_zynq.c | 2 +-
65
hw/arm/xlnx-versal-virt.c | 494 ++++++++++++++++++++++++++++++++++++
66
hw/arm/xlnx-versal.c | 323 +++++++++++++++++++++++
67
hw/char/nrf51_uart.c | 330 ++++++++++++++++++++++++
68
target/arm/cpu.c | 15 +-
69
tests/boot-serial-test.c | 19 ++
70
MAINTAINERS | 1 -
71
default-configs/aarch64-softmmu.mak | 1 +
72
hw/char/trace-events | 4 +
73
19 files changed, 1423 insertions(+), 6 deletions(-)
74
create mode 100644 include/hw/arm/xlnx-versal.h
75
create mode 100644 include/hw/char/nrf51_uart.h
76
create mode 100644 hw/arm/xlnx-versal-virt.c
77
create mode 100644 hw/arm/xlnx-versal.c
78
create mode 100644 hw/char/nrf51_uart.c
58
79
59
Vincent Dehors (1):
60
target/arm: Fix PAuth sbox functions
61
62
Makefile | 37 +-
63
tests/tcg/aarch64/Makefile.softmmu-target | 5 +-
64
tests/tcg/aarch64/Makefile.target | 3 +-
65
include/elf.h | 1 +
66
include/hw/arm/exynos4210.h | 4 +
67
include/hw/or-irq.h | 2 +-
68
target/arm/cpu.h | 25 +
69
hw/acpi/generic_event_device.c | 2 +-
70
hw/arm/exynos4210.c | 77 ++-
71
hw/arm/virt.c | 6 +-
72
hw/char/exynos4210_uart.c | 245 +++++---
73
hw/dma/pl330.c | 88 +--
74
hw/misc/stm32f4xx_syscfg.c | 2 +-
75
target/arm/arch_dump.c | 124 +++-
76
target/arm/cpu.c | 1 +
77
target/arm/kvm64.c | 24 -
78
target/arm/pauth_helper.c | 4 +-
79
tests/tcg/aarch64/pauth-1.c | 2 -
80
tests/tcg/aarch64/pauth-2.c | 2 -
81
tests/tcg/aarch64/pauth-4.c | 25 +
82
tests/tcg/aarch64/system/pauth-3.c | 40 ++
83
MAINTAINERS | 1 +
84
docs/index.html.in | 1 +
85
docs/index.rst | 2 +-
86
docs/interop/conf.py | 4 +-
87
docs/interop/index.rst | 1 +
88
docs/interop/qemu-nbd.rst | 263 ++++++++
89
docs/interop/qemu-option-trace.rst.inc | 30 +
90
docs/qemu-block-drivers.texi | 889 ---------------------------
91
docs/system/conf.py | 22 +
92
docs/system/index.rst | 17 +
93
docs/system/qemu-block-drivers.rst | 985 ++++++++++++++++++++++++++++++
94
hw/char/trace-events | 20 +
95
hw/dma/trace-events | 24 +
96
qemu-doc.texi | 18 -
97
qemu-nbd.texi | 214 -------
98
qemu-option-trace.texi | 4 +
99
qemu-options.hx | 2 +-
100
38 files changed, 1898 insertions(+), 1318 deletions(-)
101
create mode 100644 tests/tcg/aarch64/pauth-4.c
102
create mode 100644 tests/tcg/aarch64/system/pauth-3.c
103
create mode 100644 docs/interop/qemu-nbd.rst
104
create mode 100644 docs/interop/qemu-option-trace.rst.inc
105
delete mode 100644 docs/qemu-block-drivers.texi
106
create mode 100644 docs/system/conf.py
107
create mode 100644 docs/system/index.rst
108
create mode 100644 docs/system/qemu-block-drivers.rst
109
delete mode 100644 qemu-nbd.texi
110
diff view generated by jsdifflib
New patch
1
From: Eric Auger <eric.auger@redhat.com>
1
2
3
We are missing the VIRT_COMPAT_3_0 definition and setting.
4
Let's add them.
5
6
Signed-off-by: Eric Auger <eric.auger@redhat.com>
7
Reviewed-by: Andrew Jones <drjones@redhat.com>
8
Message-id: 20181024085602.16611-1-eric.auger@redhat.com
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
11
hw/arm/virt.c | 4 ++++
12
1 file changed, 4 insertions(+)
13
14
diff --git a/hw/arm/virt.c b/hw/arm/virt.c
15
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/arm/virt.c
17
+++ b/hw/arm/virt.c
18
@@ -XXX,XX +XXX,XX @@ static void virt_machine_3_1_options(MachineClass *mc)
19
}
20
DEFINE_VIRT_MACHINE_AS_LATEST(3, 1)
21
22
+#define VIRT_COMPAT_3_0 \
23
+ HW_COMPAT_3_0
24
+
25
static void virt_3_0_instance_init(Object *obj)
26
{
27
virt_3_1_instance_init(obj);
28
@@ -XXX,XX +XXX,XX @@ static void virt_3_0_instance_init(Object *obj)
29
static void virt_machine_3_0_options(MachineClass *mc)
30
{
31
virt_machine_3_1_options(mc);
32
+ SET_MACHINE_COMPAT(mc, VIRT_COMPAT_3_0);
33
}
34
DEFINE_VIRT_MACHINE(3, 0)
35
36
--
37
2.19.1
38
39
diff view generated by jsdifflib
New patch
1
From: Julia Suvorova <jusual@mail.ru>
1
2
3
Not implemented: CTS/NCTS, PSEL*.
4
5
Signed-off-by: Julia Suvorova <jusual@mail.ru>
6
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
---
9
hw/char/Makefile.objs | 1 +
10
include/hw/char/nrf51_uart.h | 78 +++++++++
11
hw/char/nrf51_uart.c | 330 +++++++++++++++++++++++++++++++++++
12
hw/char/trace-events | 4 +
13
4 files changed, 413 insertions(+)
14
create mode 100644 include/hw/char/nrf51_uart.h
15
create mode 100644 hw/char/nrf51_uart.c
16
17
diff --git a/hw/char/Makefile.objs b/hw/char/Makefile.objs
18
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/char/Makefile.objs
20
+++ b/hw/char/Makefile.objs
21
@@ -XXX,XX +XXX,XX @@
22
common-obj-$(CONFIG_IPACK) += ipoctal232.o
23
common-obj-$(CONFIG_ESCC) += escc.o
24
+common-obj-$(CONFIG_NRF51_SOC) += nrf51_uart.o
25
common-obj-$(CONFIG_PARALLEL) += parallel.o
26
common-obj-$(CONFIG_PARALLEL) += parallel-isa.o
27
common-obj-$(CONFIG_PL011) += pl011.o
28
diff --git a/include/hw/char/nrf51_uart.h b/include/hw/char/nrf51_uart.h
29
new file mode 100644
30
index XXXXXXX..XXXXXXX
31
--- /dev/null
32
+++ b/include/hw/char/nrf51_uart.h
33
@@ -XXX,XX +XXX,XX @@
34
+/*
35
+ * nRF51 SoC UART emulation
36
+ *
37
+ * Copyright (c) 2018 Julia Suvorova <jusual@mail.ru>
38
+ *
39
+ * This program is free software; you can redistribute it and/or modify
40
+ * it under the terms of the GNU General Public License version 2 or
41
+ * (at your option) any later version.
42
+ */
43
+
44
+#ifndef NRF51_UART_H
45
+#define NRF51_UART_H
46
+
47
+#include "hw/sysbus.h"
48
+#include "chardev/char-fe.h"
49
+#include "hw/registerfields.h"
50
+
51
+#define UART_FIFO_LENGTH 6
52
+#define UART_BASE 0x40002000
53
+#define UART_SIZE 0x1000
54
+
55
+#define TYPE_NRF51_UART "nrf51_soc.uart"
56
+#define NRF51_UART(obj) OBJECT_CHECK(NRF51UARTState, (obj), TYPE_NRF51_UART)
57
+
58
+REG32(UART_STARTRX, 0x000)
59
+REG32(UART_STOPRX, 0x004)
60
+REG32(UART_STARTTX, 0x008)
61
+REG32(UART_STOPTX, 0x00C)
62
+REG32(UART_SUSPEND, 0x01C)
63
+
64
+REG32(UART_CTS, 0x100)
65
+REG32(UART_NCTS, 0x104)
66
+REG32(UART_RXDRDY, 0x108)
67
+REG32(UART_TXDRDY, 0x11C)
68
+REG32(UART_ERROR, 0x124)
69
+REG32(UART_RXTO, 0x144)
70
+
71
+REG32(UART_INTEN, 0x300)
72
+ FIELD(UART_INTEN, CTS, 0, 1)
73
+ FIELD(UART_INTEN, NCTS, 1, 1)
74
+ FIELD(UART_INTEN, RXDRDY, 2, 1)
75
+ FIELD(UART_INTEN, TXDRDY, 7, 1)
76
+ FIELD(UART_INTEN, ERROR, 9, 1)
77
+ FIELD(UART_INTEN, RXTO, 17, 1)
78
+REG32(UART_INTENSET, 0x304)
79
+REG32(UART_INTENCLR, 0x308)
80
+REG32(UART_ERRORSRC, 0x480)
81
+REG32(UART_ENABLE, 0x500)
82
+REG32(UART_PSELRTS, 0x508)
83
+REG32(UART_PSELTXD, 0x50C)
84
+REG32(UART_PSELCTS, 0x510)
85
+REG32(UART_PSELRXD, 0x514)
86
+REG32(UART_RXD, 0x518)
87
+REG32(UART_TXD, 0x51C)
88
+REG32(UART_BAUDRATE, 0x524)
89
+REG32(UART_CONFIG, 0x56C)
90
+
91
+typedef struct NRF51UARTState {
92
+ SysBusDevice parent_obj;
93
+
94
+ MemoryRegion iomem;
95
+ CharBackend chr;
96
+ qemu_irq irq;
97
+ guint watch_tag;
98
+
99
+ uint8_t rx_fifo[UART_FIFO_LENGTH];
100
+ unsigned int rx_fifo_pos;
101
+ unsigned int rx_fifo_len;
102
+
103
+ uint32_t reg[0x56C];
104
+
105
+ bool rx_started;
106
+ bool tx_started;
107
+ bool pending_tx_byte;
108
+ bool enabled;
109
+} NRF51UARTState;
110
+
111
+#endif
112
diff --git a/hw/char/nrf51_uart.c b/hw/char/nrf51_uart.c
113
new file mode 100644
114
index XXXXXXX..XXXXXXX
115
--- /dev/null
116
+++ b/hw/char/nrf51_uart.c
117
@@ -XXX,XX +XXX,XX @@
118
+/*
119
+ * nRF51 SoC UART emulation
120
+ *
121
+ * See nRF51 Series Reference Manual, "29 Universal Asynchronous
122
+ * Receiver/Transmitter" for hardware specifications:
123
+ * http://infocenter.nordicsemi.com/pdf/nRF51_RM_v3.0.pdf
124
+ *
125
+ * Copyright (c) 2018 Julia Suvorova <jusual@mail.ru>
126
+ *
127
+ * This program is free software; you can redistribute it and/or modify
128
+ * it under the terms of the GNU General Public License version 2 or
129
+ * (at your option) any later version.
130
+ */
131
+
132
+#include "qemu/osdep.h"
133
+#include "qemu/log.h"
134
+#include "hw/char/nrf51_uart.h"
135
+#include "trace.h"
136
+
137
+static void nrf51_uart_update_irq(NRF51UARTState *s)
138
+{
139
+ bool irq = false;
140
+
141
+ irq |= (s->reg[R_UART_RXDRDY] &&
142
+ (s->reg[R_UART_INTEN] & R_UART_INTEN_RXDRDY_MASK));
143
+ irq |= (s->reg[R_UART_TXDRDY] &&
144
+ (s->reg[R_UART_INTEN] & R_UART_INTEN_TXDRDY_MASK));
145
+ irq |= (s->reg[R_UART_ERROR] &&
146
+ (s->reg[R_UART_INTEN] & R_UART_INTEN_ERROR_MASK));
147
+ irq |= (s->reg[R_UART_RXTO] &&
148
+ (s->reg[R_UART_INTEN] & R_UART_INTEN_RXTO_MASK));
149
+
150
+ qemu_set_irq(s->irq, irq);
151
+}
152
+
153
+static uint64_t uart_read(void *opaque, hwaddr addr, unsigned int size)
154
+{
155
+ NRF51UARTState *s = NRF51_UART(opaque);
156
+ uint64_t r;
157
+
158
+ if (!s->enabled) {
159
+ return 0;
160
+ }
161
+
162
+ switch (addr) {
163
+ case A_UART_RXD:
164
+ r = s->rx_fifo[s->rx_fifo_pos];
165
+ if (s->rx_started && s->rx_fifo_len) {
166
+ s->rx_fifo_pos = (s->rx_fifo_pos + 1) % UART_FIFO_LENGTH;
167
+ s->rx_fifo_len--;
168
+ if (s->rx_fifo_len) {
169
+ s->reg[R_UART_RXDRDY] = 1;
170
+ nrf51_uart_update_irq(s);
171
+ }
172
+ qemu_chr_fe_accept_input(&s->chr);
173
+ }
174
+ break;
175
+ case A_UART_INTENSET:
176
+ case A_UART_INTENCLR:
177
+ case A_UART_INTEN:
178
+ r = s->reg[R_UART_INTEN];
179
+ break;
180
+ default:
181
+ r = s->reg[addr / 4];
182
+ break;
183
+ }
184
+
185
+ trace_nrf51_uart_read(addr, r, size);
186
+
187
+ return r;
188
+}
189
+
190
+static gboolean uart_transmit(GIOChannel *chan, GIOCondition cond, void *opaque)
191
+{
192
+ NRF51UARTState *s = NRF51_UART(opaque);
193
+ int r;
194
+ uint8_t c = s->reg[R_UART_TXD];
195
+
196
+ s->watch_tag = 0;
197
+
198
+ r = qemu_chr_fe_write(&s->chr, &c, 1);
199
+ if (r <= 0) {
200
+ s->watch_tag = qemu_chr_fe_add_watch(&s->chr, G_IO_OUT | G_IO_HUP,
201
+ uart_transmit, s);
202
+ if (!s->watch_tag) {
203
+ /* The hardware has no transmit error reporting,
204
+ * so silently drop the byte
205
+ */
206
+ goto buffer_drained;
207
+ }
208
+ return FALSE;
209
+ }
210
+
211
+buffer_drained:
212
+ s->reg[R_UART_TXDRDY] = 1;
213
+ s->pending_tx_byte = false;
214
+ return FALSE;
215
+}
216
+
217
+static void uart_cancel_transmit(NRF51UARTState *s)
218
+{
219
+ if (s->watch_tag) {
220
+ g_source_remove(s->watch_tag);
221
+ s->watch_tag = 0;
222
+ }
223
+}
224
+
225
+static void uart_write(void *opaque, hwaddr addr,
226
+ uint64_t value, unsigned int size)
227
+{
228
+ NRF51UARTState *s = NRF51_UART(opaque);
229
+
230
+ trace_nrf51_uart_write(addr, value, size);
231
+
232
+ if (!s->enabled && (addr != A_UART_ENABLE)) {
233
+ return;
234
+ }
235
+
236
+ switch (addr) {
237
+ case A_UART_TXD:
238
+ if (!s->pending_tx_byte && s->tx_started) {
239
+ s->reg[R_UART_TXD] = value;
240
+ s->pending_tx_byte = true;
241
+ uart_transmit(NULL, G_IO_OUT, s);
242
+ }
243
+ break;
244
+ case A_UART_INTEN:
245
+ s->reg[R_UART_INTEN] = value;
246
+ break;
247
+ case A_UART_INTENSET:
248
+ s->reg[R_UART_INTEN] |= value;
249
+ break;
250
+ case A_UART_INTENCLR:
251
+ s->reg[R_UART_INTEN] &= ~value;
252
+ break;
253
+ case A_UART_TXDRDY ... A_UART_RXTO:
254
+ s->reg[addr / 4] = value;
255
+ break;
256
+ case A_UART_ERRORSRC:
257
+ s->reg[addr / 4] &= ~value;
258
+ break;
259
+ case A_UART_RXD:
260
+ break;
261
+ case A_UART_RXDRDY:
262
+ if (value == 0) {
263
+ s->reg[R_UART_RXDRDY] = 0;
264
+ }
265
+ break;
266
+ case A_UART_STARTTX:
267
+ if (value == 1) {
268
+ s->tx_started = true;
269
+ }
270
+ break;
271
+ case A_UART_STARTRX:
272
+ if (value == 1) {
273
+ s->rx_started = true;
274
+ }
275
+ break;
276
+ case A_UART_ENABLE:
277
+ if (value) {
278
+ if (value == 4) {
279
+ s->enabled = true;
280
+ }
281
+ break;
282
+ }
283
+ s->enabled = false;
284
+ value = 1;
285
+ /* fall through */
286
+ case A_UART_SUSPEND:
287
+ case A_UART_STOPTX:
288
+ if (value == 1) {
289
+ s->tx_started = false;
290
+ }
291
+ /* fall through */
292
+ case A_UART_STOPRX:
293
+ if (addr != A_UART_STOPTX && value == 1) {
294
+ s->rx_started = false;
295
+ s->reg[R_UART_RXTO] = 1;
296
+ }
297
+ break;
298
+ default:
299
+ s->reg[addr / 4] = value;
300
+ break;
301
+ }
302
+ nrf51_uart_update_irq(s);
303
+}
304
+
305
+static const MemoryRegionOps uart_ops = {
306
+ .read = uart_read,
307
+ .write = uart_write,
308
+ .endianness = DEVICE_LITTLE_ENDIAN,
309
+};
310
+
311
+static void nrf51_uart_reset(DeviceState *dev)
312
+{
313
+ NRF51UARTState *s = NRF51_UART(dev);
314
+
315
+ s->pending_tx_byte = 0;
316
+
317
+ uart_cancel_transmit(s);
318
+
319
+ memset(s->reg, 0, sizeof(s->reg));
320
+
321
+ s->reg[R_UART_PSELRTS] = 0xFFFFFFFF;
322
+ s->reg[R_UART_PSELTXD] = 0xFFFFFFFF;
323
+ s->reg[R_UART_PSELCTS] = 0xFFFFFFFF;
324
+ s->reg[R_UART_PSELRXD] = 0xFFFFFFFF;
325
+ s->reg[R_UART_BAUDRATE] = 0x4000000;
326
+
327
+ s->rx_fifo_len = 0;
328
+ s->rx_fifo_pos = 0;
329
+ s->rx_started = false;
330
+ s->tx_started = false;
331
+ s->enabled = false;
332
+}
333
+
334
+static void uart_receive(void *opaque, const uint8_t *buf, int size)
335
+{
336
+
337
+ NRF51UARTState *s = NRF51_UART(opaque);
338
+ int i;
339
+
340
+ if (size == 0 || s->rx_fifo_len >= UART_FIFO_LENGTH) {
341
+ return;
342
+ }
343
+
344
+ for (i = 0; i < size; i++) {
345
+ uint32_t pos = (s->rx_fifo_pos + s->rx_fifo_len) % UART_FIFO_LENGTH;
346
+ s->rx_fifo[pos] = buf[i];
347
+ s->rx_fifo_len++;
348
+ }
349
+
350
+ s->reg[R_UART_RXDRDY] = 1;
351
+ nrf51_uart_update_irq(s);
352
+}
353
+
354
+static int uart_can_receive(void *opaque)
355
+{
356
+ NRF51UARTState *s = NRF51_UART(opaque);
357
+
358
+ return s->rx_started ? (UART_FIFO_LENGTH - s->rx_fifo_len) : 0;
359
+}
360
+
361
+static void uart_event(void *opaque, int event)
362
+{
363
+ NRF51UARTState *s = NRF51_UART(opaque);
364
+
365
+ if (event == CHR_EVENT_BREAK) {
366
+ s->reg[R_UART_ERRORSRC] |= 3;
367
+ s->reg[R_UART_ERROR] = 1;
368
+ nrf51_uart_update_irq(s);
369
+ }
370
+}
371
+
372
+static void nrf51_uart_realize(DeviceState *dev, Error **errp)
373
+{
374
+ NRF51UARTState *s = NRF51_UART(dev);
375
+
376
+ qemu_chr_fe_set_handlers(&s->chr, uart_can_receive, uart_receive,
377
+ uart_event, NULL, s, NULL, true);
378
+}
379
+
380
+static void nrf51_uart_init(Object *obj)
381
+{
382
+ NRF51UARTState *s = NRF51_UART(obj);
383
+ SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
384
+
385
+ memory_region_init_io(&s->iomem, obj, &uart_ops, s,
386
+ "nrf51_soc.uart", UART_SIZE);
387
+ sysbus_init_mmio(sbd, &s->iomem);
388
+ sysbus_init_irq(sbd, &s->irq);
389
+}
390
+
391
+static int nrf51_uart_post_load(void *opaque, int version_id)
392
+{
393
+ NRF51UARTState *s = NRF51_UART(opaque);
394
+
395
+ if (s->pending_tx_byte) {
396
+ s->watch_tag = qemu_chr_fe_add_watch(&s->chr, G_IO_OUT | G_IO_HUP,
397
+ uart_transmit, s);
398
+ }
399
+
400
+ return 0;
401
+}
402
+
403
+static const VMStateDescription nrf51_uart_vmstate = {
404
+ .name = "nrf51_soc.uart",
405
+ .post_load = nrf51_uart_post_load,
406
+ .fields = (VMStateField[]) {
407
+ VMSTATE_UINT32_ARRAY(reg, NRF51UARTState, 0x56C),
408
+ VMSTATE_UINT8_ARRAY(rx_fifo, NRF51UARTState, UART_FIFO_LENGTH),
409
+ VMSTATE_UINT32(rx_fifo_pos, NRF51UARTState),
410
+ VMSTATE_UINT32(rx_fifo_len, NRF51UARTState),
411
+ VMSTATE_BOOL(rx_started, NRF51UARTState),
412
+ VMSTATE_BOOL(tx_started, NRF51UARTState),
413
+ VMSTATE_BOOL(pending_tx_byte, NRF51UARTState),
414
+ VMSTATE_BOOL(enabled, NRF51UARTState),
415
+ VMSTATE_END_OF_LIST()
416
+ }
417
+};
418
+
419
+static Property nrf51_uart_properties[] = {
420
+ DEFINE_PROP_CHR("chardev", NRF51UARTState, chr),
421
+ DEFINE_PROP_END_OF_LIST(),
422
+};
423
+
424
+static void nrf51_uart_class_init(ObjectClass *klass, void *data)
425
+{
426
+ DeviceClass *dc = DEVICE_CLASS(klass);
427
+
428
+ dc->reset = nrf51_uart_reset;
429
+ dc->realize = nrf51_uart_realize;
430
+ dc->props = nrf51_uart_properties;
431
+ dc->vmsd = &nrf51_uart_vmstate;
432
+}
433
+
434
+static const TypeInfo nrf51_uart_info = {
435
+ .name = TYPE_NRF51_UART,
436
+ .parent = TYPE_SYS_BUS_DEVICE,
437
+ .instance_size = sizeof(NRF51UARTState),
438
+ .instance_init = nrf51_uart_init,
439
+ .class_init = nrf51_uart_class_init
440
+};
441
+
442
+static void nrf51_uart_register_types(void)
443
+{
444
+ type_register_static(&nrf51_uart_info);
445
+}
446
+
447
+type_init(nrf51_uart_register_types)
448
diff --git a/hw/char/trace-events b/hw/char/trace-events
449
index XXXXXXX..XXXXXXX 100644
450
--- a/hw/char/trace-events
451
+++ b/hw/char/trace-events
452
@@ -XXX,XX +XXX,XX @@ cmsdk_apb_uart_receive(uint8_t c) "CMSDK APB UART: got character 0x%x from backe
453
cmsdk_apb_uart_tx_pending(void) "CMSDK APB UART: character send to backend pending"
454
cmsdk_apb_uart_tx(uint8_t c) "CMSDK APB UART: character 0x%x sent to backend"
455
cmsdk_apb_uart_set_params(int speed) "CMSDK APB UART: params set to %d 8N1"
456
+
457
+# hw/char/nrf51_uart.c
458
+nrf51_uart_read(uint64_t addr, uint64_t r, unsigned int size) "addr 0x%" PRIx64 " value 0x%" PRIx64 " size %u"
459
+nrf51_uart_write(uint64_t addr, uint64_t value, unsigned int size) "addr 0x%" PRIx64 " value 0x%" PRIx64 " size %u"
460
--
461
2.19.1
462
463
diff view generated by jsdifflib
New patch
1
From: Julia Suvorova <jusual@mail.ru>
1
2
3
Wire up nRF51 UART in the corresponding SoC.
4
5
Signed-off-by: Julia Suvorova <jusual@mail.ru>
6
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
11
include/hw/arm/nrf51_soc.h | 3 +++
12
hw/arm/microbit.c | 2 ++
13
hw/arm/nrf51_soc.c | 20 ++++++++++++++++++++
14
3 files changed, 25 insertions(+)
15
16
diff --git a/include/hw/arm/nrf51_soc.h b/include/hw/arm/nrf51_soc.h
17
index XXXXXXX..XXXXXXX 100644
18
--- a/include/hw/arm/nrf51_soc.h
19
+++ b/include/hw/arm/nrf51_soc.h
20
@@ -XXX,XX +XXX,XX @@
21
22
#include "hw/sysbus.h"
23
#include "hw/arm/armv7m.h"
24
+#include "hw/char/nrf51_uart.h"
25
26
#define TYPE_NRF51_SOC "nrf51-soc"
27
#define NRF51_SOC(obj) \
28
@@ -XXX,XX +XXX,XX @@ typedef struct NRF51State {
29
/*< public >*/
30
ARMv7MState cpu;
31
32
+ NRF51UARTState uart;
33
+
34
MemoryRegion iomem;
35
MemoryRegion sram;
36
MemoryRegion flash;
37
diff --git a/hw/arm/microbit.c b/hw/arm/microbit.c
38
index XXXXXXX..XXXXXXX 100644
39
--- a/hw/arm/microbit.c
40
+++ b/hw/arm/microbit.c
41
@@ -XXX,XX +XXX,XX @@
42
#include "qapi/error.h"
43
#include "hw/boards.h"
44
#include "hw/arm/arm.h"
45
+#include "sysemu/sysemu.h"
46
#include "exec/address-spaces.h"
47
48
#include "hw/arm/nrf51_soc.h"
49
@@ -XXX,XX +XXX,XX @@ static void microbit_init(MachineState *machine)
50
51
sysbus_init_child_obj(OBJECT(machine), "nrf51", soc, sizeof(s->nrf51),
52
TYPE_NRF51_SOC);
53
+ qdev_prop_set_chr(DEVICE(&s->nrf51), "serial0", serial_hd(0));
54
object_property_set_link(soc, OBJECT(system_memory), "memory",
55
&error_fatal);
56
object_property_set_bool(soc, true, "realized", &error_fatal);
57
diff --git a/hw/arm/nrf51_soc.c b/hw/arm/nrf51_soc.c
58
index XXXXXXX..XXXXXXX 100644
59
--- a/hw/arm/nrf51_soc.c
60
+++ b/hw/arm/nrf51_soc.c
61
@@ -XXX,XX +XXX,XX @@
62
#define NRF51822_FLASH_SIZE (256 * 1024)
63
#define NRF51822_SRAM_SIZE (16 * 1024)
64
65
+#define BASE_TO_IRQ(base) ((base >> 12) & 0x1F)
66
+
67
static void nrf51_soc_realize(DeviceState *dev_soc, Error **errp)
68
{
69
NRF51State *s = NRF51_SOC(dev_soc);
70
+ MemoryRegion *mr;
71
Error *err = NULL;
72
73
if (!s->board_memory) {
74
@@ -XXX,XX +XXX,XX @@ static void nrf51_soc_realize(DeviceState *dev_soc, Error **errp)
75
}
76
memory_region_add_subregion(&s->container, SRAM_BASE, &s->sram);
77
78
+ /* UART */
79
+ object_property_set_bool(OBJECT(&s->uart), true, "realized", &err);
80
+ if (err) {
81
+ error_propagate(errp, err);
82
+ return;
83
+ }
84
+ mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(&s->uart), 0);
85
+ memory_region_add_subregion_overlap(&s->container, UART_BASE, mr, 0);
86
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->uart), 0,
87
+ qdev_get_gpio_in(DEVICE(&s->cpu),
88
+ BASE_TO_IRQ(UART_BASE)));
89
+
90
create_unimplemented_device("nrf51_soc.io", IOMEM_BASE, IOMEM_SIZE);
91
create_unimplemented_device("nrf51_soc.ficr", FICR_BASE, FICR_SIZE);
92
create_unimplemented_device("nrf51_soc.private",
93
@@ -XXX,XX +XXX,XX @@ static void nrf51_soc_init(Object *obj)
94
qdev_prop_set_string(DEVICE(&s->cpu), "cpu-type",
95
ARM_CPU_TYPE_NAME("cortex-m0"));
96
qdev_prop_set_uint32(DEVICE(&s->cpu), "num-irq", 32);
97
+
98
+ sysbus_init_child_obj(obj, "uart", &s->uart, sizeof(s->uart),
99
+ TYPE_NRF51_UART);
100
+ object_property_add_alias(obj, "serial0", OBJECT(&s->uart), "chardev",
101
+ &error_abort);
102
}
103
104
static Property nrf51_soc_properties[] = {
105
--
106
2.19.1
107
108
diff view generated by jsdifflib
New patch
1
From: Julia Suvorova <jusual@mail.ru>
1
2
3
New mini-kernel test for nRF51 SoC UART.
4
5
Signed-off-by: Julia Suvorova <jusual@mail.ru>
6
Acked-by: Thomas Huth <thuth@redhat.com>
7
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
---
10
tests/boot-serial-test.c | 19 +++++++++++++++++++
11
1 file changed, 19 insertions(+)
12
13
diff --git a/tests/boot-serial-test.c b/tests/boot-serial-test.c
14
index XXXXXXX..XXXXXXX 100644
15
--- a/tests/boot-serial-test.c
16
+++ b/tests/boot-serial-test.c
17
@@ -XXX,XX +XXX,XX @@ static const uint8_t kernel_aarch64[] = {
18
0xfd, 0xff, 0xff, 0x17, /* b -12 (loop) */
19
};
20
21
+static const uint8_t kernel_nrf51[] = {
22
+ 0x00, 0x00, 0x00, 0x00, /* Stack top address */
23
+ 0x09, 0x00, 0x00, 0x00, /* Reset handler address */
24
+ 0x04, 0x4a, /* ldr r2, [pc, #16] Get ENABLE */
25
+ 0x04, 0x21, /* movs r1, #4 */
26
+ 0x11, 0x60, /* str r1, [r2] */
27
+ 0x04, 0x4a, /* ldr r2, [pc, #16] Get STARTTX */
28
+ 0x01, 0x21, /* movs r1, #1 */
29
+ 0x11, 0x60, /* str r1, [r2] */
30
+ 0x03, 0x4a, /* ldr r2, [pc, #12] Get TXD */
31
+ 0x54, 0x21, /* movs r1, 'T' */
32
+ 0x11, 0x60, /* str r1, [r2] */
33
+ 0xfe, 0xe7, /* b . */
34
+ 0x00, 0x25, 0x00, 0x40, /* 0x40002500 = UART ENABLE */
35
+ 0x08, 0x20, 0x00, 0x40, /* 0x40002008 = UART STARTTX */
36
+ 0x1c, 0x25, 0x00, 0x40 /* 0x4000251c = UART TXD */
37
+};
38
+
39
typedef struct testdef {
40
const char *arch; /* Target architecture */
41
const char *machine; /* Name of the machine */
42
@@ -XXX,XX +XXX,XX @@ static testdef_t tests[] = {
43
{ "hppa", "hppa", "", "SeaBIOS wants SYSTEM HALT" },
44
{ "aarch64", "virt", "-cpu cortex-a57", "TT", sizeof(kernel_aarch64),
45
kernel_aarch64 },
46
+ { "arm", "microbit", "", "T", sizeof(kernel_nrf51), kernel_nrf51 },
47
48
{ NULL }
49
};
50
--
51
2.19.1
52
53
diff view generated by jsdifflib
New patch
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
1
2
3
Shannon Zhao's email at Huawei is bouncing: remove it.
4
5
X-Failed-Recipients: zhaoshenglong@huawei.com
6
** Address not found **
7
Your message wasn't delivered to zhaoshenglong@huawei.com because the address couldn't be found, or is unable to receive mail.
8
9
Note that the section still contains his personal email (see e59f13d76bb).
10
11
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
12
Acked-by: Shannon Zhao <shannon.zhaosl@gmail.com>
13
Message-id: 20181029195931.8747-1-philmd@redhat.com
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
---
16
MAINTAINERS | 1 -
17
1 file changed, 1 deletion(-)
18
19
diff --git a/MAINTAINERS b/MAINTAINERS
20
index XXXXXXX..XXXXXXX 100644
21
--- a/MAINTAINERS
22
+++ b/MAINTAINERS
23
@@ -XXX,XX +XXX,XX @@ F: hw/*/xlnx*.c
24
F: include/hw/*/xlnx*.h
25
26
ARM ACPI Subsystem
27
-M: Shannon Zhao <zhaoshenglong@huawei.com>
28
M: Shannon Zhao <shannon.zhaosl@gmail.com>
29
L: qemu-arm@nongnu.org
30
S: Maintained
31
--
32
2.19.1
33
34
diff view generated by jsdifflib
New patch
1
From: Prasad J Pandit <pjp@fedoraproject.org>
1
2
3
The high[31:28] bits of 'direction' and 'state' registers of
4
SA-1100/SA-1110 device are reserved. Setting them may lead to
5
OOB 's->handler[]' array access issue. Mask off [31:28] bits to
6
avoid it.
7
8
Reported-by: Moguofang <moguofang@huawei.com>
9
Signed-off-by: Prasad J Pandit <pjp@fedoraproject.org>
10
Message-id: 20181030114635.31232-1-ppandit@redhat.com
11
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
---
14
hw/arm/strongarm.c | 4 ++--
15
1 file changed, 2 insertions(+), 2 deletions(-)
16
17
diff --git a/hw/arm/strongarm.c b/hw/arm/strongarm.c
18
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/arm/strongarm.c
20
+++ b/hw/arm/strongarm.c
21
@@ -XXX,XX +XXX,XX @@ static void strongarm_gpio_write(void *opaque, hwaddr offset,
22
23
switch (offset) {
24
case GPDR: /* GPIO Pin-Direction registers */
25
- s->dir = value;
26
+ s->dir = value & 0x0fffffff;
27
strongarm_gpio_handler_update(s);
28
break;
29
30
case GPSR: /* GPIO Pin-Output Set registers */
31
- s->olevel |= value;
32
+ s->olevel |= value & 0x0fffffff;
33
strongarm_gpio_handler_update(s);
34
break;
35
36
--
37
2.19.1
38
39
diff view generated by jsdifflib
New patch
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
1
2
3
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
4
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
---
9
hw/arm/xilinx_zynq.c | 2 +-
10
1 file changed, 1 insertion(+), 1 deletion(-)
11
12
diff --git a/hw/arm/xilinx_zynq.c b/hw/arm/xilinx_zynq.c
13
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/arm/xilinx_zynq.c
15
+++ b/hw/arm/xilinx_zynq.c
16
@@ -XXX,XX +XXX,XX @@ static void zynq_init(MachineState *machine)
17
busdev = SYS_BUS_DEVICE(dev);
18
sysbus_mmio_map(busdev, 0, 0xF8003000);
19
sysbus_connect_irq(busdev, 0, pic[45-IRQ_OFFSET]); /* abort irq line */
20
- for (n = 0; n < 8; ++n) { /* event irqs */
21
+ for (n = 0; n < ARRAY_SIZE(dma_irqs); ++n) { /* event irqs */
22
sysbus_connect_irq(busdev, n + 1, pic[dma_irqs[n] - IRQ_OFFSET]);
23
}
24
25
--
26
2.19.1
27
28
diff view generated by jsdifflib
New patch
1
From: Richard Henderson <richard.henderson@linaro.org>
1
2
3
When populating id registers from kvm, on a host that doesn't support
4
aarch32 mode at all, neither arm_div nor jazelle will be supported either.
5
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
8
Tested-by: Alex Bennée <alex.bennee@linaro.org>
9
Message-id: 20181102102025.3546-1-richard.henderson@linaro.org
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
13
target/arm/cpu.h | 5 +++++
14
target/arm/cpu.c | 15 +++++++++++++--
15
2 files changed, 18 insertions(+), 2 deletions(-)
16
17
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
18
index XXXXXXX..XXXXXXX 100644
19
--- a/target/arm/cpu.h
20
+++ b/target/arm/cpu.h
21
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_aa64_fp16(const ARMISARegisters *id)
22
return FIELD_EX64(id->id_aa64pfr0, ID_AA64PFR0, FP) == 1;
23
}
24
25
+static inline bool isar_feature_aa64_aa32(const ARMISARegisters *id)
26
+{
27
+ return FIELD_EX64(id->id_aa64pfr0, ID_AA64PFR0, EL0) >= 2;
28
+}
29
+
30
static inline bool isar_feature_aa64_sve(const ARMISARegisters *id)
31
{
32
return FIELD_EX64(id->id_aa64pfr0, ID_AA64PFR0, SVE) != 0;
33
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
34
index XXXXXXX..XXXXXXX 100644
35
--- a/target/arm/cpu.c
36
+++ b/target/arm/cpu.c
37
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
38
CPUARMState *env = &cpu->env;
39
int pagebits;
40
Error *local_err = NULL;
41
+ bool no_aa32 = false;
42
43
/* If we needed to query the host kernel for the CPU features
44
* then it's possible that might have failed in the initfn, but
45
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
46
set_feature(env, ARM_FEATURE_V7VE);
47
}
48
}
49
+
50
+ /*
51
+ * There exist AArch64 cpus without AArch32 support. When KVM
52
+ * queries ID_ISAR0_EL1 on such a host, the value is UNKNOWN.
53
+ * Similarly, we cannot check ID_AA64PFR0 without AArch64 support.
54
+ */
55
+ if (arm_feature(&cpu->env, ARM_FEATURE_AARCH64)) {
56
+ no_aa32 = !cpu_isar_feature(aa64_aa32, cpu);
57
+ }
58
+
59
if (arm_feature(env, ARM_FEATURE_V7VE)) {
60
/* v7 Virtualization Extensions. In real hardware this implies
61
* EL2 and also the presence of the Security Extensions.
62
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
63
* Presence of EL2 itself is ARM_FEATURE_EL2, and of the
64
* Security Extensions is ARM_FEATURE_EL3.
65
*/
66
- assert(cpu_isar_feature(arm_div, cpu));
67
+ assert(no_aa32 || cpu_isar_feature(arm_div, cpu));
68
set_feature(env, ARM_FEATURE_LPAE);
69
set_feature(env, ARM_FEATURE_V7);
70
}
71
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
72
if (arm_feature(env, ARM_FEATURE_V6)) {
73
set_feature(env, ARM_FEATURE_V5);
74
if (!arm_feature(env, ARM_FEATURE_M)) {
75
- assert(cpu_isar_feature(jazelle, cpu));
76
+ assert(no_aa32 || cpu_isar_feature(jazelle, cpu));
77
set_feature(env, ARM_FEATURE_AUXCR);
78
}
79
}
80
--
81
2.19.1
82
83
diff view generated by jsdifflib
New patch
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
2
3
Add a model of Xilinx Versal SoC.
4
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
6
Message-id: 20181102131913.1535-2-edgar.iglesias@xilinx.com
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
---
10
hw/arm/Makefile.objs | 1 +
11
include/hw/arm/xlnx-versal.h | 122 +++++++++++
12
hw/arm/xlnx-versal.c | 323 ++++++++++++++++++++++++++++
13
default-configs/aarch64-softmmu.mak | 1 +
14
4 files changed, 447 insertions(+)
15
create mode 100644 include/hw/arm/xlnx-versal.h
16
create mode 100644 hw/arm/xlnx-versal.c
17
18
diff --git a/hw/arm/Makefile.objs b/hw/arm/Makefile.objs
19
index XXXXXXX..XXXXXXX 100644
20
--- a/hw/arm/Makefile.objs
21
+++ b/hw/arm/Makefile.objs
22
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_ALLWINNER_A10) += allwinner-a10.o cubieboard.o
23
obj-$(CONFIG_RASPI) += bcm2835_peripherals.o bcm2836.o raspi.o
24
obj-$(CONFIG_STM32F205_SOC) += stm32f205_soc.o
25
obj-$(CONFIG_XLNX_ZYNQMP_ARM) += xlnx-zynqmp.o xlnx-zcu102.o
26
+obj-$(CONFIG_XLNX_VERSAL) += xlnx-versal.o
27
obj-$(CONFIG_FSL_IMX25) += fsl-imx25.o imx25_pdk.o
28
obj-$(CONFIG_FSL_IMX31) += fsl-imx31.o kzm.o
29
obj-$(CONFIG_FSL_IMX6) += fsl-imx6.o sabrelite.o
30
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
31
new file mode 100644
32
index XXXXXXX..XXXXXXX
33
--- /dev/null
34
+++ b/include/hw/arm/xlnx-versal.h
35
@@ -XXX,XX +XXX,XX @@
36
+/*
37
+ * Model of the Xilinx Versal
38
+ *
39
+ * Copyright (c) 2018 Xilinx Inc.
40
+ * Written by Edgar E. Iglesias
41
+ *
42
+ * This program is free software; you can redistribute it and/or modify
43
+ * it under the terms of the GNU General Public License version 2 or
44
+ * (at your option) any later version.
45
+ */
46
+
47
+#ifndef XLNX_VERSAL_H
48
+#define XLNX_VERSAL_H
49
+
50
+#include "hw/sysbus.h"
51
+#include "hw/arm/arm.h"
52
+#include "hw/intc/arm_gicv3.h"
53
+
54
+#define TYPE_XLNX_VERSAL "xlnx-versal"
55
+#define XLNX_VERSAL(obj) OBJECT_CHECK(Versal, (obj), TYPE_XLNX_VERSAL)
56
+
57
+#define XLNX_VERSAL_NR_ACPUS 2
58
+#define XLNX_VERSAL_NR_UARTS 2
59
+#define XLNX_VERSAL_NR_GEMS 2
60
+#define XLNX_VERSAL_NR_IRQS 256
61
+
62
+typedef struct Versal {
63
+ /*< private >*/
64
+ SysBusDevice parent_obj;
65
+
66
+ /*< public >*/
67
+ struct {
68
+ struct {
69
+ MemoryRegion mr;
70
+ ARMCPU *cpu[XLNX_VERSAL_NR_ACPUS];
71
+ GICv3State gic;
72
+ } apu;
73
+ } fpd;
74
+
75
+ MemoryRegion mr_ps;
76
+
77
+ struct {
78
+ /* 4 ranges to access DDR. */
79
+ MemoryRegion mr_ddr_ranges[4];
80
+ } noc;
81
+
82
+ struct {
83
+ MemoryRegion mr_ocm;
84
+
85
+ struct {
86
+ SysBusDevice *uart[XLNX_VERSAL_NR_UARTS];
87
+ SysBusDevice *gem[XLNX_VERSAL_NR_GEMS];
88
+ } iou;
89
+ } lpd;
90
+
91
+ struct {
92
+ MemoryRegion *mr_ddr;
93
+ uint32_t psci_conduit;
94
+ } cfg;
95
+} Versal;
96
+
97
+/* Memory-map and IRQ definitions. Copied a subset from
98
+ * auto-generated files. */
99
+
100
+#define VERSAL_GIC_MAINT_IRQ 9
101
+#define VERSAL_TIMER_VIRT_IRQ 11
102
+#define VERSAL_TIMER_S_EL1_IRQ 13
103
+#define VERSAL_TIMER_NS_EL1_IRQ 14
104
+#define VERSAL_TIMER_NS_EL2_IRQ 10
105
+
106
+#define VERSAL_UART0_IRQ_0 18
107
+#define VERSAL_UART1_IRQ_0 19
108
+#define VERSAL_GEM0_IRQ_0 56
109
+#define VERSAL_GEM0_WAKE_IRQ_0 57
110
+#define VERSAL_GEM1_IRQ_0 58
111
+#define VERSAL_GEM1_WAKE_IRQ_0 59
112
+
113
+/* Architecturally eserved IRQs suitable for virtualization. */
114
+#define VERSAL_RSVD_HIGH_IRQ_FIRST 160
115
+#define VERSAL_RSVD_HIGH_IRQ_LAST 255
116
+
117
+#define MM_TOP_RSVD 0xa0000000U
118
+#define MM_TOP_RSVD_SIZE 0x4000000
119
+#define MM_GIC_APU_DIST_MAIN 0xf9000000U
120
+#define MM_GIC_APU_DIST_MAIN_SIZE 0x10000
121
+#define MM_GIC_APU_REDIST_0 0xf9080000U
122
+#define MM_GIC_APU_REDIST_0_SIZE 0x80000
123
+
124
+#define MM_UART0 0xff000000U
125
+#define MM_UART0_SIZE 0x10000
126
+#define MM_UART1 0xff010000U
127
+#define MM_UART1_SIZE 0x10000
128
+
129
+#define MM_GEM0 0xff0c0000U
130
+#define MM_GEM0_SIZE 0x10000
131
+#define MM_GEM1 0xff0d0000U
132
+#define MM_GEM1_SIZE 0x10000
133
+
134
+#define MM_OCM 0xfffc0000U
135
+#define MM_OCM_SIZE 0x40000
136
+
137
+#define MM_TOP_DDR 0x0
138
+#define MM_TOP_DDR_SIZE 0x80000000U
139
+#define MM_TOP_DDR_2 0x800000000ULL
140
+#define MM_TOP_DDR_2_SIZE 0x800000000ULL
141
+#define MM_TOP_DDR_3 0xc000000000ULL
142
+#define MM_TOP_DDR_3_SIZE 0x4000000000ULL
143
+#define MM_TOP_DDR_4 0x10000000000ULL
144
+#define MM_TOP_DDR_4_SIZE 0xb780000000ULL
145
+
146
+#define MM_PSM_START 0xffc80000U
147
+#define MM_PSM_END 0xffcf0000U
148
+
149
+#define MM_CRL 0xff5e0000U
150
+#define MM_CRL_SIZE 0x300000
151
+#define MM_IOU_SCNTR 0xff130000U
152
+#define MM_IOU_SCNTR_SIZE 0x10000
153
+#define MM_IOU_SCNTRS 0xff140000U
154
+#define MM_IOU_SCNTRS_SIZE 0x10000
155
+#define MM_FPD_CRF 0xfd1a0000U
156
+#define MM_FPD_CRF_SIZE 0x140000
157
+#endif
158
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
159
new file mode 100644
160
index XXXXXXX..XXXXXXX
161
--- /dev/null
162
+++ b/hw/arm/xlnx-versal.c
163
@@ -XXX,XX +XXX,XX @@
164
+/*
165
+ * Xilinx Versal SoC model.
166
+ *
167
+ * Copyright (c) 2018 Xilinx Inc.
168
+ * Written by Edgar E. Iglesias
169
+ *
170
+ * This program is free software; you can redistribute it and/or modify
171
+ * it under the terms of the GNU General Public License version 2 or
172
+ * (at your option) any later version.
173
+ */
174
+
175
+#include "qemu/osdep.h"
176
+#include "qapi/error.h"
177
+#include "qemu-common.h"
178
+#include "qemu/log.h"
179
+#include "hw/sysbus.h"
180
+#include "net/net.h"
181
+#include "sysemu/sysemu.h"
182
+#include "sysemu/kvm.h"
183
+#include "hw/arm/arm.h"
184
+#include "kvm_arm.h"
185
+#include "hw/misc/unimp.h"
186
+#include "hw/intc/arm_gicv3_common.h"
187
+#include "hw/arm/xlnx-versal.h"
188
+
189
+#define XLNX_VERSAL_ACPU_TYPE ARM_CPU_TYPE_NAME("cortex-a72")
190
+#define GEM_REVISION 0x40070106
191
+
192
+static void versal_create_apu_cpus(Versal *s)
193
+{
194
+ int i;
195
+
196
+ for (i = 0; i < ARRAY_SIZE(s->fpd.apu.cpu); i++) {
197
+ Object *obj;
198
+ char *name;
199
+
200
+ obj = object_new(XLNX_VERSAL_ACPU_TYPE);
201
+ if (!obj) {
202
+ /* Secondary CPUs start in PSCI powered-down state */
203
+ error_report("Unable to create apu.cpu[%d] of type %s",
204
+ i, XLNX_VERSAL_ACPU_TYPE);
205
+ exit(EXIT_FAILURE);
206
+ }
207
+
208
+ name = g_strdup_printf("apu-cpu[%d]", i);
209
+ object_property_add_child(OBJECT(s), name, obj, &error_fatal);
210
+ g_free(name);
211
+
212
+ object_property_set_int(obj, s->cfg.psci_conduit,
213
+ "psci-conduit", &error_abort);
214
+ if (i) {
215
+ object_property_set_bool(obj, true,
216
+ "start-powered-off", &error_abort);
217
+ }
218
+
219
+ object_property_set_int(obj, ARRAY_SIZE(s->fpd.apu.cpu),
220
+ "core-count", &error_abort);
221
+ object_property_set_link(obj, OBJECT(&s->fpd.apu.mr), "memory",
222
+ &error_abort);
223
+ object_property_set_bool(obj, true, "realized", &error_fatal);
224
+ s->fpd.apu.cpu[i] = ARM_CPU(obj);
225
+ }
226
+}
227
+
228
+static void versal_create_apu_gic(Versal *s, qemu_irq *pic)
229
+{
230
+ static const uint64_t addrs[] = {
231
+ MM_GIC_APU_DIST_MAIN,
232
+ MM_GIC_APU_REDIST_0
233
+ };
234
+ SysBusDevice *gicbusdev;
235
+ DeviceState *gicdev;
236
+ int nr_apu_cpus = ARRAY_SIZE(s->fpd.apu.cpu);
237
+ int i;
238
+
239
+ sysbus_init_child_obj(OBJECT(s), "apu-gic",
240
+ &s->fpd.apu.gic, sizeof(s->fpd.apu.gic),
241
+ gicv3_class_name());
242
+ gicbusdev = SYS_BUS_DEVICE(&s->fpd.apu.gic);
243
+ gicdev = DEVICE(&s->fpd.apu.gic);
244
+ qdev_prop_set_uint32(gicdev, "revision", 3);
245
+ qdev_prop_set_uint32(gicdev, "num-cpu", 2);
246
+ qdev_prop_set_uint32(gicdev, "num-irq", XLNX_VERSAL_NR_IRQS + 32);
247
+ qdev_prop_set_uint32(gicdev, "len-redist-region-count", 1);
248
+ qdev_prop_set_uint32(gicdev, "redist-region-count[0]", 2);
249
+ qdev_prop_set_bit(gicdev, "has-security-extensions", true);
250
+
251
+ object_property_set_bool(OBJECT(&s->fpd.apu.gic), true, "realized",
252
+ &error_fatal);
253
+
254
+ for (i = 0; i < ARRAY_SIZE(addrs); i++) {
255
+ MemoryRegion *mr;
256
+
257
+ mr = sysbus_mmio_get_region(gicbusdev, i);
258
+ memory_region_add_subregion(&s->fpd.apu.mr, addrs[i], mr);
259
+ }
260
+
261
+ for (i = 0; i < nr_apu_cpus; i++) {
262
+ DeviceState *cpudev = DEVICE(s->fpd.apu.cpu[i]);
263
+ int ppibase = XLNX_VERSAL_NR_IRQS + i * GIC_INTERNAL + GIC_NR_SGIS;
264
+ qemu_irq maint_irq;
265
+ int ti;
266
+ /* Mapping from the output timer irq lines from the CPU to the
267
+ * GIC PPI inputs.
268
+ */
269
+ const int timer_irq[] = {
270
+ [GTIMER_PHYS] = VERSAL_TIMER_NS_EL1_IRQ,
271
+ [GTIMER_VIRT] = VERSAL_TIMER_VIRT_IRQ,
272
+ [GTIMER_HYP] = VERSAL_TIMER_NS_EL2_IRQ,
273
+ [GTIMER_SEC] = VERSAL_TIMER_S_EL1_IRQ,
274
+ };
275
+
276
+ for (ti = 0; ti < ARRAY_SIZE(timer_irq); ti++) {
277
+ qdev_connect_gpio_out(cpudev, ti,
278
+ qdev_get_gpio_in(gicdev,
279
+ ppibase + timer_irq[ti]));
280
+ }
281
+ maint_irq = qdev_get_gpio_in(gicdev,
282
+ ppibase + VERSAL_GIC_MAINT_IRQ);
283
+ qdev_connect_gpio_out_named(cpudev, "gicv3-maintenance-interrupt",
284
+ 0, maint_irq);
285
+ sysbus_connect_irq(gicbusdev, i, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));
286
+ sysbus_connect_irq(gicbusdev, i + nr_apu_cpus,
287
+ qdev_get_gpio_in(cpudev, ARM_CPU_FIQ));
288
+ sysbus_connect_irq(gicbusdev, i + 2 * nr_apu_cpus,
289
+ qdev_get_gpio_in(cpudev, ARM_CPU_VIRQ));
290
+ sysbus_connect_irq(gicbusdev, i + 3 * nr_apu_cpus,
291
+ qdev_get_gpio_in(cpudev, ARM_CPU_VFIQ));
292
+ }
293
+
294
+ for (i = 0; i < XLNX_VERSAL_NR_IRQS; i++) {
295
+ pic[i] = qdev_get_gpio_in(gicdev, i);
296
+ }
297
+}
298
+
299
+static void versal_create_uarts(Versal *s, qemu_irq *pic)
300
+{
301
+ int i;
302
+
303
+ for (i = 0; i < ARRAY_SIZE(s->lpd.iou.uart); i++) {
304
+ static const int irqs[] = { VERSAL_UART0_IRQ_0, VERSAL_UART1_IRQ_0};
305
+ static const uint64_t addrs[] = { MM_UART0, MM_UART1 };
306
+ char *name = g_strdup_printf("uart%d", i);
307
+ DeviceState *dev;
308
+ MemoryRegion *mr;
309
+
310
+ dev = qdev_create(NULL, "pl011");
311
+ s->lpd.iou.uart[i] = SYS_BUS_DEVICE(dev);
312
+ qdev_prop_set_chr(dev, "chardev", serial_hd(i));
313
+ object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
314
+ qdev_init_nofail(dev);
315
+
316
+ mr = sysbus_mmio_get_region(s->lpd.iou.uart[i], 0);
317
+ memory_region_add_subregion(&s->mr_ps, addrs[i], mr);
318
+
319
+ sysbus_connect_irq(s->lpd.iou.uart[i], 0, pic[irqs[i]]);
320
+ g_free(name);
321
+ }
322
+}
323
+
324
+static void versal_create_gems(Versal *s, qemu_irq *pic)
325
+{
326
+ int i;
327
+
328
+ for (i = 0; i < ARRAY_SIZE(s->lpd.iou.gem); i++) {
329
+ static const int irqs[] = { VERSAL_GEM0_IRQ_0, VERSAL_GEM1_IRQ_0};
330
+ static const uint64_t addrs[] = { MM_GEM0, MM_GEM1 };
331
+ char *name = g_strdup_printf("gem%d", i);
332
+ NICInfo *nd = &nd_table[i];
333
+ DeviceState *dev;
334
+ MemoryRegion *mr;
335
+
336
+ dev = qdev_create(NULL, "cadence_gem");
337
+ s->lpd.iou.gem[i] = SYS_BUS_DEVICE(dev);
338
+ object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
339
+ if (nd->used) {
340
+ qemu_check_nic_model(nd, "cadence_gem");
341
+ qdev_set_nic_properties(dev, nd);
342
+ }
343
+ object_property_set_int(OBJECT(s->lpd.iou.gem[i]),
344
+ 2, "num-priority-queues",
345
+ &error_abort);
346
+ object_property_set_link(OBJECT(s->lpd.iou.gem[i]),
347
+ OBJECT(&s->mr_ps), "dma",
348
+ &error_abort);
349
+ qdev_init_nofail(dev);
350
+
351
+ mr = sysbus_mmio_get_region(s->lpd.iou.gem[i], 0);
352
+ memory_region_add_subregion(&s->mr_ps, addrs[i], mr);
353
+
354
+ sysbus_connect_irq(s->lpd.iou.gem[i], 0, pic[irqs[i]]);
355
+ g_free(name);
356
+ }
357
+}
358
+
359
+/* This takes the board allocated linear DDR memory and creates aliases
360
+ * for each split DDR range/aperture on the Versal address map.
361
+ */
362
+static void versal_map_ddr(Versal *s)
363
+{
364
+ uint64_t size = memory_region_size(s->cfg.mr_ddr);
365
+ /* Describes the various split DDR access regions. */
366
+ static const struct {
367
+ uint64_t base;
368
+ uint64_t size;
369
+ } addr_ranges[] = {
370
+ { MM_TOP_DDR, MM_TOP_DDR_SIZE },
371
+ { MM_TOP_DDR_2, MM_TOP_DDR_2_SIZE },
372
+ { MM_TOP_DDR_3, MM_TOP_DDR_3_SIZE },
373
+ { MM_TOP_DDR_4, MM_TOP_DDR_4_SIZE }
374
+ };
375
+ uint64_t offset = 0;
376
+ int i;
377
+
378
+ assert(ARRAY_SIZE(addr_ranges) == ARRAY_SIZE(s->noc.mr_ddr_ranges));
379
+ for (i = 0; i < ARRAY_SIZE(addr_ranges) && size; i++) {
380
+ char *name;
381
+ uint64_t mapsize;
382
+
383
+ mapsize = size < addr_ranges[i].size ? size : addr_ranges[i].size;
384
+ name = g_strdup_printf("noc-ddr-range%d", i);
385
+ /* Create the MR alias. */
386
+ memory_region_init_alias(&s->noc.mr_ddr_ranges[i], OBJECT(s),
387
+ name, s->cfg.mr_ddr,
388
+ offset, mapsize);
389
+
390
+ /* Map it onto the NoC MR. */
391
+ memory_region_add_subregion(&s->mr_ps, addr_ranges[i].base,
392
+ &s->noc.mr_ddr_ranges[i]);
393
+ offset += mapsize;
394
+ size -= mapsize;
395
+ g_free(name);
396
+ }
397
+}
398
+
399
+static void versal_unimp_area(Versal *s, const char *name,
400
+ MemoryRegion *mr,
401
+ hwaddr base, hwaddr size)
402
+{
403
+ DeviceState *dev = qdev_create(NULL, TYPE_UNIMPLEMENTED_DEVICE);
404
+ MemoryRegion *mr_dev;
405
+
406
+ qdev_prop_set_string(dev, "name", name);
407
+ qdev_prop_set_uint64(dev, "size", size);
408
+ object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
409
+ qdev_init_nofail(dev);
410
+
411
+ mr_dev = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
412
+ memory_region_add_subregion(mr, base, mr_dev);
413
+}
414
+
415
+static void versal_unimp(Versal *s)
416
+{
417
+ versal_unimp_area(s, "psm", &s->mr_ps,
418
+ MM_PSM_START, MM_PSM_END - MM_PSM_START);
419
+ versal_unimp_area(s, "crl", &s->mr_ps,
420
+ MM_CRL, MM_CRL_SIZE);
421
+ versal_unimp_area(s, "crf", &s->mr_ps,
422
+ MM_FPD_CRF, MM_FPD_CRF_SIZE);
423
+ versal_unimp_area(s, "iou-scntr", &s->mr_ps,
424
+ MM_IOU_SCNTR, MM_IOU_SCNTR_SIZE);
425
+ versal_unimp_area(s, "iou-scntr-seucre", &s->mr_ps,
426
+ MM_IOU_SCNTRS, MM_IOU_SCNTRS_SIZE);
427
+}
428
+
429
+static void versal_realize(DeviceState *dev, Error **errp)
430
+{
431
+ Versal *s = XLNX_VERSAL(dev);
432
+ qemu_irq pic[XLNX_VERSAL_NR_IRQS];
433
+
434
+ versal_create_apu_cpus(s);
435
+ versal_create_apu_gic(s, pic);
436
+ versal_create_uarts(s, pic);
437
+ versal_create_gems(s, pic);
438
+ versal_map_ddr(s);
439
+ versal_unimp(s);
440
+
441
+ /* Create the On Chip Memory (OCM). */
442
+ memory_region_init_ram(&s->lpd.mr_ocm, OBJECT(s), "ocm",
443
+ MM_OCM_SIZE, &error_fatal);
444
+
445
+ memory_region_add_subregion_overlap(&s->mr_ps, MM_OCM, &s->lpd.mr_ocm, 0);
446
+ memory_region_add_subregion_overlap(&s->fpd.apu.mr, 0, &s->mr_ps, 0);
447
+}
448
+
449
+static void versal_init(Object *obj)
450
+{
451
+ Versal *s = XLNX_VERSAL(obj);
452
+
453
+ memory_region_init(&s->fpd.apu.mr, obj, "mr-apu", UINT64_MAX);
454
+ memory_region_init(&s->mr_ps, obj, "mr-ps-switch", UINT64_MAX);
455
+}
456
+
457
+static Property versal_properties[] = {
458
+ DEFINE_PROP_LINK("ddr", Versal, cfg.mr_ddr, TYPE_MEMORY_REGION,
459
+ MemoryRegion *),
460
+ DEFINE_PROP_UINT32("psci-conduit", Versal, cfg.psci_conduit, 0),
461
+ DEFINE_PROP_END_OF_LIST()
462
+};
463
+
464
+static void versal_class_init(ObjectClass *klass, void *data)
465
+{
466
+ DeviceClass *dc = DEVICE_CLASS(klass);
467
+
468
+ dc->realize = versal_realize;
469
+ dc->props = versal_properties;
470
+ /* No VMSD since we haven't got any top-level SoC state to save. */
471
+}
472
+
473
+static const TypeInfo versal_info = {
474
+ .name = TYPE_XLNX_VERSAL,
475
+ .parent = TYPE_SYS_BUS_DEVICE,
476
+ .instance_size = sizeof(Versal),
477
+ .instance_init = versal_init,
478
+ .class_init = versal_class_init,
479
+};
480
+
481
+static void versal_register_types(void)
482
+{
483
+ type_register_static(&versal_info);
484
+}
485
+
486
+type_init(versal_register_types);
487
diff --git a/default-configs/aarch64-softmmu.mak b/default-configs/aarch64-softmmu.mak
488
index XXXXXXX..XXXXXXX 100644
489
--- a/default-configs/aarch64-softmmu.mak
490
+++ b/default-configs/aarch64-softmmu.mak
491
@@ -XXX,XX +XXX,XX @@ CONFIG_DDC=y
492
CONFIG_DPCD=y
493
CONFIG_XLNX_ZYNQMP=y
494
CONFIG_XLNX_ZYNQMP_ARM=y
495
+CONFIG_XLNX_VERSAL=y
496
CONFIG_ARM_SMMUV3=y
497
--
498
2.19.1
499
500
diff view generated by jsdifflib
New patch
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
2
3
Add a virtual Xilinx Versal board.
4
5
This board is based on the Xilinx Versal SoC. The exact
6
details of what peripherals are attached to this board
7
will remain in control of QEMU. QEMU will generate an
8
FDT on the fly for Linux and other software to auto-discover
9
peripherals.
10
11
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
12
Message-id: 20181102131913.1535-3-edgar.iglesias@xilinx.com
13
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
---
16
hw/arm/Makefile.objs | 2 +-
17
hw/arm/xlnx-versal-virt.c | 494 ++++++++++++++++++++++++++++++++++++++
18
2 files changed, 495 insertions(+), 1 deletion(-)
19
create mode 100644 hw/arm/xlnx-versal-virt.c
20
21
diff --git a/hw/arm/Makefile.objs b/hw/arm/Makefile.objs
22
index XXXXXXX..XXXXXXX 100644
23
--- a/hw/arm/Makefile.objs
24
+++ b/hw/arm/Makefile.objs
25
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_ALLWINNER_A10) += allwinner-a10.o cubieboard.o
26
obj-$(CONFIG_RASPI) += bcm2835_peripherals.o bcm2836.o raspi.o
27
obj-$(CONFIG_STM32F205_SOC) += stm32f205_soc.o
28
obj-$(CONFIG_XLNX_ZYNQMP_ARM) += xlnx-zynqmp.o xlnx-zcu102.o
29
-obj-$(CONFIG_XLNX_VERSAL) += xlnx-versal.o
30
+obj-$(CONFIG_XLNX_VERSAL) += xlnx-versal.o xlnx-versal-virt.o
31
obj-$(CONFIG_FSL_IMX25) += fsl-imx25.o imx25_pdk.o
32
obj-$(CONFIG_FSL_IMX31) += fsl-imx31.o kzm.o
33
obj-$(CONFIG_FSL_IMX6) += fsl-imx6.o sabrelite.o
34
diff --git a/hw/arm/xlnx-versal-virt.c b/hw/arm/xlnx-versal-virt.c
35
new file mode 100644
36
index XXXXXXX..XXXXXXX
37
--- /dev/null
38
+++ b/hw/arm/xlnx-versal-virt.c
39
@@ -XXX,XX +XXX,XX @@
40
+/*
41
+ * Xilinx Versal Virtual board.
42
+ *
43
+ * Copyright (c) 2018 Xilinx Inc.
44
+ * Written by Edgar E. Iglesias
45
+ *
46
+ * This program is free software; you can redistribute it and/or modify
47
+ * it under the terms of the GNU General Public License version 2 or
48
+ * (at your option) any later version.
49
+ */
50
+
51
+#include "qemu/osdep.h"
52
+#include "qemu/log.h"
53
+#include "qemu/error-report.h"
54
+#include "qapi/error.h"
55
+#include "sysemu/device_tree.h"
56
+#include "exec/address-spaces.h"
57
+#include "hw/boards.h"
58
+#include "hw/sysbus.h"
59
+#include "hw/arm/sysbus-fdt.h"
60
+#include "hw/arm/fdt.h"
61
+#include "cpu.h"
62
+#include "hw/arm/xlnx-versal.h"
63
+
64
+#define TYPE_XLNX_VERSAL_VIRT_MACHINE MACHINE_TYPE_NAME("xlnx-versal-virt")
65
+#define XLNX_VERSAL_VIRT_MACHINE(obj) \
66
+ OBJECT_CHECK(VersalVirt, (obj), TYPE_XLNX_VERSAL_VIRT_MACHINE)
67
+
68
+typedef struct VersalVirt {
69
+ MachineState parent_obj;
70
+
71
+ Versal soc;
72
+ MemoryRegion mr_ddr;
73
+
74
+ void *fdt;
75
+ int fdt_size;
76
+ struct {
77
+ uint32_t gic;
78
+ uint32_t ethernet_phy[2];
79
+ uint32_t clk_125Mhz;
80
+ uint32_t clk_25Mhz;
81
+ } phandle;
82
+ struct arm_boot_info binfo;
83
+
84
+ struct {
85
+ bool secure;
86
+ } cfg;
87
+} VersalVirt;
88
+
89
+static void fdt_create(VersalVirt *s)
90
+{
91
+ MachineClass *mc = MACHINE_GET_CLASS(s);
92
+ int i;
93
+
94
+ s->fdt = create_device_tree(&s->fdt_size);
95
+ if (!s->fdt) {
96
+ error_report("create_device_tree() failed");
97
+ exit(1);
98
+ }
99
+
100
+ /* Allocate all phandles. */
101
+ s->phandle.gic = qemu_fdt_alloc_phandle(s->fdt);
102
+ for (i = 0; i < ARRAY_SIZE(s->phandle.ethernet_phy); i++) {
103
+ s->phandle.ethernet_phy[i] = qemu_fdt_alloc_phandle(s->fdt);
104
+ }
105
+ s->phandle.clk_25Mhz = qemu_fdt_alloc_phandle(s->fdt);
106
+ s->phandle.clk_125Mhz = qemu_fdt_alloc_phandle(s->fdt);
107
+
108
+ /* Create /chosen node for load_dtb. */
109
+ qemu_fdt_add_subnode(s->fdt, "/chosen");
110
+
111
+ /* Header */
112
+ qemu_fdt_setprop_cell(s->fdt, "/", "interrupt-parent", s->phandle.gic);
113
+ qemu_fdt_setprop_cell(s->fdt, "/", "#size-cells", 0x2);
114
+ qemu_fdt_setprop_cell(s->fdt, "/", "#address-cells", 0x2);
115
+ qemu_fdt_setprop_string(s->fdt, "/", "model", mc->desc);
116
+ qemu_fdt_setprop_string(s->fdt, "/", "compatible", "xlnx-versal-virt");
117
+}
118
+
119
+static void fdt_add_clk_node(VersalVirt *s, const char *name,
120
+ unsigned int freq_hz, uint32_t phandle)
121
+{
122
+ qemu_fdt_add_subnode(s->fdt, name);
123
+ qemu_fdt_setprop_cell(s->fdt, name, "phandle", phandle);
124
+ qemu_fdt_setprop_cell(s->fdt, name, "clock-frequency", freq_hz);
125
+ qemu_fdt_setprop_cell(s->fdt, name, "#clock-cells", 0x0);
126
+ qemu_fdt_setprop_string(s->fdt, name, "compatible", "fixed-clock");
127
+ qemu_fdt_setprop(s->fdt, name, "u-boot,dm-pre-reloc", NULL, 0);
128
+}
129
+
130
+static void fdt_add_cpu_nodes(VersalVirt *s, uint32_t psci_conduit)
131
+{
132
+ int i;
133
+
134
+ qemu_fdt_add_subnode(s->fdt, "/cpus");
135
+ qemu_fdt_setprop_cell(s->fdt, "/cpus", "#size-cells", 0x0);
136
+ qemu_fdt_setprop_cell(s->fdt, "/cpus", "#address-cells", 1);
137
+
138
+ for (i = XLNX_VERSAL_NR_ACPUS - 1; i >= 0; i--) {
139
+ char *name = g_strdup_printf("/cpus/cpu@%d", i);
140
+ ARMCPU *armcpu = ARM_CPU(qemu_get_cpu(i));
141
+
142
+ qemu_fdt_add_subnode(s->fdt, name);
143
+ qemu_fdt_setprop_cell(s->fdt, name, "reg", armcpu->mp_affinity);
144
+ if (psci_conduit != QEMU_PSCI_CONDUIT_DISABLED) {
145
+ qemu_fdt_setprop_string(s->fdt, name, "enable-method", "psci");
146
+ }
147
+ qemu_fdt_setprop_string(s->fdt, name, "device_type", "cpu");
148
+ qemu_fdt_setprop_string(s->fdt, name, "compatible",
149
+ armcpu->dtb_compatible);
150
+ g_free(name);
151
+ }
152
+}
153
+
154
+static void fdt_add_gic_nodes(VersalVirt *s)
155
+{
156
+ char *nodename;
157
+
158
+ nodename = g_strdup_printf("/gic@%x", MM_GIC_APU_DIST_MAIN);
159
+ qemu_fdt_add_subnode(s->fdt, nodename);
160
+ qemu_fdt_setprop_cell(s->fdt, nodename, "phandle", s->phandle.gic);
161
+ qemu_fdt_setprop_cells(s->fdt, nodename, "interrupts",
162
+ GIC_FDT_IRQ_TYPE_PPI, VERSAL_GIC_MAINT_IRQ,
163
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI);
164
+ qemu_fdt_setprop(s->fdt, nodename, "interrupt-controller", NULL, 0);
165
+ qemu_fdt_setprop_sized_cells(s->fdt, nodename, "reg",
166
+ 2, MM_GIC_APU_DIST_MAIN,
167
+ 2, MM_GIC_APU_DIST_MAIN_SIZE,
168
+ 2, MM_GIC_APU_REDIST_0,
169
+ 2, MM_GIC_APU_REDIST_0_SIZE);
170
+ qemu_fdt_setprop_cell(s->fdt, nodename, "#interrupt-cells", 3);
171
+ qemu_fdt_setprop_string(s->fdt, nodename, "compatible", "arm,gic-v3");
172
+}
173
+
174
+static void fdt_add_timer_nodes(VersalVirt *s)
175
+{
176
+ const char compat[] = "arm,armv8-timer";
177
+ uint32_t irqflags = GIC_FDT_IRQ_FLAGS_LEVEL_HI;
178
+
179
+ qemu_fdt_add_subnode(s->fdt, "/timer");
180
+ qemu_fdt_setprop_cells(s->fdt, "/timer", "interrupts",
181
+ GIC_FDT_IRQ_TYPE_PPI, VERSAL_TIMER_S_EL1_IRQ, irqflags,
182
+ GIC_FDT_IRQ_TYPE_PPI, VERSAL_TIMER_NS_EL1_IRQ, irqflags,
183
+ GIC_FDT_IRQ_TYPE_PPI, VERSAL_TIMER_VIRT_IRQ, irqflags,
184
+ GIC_FDT_IRQ_TYPE_PPI, VERSAL_TIMER_NS_EL2_IRQ, irqflags);
185
+ qemu_fdt_setprop(s->fdt, "/timer", "compatible",
186
+ compat, sizeof(compat));
187
+}
188
+
189
+static void fdt_add_uart_nodes(VersalVirt *s)
190
+{
191
+ uint64_t addrs[] = { MM_UART1, MM_UART0 };
192
+ unsigned int irqs[] = { VERSAL_UART1_IRQ_0, VERSAL_UART0_IRQ_0 };
193
+ const char compat[] = "arm,pl011\0arm,sbsa-uart";
194
+ const char clocknames[] = "uartclk\0apb_pclk";
195
+ int i;
196
+
197
+ for (i = 0; i < ARRAY_SIZE(addrs); i++) {
198
+ char *name = g_strdup_printf("/uart@%" PRIx64, addrs[i]);
199
+ qemu_fdt_add_subnode(s->fdt, name);
200
+ qemu_fdt_setprop_cell(s->fdt, name, "current-speed", 115200);
201
+ qemu_fdt_setprop_cells(s->fdt, name, "clocks",
202
+ s->phandle.clk_125Mhz, s->phandle.clk_125Mhz);
203
+ qemu_fdt_setprop(s->fdt, name, "clock-names",
204
+ clocknames, sizeof(clocknames));
205
+
206
+ qemu_fdt_setprop_cells(s->fdt, name, "interrupts",
207
+ GIC_FDT_IRQ_TYPE_SPI, irqs[i],
208
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI);
209
+ qemu_fdt_setprop_sized_cells(s->fdt, name, "reg",
210
+ 2, addrs[i], 2, 0x1000);
211
+ qemu_fdt_setprop(s->fdt, name, "compatible",
212
+ compat, sizeof(compat));
213
+ qemu_fdt_setprop(s->fdt, name, "u-boot,dm-pre-reloc", NULL, 0);
214
+
215
+ if (addrs[i] == MM_UART0) {
216
+ /* Select UART0. */
217
+ qemu_fdt_setprop_string(s->fdt, "/chosen", "stdout-path", name);
218
+ }
219
+ g_free(name);
220
+ }
221
+}
222
+
223
+static void fdt_add_fixed_link_nodes(VersalVirt *s, char *gemname,
224
+ uint32_t phandle)
225
+{
226
+ char *name = g_strdup_printf("%s/fixed-link", gemname);
227
+
228
+ qemu_fdt_add_subnode(s->fdt, name);
229
+ qemu_fdt_setprop_cell(s->fdt, name, "phandle", phandle);
230
+ qemu_fdt_setprop(s->fdt, name, "full-duplex", NULL, 0);
231
+ qemu_fdt_setprop_cell(s->fdt, name, "speed", 1000);
232
+ g_free(name);
233
+}
234
+
235
+static void fdt_add_gem_nodes(VersalVirt *s)
236
+{
237
+ uint64_t addrs[] = { MM_GEM1, MM_GEM0 };
238
+ unsigned int irqs[] = { VERSAL_GEM1_IRQ_0, VERSAL_GEM0_IRQ_0 };
239
+ const char clocknames[] = "pclk\0hclk\0tx_clk\0rx_clk";
240
+ const char compat_gem[] = "cdns,zynqmp-gem\0cdns,gem";
241
+ int i;
242
+
243
+ for (i = 0; i < ARRAY_SIZE(addrs); i++) {
244
+ char *name = g_strdup_printf("/ethernet@%" PRIx64, addrs[i]);
245
+ qemu_fdt_add_subnode(s->fdt, name);
246
+
247
+ fdt_add_fixed_link_nodes(s, name, s->phandle.ethernet_phy[i]);
248
+ qemu_fdt_setprop_string(s->fdt, name, "phy-mode", "rgmii-id");
249
+ qemu_fdt_setprop_cell(s->fdt, name, "phy-handle",
250
+ s->phandle.ethernet_phy[i]);
251
+ qemu_fdt_setprop_cells(s->fdt, name, "clocks",
252
+ s->phandle.clk_25Mhz, s->phandle.clk_25Mhz,
253
+ s->phandle.clk_25Mhz, s->phandle.clk_25Mhz);
254
+ qemu_fdt_setprop(s->fdt, name, "clock-names",
255
+ clocknames, sizeof(clocknames));
256
+ qemu_fdt_setprop_cells(s->fdt, name, "interrupts",
257
+ GIC_FDT_IRQ_TYPE_SPI, irqs[i],
258
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI,
259
+ GIC_FDT_IRQ_TYPE_SPI, irqs[i],
260
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI);
261
+ qemu_fdt_setprop_sized_cells(s->fdt, name, "reg",
262
+ 2, addrs[i], 2, 0x1000);
263
+ qemu_fdt_setprop(s->fdt, name, "compatible",
264
+ compat_gem, sizeof(compat_gem));
265
+ qemu_fdt_setprop_cell(s->fdt, name, "#address-cells", 1);
266
+ qemu_fdt_setprop_cell(s->fdt, name, "#size-cells", 0);
267
+ g_free(name);
268
+ }
269
+}
270
+
271
+static void fdt_nop_memory_nodes(void *fdt, Error **errp)
272
+{
273
+ Error *err = NULL;
274
+ char **node_path;
275
+ int n = 0;
276
+
277
+ node_path = qemu_fdt_node_unit_path(fdt, "memory", &err);
278
+ if (err) {
279
+ error_propagate(errp, err);
280
+ return;
281
+ }
282
+ while (node_path[n]) {
283
+ if (g_str_has_prefix(node_path[n], "/memory")) {
284
+ qemu_fdt_nop_node(fdt, node_path[n]);
285
+ }
286
+ n++;
287
+ }
288
+ g_strfreev(node_path);
289
+}
290
+
291
+static void fdt_add_memory_nodes(VersalVirt *s, void *fdt, uint64_t ram_size)
292
+{
293
+ /* Describes the various split DDR access regions. */
294
+ static const struct {
295
+ uint64_t base;
296
+ uint64_t size;
297
+ } addr_ranges[] = {
298
+ { MM_TOP_DDR, MM_TOP_DDR_SIZE },
299
+ { MM_TOP_DDR_2, MM_TOP_DDR_2_SIZE },
300
+ { MM_TOP_DDR_3, MM_TOP_DDR_3_SIZE },
301
+ { MM_TOP_DDR_4, MM_TOP_DDR_4_SIZE }
302
+ };
303
+ uint64_t mem_reg_prop[8] = {0};
304
+ uint64_t size = ram_size;
305
+ Error *err = NULL;
306
+ char *name;
307
+ int i;
308
+
309
+ fdt_nop_memory_nodes(fdt, &err);
310
+ if (err) {
311
+ error_report_err(err);
312
+ return;
313
+ }
314
+
315
+ name = g_strdup_printf("/memory@%x", MM_TOP_DDR);
316
+ for (i = 0; i < ARRAY_SIZE(addr_ranges) && size; i++) {
317
+ uint64_t mapsize;
318
+
319
+ mapsize = size < addr_ranges[i].size ? size : addr_ranges[i].size;
320
+
321
+ mem_reg_prop[i * 2] = addr_ranges[i].base;
322
+ mem_reg_prop[i * 2 + 1] = mapsize;
323
+ size -= mapsize;
324
+ }
325
+ qemu_fdt_add_subnode(fdt, name);
326
+ qemu_fdt_setprop_string(fdt, name, "device_type", "memory");
327
+
328
+ switch (i) {
329
+ case 1:
330
+ qemu_fdt_setprop_sized_cells(fdt, name, "reg",
331
+ 2, mem_reg_prop[0],
332
+ 2, mem_reg_prop[1]);
333
+ break;
334
+ case 2:
335
+ qemu_fdt_setprop_sized_cells(fdt, name, "reg",
336
+ 2, mem_reg_prop[0],
337
+ 2, mem_reg_prop[1],
338
+ 2, mem_reg_prop[2],
339
+ 2, mem_reg_prop[3]);
340
+ break;
341
+ case 3:
342
+ qemu_fdt_setprop_sized_cells(fdt, name, "reg",
343
+ 2, mem_reg_prop[0],
344
+ 2, mem_reg_prop[1],
345
+ 2, mem_reg_prop[2],
346
+ 2, mem_reg_prop[3],
347
+ 2, mem_reg_prop[4],
348
+ 2, mem_reg_prop[5]);
349
+ break;
350
+ case 4:
351
+ qemu_fdt_setprop_sized_cells(fdt, name, "reg",
352
+ 2, mem_reg_prop[0],
353
+ 2, mem_reg_prop[1],
354
+ 2, mem_reg_prop[2],
355
+ 2, mem_reg_prop[3],
356
+ 2, mem_reg_prop[4],
357
+ 2, mem_reg_prop[5],
358
+ 2, mem_reg_prop[6],
359
+ 2, mem_reg_prop[7]);
360
+ break;
361
+ default:
362
+ g_assert_not_reached();
363
+ }
364
+ g_free(name);
365
+}
366
+
367
+static void versal_virt_modify_dtb(const struct arm_boot_info *binfo,
368
+ void *fdt)
369
+{
370
+ VersalVirt *s = container_of(binfo, VersalVirt, binfo);
371
+
372
+ fdt_add_memory_nodes(s, fdt, binfo->ram_size);
373
+}
374
+
375
+static void *versal_virt_get_dtb(const struct arm_boot_info *binfo,
376
+ int *fdt_size)
377
+{
378
+ const VersalVirt *board = container_of(binfo, VersalVirt, binfo);
379
+
380
+ *fdt_size = board->fdt_size;
381
+ return board->fdt;
382
+}
383
+
384
+#define NUM_VIRTIO_TRANSPORT 32
385
+static void create_virtio_regions(VersalVirt *s)
386
+{
387
+ int virtio_mmio_size = 0x200;
388
+ int i;
389
+
390
+ for (i = 0; i < NUM_VIRTIO_TRANSPORT; i++) {
391
+ char *name = g_strdup_printf("virtio%d", i);;
392
+ hwaddr base = MM_TOP_RSVD + i * virtio_mmio_size;
393
+ int irq = VERSAL_RSVD_HIGH_IRQ_FIRST + i;
394
+ MemoryRegion *mr;
395
+ DeviceState *dev;
396
+ qemu_irq pic_irq;
397
+
398
+ pic_irq = qdev_get_gpio_in(DEVICE(&s->soc.fpd.apu.gic), irq);
399
+ dev = qdev_create(NULL, "virtio-mmio");
400
+ object_property_add_child(OBJECT(&s->soc), name, OBJECT(dev),
401
+ &error_fatal);
402
+ qdev_init_nofail(dev);
403
+ sysbus_connect_irq(SYS_BUS_DEVICE(dev), 0, pic_irq);
404
+ mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
405
+ memory_region_add_subregion(&s->soc.mr_ps, base, mr);
406
+ sysbus_create_simple("virtio-mmio", base, pic_irq);
407
+ }
408
+
409
+ for (i = 0; i < NUM_VIRTIO_TRANSPORT; i++) {
410
+ hwaddr base = MM_TOP_RSVD + i * virtio_mmio_size;
411
+ int irq = VERSAL_RSVD_HIGH_IRQ_FIRST + i;
412
+ char *name = g_strdup_printf("/virtio_mmio@%" PRIx64, base);
413
+
414
+ qemu_fdt_add_subnode(s->fdt, name);
415
+ qemu_fdt_setprop(s->fdt, name, "dma-coherent", NULL, 0);
416
+ qemu_fdt_setprop_cells(s->fdt, name, "interrupts",
417
+ GIC_FDT_IRQ_TYPE_SPI, irq,
418
+ GIC_FDT_IRQ_FLAGS_EDGE_LO_HI);
419
+ qemu_fdt_setprop_sized_cells(s->fdt, name, "reg",
420
+ 2, base, 2, virtio_mmio_size);
421
+ qemu_fdt_setprop_string(s->fdt, name, "compatible", "virtio,mmio");
422
+ g_free(name);
423
+ }
424
+}
425
+
426
+static void versal_virt_init(MachineState *machine)
427
+{
428
+ VersalVirt *s = XLNX_VERSAL_VIRT_MACHINE(machine);
429
+ int psci_conduit = QEMU_PSCI_CONDUIT_DISABLED;
430
+
431
+ /*
432
+ * If the user provides an Operating System to be loaded, we expect them
433
+ * to use the -kernel command line option.
434
+ *
435
+ * Users can load firmware or boot-loaders with the -device loader options.
436
+ *
437
+ * When loading an OS, we generate a dtb and let arm_load_kernel() select
438
+ * where it gets loaded. This dtb will be passed to the kernel in x0.
439
+ *
440
+ * If there's no -kernel option, we generate a DTB and place it at 0x1000
441
+ * for the bootloaders or firmware to pick up.
442
+ *
443
+ * If users want to provide their own DTB, they can use the -dtb option.
444
+ * These dtb's will have their memory nodes modified to match QEMU's
445
+ * selected ram_size option before they get passed to the kernel or fw.
446
+ *
447
+ * When loading an OS, we turn on QEMU's PSCI implementation with SMC
448
+ * as the PSCI conduit. When there's no -kernel, we assume the user
449
+ * provides EL3 firmware to handle PSCI.
450
+ */
451
+ if (machine->kernel_filename) {
452
+ psci_conduit = QEMU_PSCI_CONDUIT_SMC;
453
+ }
454
+
455
+ memory_region_allocate_system_memory(&s->mr_ddr, NULL, "ddr",
456
+ machine->ram_size);
457
+
458
+ sysbus_init_child_obj(OBJECT(machine), "xlnx-ve", &s->soc,
459
+ sizeof(s->soc), TYPE_XLNX_VERSAL);
460
+ object_property_set_link(OBJECT(&s->soc), OBJECT(&s->mr_ddr),
461
+ "ddr", &error_abort);
462
+ object_property_set_int(OBJECT(&s->soc), psci_conduit,
463
+ "psci-conduit", &error_abort);
464
+ object_property_set_bool(OBJECT(&s->soc), true, "realized", &error_fatal);
465
+
466
+ fdt_create(s);
467
+ create_virtio_regions(s);
468
+ fdt_add_gem_nodes(s);
469
+ fdt_add_uart_nodes(s);
470
+ fdt_add_gic_nodes(s);
471
+ fdt_add_timer_nodes(s);
472
+ fdt_add_cpu_nodes(s, psci_conduit);
473
+ fdt_add_clk_node(s, "/clk125", 125000000, s->phandle.clk_125Mhz);
474
+ fdt_add_clk_node(s, "/clk25", 25000000, s->phandle.clk_25Mhz);
475
+
476
+ /* Make the APU cpu address space visible to virtio and other
477
+ * modules unaware of muliple address-spaces. */
478
+ memory_region_add_subregion_overlap(get_system_memory(),
479
+ 0, &s->soc.fpd.apu.mr, 0);
480
+
481
+ s->binfo.ram_size = machine->ram_size;
482
+ s->binfo.kernel_filename = machine->kernel_filename;
483
+ s->binfo.kernel_cmdline = machine->kernel_cmdline;
484
+ s->binfo.initrd_filename = machine->initrd_filename;
485
+ s->binfo.loader_start = 0x0;
486
+ s->binfo.get_dtb = versal_virt_get_dtb;
487
+ s->binfo.modify_dtb = versal_virt_modify_dtb;
488
+ if (machine->kernel_filename) {
489
+ arm_load_kernel(s->soc.fpd.apu.cpu[0], &s->binfo);
490
+ } else {
491
+ AddressSpace *as = arm_boot_address_space(s->soc.fpd.apu.cpu[0],
492
+ &s->binfo);
493
+ /* Some boot-loaders (e.g u-boot) don't like blobs at address 0 (NULL).
494
+ * Offset things by 4K. */
495
+ s->binfo.loader_start = 0x1000;
496
+ s->binfo.dtb_limit = 0x1000000;
497
+ if (arm_load_dtb(s->binfo.loader_start,
498
+ &s->binfo, s->binfo.dtb_limit, as) < 0) {
499
+ exit(EXIT_FAILURE);
500
+ }
501
+ }
502
+}
503
+
504
+static void versal_virt_machine_instance_init(Object *obj)
505
+{
506
+}
507
+
508
+static void versal_virt_machine_class_init(ObjectClass *oc, void *data)
509
+{
510
+ MachineClass *mc = MACHINE_CLASS(oc);
511
+
512
+ mc->desc = "Xilinx Versal Virtual development board";
513
+ mc->init = versal_virt_init;
514
+ mc->max_cpus = XLNX_VERSAL_NR_ACPUS;
515
+ mc->default_cpus = XLNX_VERSAL_NR_ACPUS;
516
+ mc->no_cdrom = true;
517
+}
518
+
519
+static const TypeInfo versal_virt_machine_init_typeinfo = {
520
+ .name = TYPE_XLNX_VERSAL_VIRT_MACHINE,
521
+ .parent = TYPE_MACHINE,
522
+ .class_init = versal_virt_machine_class_init,
523
+ .instance_init = versal_virt_machine_instance_init,
524
+ .instance_size = sizeof(VersalVirt),
525
+};
526
+
527
+static void versal_virt_machine_init_register_types(void)
528
+{
529
+ type_register_static(&versal_virt_machine_init_typeinfo);
530
+}
531
+
532
+type_init(versal_virt_machine_init_register_types)
533
+
534
--
535
2.19.1
536
537
diff view generated by jsdifflib