1
target-arm queue: I have a lot more still in my to-review
1
More accumulated patches from during the freeze...
2
queue, but my rule of thumb is when I get to 50 patches or
3
so to send out what I have.
4
2
5
thanks
3
The following changes since commit c83fcfaf8a54d0d034bd0edf7bbb3b0d16669be9:
6
-- PMM
7
4
8
The following changes since commit 9a7beaad3dbba982f7a461d676b55a5c3851d312:
5
Merge remote-tracking branch 'remotes/armbru/tags/pull-qapi-2021-08-26' into staging (2021-08-26 13:42:34 +0100)
9
10
Merge remote-tracking branch 'remotes/alistair/tags/pull-riscv-to-apply-20210304' into staging (2021-03-05 10:47:46 +0000)
11
6
12
are available in the Git repository at:
7
are available in the Git repository at:
13
8
14
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20210305
9
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20210826
15
10
16
for you to fetch changes up to 2c669ff88ec6733420a000103a2b8b9e93df4945:
11
for you to fetch changes up to d2e6f370138a7f32bc28b20dcd55374b7a638f39:
17
12
18
hw/arm/mps2: Update old infocenter.arm.com URLs (2021-03-05 15:17:38 +0000)
13
hw/arm/xlnx-zynqmp: Add unimplemented APU mmio (2021-08-26 17:02:01 +0100)
19
14
20
----------------------------------------------------------------
15
----------------------------------------------------------------
21
* sbsa-ref: remove cortex-a53 from list of supported cpus
16
target-arm queue:
22
* sbsa-ref: add 'max' to list of allowed cpus
17
* hw/dma/xlnx-zdma, xlnx_csu_dma: Require 'dma' link property to be set
23
* target/arm: Add support for FEAT_SSBS, Speculative Store Bypass Safe
18
* hw/arm/Kconfig: no need to enable ACPI_MEMORY_HOTPLUG/ACPI_NVDIMM explicitly
24
* npcm7xx: add EMC model
19
* target/arm/cpu: Introduce sve_vq_supported bitmap
25
* xlnx-zynqmp: Remove obsolete 'has_rpu' property
20
* docs/specs: Convert ACPI spec docs to rST
26
* target/arm: Speed up aarch64 TBL/TBX
21
* arch_init: Clean up and refactoring
27
* virtio-mmio: improve virtio-mmio get_dev_path alog
22
* hw/core/loader: In gunzip(), check index is in range before use, not after
28
* target/arm: Use TCF0 and TFSRE0 for unprivileged tag checks
23
* softmmu/physmem.c: Remove unneeded NULL check in qemu_ram_alloc_from_fd()
29
* target/arm: Restrict v8M IDAU to TCG
24
* softmmu/physmem.c: Check return value from realpath()
30
* target/arm/cpu: Update coding style to make checkpatch.pl happy
25
* Zero-initialize sockaddr_in structs
31
* musicpal, tc6393xb, omap_lcdc, tcx: drop dead code for non-32-bit-RGB surfaces
26
* raspi: Use error_fatal for SoC realize errors, not error_abort
32
* Add new board: mps3-an524
27
* target/arm: Avoid assertion trying to use KVM and multiple ASes
28
* target/arm: Implement HSTR.TTEE
29
* target/arm: Implement HSTR.TJDBX
30
* target/arm: Do hflags rebuild in cpsr_write()
31
* hw/arm/xlnx-versal, xlnx-zynqmp: Add unimplemented APU mmio
33
32
34
----------------------------------------------------------------
33
----------------------------------------------------------------
35
Doug Evans (3):
34
Andrew Jones (4):
36
hw/net: Add npcm7xx emc model
35
target/arm/cpu: Introduce sve_vq_supported bitmap
37
hw/arm: Add npcm7xx emc model
36
target/arm/kvm64: Ensure sve vls map is completely clear
38
tests/qtests: Add npcm7xx emc model test
37
target/arm/cpu64: Replace kvm_supported with sve_vq_supported
38
target/arm/cpu64: Validate sve vector lengths are supported
39
39
40
Marcin Juszkiewicz (2):
40
Ani Sinha (1):
41
sbsa-ref: remove cortex-a53 from list of supported cpus
41
hw/arm/Kconfig: no need to enable ACPI_MEMORY_HOTPLUG/ACPI_NVDIMM explicitly
42
sbsa-ref: add 'max' to list of allowed cpus
43
42
44
Peter Collingbourne (1):
43
Peter Maydell (26):
45
target/arm: Use TCF0 and TFSRE0 for unprivileged tag checks
44
docs/specs/acpu_cpu_hotplug: Convert to rST
46
45
docs/specs/acpi_mem_hotplug: Convert to rST
47
Peter Maydell (34):
46
docs/specs/acpi_pci_hotplug: Convert to rST
48
hw/arm/musicpal: Remove dead code for non-32-bit-RGB surfaces
47
docs/specs/acpi_nvdimm: Convert to rST
49
hw/display/tc6393xb: Remove dead code for handling non-32bpp surfaces
48
MAINTAINERS: Add ACPI specs documents to ACPI and NVDIMM sections
50
hw/display/tc6393xb: Expand out macros in template header
49
softmmu: Use accel_find("xen") instead of xen_available()
51
hw/display/tc6393xb: Inline tc6393xb_draw_graphic32() at its callsite
50
monitor: Use accel_find("kvm") instead of kvm_available()
52
hw/display/omap_lcdc: Expand out macros in template header
51
softmmu/arch_init.c: Trim down include list
53
hw/display/omap_lcdc: Drop broken bigendian ifdef
52
meson.build: Define QEMU_ARCH in config-target.h
54
hw/display/omap_lcdc: Fix coding style issues in template header
53
arch_init.h: Add QEMU_ARCH_HEXAGON
55
hw/display/omap_lcdc: Inline template header into C file
54
arch_init.h: Move QEMU_ARCH_VIRTIO_* to qdev-monitor.c
56
hw/display/omap_lcdc: Delete unnecessary macro
55
arch_init.h: Don't include arch_init.h unnecessarily
57
hw/display/tcx: Drop unnecessary code for handling BGR format outputs
56
stubs: Remove unused arch_type.c stub
58
hw/arm/mps2-tz: Make SYSCLK frequency board-specific
57
hw/core/loader: In gunzip(), check index is in range before use, not after
59
hw/misc/mps2-scc: Support configurable number of OSCCLK values
58
softmmu/physmem.c: Remove unneeded NULL check in qemu_ram_alloc_from_fd()
60
hw/arm/mps2-tz: Correct the OSCCLK settings for mps2-an505 and mps2-an511
59
softmmu/physmem.c: Check return value from realpath()
61
hw/arm/mps2-tz: Make the OSCCLK settings be configurable per-board
60
net: Zero sockaddr_in in parse_host_port()
62
hw/misc/mps2-fpgaio: Make number of LEDs configurable by board
61
gdbstub: Zero-initialize sockaddr structs
63
hw/misc/mps2-fpgaio: Support SWITCH register
62
tests/qtest/ipmi-bt-test: Zero-initialize sockaddr struct
64
hw/arm/mps2-tz: Make FPGAIO switch and LED config per-board
63
tests/tcg/multiarch/linux-test: Zero-initialize sockaddr structs
65
hw/arm/mps2-tz: Condition IRQ splitting on number of CPUs, not board type
64
raspi: Use error_fatal for SoC realize errors, not error_abort
66
hw/arm/mps2-tz: Make number of IRQs board-specific
65
target/arm: Avoid assertion trying to use KVM and multiple ASes
67
hw/misc/mps2-scc: Implement CFG_REG5 and CFG_REG6 for MPS3 AN524
66
hw/arm/virt: Delete EL3 error checksnow provided in CPU realize
68
hw/arm/mps2-tz: Correct wrong interrupt numbers for DMA and SPI
67
target/arm: Implement HSTR.TTEE
69
hw/arm/mps2-tz: Allow PPCPortInfo structures to specify device interrupts
68
target/arm: Implement HSTR.TJDBX
70
hw/arm/mps2-tz: Move device IRQ info to data structures
69
target/arm: Do hflags rebuild in cpsr_write()
71
hw/arm/mps2-tz: Size the uart-irq-orgate based on the number of UARTs
72
hw/arm/mps2-tz: Allow boards to have different PPCInfo data
73
hw/arm/mps2-tz: Make RAM arrangement board-specific
74
hw/arm/mps2-tz: Set MachineClass default_ram info from RAMInfo data
75
hw/arm/mps2-tz: Support ROMs as well as RAMs
76
hw/arm/mps2-tz: Get armv7m_load_kernel() size argument from RAMInfo
77
hw/arm/mps2-tz: Add new mps3-an524 board
78
hw/arm/mps2-tz: Stub out USB controller for mps3-an524
79
hw/arm/mps2-tz: Provide PL031 RTC on mps3-an524
80
docs/system/arm/mps2.rst: Document the new mps3-an524 board
81
hw/arm/mps2: Update old infocenter.arm.com URLs
82
70
83
Philippe Mathieu-Daudé (4):
71
Philippe Mathieu-Daudé (4):
84
hw/arm/xlnx-zynqmp: Remove obsolete 'has_rpu' property
72
hw/arm/xlnx-zynqmp: Realize qspi controller *after* qspi_dma
85
hw/i2c/npcm7xx_smbus: Simplify npcm7xx_smbus_init()
73
hw/dma/xlnx_csu_dma: Run trivial checks early in realize()
86
target/arm: Restrict v8M IDAU to TCG
74
hw/dma/xlnx_csu_dma: Always expect 'dma' link property to be set
87
target/arm/cpu: Update coding style to make checkpatch.pl happy
75
hw/dma/xlnx-zdma Always expect 'dma' link property to be set
88
76
89
Rebecca Cran (3):
77
Tong Ho (2):
90
target/arm: Add support for FEAT_SSBS, Speculative Store Bypass Safe
78
hw/arm/xlnx-versal: Add unimplemented APU mmio
91
target/arm: Enable FEAT_SSBS for "max" AARCH64 CPU
79
hw/arm/xlnx-zynqmp: Add unimplemented APU mmio
92
target/arm: Set ID_PFR2.SSBS to 1 for "max" 32-bit CPU
93
80
94
Richard Henderson (1):
81
docs/specs/acpi_cpu_hotplug.rst | 235 +++++++++++++++++++++
95
target/arm: Speed up aarch64 TBL/TBX
82
docs/specs/acpi_cpu_hotplug.txt | 160 --------------
83
docs/specs/acpi_mem_hotplug.rst | 128 +++++++++++
84
docs/specs/acpi_mem_hotplug.txt | 94 ---------
85
docs/specs/acpi_nvdimm.rst | 228 ++++++++++++++++++++
86
docs/specs/acpi_nvdimm.txt | 188 -----------------
87
.../{acpi_pci_hotplug.txt => acpi_pci_hotplug.rst} | 37 ++--
88
docs/specs/index.rst | 4 +
89
meson.build | 2 +
90
include/hw/arm/xlnx-versal.h | 2 +
91
include/hw/arm/xlnx-zynqmp.h | 7 +
92
include/hw/dma/xlnx-zdma.h | 2 +-
93
include/hw/dma/xlnx_csu_dma.h | 2 +-
94
include/sysemu/arch_init.h | 15 +-
95
target/arm/cpu.h | 17 +-
96
target/arm/helper.h | 2 +
97
target/arm/syndrome.h | 7 +
98
blockdev.c | 1 -
99
gdbstub.c | 4 +-
100
hw/arm/raspi.c | 2 +-
101
hw/arm/virt.c | 5 -
102
hw/arm/xlnx-versal.c | 4 +
103
hw/arm/xlnx-zynqmp.c | 86 ++++++--
104
hw/core/loader.c | 35 ++-
105
hw/dma/xlnx-zdma.c | 24 +--
106
hw/dma/xlnx_csu_dma.c | 31 ++-
107
hw/i386/pc.c | 1 -
108
hw/i386/pc_piix.c | 1 -
109
hw/i386/pc_q35.c | 1 -
110
hw/mips/jazz.c | 1 -
111
hw/mips/malta.c | 1 -
112
hw/ppc/prep.c | 1 -
113
hw/riscv/sifive_e.c | 1 -
114
hw/riscv/sifive_u.c | 1 -
115
hw/riscv/spike.c | 1 -
116
hw/riscv/virt.c | 1 -
117
linux-user/arm/signal.c | 2 -
118
monitor/qmp-cmds.c | 3 +-
119
net/net.c | 2 +
120
softmmu/arch_init.c | 66 ------
121
softmmu/physmem.c | 5 +-
122
softmmu/qdev-monitor.c | 9 +
123
softmmu/vl.c | 6 +-
124
stubs/arch_type.c | 4 -
125
target/arm/cpu.c | 23 ++
126
target/arm/cpu64.c | 118 +++++------
127
target/arm/helper.c | 40 +++-
128
target/arm/kvm64.c | 2 +-
129
target/arm/op_helper.c | 16 ++
130
target/arm/translate.c | 12 ++
131
target/ppc/cpu_init.c | 1 -
132
target/s390x/cpu-sysemu.c | 1 -
133
tests/qtest/ipmi-bt-test.c | 2 +-
134
tests/tcg/multiarch/linux-test.c | 4 +-
135
MAINTAINERS | 5 +
136
hw/arm/Kconfig | 2 -
137
stubs/meson.build | 1 -
138
57 files changed, 949 insertions(+), 707 deletions(-)
139
create mode 100644 docs/specs/acpi_cpu_hotplug.rst
140
delete mode 100644 docs/specs/acpi_cpu_hotplug.txt
141
create mode 100644 docs/specs/acpi_mem_hotplug.rst
142
delete mode 100644 docs/specs/acpi_mem_hotplug.txt
143
create mode 100644 docs/specs/acpi_nvdimm.rst
144
delete mode 100644 docs/specs/acpi_nvdimm.txt
145
rename docs/specs/{acpi_pci_hotplug.txt => acpi_pci_hotplug.rst} (51%)
146
delete mode 100644 stubs/arch_type.c
96
147
97
schspa (1):
98
virtio-mmio: improve virtio-mmio get_dev_path alog
99
100
docs/system/arm/mps2.rst | 24 +-
101
docs/system/arm/nuvoton.rst | 3 +-
102
hw/display/omap_lcd_template.h | 169 --------
103
hw/display/tc6393xb_template.h | 72 ----
104
include/hw/arm/armsse.h | 4 +-
105
include/hw/arm/npcm7xx.h | 2 +
106
include/hw/arm/xlnx-zynqmp.h | 2 -
107
include/hw/misc/armsse-cpuid.h | 2 +-
108
include/hw/misc/armsse-mhu.h | 2 +-
109
include/hw/misc/iotkit-secctl.h | 2 +-
110
include/hw/misc/iotkit-sysctl.h | 2 +-
111
include/hw/misc/iotkit-sysinfo.h | 2 +-
112
include/hw/misc/mps2-fpgaio.h | 8 +-
113
include/hw/misc/mps2-scc.h | 10 +-
114
include/hw/net/npcm7xx_emc.h | 286 +++++++++++++
115
include/ui/console.h | 10 -
116
target/arm/cpu.h | 15 +-
117
target/arm/helper-a64.h | 2 +-
118
target/arm/internals.h | 6 +
119
hw/arm/mps2-tz.c | 632 +++++++++++++++++++++++-----
120
hw/arm/mps2.c | 5 +
121
hw/arm/musicpal.c | 64 ++-
122
hw/arm/npcm7xx.c | 50 ++-
123
hw/arm/sbsa-ref.c | 2 +-
124
hw/arm/xlnx-zynqmp.c | 6 -
125
hw/display/omap_lcdc.c | 129 +++++-
126
hw/display/tc6393xb.c | 48 +--
127
hw/display/tcx.c | 31 +-
128
hw/i2c/npcm7xx_smbus.c | 1 -
129
hw/misc/armsse-cpuid.c | 2 +-
130
hw/misc/armsse-mhu.c | 2 +-
131
hw/misc/iotkit-sysctl.c | 2 +-
132
hw/misc/iotkit-sysinfo.c | 2 +-
133
hw/misc/mps2-fpgaio.c | 43 +-
134
hw/misc/mps2-scc.c | 93 ++++-
135
hw/net/npcm7xx_emc.c | 857 ++++++++++++++++++++++++++++++++++++++
136
hw/virtio/virtio-mmio.c | 13 +-
137
target/arm/cpu.c | 23 +-
138
target/arm/cpu64.c | 5 +
139
target/arm/cpu_tcg.c | 8 +
140
target/arm/helper-a64.c | 32 --
141
target/arm/helper.c | 39 +-
142
target/arm/mte_helper.c | 13 +-
143
target/arm/translate-a64.c | 70 +---
144
target/arm/vec_helper.c | 48 +++
145
tests/qtest/npcm7xx_emc-test.c | 862 +++++++++++++++++++++++++++++++++++++++
146
hw/net/meson.build | 1 +
147
hw/net/trace-events | 17 +
148
tests/qtest/meson.build | 3 +-
149
49 files changed, 3098 insertions(+), 628 deletions(-)
150
delete mode 100644 hw/display/omap_lcd_template.h
151
delete mode 100644 hw/display/tc6393xb_template.h
152
create mode 100644 include/hw/net/npcm7xx_emc.h
153
create mode 100644 hw/net/npcm7xx_emc.c
154
create mode 100644 tests/qtest/npcm7xx_emc-test.c
155
diff view generated by jsdifflib
Deleted patch
1
From: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
2
1
3
Cortex-A53 supports 40bits of address space. sbsa-ref's memory starts
4
above this limit.
5
6
Signed-off-by: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Acked-by: Leif Lindholm <leif@nuviainc.com>
9
Message-id: 20210216150122.3830863-2-marcin.juszkiewicz@linaro.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
hw/arm/sbsa-ref.c | 1 -
13
1 file changed, 1 deletion(-)
14
15
diff --git a/hw/arm/sbsa-ref.c b/hw/arm/sbsa-ref.c
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/sbsa-ref.c
18
+++ b/hw/arm/sbsa-ref.c
19
@@ -XXX,XX +XXX,XX @@ static const int sbsa_ref_irqmap[] = {
20
};
21
22
static const char * const valid_cpus[] = {
23
- ARM_CPU_TYPE_NAME("cortex-a53"),
24
ARM_CPU_TYPE_NAME("cortex-a57"),
25
ARM_CPU_TYPE_NAME("cortex-a72"),
26
};
27
--
28
2.20.1
29
30
diff view generated by jsdifflib
Deleted patch
1
From: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
2
1
3
Let add 'max' cpu while work goes on adding newer CPU types than
4
Cortex-A72. This allows us to check SVE etc support.
5
6
Signed-off-by: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
7
Acked-by: Leif Lindholm <leif@nuviainc.com>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Message-id: 20210216150122.3830863-3-marcin.juszkiewicz@linaro.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
hw/arm/sbsa-ref.c | 1 +
13
1 file changed, 1 insertion(+)
14
15
diff --git a/hw/arm/sbsa-ref.c b/hw/arm/sbsa-ref.c
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/sbsa-ref.c
18
+++ b/hw/arm/sbsa-ref.c
19
@@ -XXX,XX +XXX,XX @@ static const int sbsa_ref_irqmap[] = {
20
static const char * const valid_cpus[] = {
21
ARM_CPU_TYPE_NAME("cortex-a57"),
22
ARM_CPU_TYPE_NAME("cortex-a72"),
23
+ ARM_CPU_TYPE_NAME("max"),
24
};
25
26
static bool cpu_type_valid(const char *cpu)
27
--
28
2.20.1
29
30
diff view generated by jsdifflib
1
The MPS2 board has 2 LEDs, but the MPS3 board has 10 LEDs. The
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
FPGAIO device is similar on both sets of boards, but the LED0
3
register has correspondingly more bits that have an effect. Add a
4
device property for number of LEDs.
5
2
3
If we link QOM object (a) as a property of QOM object (b),
4
we must set the property *before* (b) is realized.
5
6
Move QSPI realization *after* QSPI DMA.
7
8
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
9
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
10
Message-id: 20210819163422.2863447-2-philmd@redhat.com
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20210215115138.20465-6-peter.maydell@linaro.org
10
---
12
---
11
include/hw/misc/mps2-fpgaio.h | 5 ++++-
13
hw/arm/xlnx-zynqmp.c | 42 ++++++++++++++++++++----------------------
12
hw/misc/mps2-fpgaio.c | 31 +++++++++++++++++++++++--------
14
1 file changed, 20 insertions(+), 22 deletions(-)
13
2 files changed, 27 insertions(+), 9 deletions(-)
14
15
15
diff --git a/include/hw/misc/mps2-fpgaio.h b/include/hw/misc/mps2-fpgaio.h
16
diff --git a/hw/arm/xlnx-zynqmp.c b/hw/arm/xlnx-zynqmp.c
16
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
17
--- a/include/hw/misc/mps2-fpgaio.h
18
--- a/hw/arm/xlnx-zynqmp.c
18
+++ b/include/hw/misc/mps2-fpgaio.h
19
+++ b/hw/arm/xlnx-zynqmp.c
19
@@ -XXX,XX +XXX,XX @@
20
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
20
#define TYPE_MPS2_FPGAIO "mps2-fpgaio"
21
g_free(bus_name);
21
OBJECT_DECLARE_SIMPLE_TYPE(MPS2FPGAIO, MPS2_FPGAIO)
22
}
22
23
23
+#define MPS2FPGAIO_MAX_LEDS 32
24
- if (!sysbus_realize(SYS_BUS_DEVICE(&s->qspi), errp)) {
25
- return;
26
- }
27
- sysbus_mmio_map(SYS_BUS_DEVICE(&s->qspi), 0, QSPI_ADDR);
28
- sysbus_mmio_map(SYS_BUS_DEVICE(&s->qspi), 1, LQSPI_ADDR);
29
- sysbus_connect_irq(SYS_BUS_DEVICE(&s->qspi), 0, gic_spi[QSPI_IRQ]);
30
-
31
- for (i = 0; i < XLNX_ZYNQMP_NUM_QSPI_BUS; i++) {
32
- gchar *bus_name;
33
- gchar *target_bus;
34
-
35
- /* Alias controller SPI bus to the SoC itself */
36
- bus_name = g_strdup_printf("qspi%d", i);
37
- target_bus = g_strdup_printf("spi%d", i);
38
- object_property_add_alias(OBJECT(s), bus_name,
39
- OBJECT(&s->qspi), target_bus);
40
- g_free(bus_name);
41
- g_free(target_bus);
42
- }
43
-
44
if (!sysbus_realize(SYS_BUS_DEVICE(&s->dp), errp)) {
45
return;
46
}
47
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
48
49
sysbus_mmio_map(SYS_BUS_DEVICE(&s->qspi_dma), 0, QSPI_DMA_ADDR);
50
sysbus_connect_irq(SYS_BUS_DEVICE(&s->qspi_dma), 0, gic_spi[QSPI_IRQ]);
51
- object_property_set_link(OBJECT(&s->qspi), "stream-connected-dma",
52
- OBJECT(&s->qspi_dma), errp);
24
+
53
+
25
struct MPS2FPGAIO {
54
+ if (!object_property_set_link(OBJECT(&s->qspi), "stream-connected-dma",
26
/*< private >*/
55
+ OBJECT(&s->qspi_dma), errp)) {
27
SysBusDevice parent_obj;
56
+ return;
28
57
+ }
29
/*< public >*/
58
+ if (!sysbus_realize(SYS_BUS_DEVICE(&s->qspi), errp)) {
30
MemoryRegion iomem;
31
- LEDState *led[2];
32
+ LEDState *led[MPS2FPGAIO_MAX_LEDS];
33
+ uint32_t num_leds;
34
35
uint32_t led0;
36
uint32_t prescale;
37
diff --git a/hw/misc/mps2-fpgaio.c b/hw/misc/mps2-fpgaio.c
38
index XXXXXXX..XXXXXXX 100644
39
--- a/hw/misc/mps2-fpgaio.c
40
+++ b/hw/misc/mps2-fpgaio.c
41
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_write(void *opaque, hwaddr offset, uint64_t value,
42
43
switch (offset) {
44
case A_LED0:
45
- s->led0 = value & 0x3;
46
- led_set_state(s->led[0], value & 0x01);
47
- led_set_state(s->led[1], value & 0x02);
48
+ if (s->num_leds != 0) {
49
+ uint32_t i;
50
+
51
+ s->led0 = value & MAKE_64BIT_MASK(0, s->num_leds);
52
+ for (i = 0; i < s->num_leds; i++) {
53
+ led_set_state(s->led[i], value & (1 << i));
54
+ }
55
+ }
56
break;
57
case A_PRESCALE:
58
resync_counter(s);
59
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_reset(DeviceState *dev)
60
s->pscntr = 0;
61
s->pscntr_sync_ticks = now;
62
63
- for (size_t i = 0; i < ARRAY_SIZE(s->led); i++) {
64
+ for (size_t i = 0; i < s->num_leds; i++) {
65
device_cold_reset(DEVICE(s->led[i]));
66
}
67
}
68
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_init(Object *obj)
69
static void mps2_fpgaio_realize(DeviceState *dev, Error **errp)
70
{
71
MPS2FPGAIO *s = MPS2_FPGAIO(dev);
72
+ uint32_t i;
73
74
- s->led[0] = led_create_simple(OBJECT(dev), GPIO_POLARITY_ACTIVE_HIGH,
75
- LED_COLOR_GREEN, "USERLED0");
76
- s->led[1] = led_create_simple(OBJECT(dev), GPIO_POLARITY_ACTIVE_HIGH,
77
- LED_COLOR_GREEN, "USERLED1");
78
+ if (s->num_leds > MPS2FPGAIO_MAX_LEDS) {
79
+ error_setg(errp, "num-leds cannot be greater than %d",
80
+ MPS2FPGAIO_MAX_LEDS);
81
+ return;
59
+ return;
82
+ }
60
+ }
61
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->qspi), 0, QSPI_ADDR);
62
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->qspi), 1, LQSPI_ADDR);
63
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->qspi), 0, gic_spi[QSPI_IRQ]);
83
+
64
+
84
+ for (i = 0; i < s->num_leds; i++) {
65
+ for (i = 0; i < XLNX_ZYNQMP_NUM_QSPI_BUS; i++) {
85
+ g_autofree char *ledname = g_strdup_printf("USERLED%d", i);
66
+ g_autofree gchar *bus_name = g_strdup_printf("qspi%d", i);
86
+ s->led[i] = led_create_simple(OBJECT(dev), GPIO_POLARITY_ACTIVE_HIGH,
67
+ g_autofree gchar *target_bus = g_strdup_printf("spi%d", i);
87
+ LED_COLOR_GREEN, ledname);
68
+
69
+ /* Alias controller SPI bus to the SoC itself */
70
+ object_property_add_alias(OBJECT(s), bus_name,
71
+ OBJECT(&s->qspi), target_bus);
88
+ }
72
+ }
89
}
73
}
90
74
91
static bool mps2_fpgaio_counters_needed(void *opaque)
75
static Property xlnx_zynqmp_props[] = {
92
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription mps2_fpgaio_vmstate = {
93
static Property mps2_fpgaio_properties[] = {
94
/* Frequency of the prescale counter */
95
DEFINE_PROP_UINT32("prescale-clk", MPS2FPGAIO, prescale_clk, 20000000),
96
+ /* Number of LEDs controlled by LED0 register */
97
+ DEFINE_PROP_UINT32("num-leds", MPS2FPGAIO, num_leds, 2),
98
DEFINE_PROP_END_OF_LIST(),
99
};
100
101
--
76
--
102
2.20.1
77
2.20.1
103
78
104
79
diff view generated by jsdifflib
1
The AN505 and AN521 have the same layout of RAM; the AN524 does not.
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
Replace the current hard-coding of where the RAM is and which parts
3
of it are behind which MPCs with a data-driven approach.
4
2
3
If some property are not set, we'll return indicating a failure,
4
so it is pointless to allocate / initialize some fields too early.
5
Move the trivial checks earlier in realize().
6
7
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
8
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
9
Message-id: 20210819163422.2863447-3-philmd@redhat.com
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20210215115138.20465-17-peter.maydell@linaro.org
8
---
11
---
9
hw/arm/mps2-tz.c | 175 +++++++++++++++++++++++++++++++++++++----------
12
hw/dma/xlnx_csu_dma.c | 10 +++++-----
10
1 file changed, 138 insertions(+), 37 deletions(-)
13
1 file changed, 5 insertions(+), 5 deletions(-)
11
14
12
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
15
diff --git a/hw/dma/xlnx_csu_dma.c b/hw/dma/xlnx_csu_dma.c
13
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/arm/mps2-tz.c
17
--- a/hw/dma/xlnx_csu_dma.c
15
+++ b/hw/arm/mps2-tz.c
18
+++ b/hw/dma/xlnx_csu_dma.c
16
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ static void xlnx_csu_dma_realize(DeviceState *dev, Error **errp)
17
#include "qom/object.h"
20
XlnxCSUDMA *s = XLNX_CSU_DMA(dev);
18
21
RegisterInfoArray *reg_array;
19
#define MPS2TZ_NUMIRQ_MAX 92
22
20
+#define MPS2TZ_RAM_MAX 4
23
+ if (!s->is_dst && !s->tx_dev) {
21
24
+ error_setg(errp, "zynqmp.csu-dma: Stream not connected");
22
typedef enum MPS2TZFPGAType {
25
+ return;
23
FPGA_AN505,
24
FPGA_AN521,
25
} MPS2TZFPGAType;
26
27
+/*
28
+ * Define the layout of RAM in a board, including which parts are
29
+ * behind which MPCs.
30
+ * mrindex specifies the index into mms->ram[] to use for the backing RAM;
31
+ * -1 means "use the system RAM".
32
+ */
33
+typedef struct RAMInfo {
34
+ const char *name;
35
+ uint32_t base;
36
+ uint32_t size;
37
+ int mpc; /* MPC number, -1 for "not behind an MPC" */
38
+ int mrindex;
39
+ int flags;
40
+} RAMInfo;
41
+
42
+/*
43
+ * Flag values:
44
+ * IS_ALIAS: this RAM area is an alias to the upstream end of the
45
+ * MPC specified by its .mpc value
46
+ */
47
+#define IS_ALIAS 1
48
+
49
struct MPS2TZMachineClass {
50
MachineClass parent;
51
MPS2TZFPGAType fpga_type;
52
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineClass {
53
uint32_t fpgaio_num_leds; /* Number of LEDs in FPGAIO LED0 register */
54
bool fpgaio_has_switches; /* Does FPGAIO have SWITCH register? */
55
int numirq; /* Number of external interrupts */
56
+ const RAMInfo *raminfo;
57
const char *armsse_type;
58
};
59
60
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
61
MachineState parent;
62
63
ARMSSE iotkit;
64
- MemoryRegion ssram[3];
65
- MemoryRegion ssram1_m;
66
+ MemoryRegion ram[MPS2TZ_RAM_MAX];
67
MPS2SCC scc;
68
MPS2FPGAIO fpgaio;
69
TZPPC ppc[5];
70
- TZMPC ssram_mpc[3];
71
+ TZMPC mpc[3];
72
PL022State spi[5];
73
ArmSbconI2CState i2c[4];
74
UnimplementedDeviceState i2s_audio;
75
@@ -XXX,XX +XXX,XX @@ static const uint32_t an505_oscclk[] = {
76
25000000,
77
};
78
79
+static const RAMInfo an505_raminfo[] = { {
80
+ .name = "ssram-0",
81
+ .base = 0x00000000,
82
+ .size = 0x00400000,
83
+ .mpc = 0,
84
+ .mrindex = 0,
85
+ }, {
86
+ .name = "ssram-1",
87
+ .base = 0x28000000,
88
+ .size = 0x00200000,
89
+ .mpc = 1,
90
+ .mrindex = 1,
91
+ }, {
92
+ .name = "ssram-2",
93
+ .base = 0x28200000,
94
+ .size = 0x00200000,
95
+ .mpc = 2,
96
+ .mrindex = 2,
97
+ }, {
98
+ .name = "ssram-0-alias",
99
+ .base = 0x00400000,
100
+ .size = 0x00400000,
101
+ .mpc = 0,
102
+ .mrindex = 3,
103
+ .flags = IS_ALIAS,
104
+ }, {
105
+ /* Use the largest bit of contiguous RAM as our "system memory" */
106
+ .name = "mps.ram",
107
+ .base = 0x80000000,
108
+ .size = 16 * MiB,
109
+ .mpc = -1,
110
+ .mrindex = -1,
111
+ }, {
112
+ .name = NULL,
113
+ },
114
+};
115
+
116
+static const RAMInfo *find_raminfo_for_mpc(MPS2TZMachineState *mms, int mpc)
117
+{
118
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
119
+ const RAMInfo *p;
120
+
121
+ for (p = mmc->raminfo; p->name; p++) {
122
+ if (p->mpc == mpc && !(p->flags & IS_ALIAS)) {
123
+ return p;
124
+ }
125
+ }
126
+ /* if raminfo array doesn't have an entry for each MPC this is a bug */
127
+ g_assert_not_reached();
128
+}
129
+
130
+static MemoryRegion *mr_for_raminfo(MPS2TZMachineState *mms,
131
+ const RAMInfo *raminfo)
132
+{
133
+ /* Return an initialized MemoryRegion for the RAMInfo. */
134
+ MemoryRegion *ram;
135
+
136
+ if (raminfo->mrindex < 0) {
137
+ /* Means this RAMInfo is for QEMU's "system memory" */
138
+ MachineState *machine = MACHINE(mms);
139
+ return machine->ram;
140
+ }
26
+ }
141
+
27
+
142
+ assert(raminfo->mrindex < MPS2TZ_RAM_MAX);
28
reg_array =
143
+ ram = &mms->ram[raminfo->mrindex];
29
register_init_block32(dev, xlnx_csu_dma_regs_info[!!s->is_dst],
144
+
30
XLNX_CSU_DMA_R_MAX,
145
+ memory_region_init_ram(ram, NULL, raminfo->name,
31
@@ -XXX,XX +XXX,XX @@ static void xlnx_csu_dma_realize(DeviceState *dev, Error **errp)
146
+ raminfo->size, &error_fatal);
32
sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->iomem);
147
+ return ram;
33
sysbus_init_irq(SYS_BUS_DEVICE(dev), &s->irq);
148
+}
34
149
+
35
- if (!s->is_dst && !s->tx_dev) {
150
/* Create an alias of an entire original MemoryRegion @orig
36
- error_setg(errp, "zynqmp.csu-dma: Stream not connected");
151
* located at @base in the memory map.
37
- return;
152
*/
153
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_mpc(MPS2TZMachineState *mms, void *opaque,
154
const int *irqs)
155
{
156
TZMPC *mpc = opaque;
157
- int i = mpc - &mms->ssram_mpc[0];
158
- MemoryRegion *ssram = &mms->ssram[i];
159
+ int i = mpc - &mms->mpc[0];
160
MemoryRegion *upstream;
161
- char *mpcname = g_strdup_printf("%s-mpc", name);
162
- static uint32_t ramsize[] = { 0x00400000, 0x00200000, 0x00200000 };
163
- static uint32_t rambase[] = { 0x00000000, 0x28000000, 0x28200000 };
164
+ const RAMInfo *raminfo = find_raminfo_for_mpc(mms, i);
165
+ MemoryRegion *ram = mr_for_raminfo(mms, raminfo);
166
167
- memory_region_init_ram(ssram, NULL, name, ramsize[i], &error_fatal);
168
-
169
- object_initialize_child(OBJECT(mms), mpcname, mpc, TYPE_TZ_MPC);
170
- object_property_set_link(OBJECT(mpc), "downstream", OBJECT(ssram),
171
+ object_initialize_child(OBJECT(mms), name, mpc, TYPE_TZ_MPC);
172
+ object_property_set_link(OBJECT(mpc), "downstream", OBJECT(ram),
173
&error_fatal);
174
sysbus_realize(SYS_BUS_DEVICE(mpc), &error_fatal);
175
/* Map the upstream end of the MPC into system memory */
176
upstream = sysbus_mmio_get_region(SYS_BUS_DEVICE(mpc), 1);
177
- memory_region_add_subregion(get_system_memory(), rambase[i], upstream);
178
+ memory_region_add_subregion(get_system_memory(), raminfo->base, upstream);
179
/* and connect its interrupt to the IoTKit */
180
qdev_connect_gpio_out_named(DEVICE(mpc), "irq", 0,
181
qdev_get_gpio_in_named(DEVICE(&mms->iotkit),
182
"mpcexp_status", i));
183
184
- /* The first SSRAM is a special case as it has an alias; accesses to
185
- * the alias region at 0x00400000 must also go to the MPC upstream.
186
- */
187
- if (i == 0) {
188
- make_ram_alias(&mms->ssram1_m, "mps.ssram1_m", upstream, 0x00400000);
189
- }
38
- }
190
-
39
-
191
- g_free(mpcname);
40
s->src_timer = ptimer_init(xlnx_csu_dma_src_timeout_hit,
192
/* Return the register interface MR for our caller to map behind the PPC */
41
s, PTIMER_POLICY_DEFAULT);
193
return sysbus_mmio_get_region(SYS_BUS_DEVICE(mpc), 0);
194
}
195
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_i2c(MPS2TZMachineState *mms, void *opaque,
196
return sysbus_mmio_get_region(s, 0);
197
}
198
199
+static void create_non_mpc_ram(MPS2TZMachineState *mms)
200
+{
201
+ /*
202
+ * Handle the RAMs which are either not behind MPCs or which are
203
+ * aliases to another MPC.
204
+ */
205
+ const RAMInfo *p;
206
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
207
+
208
+ for (p = mmc->raminfo; p->name; p++) {
209
+ if (p->flags & IS_ALIAS) {
210
+ SysBusDevice *mpc_sbd = SYS_BUS_DEVICE(&mms->mpc[p->mpc]);
211
+ MemoryRegion *upstream = sysbus_mmio_get_region(mpc_sbd, 1);
212
+ make_ram_alias(&mms->ram[p->mrindex], p->name, upstream, p->base);
213
+ } else if (p->mpc == -1) {
214
+ /* RAM not behind an MPC */
215
+ MemoryRegion *mr = mr_for_raminfo(mms, p);
216
+ memory_region_add_subregion(get_system_memory(), p->base, mr);
217
+ }
218
+ }
219
+}
220
+
221
static void mps2tz_common_init(MachineState *machine)
222
{
223
MPS2TZMachineState *mms = MPS2TZ_MACHINE(machine);
224
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
225
qdev_connect_gpio_out_named(iotkitdev, "sec_resp_cfg", 0,
226
qdev_get_gpio_in(dev_splitter, 0));
227
228
- /* The IoTKit sets up much of the memory layout, including
229
+ /*
230
+ * The IoTKit sets up much of the memory layout, including
231
* the aliases between secure and non-secure regions in the
232
- * address space. The FPGA itself contains:
233
- *
234
- * 0x00000000..0x003fffff SSRAM1
235
- * 0x00400000..0x007fffff alias of SSRAM1
236
- * 0x28000000..0x283fffff 4MB SSRAM2 + SSRAM3
237
- * 0x40100000..0x4fffffff AHB Master Expansion 1 interface devices
238
- * 0x80000000..0x80ffffff 16MB PSRAM
239
- */
240
-
241
- /* The FPGA images have an odd combination of different RAMs,
242
+ * address space, and also most of the devices in the system.
243
+ * The FPGA itself contains various RAMs and some additional devices.
244
+ * The FPGA images have an odd combination of different RAMs,
245
* because in hardware they are different implementations and
246
* connected to different buses, giving varying performance/size
247
* tradeoffs. For QEMU they're all just RAM, though. We arbitrarily
248
- * call the 16MB our "system memory", as it's the largest lump.
249
+ * call the largest lump our "system memory".
250
*/
251
- memory_region_add_subregion(system_memory, 0x80000000, machine->ram);
252
253
/*
254
* The overflow IRQs for all UARTs are ORed together.
255
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
256
const PPCInfo an505_ppcs[] = { {
257
.name = "apb_ppcexp0",
258
.ports = {
259
- { "ssram-0", make_mpc, &mms->ssram_mpc[0], 0x58007000, 0x1000 },
260
- { "ssram-1", make_mpc, &mms->ssram_mpc[1], 0x58008000, 0x1000 },
261
- { "ssram-2", make_mpc, &mms->ssram_mpc[2], 0x58009000, 0x1000 },
262
+ { "ssram-0-mpc", make_mpc, &mms->mpc[0], 0x58007000, 0x1000 },
263
+ { "ssram-1-mpc", make_mpc, &mms->mpc[1], 0x58008000, 0x1000 },
264
+ { "ssram-2-mpc", make_mpc, &mms->mpc[2], 0x58009000, 0x1000 },
265
},
266
}, {
267
.name = "apb_ppcexp1",
268
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
269
270
create_unimplemented_device("FPGA NS PC", 0x48007000, 0x1000);
271
272
+ create_non_mpc_ram(mms);
273
+
274
armv7m_load_kernel(ARM_CPU(first_cpu), machine->kernel_filename, 0x400000);
275
}
276
277
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
278
mmc->fpgaio_num_leds = 2;
279
mmc->fpgaio_has_switches = false;
280
mmc->numirq = 92;
281
+ mmc->raminfo = an505_raminfo;
282
mmc->armsse_type = TYPE_IOTKIT;
283
}
284
285
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
286
mmc->fpgaio_num_leds = 2;
287
mmc->fpgaio_has_switches = false;
288
mmc->numirq = 92;
289
+ mmc->raminfo = an505_raminfo; /* AN521 is the same as AN505 here */
290
mmc->armsse_type = TYPE_SSE200;
291
}
292
42
293
--
43
--
294
2.20.1
44
2.20.1
295
45
296
46
diff view generated by jsdifflib
1
From: Doug Evans <dje@google.com>
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
2
3
This is a 10/100 ethernet device that has several features.
3
Simplify by always passing a MemoryRegion property to the device.
4
Only the ones needed by the Linux driver have been implemented.
4
Doing so we can move the AddressSpace field to the device struct,
5
See npcm7xx_emc.c for a list of unimplemented features.
5
removing need for heap allocation.
6
6
7
Reviewed-by: Hao Wu <wuhaotsh@google.com>
7
Update the Xilinx ZynqMP SoC model to pass the default system
8
Reviewed-by: Avi Fishman <avi.fishman@nuvoton.com>
8
memory instead of a NULL value.
9
Signed-off-by: Doug Evans <dje@google.com>
9
10
Message-id: 20210218212453.831406-2-dje@google.com
10
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
12
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
13
Message-id: 20210819163422.2863447-4-philmd@redhat.com
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
15
---
13
include/hw/net/npcm7xx_emc.h | 286 ++++++++++++
16
include/hw/dma/xlnx_csu_dma.h | 2 +-
14
hw/net/npcm7xx_emc.c | 857 +++++++++++++++++++++++++++++++++++
17
hw/arm/xlnx-zynqmp.c | 4 ++++
15
hw/net/meson.build | 1 +
18
hw/dma/xlnx_csu_dma.c | 21 ++++++++++-----------
16
hw/net/trace-events | 17 +
19
3 files changed, 15 insertions(+), 12 deletions(-)
17
4 files changed, 1161 insertions(+)
18
create mode 100644 include/hw/net/npcm7xx_emc.h
19
create mode 100644 hw/net/npcm7xx_emc.c
20
20
21
diff --git a/include/hw/net/npcm7xx_emc.h b/include/hw/net/npcm7xx_emc.h
21
diff --git a/include/hw/dma/xlnx_csu_dma.h b/include/hw/dma/xlnx_csu_dma.h
22
new file mode 100644
22
index XXXXXXX..XXXXXXX 100644
23
index XXXXXXX..XXXXXXX
23
--- a/include/hw/dma/xlnx_csu_dma.h
24
--- /dev/null
24
+++ b/include/hw/dma/xlnx_csu_dma.h
25
+++ b/include/hw/net/npcm7xx_emc.h
25
@@ -XXX,XX +XXX,XX @@ typedef struct XlnxCSUDMA {
26
@@ -XXX,XX +XXX,XX @@
26
MemoryRegion iomem;
27
+/*
27
MemTxAttrs attr;
28
+ * Nuvoton NPCM7xx EMC Module
28
MemoryRegion *dma_mr;
29
+ *
29
- AddressSpace *dma_as;
30
+ * Copyright 2020 Google LLC
30
+ AddressSpace dma_as;
31
+ *
31
qemu_irq irq;
32
+ * This program is free software; you can redistribute it and/or modify it
32
StreamSink *tx_dev; /* Used as generic StreamSink */
33
+ * under the terms of the GNU General Public License as published by the
33
ptimer_state *src_timer;
34
+ * Free Software Foundation; either version 2 of the License, or
34
diff --git a/hw/arm/xlnx-zynqmp.c b/hw/arm/xlnx-zynqmp.c
35
+ * (at your option) any later version.
35
index XXXXXXX..XXXXXXX 100644
36
+ *
36
--- a/hw/arm/xlnx-zynqmp.c
37
+ * This program is distributed in the hope that it will be useful, but WITHOUT
37
+++ b/hw/arm/xlnx-zynqmp.c
38
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
38
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
39
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
39
gic_spi[adma_ch_intr[i]]);
40
+ * for more details.
40
}
41
+ */
41
42
+
42
+ if (!object_property_set_link(OBJECT(&s->qspi_dma), "dma",
43
+#ifndef NPCM7XX_EMC_H
43
+ OBJECT(system_memory), errp)) {
44
+#define NPCM7XX_EMC_H
45
+
46
+#include "hw/irq.h"
47
+#include "hw/sysbus.h"
48
+#include "net/net.h"
49
+
50
+/* 32-bit register indices. */
51
+enum NPCM7xxPWMRegister {
52
+ /* Control registers. */
53
+ REG_CAMCMR,
54
+ REG_CAMEN,
55
+
56
+ /* There are 16 CAMn[ML] registers. */
57
+ REG_CAMM_BASE,
58
+ REG_CAML_BASE,
59
+ REG_CAMML_LAST = 0x21,
60
+
61
+ REG_TXDLSA = 0x22,
62
+ REG_RXDLSA,
63
+ REG_MCMDR,
64
+ REG_MIID,
65
+ REG_MIIDA,
66
+ REG_FFTCR,
67
+ REG_TSDR,
68
+ REG_RSDR,
69
+ REG_DMARFC,
70
+ REG_MIEN,
71
+
72
+ /* Status registers. */
73
+ REG_MISTA,
74
+ REG_MGSTA,
75
+ REG_MPCNT,
76
+ REG_MRPC,
77
+ REG_MRPCC,
78
+ REG_MREPC,
79
+ REG_DMARFS,
80
+ REG_CTXDSA,
81
+ REG_CTXBSA,
82
+ REG_CRXDSA,
83
+ REG_CRXBSA,
84
+
85
+ NPCM7XX_NUM_EMC_REGS,
86
+};
87
+
88
+/* REG_CAMCMR fields */
89
+/* Enable CAM Compare */
90
+#define REG_CAMCMR_ECMP (1 << 4)
91
+/* Complement CAM Compare */
92
+#define REG_CAMCMR_CCAM (1 << 3)
93
+/* Accept Broadcast Packet */
94
+#define REG_CAMCMR_ABP (1 << 2)
95
+/* Accept Multicast Packet */
96
+#define REG_CAMCMR_AMP (1 << 1)
97
+/* Accept Unicast Packet */
98
+#define REG_CAMCMR_AUP (1 << 0)
99
+
100
+/* REG_MCMDR fields */
101
+/* Software Reset */
102
+#define REG_MCMDR_SWR (1 << 24)
103
+/* Internal Loopback Select */
104
+#define REG_MCMDR_LBK (1 << 21)
105
+/* Operation Mode Select */
106
+#define REG_MCMDR_OPMOD (1 << 20)
107
+/* Enable MDC Clock Generation */
108
+#define REG_MCMDR_ENMDC (1 << 19)
109
+/* Full-Duplex Mode Select */
110
+#define REG_MCMDR_FDUP (1 << 18)
111
+/* Enable SQE Checking */
112
+#define REG_MCMDR_ENSEQ (1 << 17)
113
+/* Send PAUSE Frame */
114
+#define REG_MCMDR_SDPZ (1 << 16)
115
+/* No Defer */
116
+#define REG_MCMDR_NDEF (1 << 9)
117
+/* Frame Transmission On */
118
+#define REG_MCMDR_TXON (1 << 8)
119
+/* Strip CRC Checksum */
120
+#define REG_MCMDR_SPCRC (1 << 5)
121
+/* Accept CRC Error Packet */
122
+#define REG_MCMDR_AEP (1 << 4)
123
+/* Accept Control Packet */
124
+#define REG_MCMDR_ACP (1 << 3)
125
+/* Accept Runt Packet */
126
+#define REG_MCMDR_ARP (1 << 2)
127
+/* Accept Long Packet */
128
+#define REG_MCMDR_ALP (1 << 1)
129
+/* Frame Reception On */
130
+#define REG_MCMDR_RXON (1 << 0)
131
+
132
+/* REG_MIEN fields */
133
+/* Enable Transmit Descriptor Unavailable Interrupt */
134
+#define REG_MIEN_ENTDU (1 << 23)
135
+/* Enable Transmit Completion Interrupt */
136
+#define REG_MIEN_ENTXCP (1 << 18)
137
+/* Enable Transmit Interrupt */
138
+#define REG_MIEN_ENTXINTR (1 << 16)
139
+/* Enable Receive Descriptor Unavailable Interrupt */
140
+#define REG_MIEN_ENRDU (1 << 10)
141
+/* Enable Receive Good Interrupt */
142
+#define REG_MIEN_ENRXGD (1 << 4)
143
+/* Enable Receive Interrupt */
144
+#define REG_MIEN_ENRXINTR (1 << 0)
145
+
146
+/* REG_MISTA fields */
147
+/* TODO: Add error fields and support simulated errors? */
148
+/* Transmit Bus Error Interrupt */
149
+#define REG_MISTA_TXBERR (1 << 24)
150
+/* Transmit Descriptor Unavailable Interrupt */
151
+#define REG_MISTA_TDU (1 << 23)
152
+/* Transmit Completion Interrupt */
153
+#define REG_MISTA_TXCP (1 << 18)
154
+/* Transmit Interrupt */
155
+#define REG_MISTA_TXINTR (1 << 16)
156
+/* Receive Bus Error Interrupt */
157
+#define REG_MISTA_RXBERR (1 << 11)
158
+/* Receive Descriptor Unavailable Interrupt */
159
+#define REG_MISTA_RDU (1 << 10)
160
+/* DMA Early Notification Interrupt */
161
+#define REG_MISTA_DENI (1 << 9)
162
+/* Maximum Frame Length Interrupt */
163
+#define REG_MISTA_DFOI (1 << 8)
164
+/* Receive Good Interrupt */
165
+#define REG_MISTA_RXGD (1 << 4)
166
+/* Packet Too Long Interrupt */
167
+#define REG_MISTA_PTLE (1 << 3)
168
+/* Receive Interrupt */
169
+#define REG_MISTA_RXINTR (1 << 0)
170
+
171
+/* REG_MGSTA fields */
172
+/* Transmission Halted */
173
+#define REG_MGSTA_TXHA (1 << 11)
174
+/* Receive Halted */
175
+#define REG_MGSTA_RXHA (1 << 11)
176
+
177
+/* REG_DMARFC fields */
178
+/* Maximum Receive Frame Length */
179
+#define REG_DMARFC_RXMS(word) extract32((word), 0, 16)
180
+
181
+/* REG MIIDA fields */
182
+/* Busy Bit */
183
+#define REG_MIIDA_BUSY (1 << 17)
184
+
185
+/* Transmit and receive descriptors */
186
+typedef struct NPCM7xxEMCTxDesc NPCM7xxEMCTxDesc;
187
+typedef struct NPCM7xxEMCRxDesc NPCM7xxEMCRxDesc;
188
+
189
+struct NPCM7xxEMCTxDesc {
190
+ uint32_t flags;
191
+ uint32_t txbsa;
192
+ uint32_t status_and_length;
193
+ uint32_t ntxdsa;
194
+};
195
+
196
+struct NPCM7xxEMCRxDesc {
197
+ uint32_t status_and_length;
198
+ uint32_t rxbsa;
199
+ uint32_t reserved;
200
+ uint32_t nrxdsa;
201
+};
202
+
203
+/* NPCM7xxEMCTxDesc.flags values */
204
+/* Owner: 0 = cpu, 1 = emc */
205
+#define TX_DESC_FLAG_OWNER_MASK (1 << 31)
206
+/* Transmit interrupt enable */
207
+#define TX_DESC_FLAG_INTEN (1 << 2)
208
+/* CRC append */
209
+#define TX_DESC_FLAG_CRCAPP (1 << 1)
210
+/* Padding enable */
211
+#define TX_DESC_FLAG_PADEN (1 << 0)
212
+
213
+/* NPCM7xxEMCTxDesc.status_and_length values */
214
+/* Collision count */
215
+#define TX_DESC_STATUS_CCNT_SHIFT 28
216
+#define TX_DESC_STATUS_CCNT_BITSIZE 4
217
+/* SQE error */
218
+#define TX_DESC_STATUS_SQE (1 << 26)
219
+/* Transmission paused */
220
+#define TX_DESC_STATUS_PAU (1 << 25)
221
+/* P transmission halted */
222
+#define TX_DESC_STATUS_TXHA (1 << 24)
223
+/* Late collision */
224
+#define TX_DESC_STATUS_LC (1 << 23)
225
+/* Transmission abort */
226
+#define TX_DESC_STATUS_TXABT (1 << 22)
227
+/* No carrier sense */
228
+#define TX_DESC_STATUS_NCS (1 << 21)
229
+/* Defer exceed */
230
+#define TX_DESC_STATUS_EXDEF (1 << 20)
231
+/* Transmission complete */
232
+#define TX_DESC_STATUS_TXCP (1 << 19)
233
+/* Transmission deferred */
234
+#define TX_DESC_STATUS_DEF (1 << 17)
235
+/* Transmit interrupt */
236
+#define TX_DESC_STATUS_TXINTR (1 << 16)
237
+
238
+#define TX_DESC_PKT_LEN(word) extract32((word), 0, 16)
239
+
240
+/* Transmit buffer start address */
241
+#define TX_DESC_TXBSA(word) ((uint32_t) (word) & ~3u)
242
+
243
+/* Next transmit descriptor start address */
244
+#define TX_DESC_NTXDSA(word) ((uint32_t) (word) & ~3u)
245
+
246
+/* NPCM7xxEMCRxDesc.status_and_length values */
247
+/* Owner: 0b00 = cpu, 0b01 = undefined, 0b10 = emc, 0b11 = undefined */
248
+#define RX_DESC_STATUS_OWNER_SHIFT 30
249
+#define RX_DESC_STATUS_OWNER_BITSIZE 2
250
+#define RX_DESC_STATUS_OWNER_MASK (3 << RX_DESC_STATUS_OWNER_SHIFT)
251
+/* Runt packet */
252
+#define RX_DESC_STATUS_RP (1 << 22)
253
+/* Alignment error */
254
+#define RX_DESC_STATUS_ALIE (1 << 21)
255
+/* Frame reception complete */
256
+#define RX_DESC_STATUS_RXGD (1 << 20)
257
+/* Packet too long */
258
+#define RX_DESC_STATUS_PTLE (1 << 19)
259
+/* CRC error */
260
+#define RX_DESC_STATUS_CRCE (1 << 17)
261
+/* Receive interrupt */
262
+#define RX_DESC_STATUS_RXINTR (1 << 16)
263
+
264
+#define RX_DESC_PKT_LEN(word) extract32((word), 0, 16)
265
+
266
+/* Receive buffer start address */
267
+#define RX_DESC_RXBSA(word) ((uint32_t) (word) & ~3u)
268
+
269
+/* Next receive descriptor start address */
270
+#define RX_DESC_NRXDSA(word) ((uint32_t) (word) & ~3u)
271
+
272
+/* Minimum packet length, when TX_DESC_FLAG_PADEN is set. */
273
+#define MIN_PACKET_LENGTH 64
274
+
275
+struct NPCM7xxEMCState {
276
+ /*< private >*/
277
+ SysBusDevice parent;
278
+ /*< public >*/
279
+
280
+ MemoryRegion iomem;
281
+
282
+ qemu_irq tx_irq;
283
+ qemu_irq rx_irq;
284
+
285
+ NICState *nic;
286
+ NICConf conf;
287
+
288
+ /* 0 or 1, for log messages */
289
+ uint8_t emc_num;
290
+
291
+ uint32_t regs[NPCM7XX_NUM_EMC_REGS];
292
+
293
+ /*
294
+ * tx is active. Set to true by TSDR and then switches off when out of
295
+ * descriptors. If the TXON bit in REG_MCMDR is off then this is off.
296
+ */
297
+ bool tx_active;
298
+
299
+ /*
300
+ * rx is active. Set to true by RSDR and then switches off when out of
301
+ * descriptors. If the RXON bit in REG_MCMDR is off then this is off.
302
+ */
303
+ bool rx_active;
304
+};
305
+
306
+typedef struct NPCM7xxEMCState NPCM7xxEMCState;
307
+
308
+#define TYPE_NPCM7XX_EMC "npcm7xx-emc"
309
+#define NPCM7XX_EMC(obj) \
310
+ OBJECT_CHECK(NPCM7xxEMCState, (obj), TYPE_NPCM7XX_EMC)
311
+
312
+#endif /* NPCM7XX_EMC_H */
313
diff --git a/hw/net/npcm7xx_emc.c b/hw/net/npcm7xx_emc.c
314
new file mode 100644
315
index XXXXXXX..XXXXXXX
316
--- /dev/null
317
+++ b/hw/net/npcm7xx_emc.c
318
@@ -XXX,XX +XXX,XX @@
319
+/*
320
+ * Nuvoton NPCM7xx EMC Module
321
+ *
322
+ * Copyright 2020 Google LLC
323
+ *
324
+ * This program is free software; you can redistribute it and/or modify it
325
+ * under the terms of the GNU General Public License as published by the
326
+ * Free Software Foundation; either version 2 of the License, or
327
+ * (at your option) any later version.
328
+ *
329
+ * This program is distributed in the hope that it will be useful, but WITHOUT
330
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
331
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
332
+ * for more details.
333
+ *
334
+ * Unsupported/unimplemented features:
335
+ * - MCMDR.FDUP (full duplex) is ignored, half duplex is not supported
336
+ * - Only CAM0 is supported, CAM[1-15] are not
337
+ * - writes to CAMEN.[1-15] are ignored, these bits always read as zeroes
338
+ * - MII is not implemented, MIIDA.BUSY and MIID always return zero
339
+ * - MCMDR.LBK is not implemented
340
+ * - MCMDR.{OPMOD,ENSQE,AEP,ARP} are not supported
341
+ * - H/W FIFOs are not supported, MCMDR.FFTCR is ignored
342
+ * - MGSTA.SQE is not supported
343
+ * - pause and control frames are not implemented
344
+ * - MGSTA.CCNT is not supported
345
+ * - MPCNT, DMARFS are not implemented
346
+ */
347
+
348
+#include "qemu/osdep.h"
349
+
350
+/* For crc32 */
351
+#include <zlib.h>
352
+
353
+#include "qemu-common.h"
354
+#include "hw/irq.h"
355
+#include "hw/qdev-clock.h"
356
+#include "hw/qdev-properties.h"
357
+#include "hw/net/npcm7xx_emc.h"
358
+#include "net/eth.h"
359
+#include "migration/vmstate.h"
360
+#include "qemu/bitops.h"
361
+#include "qemu/error-report.h"
362
+#include "qemu/log.h"
363
+#include "qemu/module.h"
364
+#include "qemu/units.h"
365
+#include "sysemu/dma.h"
366
+#include "trace.h"
367
+
368
+#define CRC_LENGTH 4
369
+
370
+/*
371
+ * The maximum size of a (layer 2) ethernet frame as defined by 802.3.
372
+ * 1518 = 6(dest macaddr) + 6(src macaddr) + 2(proto) + 4(crc) + 1500(payload)
373
+ * This does not include an additional 4 for the vlan field (802.1q).
374
+ */
375
+#define MAX_ETH_FRAME_SIZE 1518
376
+
377
+static const char *emc_reg_name(int regno)
378
+{
379
+#define REG(name) case REG_ ## name: return #name;
380
+ switch (regno) {
381
+ REG(CAMCMR)
382
+ REG(CAMEN)
383
+ REG(TXDLSA)
384
+ REG(RXDLSA)
385
+ REG(MCMDR)
386
+ REG(MIID)
387
+ REG(MIIDA)
388
+ REG(FFTCR)
389
+ REG(TSDR)
390
+ REG(RSDR)
391
+ REG(DMARFC)
392
+ REG(MIEN)
393
+ REG(MISTA)
394
+ REG(MGSTA)
395
+ REG(MPCNT)
396
+ REG(MRPC)
397
+ REG(MRPCC)
398
+ REG(MREPC)
399
+ REG(DMARFS)
400
+ REG(CTXDSA)
401
+ REG(CTXBSA)
402
+ REG(CRXDSA)
403
+ REG(CRXBSA)
404
+ case REG_CAMM_BASE + 0: return "CAM0M";
405
+ case REG_CAML_BASE + 0: return "CAM0L";
406
+ case REG_CAMM_BASE + 2 ... REG_CAMML_LAST:
407
+ /* Only CAM0 is supported, fold the others into something simple. */
408
+ if (regno & 1) {
409
+ return "CAM<n>L";
410
+ } else {
411
+ return "CAM<n>M";
412
+ }
413
+ default: return "UNKNOWN";
414
+ }
415
+#undef REG
416
+}
417
+
418
+static void emc_reset(NPCM7xxEMCState *emc)
419
+{
420
+ trace_npcm7xx_emc_reset(emc->emc_num);
421
+
422
+ memset(&emc->regs[0], 0, sizeof(emc->regs));
423
+
424
+ /* These regs have non-zero reset values. */
425
+ emc->regs[REG_TXDLSA] = 0xfffffffc;
426
+ emc->regs[REG_RXDLSA] = 0xfffffffc;
427
+ emc->regs[REG_MIIDA] = 0x00900000;
428
+ emc->regs[REG_FFTCR] = 0x0101;
429
+ emc->regs[REG_DMARFC] = 0x0800;
430
+ emc->regs[REG_MPCNT] = 0x7fff;
431
+
432
+ emc->tx_active = false;
433
+ emc->rx_active = false;
434
+}
435
+
436
+static void npcm7xx_emc_reset(DeviceState *dev)
437
+{
438
+ NPCM7xxEMCState *emc = NPCM7XX_EMC(dev);
439
+ emc_reset(emc);
440
+}
441
+
442
+static void emc_soft_reset(NPCM7xxEMCState *emc)
443
+{
444
+ /*
445
+ * The docs say at least MCMDR.{LBK,OPMOD} bits are not changed during a
446
+ * soft reset, but does not go into further detail. For now, KISS.
447
+ */
448
+ uint32_t mcmdr = emc->regs[REG_MCMDR];
449
+ emc_reset(emc);
450
+ emc->regs[REG_MCMDR] = mcmdr & (REG_MCMDR_LBK | REG_MCMDR_OPMOD);
451
+
452
+ qemu_set_irq(emc->tx_irq, 0);
453
+ qemu_set_irq(emc->rx_irq, 0);
454
+}
455
+
456
+static void emc_set_link(NetClientState *nc)
457
+{
458
+ /* Nothing to do yet. */
459
+}
460
+
461
+/* MISTA.TXINTR is the union of the individual bits with their enables. */
462
+static void emc_update_mista_txintr(NPCM7xxEMCState *emc)
463
+{
464
+ /* Only look at the bits we support. */
465
+ uint32_t mask = (REG_MISTA_TXBERR |
466
+ REG_MISTA_TDU |
467
+ REG_MISTA_TXCP);
468
+ if (emc->regs[REG_MISTA] & emc->regs[REG_MIEN] & mask) {
469
+ emc->regs[REG_MISTA] |= REG_MISTA_TXINTR;
470
+ } else {
471
+ emc->regs[REG_MISTA] &= ~REG_MISTA_TXINTR;
472
+ }
473
+}
474
+
475
+/* MISTA.RXINTR is the union of the individual bits with their enables. */
476
+static void emc_update_mista_rxintr(NPCM7xxEMCState *emc)
477
+{
478
+ /* Only look at the bits we support. */
479
+ uint32_t mask = (REG_MISTA_RXBERR |
480
+ REG_MISTA_RDU |
481
+ REG_MISTA_RXGD);
482
+ if (emc->regs[REG_MISTA] & emc->regs[REG_MIEN] & mask) {
483
+ emc->regs[REG_MISTA] |= REG_MISTA_RXINTR;
484
+ } else {
485
+ emc->regs[REG_MISTA] &= ~REG_MISTA_RXINTR;
486
+ }
487
+}
488
+
489
+/* N.B. emc_update_mista_txintr must have already been called. */
490
+static void emc_update_tx_irq(NPCM7xxEMCState *emc)
491
+{
492
+ int level = !!(emc->regs[REG_MISTA] &
493
+ emc->regs[REG_MIEN] &
494
+ REG_MISTA_TXINTR);
495
+ trace_npcm7xx_emc_update_tx_irq(level);
496
+ qemu_set_irq(emc->tx_irq, level);
497
+}
498
+
499
+/* N.B. emc_update_mista_rxintr must have already been called. */
500
+static void emc_update_rx_irq(NPCM7xxEMCState *emc)
501
+{
502
+ int level = !!(emc->regs[REG_MISTA] &
503
+ emc->regs[REG_MIEN] &
504
+ REG_MISTA_RXINTR);
505
+ trace_npcm7xx_emc_update_rx_irq(level);
506
+ qemu_set_irq(emc->rx_irq, level);
507
+}
508
+
509
+/* Update IRQ states due to changes in MIEN,MISTA. */
510
+static void emc_update_irq_from_reg_change(NPCM7xxEMCState *emc)
511
+{
512
+ emc_update_mista_txintr(emc);
513
+ emc_update_tx_irq(emc);
514
+
515
+ emc_update_mista_rxintr(emc);
516
+ emc_update_rx_irq(emc);
517
+}
518
+
519
+static int emc_read_tx_desc(dma_addr_t addr, NPCM7xxEMCTxDesc *desc)
520
+{
521
+ if (dma_memory_read(&address_space_memory, addr, desc, sizeof(*desc))) {
522
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Failed to read descriptor @ 0x%"
523
+ HWADDR_PRIx "\n", __func__, addr);
524
+ return -1;
525
+ }
526
+ desc->flags = le32_to_cpu(desc->flags);
527
+ desc->txbsa = le32_to_cpu(desc->txbsa);
528
+ desc->status_and_length = le32_to_cpu(desc->status_and_length);
529
+ desc->ntxdsa = le32_to_cpu(desc->ntxdsa);
530
+ return 0;
531
+}
532
+
533
+static int emc_write_tx_desc(const NPCM7xxEMCTxDesc *desc, dma_addr_t addr)
534
+{
535
+ NPCM7xxEMCTxDesc le_desc;
536
+
537
+ le_desc.flags = cpu_to_le32(desc->flags);
538
+ le_desc.txbsa = cpu_to_le32(desc->txbsa);
539
+ le_desc.status_and_length = cpu_to_le32(desc->status_and_length);
540
+ le_desc.ntxdsa = cpu_to_le32(desc->ntxdsa);
541
+ if (dma_memory_write(&address_space_memory, addr, &le_desc,
542
+ sizeof(le_desc))) {
543
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Failed to write descriptor @ 0x%"
544
+ HWADDR_PRIx "\n", __func__, addr);
545
+ return -1;
546
+ }
547
+ return 0;
548
+}
549
+
550
+static int emc_read_rx_desc(dma_addr_t addr, NPCM7xxEMCRxDesc *desc)
551
+{
552
+ if (dma_memory_read(&address_space_memory, addr, desc, sizeof(*desc))) {
553
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Failed to read descriptor @ 0x%"
554
+ HWADDR_PRIx "\n", __func__, addr);
555
+ return -1;
556
+ }
557
+ desc->status_and_length = le32_to_cpu(desc->status_and_length);
558
+ desc->rxbsa = le32_to_cpu(desc->rxbsa);
559
+ desc->reserved = le32_to_cpu(desc->reserved);
560
+ desc->nrxdsa = le32_to_cpu(desc->nrxdsa);
561
+ return 0;
562
+}
563
+
564
+static int emc_write_rx_desc(const NPCM7xxEMCRxDesc *desc, dma_addr_t addr)
565
+{
566
+ NPCM7xxEMCRxDesc le_desc;
567
+
568
+ le_desc.status_and_length = cpu_to_le32(desc->status_and_length);
569
+ le_desc.rxbsa = cpu_to_le32(desc->rxbsa);
570
+ le_desc.reserved = cpu_to_le32(desc->reserved);
571
+ le_desc.nrxdsa = cpu_to_le32(desc->nrxdsa);
572
+ if (dma_memory_write(&address_space_memory, addr, &le_desc,
573
+ sizeof(le_desc))) {
574
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Failed to write descriptor @ 0x%"
575
+ HWADDR_PRIx "\n", __func__, addr);
576
+ return -1;
577
+ }
578
+ return 0;
579
+}
580
+
581
+static void emc_set_mista(NPCM7xxEMCState *emc, uint32_t flags)
582
+{
583
+ trace_npcm7xx_emc_set_mista(flags);
584
+ emc->regs[REG_MISTA] |= flags;
585
+ if (extract32(flags, 16, 16)) {
586
+ emc_update_mista_txintr(emc);
587
+ }
588
+ if (extract32(flags, 0, 16)) {
589
+ emc_update_mista_rxintr(emc);
590
+ }
591
+}
592
+
593
+static void emc_halt_tx(NPCM7xxEMCState *emc, uint32_t mista_flag)
594
+{
595
+ emc->tx_active = false;
596
+ emc_set_mista(emc, mista_flag);
597
+}
598
+
599
+static void emc_halt_rx(NPCM7xxEMCState *emc, uint32_t mista_flag)
600
+{
601
+ emc->rx_active = false;
602
+ emc_set_mista(emc, mista_flag);
603
+}
604
+
605
+static void emc_set_next_tx_descriptor(NPCM7xxEMCState *emc,
606
+ const NPCM7xxEMCTxDesc *tx_desc,
607
+ uint32_t desc_addr)
608
+{
609
+ /* Update the current descriptor, if only to reset the owner flag. */
610
+ if (emc_write_tx_desc(tx_desc, desc_addr)) {
611
+ /*
612
+ * We just read it so this shouldn't generally happen.
613
+ * Error already reported.
614
+ */
615
+ emc_set_mista(emc, REG_MISTA_TXBERR);
616
+ }
617
+ emc->regs[REG_CTXDSA] = TX_DESC_NTXDSA(tx_desc->ntxdsa);
618
+}
619
+
620
+static void emc_set_next_rx_descriptor(NPCM7xxEMCState *emc,
621
+ const NPCM7xxEMCRxDesc *rx_desc,
622
+ uint32_t desc_addr)
623
+{
624
+ /* Update the current descriptor, if only to reset the owner flag. */
625
+ if (emc_write_rx_desc(rx_desc, desc_addr)) {
626
+ /*
627
+ * We just read it so this shouldn't generally happen.
628
+ * Error already reported.
629
+ */
630
+ emc_set_mista(emc, REG_MISTA_RXBERR);
631
+ }
632
+ emc->regs[REG_CRXDSA] = RX_DESC_NRXDSA(rx_desc->nrxdsa);
633
+}
634
+
635
+static void emc_try_send_next_packet(NPCM7xxEMCState *emc)
636
+{
637
+ /* Working buffer for sending out packets. Most packets fit in this. */
638
+#define TX_BUFFER_SIZE 2048
639
+ uint8_t tx_send_buffer[TX_BUFFER_SIZE];
640
+ uint32_t desc_addr = TX_DESC_NTXDSA(emc->regs[REG_CTXDSA]);
641
+ NPCM7xxEMCTxDesc tx_desc;
642
+ uint32_t next_buf_addr, length;
643
+ uint8_t *buf;
644
+ g_autofree uint8_t *malloced_buf = NULL;
645
+
646
+ if (emc_read_tx_desc(desc_addr, &tx_desc)) {
647
+ /* Error reading descriptor, already reported. */
648
+ emc_halt_tx(emc, REG_MISTA_TXBERR);
649
+ emc_update_tx_irq(emc);
650
+ return;
44
+ return;
651
+ }
45
+ }
652
+
46
if (!sysbus_realize(SYS_BUS_DEVICE(&s->qspi_dma), errp)) {
653
+ /* Nothing we can do if we don't own the descriptor. */
47
return;
654
+ if (!(tx_desc.flags & TX_DESC_FLAG_OWNER_MASK)) {
48
}
655
+ trace_npcm7xx_emc_cpu_owned_desc(desc_addr);
49
diff --git a/hw/dma/xlnx_csu_dma.c b/hw/dma/xlnx_csu_dma.c
656
+ emc_halt_tx(emc, REG_MISTA_TDU);
50
index XXXXXXX..XXXXXXX 100644
657
+ emc_update_tx_irq(emc);
51
--- a/hw/dma/xlnx_csu_dma.c
658
+ return;
52
+++ b/hw/dma/xlnx_csu_dma.c
659
+ }
53
@@ -XXX,XX +XXX,XX @@ static uint32_t xlnx_csu_dma_read(XlnxCSUDMA *s, uint8_t *buf, uint32_t len)
660
+
54
for (i = 0; i < len && (result == MEMTX_OK); i += s->width) {
661
+ /* Give the descriptor back regardless of what happens. */
55
uint32_t mlen = MIN(len - i, s->width);
662
+ tx_desc.flags &= ~TX_DESC_FLAG_OWNER_MASK;
56
663
+ tx_desc.status_and_length &= 0xffff;
57
- result = address_space_rw(s->dma_as, addr, s->attr,
664
+
58
+ result = address_space_rw(&s->dma_as, addr, s->attr,
665
+ /*
59
buf + i, mlen, false);
666
+ * Despite the h/w documentation saying the tx buffer is word aligned,
60
}
667
+ * the linux driver does not word align the buffer. There is value in not
61
} else {
668
+ * aligning the buffer: See the description of NET_IP_ALIGN in linux
62
- result = address_space_rw(s->dma_as, addr, s->attr, buf, len, false);
669
+ * kernel sources.
63
+ result = address_space_rw(&s->dma_as, addr, s->attr, buf, len, false);
670
+ */
64
}
671
+ next_buf_addr = tx_desc.txbsa;
65
672
+ emc->regs[REG_CTXBSA] = next_buf_addr;
66
if (result == MEMTX_OK) {
673
+ length = TX_DESC_PKT_LEN(tx_desc.status_and_length);
67
@@ -XXX,XX +XXX,XX @@ static uint32_t xlnx_csu_dma_write(XlnxCSUDMA *s, uint8_t *buf, uint32_t len)
674
+ buf = &tx_send_buffer[0];
68
for (i = 0; i < len && (result == MEMTX_OK); i += s->width) {
675
+
69
uint32_t mlen = MIN(len - i, s->width);
676
+ if (length > sizeof(tx_send_buffer)) {
70
677
+ malloced_buf = g_malloc(length);
71
- result = address_space_rw(s->dma_as, addr, s->attr,
678
+ buf = malloced_buf;
72
+ result = address_space_rw(&s->dma_as, addr, s->attr,
679
+ }
73
buf, mlen, true);
680
+
74
buf += mlen;
681
+ if (dma_memory_read(&address_space_memory, next_buf_addr, buf, length)) {
75
}
682
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Failed to read packet @ 0x%x\n",
76
} else {
683
+ __func__, next_buf_addr);
77
- result = address_space_rw(s->dma_as, addr, s->attr, buf, len, true);
684
+ emc_set_mista(emc, REG_MISTA_TXBERR);
78
+ result = address_space_rw(&s->dma_as, addr, s->attr, buf, len, true);
685
+ emc_set_next_tx_descriptor(emc, &tx_desc, desc_addr);
79
}
686
+ emc_update_tx_irq(emc);
80
687
+ trace_npcm7xx_emc_tx_done(emc->regs[REG_CTXDSA]);
81
if (result != MEMTX_OK) {
82
@@ -XXX,XX +XXX,XX @@ static void xlnx_csu_dma_realize(DeviceState *dev, Error **errp)
83
return;
84
}
85
86
+ if (!s->dma_mr) {
87
+ error_setg(errp, TYPE_XLNX_CSU_DMA " 'dma' link not set");
688
+ return;
88
+ return;
689
+ }
89
+ }
90
+ address_space_init(&s->dma_as, s->dma_mr, "csu-dma");
690
+
91
+
691
+ if ((tx_desc.flags & TX_DESC_FLAG_PADEN) && (length < MIN_PACKET_LENGTH)) {
92
reg_array =
692
+ memset(buf + length, 0, MIN_PACKET_LENGTH - length);
93
register_init_block32(dev, xlnx_csu_dma_regs_info[!!s->is_dst],
693
+ length = MIN_PACKET_LENGTH;
94
XLNX_CSU_DMA_R_MAX,
694
+ }
95
@@ -XXX,XX +XXX,XX @@ static void xlnx_csu_dma_realize(DeviceState *dev, Error **errp)
695
+
96
s->src_timer = ptimer_init(xlnx_csu_dma_src_timeout_hit,
696
+ /* N.B. emc_receive can get called here. */
97
s, PTIMER_POLICY_DEFAULT);
697
+ qemu_send_packet(qemu_get_queue(emc->nic), buf, length);
98
698
+ trace_npcm7xx_emc_sent_packet(length);
99
- if (s->dma_mr) {
699
+
100
- s->dma_as = g_malloc0(sizeof(AddressSpace));
700
+ tx_desc.status_and_length |= TX_DESC_STATUS_TXCP;
101
- address_space_init(s->dma_as, s->dma_mr, NULL);
701
+ if (tx_desc.flags & TX_DESC_FLAG_INTEN) {
102
- } else {
702
+ emc_set_mista(emc, REG_MISTA_TXCP);
103
- s->dma_as = &address_space_memory;
703
+ }
104
- }
704
+ if (emc->regs[REG_MISTA] & emc->regs[REG_MIEN] & REG_MISTA_TXINTR) {
105
-
705
+ tx_desc.status_and_length |= TX_DESC_STATUS_TXINTR;
106
s->attr = MEMTXATTRS_UNSPECIFIED;
706
+ }
107
707
+
108
s->r_size_last_word = 0;
708
+ emc_set_next_tx_descriptor(emc, &tx_desc, desc_addr);
709
+ emc_update_tx_irq(emc);
710
+ trace_npcm7xx_emc_tx_done(emc->regs[REG_CTXDSA]);
711
+}
712
+
713
+static bool emc_can_receive(NetClientState *nc)
714
+{
715
+ NPCM7xxEMCState *emc = NPCM7XX_EMC(qemu_get_nic_opaque(nc));
716
+
717
+ bool can_receive = emc->rx_active;
718
+ trace_npcm7xx_emc_can_receive(can_receive);
719
+ return can_receive;
720
+}
721
+
722
+/* If result is false then *fail_reason contains the reason. */
723
+static bool emc_receive_filter1(NPCM7xxEMCState *emc, const uint8_t *buf,
724
+ size_t len, const char **fail_reason)
725
+{
726
+ eth_pkt_types_e pkt_type = get_eth_packet_type(PKT_GET_ETH_HDR(buf));
727
+
728
+ switch (pkt_type) {
729
+ case ETH_PKT_BCAST:
730
+ if (emc->regs[REG_CAMCMR] & REG_CAMCMR_CCAM) {
731
+ return true;
732
+ } else {
733
+ *fail_reason = "Broadcast packet disabled";
734
+ return !!(emc->regs[REG_CAMCMR] & REG_CAMCMR_ABP);
735
+ }
736
+ case ETH_PKT_MCAST:
737
+ if (emc->regs[REG_CAMCMR] & REG_CAMCMR_CCAM) {
738
+ return true;
739
+ } else {
740
+ *fail_reason = "Multicast packet disabled";
741
+ return !!(emc->regs[REG_CAMCMR] & REG_CAMCMR_AMP);
742
+ }
743
+ case ETH_PKT_UCAST: {
744
+ bool matches;
745
+ if (emc->regs[REG_CAMCMR] & REG_CAMCMR_AUP) {
746
+ return true;
747
+ }
748
+ matches = ((emc->regs[REG_CAMCMR] & REG_CAMCMR_ECMP) &&
749
+ /* We only support one CAM register, CAM0. */
750
+ (emc->regs[REG_CAMEN] & (1 << 0)) &&
751
+ memcmp(buf, emc->conf.macaddr.a, ETH_ALEN) == 0);
752
+ if (emc->regs[REG_CAMCMR] & REG_CAMCMR_CCAM) {
753
+ *fail_reason = "MACADDR matched, comparison complemented";
754
+ return !matches;
755
+ } else {
756
+ *fail_reason = "MACADDR didn't match";
757
+ return matches;
758
+ }
759
+ }
760
+ default:
761
+ g_assert_not_reached();
762
+ }
763
+}
764
+
765
+static bool emc_receive_filter(NPCM7xxEMCState *emc, const uint8_t *buf,
766
+ size_t len)
767
+{
768
+ const char *fail_reason = NULL;
769
+ bool ok = emc_receive_filter1(emc, buf, len, &fail_reason);
770
+ if (!ok) {
771
+ trace_npcm7xx_emc_packet_filtered_out(fail_reason);
772
+ }
773
+ return ok;
774
+}
775
+
776
+static ssize_t emc_receive(NetClientState *nc, const uint8_t *buf, size_t len1)
777
+{
778
+ NPCM7xxEMCState *emc = NPCM7XX_EMC(qemu_get_nic_opaque(nc));
779
+ const uint32_t len = len1;
780
+ size_t max_frame_len;
781
+ bool long_frame;
782
+ uint32_t desc_addr;
783
+ NPCM7xxEMCRxDesc rx_desc;
784
+ uint32_t crc;
785
+ uint8_t *crc_ptr;
786
+ uint32_t buf_addr;
787
+
788
+ trace_npcm7xx_emc_receiving_packet(len);
789
+
790
+ if (!emc_can_receive(nc)) {
791
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Unexpected packet\n", __func__);
792
+ return -1;
793
+ }
794
+
795
+ if (len < ETH_HLEN ||
796
+ /* Defensive programming: drop unsupportable large packets. */
797
+ len > 0xffff - CRC_LENGTH) {
798
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Dropped frame of %u bytes\n",
799
+ __func__, len);
800
+ return len;
801
+ }
802
+
803
+ /*
804
+ * DENI is set if EMC received the Length/Type field of the incoming
805
+ * packet, so it will be set regardless of what happens next.
806
+ */
807
+ emc_set_mista(emc, REG_MISTA_DENI);
808
+
809
+ if (!emc_receive_filter(emc, buf, len)) {
810
+ emc_update_rx_irq(emc);
811
+ return len;
812
+ }
813
+
814
+ /* Huge frames (> DMARFC) are dropped. */
815
+ max_frame_len = REG_DMARFC_RXMS(emc->regs[REG_DMARFC]);
816
+ if (len + CRC_LENGTH > max_frame_len) {
817
+ trace_npcm7xx_emc_packet_dropped(len);
818
+ emc_set_mista(emc, REG_MISTA_DFOI);
819
+ emc_update_rx_irq(emc);
820
+ return len;
821
+ }
822
+
823
+ /*
824
+ * Long Frames (> MAX_ETH_FRAME_SIZE) are also dropped, unless MCMDR.ALP
825
+ * is set.
826
+ */
827
+ long_frame = false;
828
+ if (len + CRC_LENGTH > MAX_ETH_FRAME_SIZE) {
829
+ if (emc->regs[REG_MCMDR] & REG_MCMDR_ALP) {
830
+ long_frame = true;
831
+ } else {
832
+ trace_npcm7xx_emc_packet_dropped(len);
833
+ emc_set_mista(emc, REG_MISTA_PTLE);
834
+ emc_update_rx_irq(emc);
835
+ return len;
836
+ }
837
+ }
838
+
839
+ desc_addr = RX_DESC_NRXDSA(emc->regs[REG_CRXDSA]);
840
+ if (emc_read_rx_desc(desc_addr, &rx_desc)) {
841
+ /* Error reading descriptor, already reported. */
842
+ emc_halt_rx(emc, REG_MISTA_RXBERR);
843
+ emc_update_rx_irq(emc);
844
+ return len;
845
+ }
846
+
847
+ /* Nothing we can do if we don't own the descriptor. */
848
+ if (!(rx_desc.status_and_length & RX_DESC_STATUS_OWNER_MASK)) {
849
+ trace_npcm7xx_emc_cpu_owned_desc(desc_addr);
850
+ emc_halt_rx(emc, REG_MISTA_RDU);
851
+ emc_update_rx_irq(emc);
852
+ return len;
853
+ }
854
+
855
+ crc = 0;
856
+ crc_ptr = (uint8_t *) &crc;
857
+ if (!(emc->regs[REG_MCMDR] & REG_MCMDR_SPCRC)) {
858
+ crc = cpu_to_be32(crc32(~0, buf, len));
859
+ }
860
+
861
+ /* Give the descriptor back regardless of what happens. */
862
+ rx_desc.status_and_length &= ~RX_DESC_STATUS_OWNER_MASK;
863
+
864
+ buf_addr = rx_desc.rxbsa;
865
+ emc->regs[REG_CRXBSA] = buf_addr;
866
+ if (dma_memory_write(&address_space_memory, buf_addr, buf, len) ||
867
+ (!(emc->regs[REG_MCMDR] & REG_MCMDR_SPCRC) &&
868
+ dma_memory_write(&address_space_memory, buf_addr + len, crc_ptr,
869
+ 4))) {
870
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: Bus error writing packet\n",
871
+ __func__);
872
+ emc_set_mista(emc, REG_MISTA_RXBERR);
873
+ emc_set_next_rx_descriptor(emc, &rx_desc, desc_addr);
874
+ emc_update_rx_irq(emc);
875
+ trace_npcm7xx_emc_rx_done(emc->regs[REG_CRXDSA]);
876
+ return len;
877
+ }
878
+
879
+ trace_npcm7xx_emc_received_packet(len);
880
+
881
+ /* Note: We've already verified len+4 <= 0xffff. */
882
+ rx_desc.status_and_length = len;
883
+ if (!(emc->regs[REG_MCMDR] & REG_MCMDR_SPCRC)) {
884
+ rx_desc.status_and_length += 4;
885
+ }
886
+ rx_desc.status_and_length |= RX_DESC_STATUS_RXGD;
887
+ emc_set_mista(emc, REG_MISTA_RXGD);
888
+
889
+ if (emc->regs[REG_MISTA] & emc->regs[REG_MIEN] & REG_MISTA_RXINTR) {
890
+ rx_desc.status_and_length |= RX_DESC_STATUS_RXINTR;
891
+ }
892
+ if (long_frame) {
893
+ rx_desc.status_and_length |= RX_DESC_STATUS_PTLE;
894
+ }
895
+
896
+ emc_set_next_rx_descriptor(emc, &rx_desc, desc_addr);
897
+ emc_update_rx_irq(emc);
898
+ trace_npcm7xx_emc_rx_done(emc->regs[REG_CRXDSA]);
899
+ return len;
900
+}
901
+
902
+static void emc_try_receive_next_packet(NPCM7xxEMCState *emc)
903
+{
904
+ if (emc_can_receive(qemu_get_queue(emc->nic))) {
905
+ qemu_flush_queued_packets(qemu_get_queue(emc->nic));
906
+ }
907
+}
908
+
909
+static uint64_t npcm7xx_emc_read(void *opaque, hwaddr offset, unsigned size)
910
+{
911
+ NPCM7xxEMCState *emc = opaque;
912
+ uint32_t reg = offset / sizeof(uint32_t);
913
+ uint32_t result;
914
+
915
+ if (reg >= NPCM7XX_NUM_EMC_REGS) {
916
+ qemu_log_mask(LOG_GUEST_ERROR,
917
+ "%s: Invalid offset 0x%04" HWADDR_PRIx "\n",
918
+ __func__, offset);
919
+ return 0;
920
+ }
921
+
922
+ switch (reg) {
923
+ case REG_MIID:
924
+ /*
925
+ * We don't implement MII. For determinism, always return zero as
926
+ * writes record the last value written for debugging purposes.
927
+ */
928
+ qemu_log_mask(LOG_UNIMP, "%s: Read of MIID, returning 0\n", __func__);
929
+ result = 0;
930
+ break;
931
+ case REG_TSDR:
932
+ case REG_RSDR:
933
+ qemu_log_mask(LOG_GUEST_ERROR,
934
+ "%s: Read of write-only reg, %s/%d\n",
935
+ __func__, emc_reg_name(reg), reg);
936
+ return 0;
937
+ default:
938
+ result = emc->regs[reg];
939
+ break;
940
+ }
941
+
942
+ trace_npcm7xx_emc_reg_read(emc->emc_num, result, emc_reg_name(reg), reg);
943
+ return result;
944
+}
945
+
946
+static void npcm7xx_emc_write(void *opaque, hwaddr offset,
947
+ uint64_t v, unsigned size)
948
+{
949
+ NPCM7xxEMCState *emc = opaque;
950
+ uint32_t reg = offset / sizeof(uint32_t);
951
+ uint32_t value = v;
952
+
953
+ g_assert(size == sizeof(uint32_t));
954
+
955
+ if (reg >= NPCM7XX_NUM_EMC_REGS) {
956
+ qemu_log_mask(LOG_GUEST_ERROR,
957
+ "%s: Invalid offset 0x%04" HWADDR_PRIx "\n",
958
+ __func__, offset);
959
+ return;
960
+ }
961
+
962
+ trace_npcm7xx_emc_reg_write(emc->emc_num, emc_reg_name(reg), reg, value);
963
+
964
+ switch (reg) {
965
+ case REG_CAMCMR:
966
+ emc->regs[reg] = value;
967
+ break;
968
+ case REG_CAMEN:
969
+ /* Only CAM0 is supported, don't pretend otherwise. */
970
+ if (value & ~1) {
971
+ qemu_log_mask(LOG_GUEST_ERROR,
972
+ "%s: Only CAM0 is supported, cannot enable others"
973
+ ": 0x%x\n",
974
+ __func__, value);
975
+ }
976
+ emc->regs[reg] = value & 1;
977
+ break;
978
+ case REG_CAMM_BASE + 0:
979
+ emc->regs[reg] = value;
980
+ emc->conf.macaddr.a[0] = value >> 24;
981
+ emc->conf.macaddr.a[1] = value >> 16;
982
+ emc->conf.macaddr.a[2] = value >> 8;
983
+ emc->conf.macaddr.a[3] = value >> 0;
984
+ break;
985
+ case REG_CAML_BASE + 0:
986
+ emc->regs[reg] = value;
987
+ emc->conf.macaddr.a[4] = value >> 24;
988
+ emc->conf.macaddr.a[5] = value >> 16;
989
+ break;
990
+ case REG_MCMDR: {
991
+ uint32_t prev;
992
+ if (value & REG_MCMDR_SWR) {
993
+ emc_soft_reset(emc);
994
+ /* On h/w the reset happens over multiple cycles. For now KISS. */
995
+ break;
996
+ }
997
+ prev = emc->regs[reg];
998
+ emc->regs[reg] = value;
999
+ /* Update tx state. */
1000
+ if (!(prev & REG_MCMDR_TXON) &&
1001
+ (value & REG_MCMDR_TXON)) {
1002
+ emc->regs[REG_CTXDSA] = emc->regs[REG_TXDLSA];
1003
+ /*
1004
+ * Linux kernel turns TX on with CPU still holding descriptor,
1005
+ * which suggests we should wait for a write to TSDR before trying
1006
+ * to send a packet: so we don't send one here.
1007
+ */
1008
+ } else if ((prev & REG_MCMDR_TXON) &&
1009
+ !(value & REG_MCMDR_TXON)) {
1010
+ emc->regs[REG_MGSTA] |= REG_MGSTA_TXHA;
1011
+ }
1012
+ if (!(value & REG_MCMDR_TXON)) {
1013
+ emc_halt_tx(emc, 0);
1014
+ }
1015
+ /* Update rx state. */
1016
+ if (!(prev & REG_MCMDR_RXON) &&
1017
+ (value & REG_MCMDR_RXON)) {
1018
+ emc->regs[REG_CRXDSA] = emc->regs[REG_RXDLSA];
1019
+ } else if ((prev & REG_MCMDR_RXON) &&
1020
+ !(value & REG_MCMDR_RXON)) {
1021
+ emc->regs[REG_MGSTA] |= REG_MGSTA_RXHA;
1022
+ }
1023
+ if (!(value & REG_MCMDR_RXON)) {
1024
+ emc_halt_rx(emc, 0);
1025
+ }
1026
+ break;
1027
+ }
1028
+ case REG_TXDLSA:
1029
+ case REG_RXDLSA:
1030
+ case REG_DMARFC:
1031
+ case REG_MIID:
1032
+ emc->regs[reg] = value;
1033
+ break;
1034
+ case REG_MIEN:
1035
+ emc->regs[reg] = value;
1036
+ emc_update_irq_from_reg_change(emc);
1037
+ break;
1038
+ case REG_MISTA:
1039
+ /* Clear the bits that have 1 in "value". */
1040
+ emc->regs[reg] &= ~value;
1041
+ emc_update_irq_from_reg_change(emc);
1042
+ break;
1043
+ case REG_MGSTA:
1044
+ /* Clear the bits that have 1 in "value". */
1045
+ emc->regs[reg] &= ~value;
1046
+ break;
1047
+ case REG_TSDR:
1048
+ if (emc->regs[REG_MCMDR] & REG_MCMDR_TXON) {
1049
+ emc->tx_active = true;
1050
+ /* Keep trying to send packets until we run out. */
1051
+ while (emc->tx_active) {
1052
+ emc_try_send_next_packet(emc);
1053
+ }
1054
+ }
1055
+ break;
1056
+ case REG_RSDR:
1057
+ if (emc->regs[REG_MCMDR] & REG_MCMDR_RXON) {
1058
+ emc->rx_active = true;
1059
+ emc_try_receive_next_packet(emc);
1060
+ }
1061
+ break;
1062
+ case REG_MIIDA:
1063
+ emc->regs[reg] = value & ~REG_MIIDA_BUSY;
1064
+ break;
1065
+ case REG_MRPC:
1066
+ case REG_MRPCC:
1067
+ case REG_MREPC:
1068
+ case REG_CTXDSA:
1069
+ case REG_CTXBSA:
1070
+ case REG_CRXDSA:
1071
+ case REG_CRXBSA:
1072
+ qemu_log_mask(LOG_GUEST_ERROR,
1073
+ "%s: Write to read-only reg %s/%d\n",
1074
+ __func__, emc_reg_name(reg), reg);
1075
+ break;
1076
+ default:
1077
+ qemu_log_mask(LOG_UNIMP, "%s: Write to unimplemented reg %s/%d\n",
1078
+ __func__, emc_reg_name(reg), reg);
1079
+ break;
1080
+ }
1081
+}
1082
+
1083
+static const struct MemoryRegionOps npcm7xx_emc_ops = {
1084
+ .read = npcm7xx_emc_read,
1085
+ .write = npcm7xx_emc_write,
1086
+ .endianness = DEVICE_LITTLE_ENDIAN,
1087
+ .valid = {
1088
+ .min_access_size = 4,
1089
+ .max_access_size = 4,
1090
+ .unaligned = false,
1091
+ },
1092
+};
1093
+
1094
+static void emc_cleanup(NetClientState *nc)
1095
+{
1096
+ /* Nothing to do yet. */
1097
+}
1098
+
1099
+static NetClientInfo net_npcm7xx_emc_info = {
1100
+ .type = NET_CLIENT_DRIVER_NIC,
1101
+ .size = sizeof(NICState),
1102
+ .can_receive = emc_can_receive,
1103
+ .receive = emc_receive,
1104
+ .cleanup = emc_cleanup,
1105
+ .link_status_changed = emc_set_link,
1106
+};
1107
+
1108
+static void npcm7xx_emc_realize(DeviceState *dev, Error **errp)
1109
+{
1110
+ NPCM7xxEMCState *emc = NPCM7XX_EMC(dev);
1111
+ SysBusDevice *sbd = SYS_BUS_DEVICE(emc);
1112
+
1113
+ memory_region_init_io(&emc->iomem, OBJECT(emc), &npcm7xx_emc_ops, emc,
1114
+ TYPE_NPCM7XX_EMC, 4 * KiB);
1115
+ sysbus_init_mmio(sbd, &emc->iomem);
1116
+ sysbus_init_irq(sbd, &emc->tx_irq);
1117
+ sysbus_init_irq(sbd, &emc->rx_irq);
1118
+
1119
+ qemu_macaddr_default_if_unset(&emc->conf.macaddr);
1120
+ emc->nic = qemu_new_nic(&net_npcm7xx_emc_info, &emc->conf,
1121
+ object_get_typename(OBJECT(dev)), dev->id, emc);
1122
+ qemu_format_nic_info_str(qemu_get_queue(emc->nic), emc->conf.macaddr.a);
1123
+}
1124
+
1125
+static void npcm7xx_emc_unrealize(DeviceState *dev)
1126
+{
1127
+ NPCM7xxEMCState *emc = NPCM7XX_EMC(dev);
1128
+
1129
+ qemu_del_nic(emc->nic);
1130
+}
1131
+
1132
+static const VMStateDescription vmstate_npcm7xx_emc = {
1133
+ .name = TYPE_NPCM7XX_EMC,
1134
+ .version_id = 0,
1135
+ .minimum_version_id = 0,
1136
+ .fields = (VMStateField[]) {
1137
+ VMSTATE_UINT8(emc_num, NPCM7xxEMCState),
1138
+ VMSTATE_UINT32_ARRAY(regs, NPCM7xxEMCState, NPCM7XX_NUM_EMC_REGS),
1139
+ VMSTATE_BOOL(tx_active, NPCM7xxEMCState),
1140
+ VMSTATE_BOOL(rx_active, NPCM7xxEMCState),
1141
+ VMSTATE_END_OF_LIST(),
1142
+ },
1143
+};
1144
+
1145
+static Property npcm7xx_emc_properties[] = {
1146
+ DEFINE_NIC_PROPERTIES(NPCM7xxEMCState, conf),
1147
+ DEFINE_PROP_END_OF_LIST(),
1148
+};
1149
+
1150
+static void npcm7xx_emc_class_init(ObjectClass *klass, void *data)
1151
+{
1152
+ DeviceClass *dc = DEVICE_CLASS(klass);
1153
+
1154
+ set_bit(DEVICE_CATEGORY_NETWORK, dc->categories);
1155
+ dc->desc = "NPCM7xx EMC Controller";
1156
+ dc->realize = npcm7xx_emc_realize;
1157
+ dc->unrealize = npcm7xx_emc_unrealize;
1158
+ dc->reset = npcm7xx_emc_reset;
1159
+ dc->vmsd = &vmstate_npcm7xx_emc;
1160
+ device_class_set_props(dc, npcm7xx_emc_properties);
1161
+}
1162
+
1163
+static const TypeInfo npcm7xx_emc_info = {
1164
+ .name = TYPE_NPCM7XX_EMC,
1165
+ .parent = TYPE_SYS_BUS_DEVICE,
1166
+ .instance_size = sizeof(NPCM7xxEMCState),
1167
+ .class_init = npcm7xx_emc_class_init,
1168
+};
1169
+
1170
+static void npcm7xx_emc_register_type(void)
1171
+{
1172
+ type_register_static(&npcm7xx_emc_info);
1173
+}
1174
+
1175
+type_init(npcm7xx_emc_register_type)
1176
diff --git a/hw/net/meson.build b/hw/net/meson.build
1177
index XXXXXXX..XXXXXXX 100644
1178
--- a/hw/net/meson.build
1179
+++ b/hw/net/meson.build
1180
@@ -XXX,XX +XXX,XX @@ softmmu_ss.add(when: 'CONFIG_I82596_COMMON', if_true: files('i82596.c'))
1181
softmmu_ss.add(when: 'CONFIG_SUNHME', if_true: files('sunhme.c'))
1182
softmmu_ss.add(when: 'CONFIG_FTGMAC100', if_true: files('ftgmac100.c'))
1183
softmmu_ss.add(when: 'CONFIG_SUNGEM', if_true: files('sungem.c'))
1184
+softmmu_ss.add(when: 'CONFIG_NPCM7XX', if_true: files('npcm7xx_emc.c'))
1185
1186
softmmu_ss.add(when: 'CONFIG_ETRAXFS', if_true: files('etraxfs_eth.c'))
1187
softmmu_ss.add(when: 'CONFIG_COLDFIRE', if_true: files('mcf_fec.c'))
1188
diff --git a/hw/net/trace-events b/hw/net/trace-events
1189
index XXXXXXX..XXXXXXX 100644
1190
--- a/hw/net/trace-events
1191
+++ b/hw/net/trace-events
1192
@@ -XXX,XX +XXX,XX @@ imx_fec_receive_last(int last) "rx frame flags 0x%04x"
1193
imx_enet_receive(size_t size) "len %zu"
1194
imx_enet_receive_len(uint64_t addr, int len) "rx_bd 0x%"PRIx64" length %d"
1195
imx_enet_receive_last(int last) "rx frame flags 0x%04x"
1196
+
1197
+# npcm7xx_emc.c
1198
+npcm7xx_emc_reset(int emc_num) "Resetting emc%d"
1199
+npcm7xx_emc_update_tx_irq(int level) "Setting tx irq to %d"
1200
+npcm7xx_emc_update_rx_irq(int level) "Setting rx irq to %d"
1201
+npcm7xx_emc_set_mista(uint32_t flags) "ORing 0x%x into MISTA"
1202
+npcm7xx_emc_cpu_owned_desc(uint32_t addr) "Can't process cpu-owned descriptor @0x%x"
1203
+npcm7xx_emc_sent_packet(uint32_t len) "Sent %u byte packet"
1204
+npcm7xx_emc_tx_done(uint32_t ctxdsa) "TX done, CTXDSA=0x%x"
1205
+npcm7xx_emc_can_receive(int can_receive) "Can receive: %d"
1206
+npcm7xx_emc_packet_filtered_out(const char* fail_reason) "Packet filtered out: %s"
1207
+npcm7xx_emc_packet_dropped(uint32_t len) "%u byte packet dropped"
1208
+npcm7xx_emc_receiving_packet(uint32_t len) "Receiving %u byte packet"
1209
+npcm7xx_emc_received_packet(uint32_t len) "Received %u byte packet"
1210
+npcm7xx_emc_rx_done(uint32_t crxdsa) "RX done, CRXDSA=0x%x"
1211
+npcm7xx_emc_reg_read(int emc_num, uint32_t result, const char *name, int regno) "emc%d: 0x%x = reg[%s/%d]"
1212
+npcm7xx_emc_reg_write(int emc_num, const char *name, int regno, uint32_t value) "emc%d: reg[%s/%d] = 0x%x"
1213
--
109
--
1214
2.20.1
110
2.20.1
1215
111
1216
112
diff view generated by jsdifflib
1
The AN505 and AN521 don't have any read-only memory, but the AN524
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
does; add a flag to ROMInfo to mark a region as ROM.
3
2
3
Simplify by always passing a MemoryRegion property to the device.
4
Doing so we can move the AddressSpace field to the device struct,
5
removing need for heap allocation.
6
7
Update the Xilinx ZynqMP / Versal SoC models to pass the default
8
system memory instead of a NULL value.
9
10
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
12
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
13
Message-id: 20210819163422.2863447-5-philmd@redhat.com
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20210215115138.20465-19-peter.maydell@linaro.org
7
---
15
---
8
hw/arm/mps2-tz.c | 6 ++++++
16
include/hw/dma/xlnx-zdma.h | 2 +-
9
1 file changed, 6 insertions(+)
17
hw/arm/xlnx-versal.c | 2 ++
18
hw/arm/xlnx-zynqmp.c | 8 ++++++++
19
hw/dma/xlnx-zdma.c | 24 ++++++++++++------------
20
4 files changed, 23 insertions(+), 13 deletions(-)
10
21
11
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
22
diff --git a/include/hw/dma/xlnx-zdma.h b/include/hw/dma/xlnx-zdma.h
12
index XXXXXXX..XXXXXXX 100644
23
index XXXXXXX..XXXXXXX 100644
13
--- a/hw/arm/mps2-tz.c
24
--- a/include/hw/dma/xlnx-zdma.h
14
+++ b/hw/arm/mps2-tz.c
25
+++ b/include/hw/dma/xlnx-zdma.h
15
@@ -XXX,XX +XXX,XX @@ typedef struct RAMInfo {
26
@@ -XXX,XX +XXX,XX @@ struct XlnxZDMA {
16
* Flag values:
27
MemoryRegion iomem;
17
* IS_ALIAS: this RAM area is an alias to the upstream end of the
28
MemTxAttrs attr;
18
* MPC specified by its .mpc value
29
MemoryRegion *dma_mr;
19
+ * IS_ROM: this RAM area is read-only
30
- AddressSpace *dma_as;
20
*/
31
+ AddressSpace dma_as;
21
#define IS_ALIAS 1
32
qemu_irq irq_zdma_ch_imr;
22
+#define IS_ROM 2
33
23
34
struct {
24
struct MPS2TZMachineClass {
35
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
25
MachineClass parent;
36
index XXXXXXX..XXXXXXX 100644
26
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *mr_for_raminfo(MPS2TZMachineState *mms,
37
--- a/hw/arm/xlnx-versal.c
27
if (raminfo->mrindex < 0) {
38
+++ b/hw/arm/xlnx-versal.c
28
/* Means this RAMInfo is for QEMU's "system memory" */
39
@@ -XXX,XX +XXX,XX @@ static void versal_create_admas(Versal *s, qemu_irq *pic)
29
MachineState *machine = MACHINE(mms);
40
TYPE_XLNX_ZDMA);
30
+ assert(!(raminfo->flags & IS_ROM));
41
dev = DEVICE(&s->lpd.iou.adma[i]);
31
return machine->ram;
42
object_property_set_int(OBJECT(dev), "bus-width", 128, &error_abort);
43
+ object_property_set_link(OBJECT(dev), "dma",
44
+ OBJECT(get_system_memory()), &error_fatal);
45
sysbus_realize(SYS_BUS_DEVICE(dev), &error_fatal);
46
47
mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
48
diff --git a/hw/arm/xlnx-zynqmp.c b/hw/arm/xlnx-zynqmp.c
49
index XXXXXXX..XXXXXXX 100644
50
--- a/hw/arm/xlnx-zynqmp.c
51
+++ b/hw/arm/xlnx-zynqmp.c
52
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
53
errp)) {
54
return;
55
}
56
+ if (!object_property_set_link(OBJECT(&s->gdma[i]), "dma",
57
+ OBJECT(system_memory), errp)) {
58
+ return;
59
+ }
60
if (!sysbus_realize(SYS_BUS_DEVICE(&s->gdma[i]), errp)) {
61
return;
62
}
63
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
32
}
64
}
33
65
34
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *mr_for_raminfo(MPS2TZMachineState *mms,
66
for (i = 0; i < XLNX_ZYNQMP_NUM_ADMA_CH; i++) {
35
67
+ if (!object_property_set_link(OBJECT(&s->adma[i]), "dma",
36
memory_region_init_ram(ram, NULL, raminfo->name,
68
+ OBJECT(system_memory), errp)) {
37
raminfo->size, &error_fatal);
69
+ return;
38
+ if (raminfo->flags & IS_ROM) {
70
+ }
39
+ memory_region_set_readonly(ram, true);
71
if (!sysbus_realize(SYS_BUS_DEVICE(&s->adma[i]), errp)) {
72
return;
73
}
74
diff --git a/hw/dma/xlnx-zdma.c b/hw/dma/xlnx-zdma.c
75
index XXXXXXX..XXXXXXX 100644
76
--- a/hw/dma/xlnx-zdma.c
77
+++ b/hw/dma/xlnx-zdma.c
78
@@ -XXX,XX +XXX,XX @@ static bool zdma_load_descriptor(XlnxZDMA *s, uint64_t addr,
79
return false;
80
}
81
82
- descr->addr = address_space_ldq_le(s->dma_as, addr, s->attr, NULL);
83
- descr->size = address_space_ldl_le(s->dma_as, addr + 8, s->attr, NULL);
84
- descr->attr = address_space_ldl_le(s->dma_as, addr + 12, s->attr, NULL);
85
+ descr->addr = address_space_ldq_le(&s->dma_as, addr, s->attr, NULL);
86
+ descr->size = address_space_ldl_le(&s->dma_as, addr + 8, s->attr, NULL);
87
+ descr->attr = address_space_ldl_le(&s->dma_as, addr + 12, s->attr, NULL);
88
return true;
89
}
90
91
@@ -XXX,XX +XXX,XX @@ static void zdma_update_descr_addr(XlnxZDMA *s, bool type,
92
} else {
93
addr = zdma_get_regaddr64(s, basereg);
94
addr += sizeof(s->dsc_dst);
95
- next = address_space_ldq_le(s->dma_as, addr, s->attr, NULL);
96
+ next = address_space_ldq_le(&s->dma_as, addr, s->attr, NULL);
97
}
98
99
zdma_put_regaddr64(s, basereg, next);
100
@@ -XXX,XX +XXX,XX @@ static void zdma_write_dst(XlnxZDMA *s, uint8_t *buf, uint32_t len)
101
}
102
}
103
104
- address_space_write(s->dma_as, s->dsc_dst.addr, s->attr, buf, dlen);
105
+ address_space_write(&s->dma_as, s->dsc_dst.addr, s->attr, buf, dlen);
106
if (burst_type == AXI_BURST_INCR) {
107
s->dsc_dst.addr += dlen;
108
}
109
@@ -XXX,XX +XXX,XX @@ static void zdma_process_descr(XlnxZDMA *s)
110
len = s->cfg.bus_width / 8;
111
}
112
} else {
113
- address_space_read(s->dma_as, src_addr, s->attr, s->buf, len);
114
+ address_space_read(&s->dma_as, src_addr, s->attr, s->buf, len);
115
if (burst_type == AXI_BURST_INCR) {
116
src_addr += len;
117
}
118
@@ -XXX,XX +XXX,XX @@ static void zdma_realize(DeviceState *dev, Error **errp)
119
XlnxZDMA *s = XLNX_ZDMA(dev);
120
unsigned int i;
121
122
+ if (!s->dma_mr) {
123
+ error_setg(errp, TYPE_XLNX_ZDMA " 'dma' link not set");
124
+ return;
40
+ }
125
+ }
41
return ram;
126
+ address_space_init(&s->dma_as, s->dma_mr, "zdma-dma");
127
+
128
for (i = 0; i < ARRAY_SIZE(zdma_regs_info); ++i) {
129
RegisterInfo *r = &s->regs_info[zdma_regs_info[i].addr / 4];
130
131
@@ -XXX,XX +XXX,XX @@ static void zdma_realize(DeviceState *dev, Error **errp)
132
};
133
}
134
135
- if (s->dma_mr) {
136
- s->dma_as = g_malloc0(sizeof(AddressSpace));
137
- address_space_init(s->dma_as, s->dma_mr, NULL);
138
- } else {
139
- s->dma_as = &address_space_memory;
140
- }
141
s->attr = MEMTXATTRS_UNSPECIFIED;
42
}
142
}
43
143
44
--
144
--
45
2.20.1
145
2.20.1
46
146
47
147
diff view generated by jsdifflib
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
1
From: Ani Sinha <ani@anisinha.ca>
2
2
3
We will move this code in the next commit. Clean it up
3
Since commit
4
first to avoid checkpatch.pl errors.
4
36b79e3219d ("hw/acpi/Kconfig: Add missing Kconfig dependencies (build error)"),
5
ACPI_MEMORY_HOTPLUG and ACPI_NVDIMM is implicitly turned on when
6
ACPI_HW_REDUCED is selected. ACPI_HW_REDUCED is already enabled. No need to
7
turn on ACPI_MEMORY_HOTPLUG or ACPI_NVDIMM explicitly. This is a minor cleanup.
5
8
6
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Signed-off-by: Ani Sinha <ani@anisinha.ca>
7
Message-id: 20210221222617.2579610-3-f4bug@amsat.org
10
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Message-id: 20210819162637.518507-1-ani@anisinha.ca
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
13
---
11
target/arm/cpu.c | 12 ++++++++----
14
hw/arm/Kconfig | 2 --
12
1 file changed, 8 insertions(+), 4 deletions(-)
15
1 file changed, 2 deletions(-)
13
16
14
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
17
diff --git a/hw/arm/Kconfig b/hw/arm/Kconfig
15
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/cpu.c
19
--- a/hw/arm/Kconfig
17
+++ b/target/arm/cpu.c
20
+++ b/hw/arm/Kconfig
18
@@ -XXX,XX +XXX,XX @@ static void cortex_a8_initfn(Object *obj)
21
@@ -XXX,XX +XXX,XX @@ config ARM_VIRT
19
}
22
select ACPI_PCI
20
23
select MEM_DEVICE
21
static const ARMCPRegInfo cortexa9_cp_reginfo[] = {
24
select DIMM
22
- /* power_control should be set to maximum latency. Again,
25
- select ACPI_MEMORY_HOTPLUG
23
+ /*
26
select ACPI_HW_REDUCED
24
+ * power_control should be set to maximum latency. Again,
27
- select ACPI_NVDIMM
25
* default to 0 and set by private hook
28
select ACPI_APEI
26
*/
29
27
{ .name = "A9_PWRCTL", .cp = 15, .crn = 15, .crm = 0, .opc1 = 0, .opc2 = 0,
30
config CHEETAH
28
@@ -XXX,XX +XXX,XX @@ static void cortex_a9_initfn(Object *obj)
29
set_feature(&cpu->env, ARM_FEATURE_NEON);
30
set_feature(&cpu->env, ARM_FEATURE_THUMB2EE);
31
set_feature(&cpu->env, ARM_FEATURE_EL3);
32
- /* Note that A9 supports the MP extensions even for
33
+ /*
34
+ * Note that A9 supports the MP extensions even for
35
* A9UP and single-core A9MP (which are both different
36
* and valid configurations; we don't model A9UP).
37
*/
38
@@ -XXX,XX +XXX,XX @@ static uint64_t a15_l2ctlr_read(CPUARMState *env, const ARMCPRegInfo *ri)
39
{
40
MachineState *ms = MACHINE(qdev_get_machine());
41
42
- /* Linux wants the number of processors from here.
43
+ /*
44
+ * Linux wants the number of processors from here.
45
* Might as well set the interrupt-controller bit too.
46
*/
47
return ((ms->smp.cpus - 1) << 24) | (1 << 23);
48
@@ -XXX,XX +XXX,XX @@ static void cortex_a7_initfn(Object *obj)
49
cpu->isar.id_mmfr1 = 0x40000000;
50
cpu->isar.id_mmfr2 = 0x01240000;
51
cpu->isar.id_mmfr3 = 0x02102211;
52
- /* a7_mpcore_r0p5_trm, page 4-4 gives 0x01101110; but
53
+ /*
54
+ * a7_mpcore_r0p5_trm, page 4-4 gives 0x01101110; but
55
* table 4-41 gives 0x02101110, which includes the arm div insns.
56
*/
57
cpu->isar.id_isar0 = 0x02101110;
58
--
31
--
59
2.20.1
32
2.20.1
60
33
61
34
diff view generated by jsdifflib
1
From: Richard Henderson <richard.henderson@linaro.org>
1
From: Andrew Jones <drjones@redhat.com>
2
2
3
Always perform one call instead of two for 16-byte operands.
3
Allow CPUs that support SVE to specify which SVE vector lengths they
4
Use byte loads/stores directly into the vector register file
4
support by setting them in this bitmap. Currently only the 'max' and
5
instead of extractions and deposits to a 64-bit local variable.
5
'host' CPU types supports SVE and 'host' requires KVM which obtains
6
its supported bitmap from the host. So, we only need to initialize the
7
bitmap for 'max' with TCG. And, since 'max' should support all SVE
8
vector lengths we simply fill the bitmap. Future CPU types may have
9
less trivial maps though.
6
10
7
In order to easily receive pointers into the vector register file,
11
Signed-off-by: Andrew Jones <drjones@redhat.com>
8
convert the helper to the gvec out-of-line signature. Move the
12
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
9
helper into vec_helper.c, where it can make use of H1 and clear_tail.
13
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
14
Message-id: 20210823160647.34028-2-drjones@redhat.com
11
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
12
Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
13
Tested-by: Alex Bennée <alex.bennee@linaro.org>
14
Message-id: 20210224230532.276878-1-richard.henderson@linaro.org
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
16
---
16
---
17
target/arm/helper-a64.h | 2 +-
17
target/arm/cpu.h | 4 ++++
18
target/arm/helper-a64.c | 32 ---------------------
18
target/arm/cpu64.c | 2 ++
19
target/arm/translate-a64.c | 58 +++++---------------------------------
19
2 files changed, 6 insertions(+)
20
target/arm/vec_helper.c | 48 +++++++++++++++++++++++++++++++
21
4 files changed, 56 insertions(+), 84 deletions(-)
22
20
23
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
21
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
24
index XXXXXXX..XXXXXXX 100644
22
index XXXXXXX..XXXXXXX 100644
25
--- a/target/arm/helper-a64.h
23
--- a/target/arm/cpu.h
26
+++ b/target/arm/helper-a64.h
24
+++ b/target/arm/cpu.h
27
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_3(vfp_cmps_a64, i64, f32, f32, ptr)
25
@@ -XXX,XX +XXX,XX @@ struct ARMCPU {
28
DEF_HELPER_3(vfp_cmpes_a64, i64, f32, f32, ptr)
26
* While processing properties during initialization, corresponding
29
DEF_HELPER_3(vfp_cmpd_a64, i64, f64, f64, ptr)
27
* sve_vq_init bits are set for bits in sve_vq_map that have been
30
DEF_HELPER_3(vfp_cmped_a64, i64, f64, f64, ptr)
28
* set by properties.
31
-DEF_HELPER_FLAGS_5(simd_tbl, TCG_CALL_NO_RWG_SE, i64, env, i64, i64, i32, i32)
29
+ *
32
+DEF_HELPER_FLAGS_4(simd_tblx, TCG_CALL_NO_RWG, void, ptr, ptr, ptr, i32)
30
+ * Bits set in sve_vq_supported represent valid vector lengths for
33
DEF_HELPER_FLAGS_3(vfp_mulxs, TCG_CALL_NO_RWG, f32, f32, f32, ptr)
31
+ * the CPU type.
34
DEF_HELPER_FLAGS_3(vfp_mulxd, TCG_CALL_NO_RWG, f64, f64, f64, ptr)
32
*/
35
DEF_HELPER_FLAGS_3(neon_ceq_f64, TCG_CALL_NO_RWG, i64, i64, i64, ptr)
33
DECLARE_BITMAP(sve_vq_map, ARM_MAX_VQ);
36
diff --git a/target/arm/helper-a64.c b/target/arm/helper-a64.c
34
DECLARE_BITMAP(sve_vq_init, ARM_MAX_VQ);
35
+ DECLARE_BITMAP(sve_vq_supported, ARM_MAX_VQ);
36
37
/* Generic timer counter frequency, in Hz */
38
uint64_t gt_cntfrq_hz;
39
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
37
index XXXXXXX..XXXXXXX 100644
40
index XXXXXXX..XXXXXXX 100644
38
--- a/target/arm/helper-a64.c
41
--- a/target/arm/cpu64.c
39
+++ b/target/arm/helper-a64.c
42
+++ b/target/arm/cpu64.c
40
@@ -XXX,XX +XXX,XX @@ float64 HELPER(vfp_mulxd)(float64 a, float64 b, void *fpstp)
43
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
41
return float64_mul(a, b, fpst);
44
/* Default to PAUTH on, with the architected algorithm. */
42
}
45
qdev_property_add_static(DEVICE(obj), &arm_cpu_pauth_property);
43
46
qdev_property_add_static(DEVICE(obj), &arm_cpu_pauth_impdef_property);
44
-uint64_t HELPER(simd_tbl)(CPUARMState *env, uint64_t result, uint64_t indices,
47
+
45
- uint32_t rn, uint32_t numregs)
48
+ bitmap_fill(cpu->sve_vq_supported, ARM_MAX_VQ);
46
-{
47
- /* Helper function for SIMD TBL and TBX. We have to do the table
48
- * lookup part for the 64 bits worth of indices we're passed in.
49
- * result is the initial results vector (either zeroes for TBL
50
- * or some guest values for TBX), rn the register number where
51
- * the table starts, and numregs the number of registers in the table.
52
- * We return the results of the lookups.
53
- */
54
- int shift;
55
-
56
- for (shift = 0; shift < 64; shift += 8) {
57
- int index = extract64(indices, shift, 8);
58
- if (index < 16 * numregs) {
59
- /* Convert index (a byte offset into the virtual table
60
- * which is a series of 128-bit vectors concatenated)
61
- * into the correct register element plus a bit offset
62
- * into that element, bearing in mind that the table
63
- * can wrap around from V31 to V0.
64
- */
65
- int elt = (rn * 2 + (index >> 3)) % 64;
66
- int bitidx = (index & 7) * 8;
67
- uint64_t *q = aa64_vfp_qreg(env, elt >> 1);
68
- uint64_t val = extract64(q[elt & 1], bitidx, 8);
69
-
70
- result = deposit64(result, shift, 8, val);
71
- }
72
- }
73
- return result;
74
-}
75
-
76
/* 64bit/double versions of the neon float compare functions */
77
uint64_t HELPER(neon_ceq_f64)(float64 a, float64 b, void *fpstp)
78
{
79
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
80
index XXXXXXX..XXXXXXX 100644
81
--- a/target/arm/translate-a64.c
82
+++ b/target/arm/translate-a64.c
83
@@ -XXX,XX +XXX,XX @@ static void disas_simd_tb(DisasContext *s, uint32_t insn)
84
int rm = extract32(insn, 16, 5);
85
int rn = extract32(insn, 5, 5);
86
int rd = extract32(insn, 0, 5);
87
- int is_tblx = extract32(insn, 12, 1);
88
- int len = extract32(insn, 13, 2);
89
- TCGv_i64 tcg_resl, tcg_resh, tcg_idx;
90
- TCGv_i32 tcg_regno, tcg_numregs;
91
+ int is_tbx = extract32(insn, 12, 1);
92
+ int len = (extract32(insn, 13, 2) + 1) * 16;
93
94
if (op2 != 0) {
95
unallocated_encoding(s);
96
@@ -XXX,XX +XXX,XX @@ static void disas_simd_tb(DisasContext *s, uint32_t insn)
97
return;
98
}
49
}
99
50
100
- /* This does a table lookup: for every byte element in the input
51
aarch64_add_sve_properties(obj);
101
- * we index into a table formed from up to four vector registers,
102
- * and then the output is the result of the lookups. Our helper
103
- * function does the lookup operation for a single 64 bit part of
104
- * the input.
105
- */
106
- tcg_resl = tcg_temp_new_i64();
107
- tcg_resh = NULL;
108
-
109
- if (is_tblx) {
110
- read_vec_element(s, tcg_resl, rd, 0, MO_64);
111
- } else {
112
- tcg_gen_movi_i64(tcg_resl, 0);
113
- }
114
-
115
- if (is_q) {
116
- tcg_resh = tcg_temp_new_i64();
117
- if (is_tblx) {
118
- read_vec_element(s, tcg_resh, rd, 1, MO_64);
119
- } else {
120
- tcg_gen_movi_i64(tcg_resh, 0);
121
- }
122
- }
123
-
124
- tcg_idx = tcg_temp_new_i64();
125
- tcg_regno = tcg_const_i32(rn);
126
- tcg_numregs = tcg_const_i32(len + 1);
127
- read_vec_element(s, tcg_idx, rm, 0, MO_64);
128
- gen_helper_simd_tbl(tcg_resl, cpu_env, tcg_resl, tcg_idx,
129
- tcg_regno, tcg_numregs);
130
- if (is_q) {
131
- read_vec_element(s, tcg_idx, rm, 1, MO_64);
132
- gen_helper_simd_tbl(tcg_resh, cpu_env, tcg_resh, tcg_idx,
133
- tcg_regno, tcg_numregs);
134
- }
135
- tcg_temp_free_i64(tcg_idx);
136
- tcg_temp_free_i32(tcg_regno);
137
- tcg_temp_free_i32(tcg_numregs);
138
-
139
- write_vec_element(s, tcg_resl, rd, 0, MO_64);
140
- tcg_temp_free_i64(tcg_resl);
141
-
142
- if (is_q) {
143
- write_vec_element(s, tcg_resh, rd, 1, MO_64);
144
- tcg_temp_free_i64(tcg_resh);
145
- }
146
- clear_vec_high(s, is_q, rd);
147
+ tcg_gen_gvec_2_ptr(vec_full_reg_offset(s, rd),
148
+ vec_full_reg_offset(s, rm), cpu_env,
149
+ is_q ? 16 : 8, vec_full_reg_size(s),
150
+ (len << 6) | (is_tbx << 5) | rn,
151
+ gen_helper_simd_tblx);
152
}
153
154
/* ZIP/UZP/TRN
155
diff --git a/target/arm/vec_helper.c b/target/arm/vec_helper.c
156
index XXXXXXX..XXXXXXX 100644
157
--- a/target/arm/vec_helper.c
158
+++ b/target/arm/vec_helper.c
159
@@ -XXX,XX +XXX,XX @@ DO_VRINT_RMODE(gvec_vrint_rm_h, helper_rinth, uint16_t)
160
DO_VRINT_RMODE(gvec_vrint_rm_s, helper_rints, uint32_t)
161
162
#undef DO_VRINT_RMODE
163
+
164
+#ifdef TARGET_AARCH64
165
+void HELPER(simd_tblx)(void *vd, void *vm, void *venv, uint32_t desc)
166
+{
167
+ const uint8_t *indices = vm;
168
+ CPUARMState *env = venv;
169
+ size_t oprsz = simd_oprsz(desc);
170
+ uint32_t rn = extract32(desc, SIMD_DATA_SHIFT, 5);
171
+ bool is_tbx = extract32(desc, SIMD_DATA_SHIFT + 5, 1);
172
+ uint32_t table_len = desc >> (SIMD_DATA_SHIFT + 6);
173
+ union {
174
+ uint8_t b[16];
175
+ uint64_t d[2];
176
+ } result;
177
+
178
+ /*
179
+ * We must construct the final result in a temp, lest the output
180
+ * overlaps the input table. For TBL, begin with zero; for TBX,
181
+ * begin with the original register contents. Note that we always
182
+ * copy 16 bytes here to avoid an extra branch; clearing the high
183
+ * bits of the register for oprsz == 8 is handled below.
184
+ */
185
+ if (is_tbx) {
186
+ memcpy(&result, vd, 16);
187
+ } else {
188
+ memset(&result, 0, 16);
189
+ }
190
+
191
+ for (size_t i = 0; i < oprsz; ++i) {
192
+ uint32_t index = indices[H1(i)];
193
+
194
+ if (index < table_len) {
195
+ /*
196
+ * Convert index (a byte offset into the virtual table
197
+ * which is a series of 128-bit vectors concatenated)
198
+ * into the correct register element, bearing in mind
199
+ * that the table can wrap around from V31 to V0.
200
+ */
201
+ const uint8_t *table = (const uint8_t *)
202
+ aa64_vfp_qreg(env, (rn + (index >> 4)) % 32);
203
+ result.b[H1(i)] = table[H1(index % 16)];
204
+ }
205
+ }
206
+
207
+ memcpy(vd, &result, 16);
208
+ clear_tail(vd, oprsz, simd_maxsz(desc));
209
+}
210
+#endif
211
--
52
--
212
2.20.1
53
2.20.1
213
54
214
55
diff view generated by jsdifflib
1
In the mps2-tz board code, we handle devices whose interrupt lines
1
From: Andrew Jones <drjones@redhat.com>
2
must be wired to all CPUs by creating IRQ splitter devices for the
3
AN521, because it has 2 CPUs, but wiring the device IRQ directly to
4
the SSE/IoTKit input for the AN505, which has only 1 CPU.
5
2
6
We can avoid making an explicit check on the board type constant by
3
bitmap_clear() only clears the given range. While the given
7
instead creating and using the IRQ splitters for any board with more
4
range should be sufficient in this case we might as well be
8
than 1 CPU. This avoids having to add extra cases to the
5
100% sure all bits are zeroed by using bitmap_zero().
9
conditionals every time we add new boards.
10
6
7
Signed-off-by: Andrew Jones <drjones@redhat.com>
8
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Message-id: 20210823160647.34028-3-drjones@redhat.com
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
13
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
14
Message-id: 20210215115138.20465-9-peter.maydell@linaro.org
15
---
12
---
16
hw/arm/mps2-tz.c | 19 +++++++++----------
13
target/arm/kvm64.c | 2 +-
17
1 file changed, 9 insertions(+), 10 deletions(-)
14
1 file changed, 1 insertion(+), 1 deletion(-)
18
15
19
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
16
diff --git a/target/arm/kvm64.c b/target/arm/kvm64.c
20
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
21
--- a/hw/arm/mps2-tz.c
18
--- a/target/arm/kvm64.c
22
+++ b/hw/arm/mps2-tz.c
19
+++ b/target/arm/kvm64.c
23
@@ -XXX,XX +XXX,XX @@ static void make_ram_alias(MemoryRegion *mr, const char *name,
20
@@ -XXX,XX +XXX,XX @@ void kvm_arm_sve_get_vls(CPUState *cs, unsigned long *map)
24
static qemu_irq get_sse_irq_in(MPS2TZMachineState *mms, int irqno)
21
uint32_t vq = 0;
25
{
22
int i, j;
26
/* Return a qemu_irq which will signal IRQ n to all CPUs in the SSE. */
23
27
- MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
24
- bitmap_clear(map, 0, ARM_MAX_VQ);
28
+ MachineClass *mc = MACHINE_GET_CLASS(mms);
25
+ bitmap_zero(map, ARM_MAX_VQ);
29
30
assert(irqno < MPS2TZ_NUMIRQ);
31
32
- switch (mmc->fpga_type) {
33
- case FPGA_AN505:
34
- return qdev_get_gpio_in_named(DEVICE(&mms->iotkit), "EXP_IRQ", irqno);
35
- case FPGA_AN521:
36
+ if (mc->max_cpus > 1) {
37
return qdev_get_gpio_in(DEVICE(&mms->cpu_irq_splitter[irqno]), 0);
38
- default:
39
- g_assert_not_reached();
40
+ } else {
41
+ return qdev_get_gpio_in_named(DEVICE(&mms->iotkit), "EXP_IRQ", irqno);
42
}
43
}
44
45
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
46
sysbus_realize(SYS_BUS_DEVICE(&mms->iotkit), &error_fatal);
47
26
48
/*
27
/*
49
- * The AN521 needs us to create splitters to feed the IRQ inputs
28
* KVM ensures all host CPUs support the same set of vector lengths.
50
- * for each CPU in the SSE-200 from each device in the board.
51
+ * If this board has more than one CPU, then we need to create splitters
52
+ * to feed the IRQ inputs for each CPU in the SSE from each device in the
53
+ * board. If there is only one CPU, we can just wire the device IRQ
54
+ * directly to the SSE's IRQ input.
55
*/
56
- if (mmc->fpga_type == FPGA_AN521) {
57
+ if (mc->max_cpus > 1) {
58
for (i = 0; i < MPS2TZ_NUMIRQ; i++) {
59
char *name = g_strdup_printf("mps2-irq-splitter%d", i);
60
SplitIRQ *splitter = &mms->cpu_irq_splitter[i];
61
--
29
--
62
2.20.1
30
2.20.1
63
31
64
32
diff view generated by jsdifflib
1
From: Rebecca Cran <rebecca@nuviainc.com>
1
From: Andrew Jones <drjones@redhat.com>
2
2
3
Set ID_AA64PFR1_EL1.SSBS to 2 and ID_PFR2.SSBS to 1.
3
Now that we have an ARMCPU member sve_vq_supported we no longer
4
need the local kvm_supported bitmap for KVM's supported vector
5
lengths.
4
6
5
Signed-off-by: Rebecca Cran <rebecca@nuviainc.com>
7
Signed-off-by: Andrew Jones <drjones@redhat.com>
8
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20210216224543.16142-3-rebecca@nuviainc.com
10
Message-id: 20210823160647.34028-4-drjones@redhat.com
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
---
12
---
10
target/arm/cpu64.c | 5 +++++
13
target/arm/cpu64.c | 19 +++++++++++--------
11
1 file changed, 5 insertions(+)
14
1 file changed, 11 insertions(+), 8 deletions(-)
12
15
13
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
16
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
14
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/cpu64.c
18
--- a/target/arm/cpu64.c
16
+++ b/target/arm/cpu64.c
19
+++ b/target/arm/cpu64.c
17
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
20
@@ -XXX,XX +XXX,XX @@ void arm_cpu_sve_finalize(ARMCPU *cpu, Error **errp)
18
21
* any of the above. Finally, if SVE is not disabled, then at least one
19
t = cpu->isar.id_aa64pfr1;
22
* vector length must be enabled.
20
t = FIELD_DP64(t, ID_AA64PFR1, BT, 1);
23
*/
21
+ t = FIELD_DP64(t, ID_AA64PFR1, SSBS, 2);
24
- DECLARE_BITMAP(kvm_supported, ARM_MAX_VQ);
22
/*
25
DECLARE_BITMAP(tmp, ARM_MAX_VQ);
23
* Begin with full support for MTE. This will be downgraded to MTE=0
26
uint32_t vq, max_vq = 0;
24
* during realize if the board provides no tag memory, much like
27
25
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
28
- /* Collect the set of vector lengths supported by KVM. */
26
u = FIELD_DP32(u, ID_PFR0, DIT, 1);
29
- bitmap_zero(kvm_supported, ARM_MAX_VQ);
27
cpu->isar.id_pfr0 = u;
30
+ /*
28
31
+ * CPU models specify a set of supported vector lengths which are
29
+ u = cpu->isar.id_pfr2;
32
+ * enabled by default. Attempting to enable any vector length not set
30
+ u = FIELD_DP32(u, ID_PFR2, SSBS, 1);
33
+ * in the supported bitmap results in an error. When KVM is enabled we
31
+ cpu->isar.id_pfr2 = u;
34
+ * fetch the supported bitmap from the host.
32
+
35
+ */
33
u = cpu->isar.id_mmfr3;
36
if (kvm_enabled() && kvm_arm_sve_supported()) {
34
u = FIELD_DP32(u, ID_MMFR3, PAN, 2); /* ATS1E1 */
37
- kvm_arm_sve_get_vls(CPU(cpu), kvm_supported);
35
cpu->isar.id_mmfr3 = u;
38
+ kvm_arm_sve_get_vls(CPU(cpu), cpu->sve_vq_supported);
39
} else if (kvm_enabled()) {
40
assert(!cpu_isar_feature(aa64_sve, cpu));
41
}
42
@@ -XXX,XX +XXX,XX @@ void arm_cpu_sve_finalize(ARMCPU *cpu, Error **errp)
43
* For KVM we have to automatically enable all supported unitialized
44
* lengths, even when the smaller lengths are not all powers-of-two.
45
*/
46
- bitmap_andnot(tmp, kvm_supported, cpu->sve_vq_init, max_vq);
47
+ bitmap_andnot(tmp, cpu->sve_vq_supported, cpu->sve_vq_init, max_vq);
48
bitmap_or(cpu->sve_vq_map, cpu->sve_vq_map, tmp, max_vq);
49
} else {
50
/* Propagate enabled bits down through required powers-of-two. */
51
@@ -XXX,XX +XXX,XX @@ void arm_cpu_sve_finalize(ARMCPU *cpu, Error **errp)
52
/* Disabling a supported length disables all larger lengths. */
53
for (vq = 1; vq <= ARM_MAX_VQ; ++vq) {
54
if (test_bit(vq - 1, cpu->sve_vq_init) &&
55
- test_bit(vq - 1, kvm_supported)) {
56
+ test_bit(vq - 1, cpu->sve_vq_supported)) {
57
break;
58
}
59
}
60
max_vq = vq <= ARM_MAX_VQ ? vq - 1 : ARM_MAX_VQ;
61
- bitmap_andnot(cpu->sve_vq_map, kvm_supported,
62
+ bitmap_andnot(cpu->sve_vq_map, cpu->sve_vq_supported,
63
cpu->sve_vq_init, max_vq);
64
if (max_vq == 0 || bitmap_empty(cpu->sve_vq_map, max_vq)) {
65
error_setg(errp, "cannot disable sve%d", vq * 128);
66
@@ -XXX,XX +XXX,XX @@ void arm_cpu_sve_finalize(ARMCPU *cpu, Error **errp)
67
68
if (kvm_enabled()) {
69
/* Ensure the set of lengths matches what KVM supports. */
70
- bitmap_xor(tmp, cpu->sve_vq_map, kvm_supported, max_vq);
71
+ bitmap_xor(tmp, cpu->sve_vq_map, cpu->sve_vq_supported, max_vq);
72
if (!bitmap_empty(tmp, max_vq)) {
73
vq = find_last_bit(tmp, max_vq) + 1;
74
if (test_bit(vq - 1, cpu->sve_vq_map)) {
36
--
75
--
37
2.20.1
76
2.20.1
38
77
39
78
diff view generated by jsdifflib
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
1
From: Andrew Jones <drjones@redhat.com>
2
2
3
IDAU is specific to M-profile. KVM only supports A-profile.
3
Future CPU types may specify which vector lengths are supported.
4
Restrict this interface to TCG, as it is pointless (and
4
We can apply nearly the same logic to validate those lengths
5
confusing) on a KVM-only build.
5
as we do for KVM's supported vector lengths. We merge the code
6
where we can, but unfortunately can't completely merge it because
7
KVM requires all vector lengths, power-of-two or not, smaller than
8
the maximum enabled length to also be enabled. The architecture
9
only requires all the power-of-two lengths, though, so TCG will
10
only enforce that.
6
11
12
Signed-off-by: Andrew Jones <drjones@redhat.com>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
13
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
14
Message-id: 20210823160647.34028-5-drjones@redhat.com
9
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
10
Message-id: 20210221222617.2579610-2-f4bug@amsat.org
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
16
---
13
target/arm/cpu.c | 7 -------
17
target/arm/cpu64.c | 101 ++++++++++++++++++++-------------------------
14
target/arm/cpu_tcg.c | 8 ++++++++
18
1 file changed, 45 insertions(+), 56 deletions(-)
15
2 files changed, 8 insertions(+), 7 deletions(-)
16
19
17
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
20
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
18
index XXXXXXX..XXXXXXX 100644
21
index XXXXXXX..XXXXXXX 100644
19
--- a/target/arm/cpu.c
22
--- a/target/arm/cpu64.c
20
+++ b/target/arm/cpu.c
23
+++ b/target/arm/cpu64.c
21
@@ -XXX,XX +XXX,XX @@ static const TypeInfo arm_cpu_type_info = {
24
@@ -XXX,XX +XXX,XX @@ void arm_cpu_sve_finalize(ARMCPU *cpu, Error **errp)
22
.class_init = arm_cpu_class_init,
25
break;
23
};
26
}
24
27
}
25
-static const TypeInfo idau_interface_type_info = {
28
- max_vq = vq <= ARM_MAX_VQ ? vq - 1 : ARM_MAX_VQ;
26
- .name = TYPE_IDAU_INTERFACE,
29
- bitmap_andnot(cpu->sve_vq_map, cpu->sve_vq_supported,
27
- .parent = TYPE_INTERFACE,
30
- cpu->sve_vq_init, max_vq);
28
- .class_size = sizeof(IDAUInterfaceClass),
31
- if (max_vq == 0 || bitmap_empty(cpu->sve_vq_map, max_vq)) {
29
-};
32
- error_setg(errp, "cannot disable sve%d", vq * 128);
30
-
33
- error_append_hint(errp, "Disabling sve%d results in all "
31
static void arm_cpu_register_types(void)
34
- "vector lengths being disabled.\n",
32
{
35
- vq * 128);
33
const size_t cpu_count = ARRAY_SIZE(arm_cpus);
36
- error_append_hint(errp, "With SVE enabled, at least one "
34
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_register_types(void)
37
- "vector length must be enabled.\n");
35
if (cpu_count) {
38
- return;
36
size_t i;
39
- }
37
40
} else {
38
- type_register_static(&idau_interface_type_info);
41
/* Disabling a power-of-two disables all larger lengths. */
39
for (i = 0; i < cpu_count; ++i) {
42
- if (test_bit(0, cpu->sve_vq_init)) {
40
arm_cpu_register(&arm_cpus[i]);
43
- error_setg(errp, "cannot disable sve128");
44
- error_append_hint(errp, "Disabling sve128 results in all "
45
- "vector lengths being disabled.\n");
46
- error_append_hint(errp, "With SVE enabled, at least one "
47
- "vector length must be enabled.\n");
48
- return;
49
- }
50
- for (vq = 2; vq <= ARM_MAX_VQ; vq <<= 1) {
51
+ for (vq = 1; vq <= ARM_MAX_VQ; vq <<= 1) {
52
if (test_bit(vq - 1, cpu->sve_vq_init)) {
53
break;
54
}
55
}
56
- max_vq = vq <= ARM_MAX_VQ ? vq - 1 : ARM_MAX_VQ;
57
- bitmap_complement(cpu->sve_vq_map, cpu->sve_vq_init, max_vq);
58
+ }
59
+
60
+ max_vq = vq <= ARM_MAX_VQ ? vq - 1 : ARM_MAX_VQ;
61
+ bitmap_andnot(cpu->sve_vq_map, cpu->sve_vq_supported,
62
+ cpu->sve_vq_init, max_vq);
63
+ if (max_vq == 0 || bitmap_empty(cpu->sve_vq_map, max_vq)) {
64
+ error_setg(errp, "cannot disable sve%d", vq * 128);
65
+ error_append_hint(errp, "Disabling sve%d results in all "
66
+ "vector lengths being disabled.\n",
67
+ vq * 128);
68
+ error_append_hint(errp, "With SVE enabled, at least one "
69
+ "vector length must be enabled.\n");
70
+ return;
41
}
71
}
42
diff --git a/target/arm/cpu_tcg.c b/target/arm/cpu_tcg.c
72
43
index XXXXXXX..XXXXXXX 100644
73
max_vq = find_last_bit(cpu->sve_vq_map, max_vq) + 1;
44
--- a/target/arm/cpu_tcg.c
74
@@ -XXX,XX +XXX,XX @@ void arm_cpu_sve_finalize(ARMCPU *cpu, Error **errp)
45
+++ b/target/arm/cpu_tcg.c
75
assert(max_vq != 0);
46
@@ -XXX,XX +XXX,XX @@
76
bitmap_clear(cpu->sve_vq_map, max_vq, ARM_MAX_VQ - max_vq);
47
#include "hw/core/tcg-cpu-ops.h"
77
48
#endif /* CONFIG_TCG */
78
- if (kvm_enabled()) {
49
#include "internals.h"
79
- /* Ensure the set of lengths matches what KVM supports. */
50
+#include "target/arm/idau.h"
80
- bitmap_xor(tmp, cpu->sve_vq_map, cpu->sve_vq_supported, max_vq);
51
81
- if (!bitmap_empty(tmp, max_vq)) {
52
/* CPU models. These are not needed for the AArch64 linux-user build. */
82
- vq = find_last_bit(tmp, max_vq) + 1;
53
#if !defined(CONFIG_USER_ONLY) || !defined(TARGET_AARCH64)
83
- if (test_bit(vq - 1, cpu->sve_vq_map)) {
54
@@ -XXX,XX +XXX,XX @@ static const ARMCPUInfo arm_tcg_cpus[] = {
84
- if (cpu->sve_max_vq) {
55
{ .name = "pxa270-c5", .initfn = pxa270c5_initfn },
85
- error_setg(errp, "cannot set sve-max-vq=%d",
56
};
86
- cpu->sve_max_vq);
57
87
- error_append_hint(errp, "This KVM host does not support "
58
+static const TypeInfo idau_interface_type_info = {
88
- "the vector length %d-bits.\n",
59
+ .name = TYPE_IDAU_INTERFACE,
89
- vq * 128);
60
+ .parent = TYPE_INTERFACE,
90
- error_append_hint(errp, "It may not be possible to use "
61
+ .class_size = sizeof(IDAUInterfaceClass),
91
- "sve-max-vq with this KVM host. Try "
62
+};
92
- "using only sve<N> properties.\n");
63
+
93
- } else {
64
static void arm_tcg_cpu_register_types(void)
94
- error_setg(errp, "cannot enable sve%d", vq * 128);
65
{
95
- error_append_hint(errp, "This KVM host does not support "
66
size_t i;
96
- "the vector length %d-bits.\n",
67
97
- vq * 128);
68
+ type_register_static(&idau_interface_type_info);
98
- }
69
for (i = 0; i < ARRAY_SIZE(arm_tcg_cpus); ++i) {
99
+ /* Ensure the set of lengths matches what is supported. */
70
arm_cpu_register(&arm_tcg_cpus[i]);
100
+ bitmap_xor(tmp, cpu->sve_vq_map, cpu->sve_vq_supported, max_vq);
101
+ if (!bitmap_empty(tmp, max_vq)) {
102
+ vq = find_last_bit(tmp, max_vq) + 1;
103
+ if (test_bit(vq - 1, cpu->sve_vq_map)) {
104
+ if (cpu->sve_max_vq) {
105
+ error_setg(errp, "cannot set sve-max-vq=%d", cpu->sve_max_vq);
106
+ error_append_hint(errp, "This CPU does not support "
107
+ "the vector length %d-bits.\n", vq * 128);
108
+ error_append_hint(errp, "It may not be possible to use "
109
+ "sve-max-vq with this CPU. Try "
110
+ "using only sve<N> properties.\n");
111
} else {
112
+ error_setg(errp, "cannot enable sve%d", vq * 128);
113
+ error_append_hint(errp, "This CPU does not support "
114
+ "the vector length %d-bits.\n", vq * 128);
115
+ }
116
+ return;
117
+ } else {
118
+ if (kvm_enabled()) {
119
error_setg(errp, "cannot disable sve%d", vq * 128);
120
error_append_hint(errp, "The KVM host requires all "
121
"supported vector lengths smaller "
122
"than %d bits to also be enabled.\n",
123
max_vq * 128);
124
- }
125
- return;
126
- }
127
- } else {
128
- /* Ensure all required powers-of-two are enabled. */
129
- for (vq = pow2floor(max_vq); vq >= 1; vq >>= 1) {
130
- if (!test_bit(vq - 1, cpu->sve_vq_map)) {
131
- error_setg(errp, "cannot disable sve%d", vq * 128);
132
- error_append_hint(errp, "sve%d is required as it "
133
- "is a power-of-two length smaller than "
134
- "the maximum, sve%d\n",
135
- vq * 128, max_vq * 128);
136
return;
137
+ } else {
138
+ /* Ensure all required powers-of-two are enabled. */
139
+ for (vq = pow2floor(max_vq); vq >= 1; vq >>= 1) {
140
+ if (!test_bit(vq - 1, cpu->sve_vq_map)) {
141
+ error_setg(errp, "cannot disable sve%d", vq * 128);
142
+ error_append_hint(errp, "sve%d is required as it "
143
+ "is a power-of-two length smaller "
144
+ "than the maximum, sve%d\n",
145
+ vq * 128, max_vq * 128);
146
+ return;
147
+ }
148
+ }
149
}
150
}
71
}
151
}
72
--
152
--
73
2.20.1
153
2.20.1
74
154
75
155
diff view generated by jsdifflib
1
From: Doug Evans <dje@google.com>
1
Do a basic conversion of the acpi_cpu_hotplug spec document to rST.
2
2
3
Reviewed-by: Hao Wu <wuhaotsh@google.com>
4
Reviewed-by: Avi Fishman <avi.fishman@nuvoton.com>
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Doug Evans <dje@google.com>
7
Message-id: 20210218212453.831406-4-dje@google.com
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Igor Mammedov <imammedo@redhat.com>
5
Message-id: 20210727170414.3368-2-peter.maydell@linaro.org
9
---
6
---
10
tests/qtest/npcm7xx_emc-test.c | 862 +++++++++++++++++++++++++++++++++
7
docs/specs/acpi_cpu_hotplug.rst | 235 ++++++++++++++++++++++++++++++++
11
tests/qtest/meson.build | 3 +-
8
docs/specs/acpi_cpu_hotplug.txt | 160 ----------------------
12
2 files changed, 864 insertions(+), 1 deletion(-)
9
docs/specs/index.rst | 1 +
13
create mode 100644 tests/qtest/npcm7xx_emc-test.c
10
3 files changed, 236 insertions(+), 160 deletions(-)
11
create mode 100644 docs/specs/acpi_cpu_hotplug.rst
12
delete mode 100644 docs/specs/acpi_cpu_hotplug.txt
14
13
15
diff --git a/tests/qtest/npcm7xx_emc-test.c b/tests/qtest/npcm7xx_emc-test.c
14
diff --git a/docs/specs/acpi_cpu_hotplug.rst b/docs/specs/acpi_cpu_hotplug.rst
16
new file mode 100644
15
new file mode 100644
17
index XXXXXXX..XXXXXXX
16
index XXXXXXX..XXXXXXX
18
--- /dev/null
17
--- /dev/null
19
+++ b/tests/qtest/npcm7xx_emc-test.c
18
+++ b/docs/specs/acpi_cpu_hotplug.rst
20
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@
21
+/*
20
+QEMU<->ACPI BIOS CPU hotplug interface
22
+ * QTests for Nuvoton NPCM7xx EMC Modules.
21
+======================================
23
+ *
22
+
24
+ * Copyright 2020 Google LLC
23
+QEMU supports CPU hotplug via ACPI. This document
25
+ *
24
+describes the interface between QEMU and the ACPI BIOS.
26
+ * This program is free software; you can redistribute it and/or modify it
25
+
27
+ * under the terms of the GNU General Public License as published by the
26
+ACPI BIOS GPE.2 handler is dedicated for notifying OS about CPU hot-add
28
+ * Free Software Foundation; either version 2 of the License, or
27
+and hot-remove events.
29
+ * (at your option) any later version.
28
+
30
+ *
29
+
31
+ * This program is distributed in the hope that it will be useful, but WITHOUT
30
+Legacy ACPI CPU hotplug interface registers
32
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
31
+-------------------------------------------
33
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
32
+
34
+ * for more details.
33
+CPU present bitmap for:
35
+ */
34
+
36
+
35
+- ICH9-LPC (IO port 0x0cd8-0xcf7, 1-byte access)
37
+#include "qemu/osdep.h"
36
+- PIIX-PM (IO port 0xaf00-0xaf1f, 1-byte access)
38
+#include "qemu-common.h"
37
+- One bit per CPU. Bit position reflects corresponding CPU APIC ID. Read-only.
39
+#include "libqos/libqos.h"
38
+- The first DWORD in bitmap is used in write mode to switch from legacy
40
+#include "qapi/qmp/qdict.h"
39
+ to modern CPU hotplug interface, write 0 into it to do switch.
41
+#include "qapi/qmp/qnum.h"
40
+
42
+#include "qemu/bitops.h"
41
+QEMU sets corresponding CPU bit on hot-add event and issues SCI
43
+#include "qemu/iov.h"
42
+with GPE.2 event set. CPU present map is read by ACPI BIOS GPE.2 handler
44
+
43
+to notify OS about CPU hot-add events. CPU hot-remove isn't supported.
45
+/* Name of the emc device. */
44
+
46
+#define TYPE_NPCM7XX_EMC "npcm7xx-emc"
45
+
47
+
46
+Modern ACPI CPU hotplug interface registers
48
+/* Timeout for various operations, in seconds. */
47
+-------------------------------------------
49
+#define TIMEOUT_SECONDS 10
48
+
50
+
49
+Register block base address:
51
+/* Address in memory of the descriptor. */
50
+
52
+#define DESC_ADDR (1 << 20) /* 1 MiB */
51
+- ICH9-LPC IO port 0x0cd8
53
+
52
+- PIIX-PM IO port 0xaf00
54
+/* Address in memory of the data packet. */
53
+
55
+#define DATA_ADDR (DESC_ADDR + 4096)
54
+Register block size:
56
+
55
+
57
+#define CRC_LENGTH 4
56
+- ACPI_CPU_HOTPLUG_REG_LEN = 12
58
+
57
+
59
+#define NUM_TX_DESCRIPTORS 3
58
+All accesses to registers described below, imply little-endian byte order.
60
+#define NUM_RX_DESCRIPTORS 2
59
+
61
+
60
+Reserved registers behavior:
62
+/* Size of tx,rx test buffers. */
61
+
63
+#define TX_DATA_LEN 64
62
+- write accesses are ignored
64
+#define RX_DATA_LEN 64
63
+- read accesses return all bits set to 0.
65
+
64
+
66
+#define TX_STEP_COUNT 10000
65
+The last stored value in 'CPU selector' must refer to a possible CPU, otherwise
67
+#define RX_STEP_COUNT 10000
66
+
68
+
67
+- reads from any register return 0
69
+/* 32-bit register indices. */
68
+- writes to any other register are ignored until valid value is stored into it
70
+typedef enum NPCM7xxPWMRegister {
69
+
71
+ /* Control registers. */
70
+On QEMU start, 'CPU selector' is initialized to a valid value, on reset it
72
+ REG_CAMCMR,
71
+keeps the current value.
73
+ REG_CAMEN,
72
+
74
+
73
+Read access behavior
75
+ /* There are 16 CAMn[ML] registers. */
74
+^^^^^^^^^^^^^^^^^^^^
76
+ REG_CAMM_BASE,
75
+
77
+ REG_CAML_BASE,
76
+offset [0x0-0x3]
78
+
77
+ Command data 2: (DWORD access)
79
+ REG_TXDLSA = 0x22,
78
+
80
+ REG_RXDLSA,
79
+ If value last stored in 'Command field' is:
81
+ REG_MCMDR,
80
+
82
+ REG_MIID,
81
+ 0:
83
+ REG_MIIDA,
82
+ reads as 0x0
84
+ REG_FFTCR,
83
+ 3:
85
+ REG_TSDR,
84
+ upper 32 bits of architecture specific CPU ID value
86
+ REG_RSDR,
85
+ other values:
87
+ REG_DMARFC,
86
+ reserved
88
+ REG_MIEN,
87
+
89
+
88
+offset [0x4]
90
+ /* Status registers. */
89
+ CPU device status fields: (1 byte access)
91
+ REG_MISTA,
90
+
92
+ REG_MGSTA,
91
+ bits:
93
+ REG_MPCNT,
92
+
94
+ REG_MRPC,
93
+ 0:
95
+ REG_MRPCC,
94
+ Device is enabled and may be used by guest
96
+ REG_MREPC,
95
+ 1:
97
+ REG_DMARFS,
96
+ Device insert event, used to distinguish device for which
98
+ REG_CTXDSA,
97
+ no device check event to OSPM was issued.
99
+ REG_CTXBSA,
98
+ It's valid only when bit 0 is set.
100
+ REG_CRXDSA,
99
+ 2:
101
+ REG_CRXBSA,
100
+ Device remove event, used to distinguish device for which
102
+
101
+ no device eject request to OSPM was issued. Firmware must
103
+ NPCM7XX_NUM_EMC_REGS,
102
+ ignore this bit.
104
+} NPCM7xxPWMRegister;
103
+ 3:
105
+
104
+ reserved and should be ignored by OSPM
106
+enum { NUM_CAMML_REGS = 16 };
105
+ 4:
107
+
106
+ if set to 1, OSPM requests firmware to perform device eject.
108
+/* REG_CAMCMR fields */
107
+ 5-7:
109
+/* Enable CAM Compare */
108
+ reserved and should be ignored by OSPM
110
+#define REG_CAMCMR_ECMP (1 << 4)
109
+
111
+/* Accept Unicast Packet */
110
+offset [0x5-0x7]
112
+#define REG_CAMCMR_AUP (1 << 0)
111
+ reserved
113
+
112
+
114
+/* REG_MCMDR fields */
113
+offset [0x8]
115
+/* Software Reset */
114
+ Command data: (DWORD access)
116
+#define REG_MCMDR_SWR (1 << 24)
115
+
117
+/* Frame Transmission On */
116
+ If value last stored in 'Command field' is one of:
118
+#define REG_MCMDR_TXON (1 << 8)
117
+
119
+/* Accept Long Packet */
118
+ 0:
120
+#define REG_MCMDR_ALP (1 << 1)
119
+ contains 'CPU selector' value of a CPU with pending event[s]
121
+/* Frame Reception On */
120
+ 3:
122
+#define REG_MCMDR_RXON (1 << 0)
121
+ lower 32 bits of architecture specific CPU ID value
123
+
122
+ (in x86 case: APIC ID)
124
+/* REG_MIEN fields */
123
+ otherwise:
125
+/* Enable Transmit Completion Interrupt */
124
+ contains 0
126
+#define REG_MIEN_ENTXCP (1 << 18)
125
+
127
+/* Enable Transmit Interrupt */
126
+Write access behavior
128
+#define REG_MIEN_ENTXINTR (1 << 16)
127
+^^^^^^^^^^^^^^^^^^^^^
129
+/* Enable Receive Good Interrupt */
128
+
130
+#define REG_MIEN_ENRXGD (1 << 4)
129
+offset [0x0-0x3]
131
+/* ENable Receive Interrupt */
130
+ CPU selector: (DWORD access)
132
+#define REG_MIEN_ENRXINTR (1 << 0)
131
+
133
+
132
+ Selects active CPU device. All following accesses to other
134
+/* REG_MISTA fields */
133
+ registers will read/store data from/to selected CPU.
135
+/* Transmit Bus Error Interrupt */
134
+ Valid values: [0 .. max_cpus)
136
+#define REG_MISTA_TXBERR (1 << 24)
135
+
137
+/* Transmit Descriptor Unavailable Interrupt */
136
+offset [0x4]
138
+#define REG_MISTA_TDU (1 << 23)
137
+ CPU device control fields: (1 byte access)
139
+/* Transmit Completion Interrupt */
138
+
140
+#define REG_MISTA_TXCP (1 << 18)
139
+ bits:
141
+/* Transmit Interrupt */
140
+
142
+#define REG_MISTA_TXINTR (1 << 16)
141
+ 0:
143
+/* Receive Bus Error Interrupt */
142
+ reserved, OSPM must clear it before writing to register.
144
+#define REG_MISTA_RXBERR (1 << 11)
143
+ 1:
145
+/* Receive Descriptor Unavailable Interrupt */
144
+ if set to 1 clears device insert event, set by OSPM
146
+#define REG_MISTA_RDU (1 << 10)
145
+ after it has emitted device check event for the
147
+/* DMA Early Notification Interrupt */
146
+ selected CPU device
148
+#define REG_MISTA_DENI (1 << 9)
147
+ 2:
149
+/* Maximum Frame Length Interrupt */
148
+ if set to 1 clears device remove event, set by OSPM
150
+#define REG_MISTA_DFOI (1 << 8)
149
+ after it has emitted device eject request for the
151
+/* Receive Good Interrupt */
150
+ selected CPU device.
152
+#define REG_MISTA_RXGD (1 << 4)
151
+ 3:
153
+/* Packet Too Long Interrupt */
152
+ if set to 1 initiates device eject, set by OSPM when it
154
+#define REG_MISTA_PTLE (1 << 3)
153
+ triggers CPU device removal and calls _EJ0 method or by firmware
155
+/* Receive Interrupt */
154
+ when bit #4 is set. In case bit #4 were set, it's cleared as
156
+#define REG_MISTA_RXINTR (1 << 0)
155
+ part of device eject.
157
+
156
+ 4:
158
+typedef struct NPCM7xxEMCTxDesc NPCM7xxEMCTxDesc;
157
+ if set to 1, OSPM hands over device eject to firmware.
159
+typedef struct NPCM7xxEMCRxDesc NPCM7xxEMCRxDesc;
158
+ Firmware shall issue device eject request as described above
160
+
159
+ (bit #3) and OSPM should not touch device eject bit (#3) in case
161
+struct NPCM7xxEMCTxDesc {
160
+ it's asked firmware to perform CPU device eject.
162
+ uint32_t flags;
161
+ 5-7:
163
+ uint32_t txbsa;
162
+ reserved, OSPM must clear them before writing to register
164
+ uint32_t status_and_length;
163
+
165
+ uint32_t ntxdsa;
164
+offset[0x5]
166
+};
165
+ Command field: (1 byte access)
167
+
166
+
168
+struct NPCM7xxEMCRxDesc {
167
+ value:
169
+ uint32_t status_and_length;
168
+
170
+ uint32_t rxbsa;
169
+ 0:
171
+ uint32_t reserved;
170
+ selects a CPU device with inserting/removing events and
172
+ uint32_t nrxdsa;
171
+ following reads from 'Command data' register return
173
+};
172
+ selected CPU ('CPU selector' value).
174
+
173
+ If no CPU with events found, the current 'CPU selector' doesn't
175
+/* NPCM7xxEMCTxDesc.flags values */
174
+ change and corresponding insert/remove event flags are not modified.
176
+/* Owner: 0 = cpu, 1 = emc */
175
+
177
+#define TX_DESC_FLAG_OWNER_MASK (1 << 31)
176
+ 1:
178
+/* Transmit interrupt enable */
177
+ following writes to 'Command data' register set OST event
179
+#define TX_DESC_FLAG_INTEN (1 << 2)
178
+ register in QEMU
180
+
179
+ 2:
181
+/* NPCM7xxEMCTxDesc.status_and_length values */
180
+ following writes to 'Command data' register set OST status
182
+/* Transmission complete */
181
+ register in QEMU
183
+#define TX_DESC_STATUS_TXCP (1 << 19)
182
+ 3:
184
+/* Transmit interrupt */
183
+ following reads from 'Command data' and 'Command data 2' return
185
+#define TX_DESC_STATUS_TXINTR (1 << 16)
184
+ architecture specific CPU ID value for currently selected CPU.
186
+
185
+ other values:
187
+/* NPCM7xxEMCRxDesc.status_and_length values */
186
+ reserved
188
+/* Owner: 0b00 = cpu, 0b10 = emc */
187
+
189
+#define RX_DESC_STATUS_OWNER_SHIFT 30
188
+offset [0x6-0x7]
190
+#define RX_DESC_STATUS_OWNER_MASK 0xc0000000
189
+ reserved
191
+/* Frame Reception Complete */
190
+
192
+#define RX_DESC_STATUS_RXGD (1 << 20)
191
+offset [0x8]
193
+/* Packet too long */
192
+ Command data: (DWORD access)
194
+#define RX_DESC_STATUS_PTLE (1 << 19)
193
+
195
+/* Receive Interrupt */
194
+ If last stored 'Command field' value is:
196
+#define RX_DESC_STATUS_RXINTR (1 << 16)
195
+
197
+
196
+ 1:
198
+#define RX_DESC_PKT_LEN(word) ((uint32_t) (word) & 0xffff)
197
+ stores value into OST event register
199
+
198
+ 2:
200
+typedef struct EMCModule {
199
+ stores value into OST status register, triggers
201
+ int rx_irq;
200
+ ACPI_DEVICE_OST QMP event from QEMU to external applications
202
+ int tx_irq;
201
+ with current values of OST event and status registers.
203
+ uint64_t base_addr;
202
+ other values:
204
+} EMCModule;
203
+ reserved
205
+
204
+
206
+typedef struct TestData {
205
+Typical usecases
207
+ const EMCModule *module;
206
+----------------
208
+} TestData;
207
+
209
+
208
+(x86) Detecting and enabling modern CPU hotplug interface
210
+static const EMCModule emc_module_list[] = {
209
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
211
+ {
210
+
212
+ .rx_irq = 15,
211
+QEMU starts with legacy CPU hotplug interface enabled. Detecting and
213
+ .tx_irq = 16,
212
+switching to modern interface is based on the 2 legacy CPU hotplug features:
214
+ .base_addr = 0xf0825000
213
+
215
+ },
214
+#. Writes into CPU bitmap are ignored.
216
+ {
215
+#. CPU bitmap always has bit #0 set, corresponding to boot CPU.
217
+ .rx_irq = 114,
216
+
218
+ .tx_irq = 115,
217
+Use following steps to detect and enable modern CPU hotplug interface:
219
+ .base_addr = 0xf0826000
218
+
220
+ }
219
+#. Store 0x0 to the 'CPU selector' register, attempting to switch to modern mode
221
+};
220
+#. Store 0x0 to the 'CPU selector' register, to ensure valid selector value
222
+
221
+#. Store 0x0 to the 'Command field' register
223
+/* Returns the index of the EMC module. */
222
+#. Read the 'Command data 2' register.
224
+static int emc_module_index(const EMCModule *mod)
223
+ If read value is 0x0, the modern interface is enabled.
225
+{
224
+ Otherwise legacy or no CPU hotplug interface available
226
+ ptrdiff_t diff = mod - emc_module_list;
225
+
227
+
226
+Get a cpu with pending event
228
+ g_assert_true(diff >= 0 && diff < ARRAY_SIZE(emc_module_list));
227
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^
229
+
228
+
230
+ return diff;
229
+#. Store 0x0 to the 'CPU selector' register.
231
+}
230
+#. Store 0x0 to the 'Command field' register.
232
+
231
+#. Read the 'CPU device status fields' register.
233
+static void packet_test_clear(void *sockets)
232
+#. If both bit #1 and bit #2 are clear in the value read, there is no CPU
234
+{
233
+ with a pending event and selected CPU remains unchanged.
235
+ int *test_sockets = sockets;
234
+#. Otherwise, read the 'Command data' register. The value read is the
236
+
235
+ selector of the CPU with the pending event (which is already selected).
237
+ close(test_sockets[0]);
236
+
238
+ g_free(test_sockets);
237
+Enumerate CPUs present/non present CPUs
239
+}
238
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
240
+
239
+
241
+static int *packet_test_init(int module_num, GString *cmd_line)
240
+#. Set the present CPU count to 0.
242
+{
241
+#. Set the iterator to 0.
243
+ int *test_sockets = g_new(int, 2);
242
+#. Store 0x0 to the 'CPU selector' register, to ensure that it's in
244
+ int ret = socketpair(PF_UNIX, SOCK_STREAM, 0, test_sockets);
243
+ a valid state and that access to other registers won't be ignored.
245
+ g_assert_cmpint(ret, != , -1);
244
+#. Store 0x0 to the 'Command field' register to make 'Command data'
246
+
245
+ register return 'CPU selector' value of selected CPU
247
+ /*
246
+#. Read the 'CPU device status fields' register.
248
+ * KISS and use -nic. We specify two nics (both emc{0,1}) because there's
247
+#. If bit #0 is set, increment the present CPU count.
249
+ * currently no way to specify only emc1: The driver implicitly relies on
248
+#. Increment the iterator.
250
+ * emc[i] == nd_table[i].
249
+#. Store the iterator to the 'CPU selector' register.
251
+ */
250
+#. Read the 'Command data' register.
252
+ if (module_num == 0) {
251
+#. If the value read is not zero, goto 05.
253
+ g_string_append_printf(cmd_line,
252
+#. Otherwise store 0x0 to the 'CPU selector' register, to put it
254
+ " -nic socket,fd=%d,model=" TYPE_NPCM7XX_EMC " "
253
+ into a valid state and exit.
255
+ " -nic user,model=" TYPE_NPCM7XX_EMC " ",
254
+ The iterator at this point equals "max_cpus".
256
+ test_sockets[1]);
255
diff --git a/docs/specs/acpi_cpu_hotplug.txt b/docs/specs/acpi_cpu_hotplug.txt
257
+ } else {
256
deleted file mode 100644
258
+ g_string_append_printf(cmd_line,
257
index XXXXXXX..XXXXXXX
259
+ " -nic user,model=" TYPE_NPCM7XX_EMC " "
258
--- a/docs/specs/acpi_cpu_hotplug.txt
260
+ " -nic socket,fd=%d,model=" TYPE_NPCM7XX_EMC " ",
259
+++ /dev/null
261
+ test_sockets[1]);
260
@@ -XXX,XX +XXX,XX @@
262
+ }
261
-QEMU<->ACPI BIOS CPU hotplug interface
263
+
262
---------------------------------------
264
+ g_test_queue_destroy(packet_test_clear, test_sockets);
263
-
265
+ return test_sockets;
264
-QEMU supports CPU hotplug via ACPI. This document
266
+}
265
-describes the interface between QEMU and the ACPI BIOS.
267
+
266
-
268
+static uint32_t emc_read(QTestState *qts, const EMCModule *mod,
267
-ACPI BIOS GPE.2 handler is dedicated for notifying OS about CPU hot-add
269
+ NPCM7xxPWMRegister regno)
268
-and hot-remove events.
270
+{
269
-
271
+ return qtest_readl(qts, mod->base_addr + regno * sizeof(uint32_t));
270
-============================================
272
+}
271
-Legacy ACPI CPU hotplug interface registers:
273
+
272
---------------------------------------------
274
+static void emc_write(QTestState *qts, const EMCModule *mod,
273
-CPU present bitmap for:
275
+ NPCM7xxPWMRegister regno, uint32_t value)
274
- ICH9-LPC (IO port 0x0cd8-0xcf7, 1-byte access)
276
+{
275
- PIIX-PM (IO port 0xaf00-0xaf1f, 1-byte access)
277
+ qtest_writel(qts, mod->base_addr + regno * sizeof(uint32_t), value);
276
- One bit per CPU. Bit position reflects corresponding CPU APIC ID. Read-only.
278
+}
277
- The first DWORD in bitmap is used in write mode to switch from legacy
279
+
278
- to modern CPU hotplug interface, write 0 into it to do switch.
280
+static void emc_read_tx_desc(QTestState *qts, uint32_t addr,
279
----------------------------------------------------------------
281
+ NPCM7xxEMCTxDesc *desc)
280
-QEMU sets corresponding CPU bit on hot-add event and issues SCI
282
+{
281
-with GPE.2 event set. CPU present map is read by ACPI BIOS GPE.2 handler
283
+ qtest_memread(qts, addr, desc, sizeof(*desc));
282
-to notify OS about CPU hot-add events. CPU hot-remove isn't supported.
284
+ desc->flags = le32_to_cpu(desc->flags);
283
-
285
+ desc->txbsa = le32_to_cpu(desc->txbsa);
284
-=====================================
286
+ desc->status_and_length = le32_to_cpu(desc->status_and_length);
285
-Modern ACPI CPU hotplug interface registers:
287
+ desc->ntxdsa = le32_to_cpu(desc->ntxdsa);
286
--------------------------------------
288
+}
287
-Register block base address:
289
+
288
- ICH9-LPC IO port 0x0cd8
290
+static void emc_write_tx_desc(QTestState *qts, const NPCM7xxEMCTxDesc *desc,
289
- PIIX-PM IO port 0xaf00
291
+ uint32_t addr)
290
-Register block size:
292
+{
291
- ACPI_CPU_HOTPLUG_REG_LEN = 12
293
+ NPCM7xxEMCTxDesc le_desc;
292
-
294
+
293
-All accesses to registers described below, imply little-endian byte order.
295
+ le_desc.flags = cpu_to_le32(desc->flags);
294
-
296
+ le_desc.txbsa = cpu_to_le32(desc->txbsa);
295
-Reserved resisters behavior:
297
+ le_desc.status_and_length = cpu_to_le32(desc->status_and_length);
296
- - write accesses are ignored
298
+ le_desc.ntxdsa = cpu_to_le32(desc->ntxdsa);
297
- - read accesses return all bits set to 0.
299
+ qtest_memwrite(qts, addr, &le_desc, sizeof(le_desc));
298
-
300
+}
299
-The last stored value in 'CPU selector' must refer to a possible CPU, otherwise
301
+
300
- - reads from any register return 0
302
+static void emc_read_rx_desc(QTestState *qts, uint32_t addr,
301
- - writes to any other register are ignored until valid value is stored into it
303
+ NPCM7xxEMCRxDesc *desc)
302
-On QEMU start, 'CPU selector' is initialized to a valid value, on reset it
304
+{
303
-keeps the current value.
305
+ qtest_memread(qts, addr, desc, sizeof(*desc));
304
-
306
+ desc->status_and_length = le32_to_cpu(desc->status_and_length);
305
-read access:
307
+ desc->rxbsa = le32_to_cpu(desc->rxbsa);
306
- offset:
308
+ desc->reserved = le32_to_cpu(desc->reserved);
307
- [0x0-0x3] Command data 2: (DWORD access)
309
+ desc->nrxdsa = le32_to_cpu(desc->nrxdsa);
308
- if value last stored in 'Command field':
310
+}
309
- 0: reads as 0x0
311
+
310
- 3: upper 32 bits of architecture specific CPU ID value
312
+static void emc_write_rx_desc(QTestState *qts, const NPCM7xxEMCRxDesc *desc,
311
- other values: reserved
313
+ uint32_t addr)
312
- [0x4] CPU device status fields: (1 byte access)
314
+{
313
- bits:
315
+ NPCM7xxEMCRxDesc le_desc;
314
- 0: Device is enabled and may be used by guest
316
+
315
- 1: Device insert event, used to distinguish device for which
317
+ le_desc.status_and_length = cpu_to_le32(desc->status_and_length);
316
- no device check event to OSPM was issued.
318
+ le_desc.rxbsa = cpu_to_le32(desc->rxbsa);
317
- It's valid only when bit 0 is set.
319
+ le_desc.reserved = cpu_to_le32(desc->reserved);
318
- 2: Device remove event, used to distinguish device for which
320
+ le_desc.nrxdsa = cpu_to_le32(desc->nrxdsa);
319
- no device eject request to OSPM was issued. Firmware must
321
+ qtest_memwrite(qts, addr, &le_desc, sizeof(le_desc));
320
- ignore this bit.
322
+}
321
- 3: reserved and should be ignored by OSPM
323
+
322
- 4: if set to 1, OSPM requests firmware to perform device eject.
324
+/*
323
- 5-7: reserved and should be ignored by OSPM
325
+ * Reset the EMC module.
324
- [0x5-0x7] reserved
326
+ * The module must be reset before, e.g., TXDLSA,RXDLSA are changed.
325
- [0x8] Command data: (DWORD access)
327
+ */
326
- contains 0 unless value last stored in 'Command field' is one of:
328
+static bool emc_soft_reset(QTestState *qts, const EMCModule *mod)
327
- 0: contains 'CPU selector' value of a CPU with pending event[s]
329
+{
328
- 3: lower 32 bits of architecture specific CPU ID value
330
+ uint32_t val;
329
- (in x86 case: APIC ID)
331
+ uint64_t end_time;
330
-
332
+
331
-write access:
333
+ emc_write(qts, mod, REG_MCMDR, REG_MCMDR_SWR);
332
- offset:
334
+
333
- [0x0-0x3] CPU selector: (DWORD access)
335
+ /*
334
- selects active CPU device. All following accesses to other
336
+ * Wait for device to reset as the linux driver does.
335
- registers will read/store data from/to selected CPU.
337
+ * During reset the AHB reads 0 for all registers. So first wait for
336
- Valid values: [0 .. max_cpus)
338
+ * something that resets to non-zero, and then wait for SWR becoming 0.
337
- [0x4] CPU device control fields: (1 byte access)
339
+ */
338
- bits:
340
+ end_time = g_get_monotonic_time() + TIMEOUT_SECONDS * G_TIME_SPAN_SECOND;
339
- 0: reserved, OSPM must clear it before writing to register.
341
+
340
- 1: if set to 1 clears device insert event, set by OSPM
342
+ do {
341
- after it has emitted device check event for the
343
+ qtest_clock_step(qts, 100);
342
- selected CPU device
344
+ val = emc_read(qts, mod, REG_FFTCR);
343
- 2: if set to 1 clears device remove event, set by OSPM
345
+ } while (val == 0 && g_get_monotonic_time() < end_time);
344
- after it has emitted device eject request for the
346
+ if (val != 0) {
345
- selected CPU device.
347
+ do {
346
- 3: if set to 1 initiates device eject, set by OSPM when it
348
+ qtest_clock_step(qts, 100);
347
- triggers CPU device removal and calls _EJ0 method or by firmware
349
+ val = emc_read(qts, mod, REG_MCMDR);
348
- when bit #4 is set. In case bit #4 were set, it's cleared as
350
+ if ((val & REG_MCMDR_SWR) == 0) {
349
- part of device eject.
351
+ /*
350
- 4: if set to 1, OSPM hands over device eject to firmware.
352
+ * N.B. The CAMs have been reset here, so macaddr matching of
351
- Firmware shall issue device eject request as described above
353
+ * incoming packets will not work.
352
- (bit #3) and OSPM should not touch device eject bit (#3) in case
354
+ */
353
- it's asked firmware to perform CPU device eject.
355
+ return true;
354
- 5-7: reserved, OSPM must clear them before writing to register
356
+ }
355
- [0x5] Command field: (1 byte access)
357
+ } while (g_get_monotonic_time() < end_time);
356
- value:
358
+ }
357
- 0: selects a CPU device with inserting/removing events and
359
+
358
- following reads from 'Command data' register return
360
+ g_message("%s: Timeout expired", __func__);
359
- selected CPU ('CPU selector' value).
361
+ return false;
360
- If no CPU with events found, the current 'CPU selector' doesn't
362
+}
361
- change and corresponding insert/remove event flags are not modified.
363
+
362
- 1: following writes to 'Command data' register set OST event
364
+/* Check emc registers are reset to default value. */
363
- register in QEMU
365
+static void test_init(gconstpointer test_data)
364
- 2: following writes to 'Command data' register set OST status
366
+{
365
- register in QEMU
367
+ const TestData *td = test_data;
366
- 3: following reads from 'Command data' and 'Command data 2' return
368
+ const EMCModule *mod = td->module;
367
- architecture specific CPU ID value for currently selected CPU.
369
+ QTestState *qts = qtest_init("-machine quanta-gsj");
368
- other values: reserved
370
+ int i;
369
- [0x6-0x7] reserved
371
+
370
- [0x8] Command data: (DWORD access)
372
+#define CHECK_REG(regno, value) \
371
- if last stored 'Command field' value:
373
+ do { \
372
- 1: stores value into OST event register
374
+ g_assert_cmphex(emc_read(qts, mod, (regno)), ==, (value)); \
373
- 2: stores value into OST status register, triggers
375
+ } while (0)
374
- ACPI_DEVICE_OST QMP event from QEMU to external applications
376
+
375
- with current values of OST event and status registers.
377
+ CHECK_REG(REG_CAMCMR, 0);
376
- other values: reserved
378
+ CHECK_REG(REG_CAMEN, 0);
377
-
379
+ CHECK_REG(REG_TXDLSA, 0xfffffffc);
378
-Typical usecases:
380
+ CHECK_REG(REG_RXDLSA, 0xfffffffc);
379
- - (x86) Detecting and enabling modern CPU hotplug interface.
381
+ CHECK_REG(REG_MCMDR, 0);
380
- QEMU starts with legacy CPU hotplug interface enabled. Detecting and
382
+ CHECK_REG(REG_MIID, 0);
381
- switching to modern interface is based on the 2 legacy CPU hotplug features:
383
+ CHECK_REG(REG_MIIDA, 0x00900000);
382
- 1. Writes into CPU bitmap are ignored.
384
+ CHECK_REG(REG_FFTCR, 0x0101);
383
- 2. CPU bitmap always has bit#0 set, corresponding to boot CPU.
385
+ CHECK_REG(REG_DMARFC, 0x0800);
384
-
386
+ CHECK_REG(REG_MIEN, 0);
385
- Use following steps to detect and enable modern CPU hotplug interface:
387
+ CHECK_REG(REG_MISTA, 0);
386
- 1. Store 0x0 to the 'CPU selector' register,
388
+ CHECK_REG(REG_MGSTA, 0);
387
- attempting to switch to modern mode
389
+ CHECK_REG(REG_MPCNT, 0x7fff);
388
- 2. Store 0x0 to the 'CPU selector' register,
390
+ CHECK_REG(REG_MRPC, 0);
389
- to ensure valid selector value
391
+ CHECK_REG(REG_MRPCC, 0);
390
- 3. Store 0x0 to the 'Command field' register,
392
+ CHECK_REG(REG_MREPC, 0);
391
- 4. Read the 'Command data 2' register.
393
+ CHECK_REG(REG_DMARFS, 0);
392
- If read value is 0x0, the modern interface is enabled.
394
+ CHECK_REG(REG_CTXDSA, 0);
393
- Otherwise legacy or no CPU hotplug interface available
395
+ CHECK_REG(REG_CTXBSA, 0);
394
-
396
+ CHECK_REG(REG_CRXDSA, 0);
395
- - Get a cpu with pending event
397
+ CHECK_REG(REG_CRXBSA, 0);
396
- 1. Store 0x0 to the 'CPU selector' register.
398
+
397
- 2. Store 0x0 to the 'Command field' register.
399
+#undef CHECK_REG
398
- 3. Read the 'CPU device status fields' register.
400
+
399
- 4. If both bit#1 and bit#2 are clear in the value read, there is no CPU
401
+ for (i = 0; i < NUM_CAMML_REGS; ++i) {
400
- with a pending event and selected CPU remains unchanged.
402
+ g_assert_cmpuint(emc_read(qts, mod, REG_CAMM_BASE + i * 2), ==,
401
- 5. Otherwise, read the 'Command data' register. The value read is the
403
+ 0);
402
- selector of the CPU with the pending event (which is already
404
+ g_assert_cmpuint(emc_read(qts, mod, REG_CAML_BASE + i * 2), ==,
403
- selected).
405
+ 0);
404
-
406
+ }
405
- - Enumerate CPUs present/non present CPUs
407
+
406
- 01. Set the present CPU count to 0.
408
+ qtest_quit(qts);
407
- 02. Set the iterator to 0.
409
+}
408
- 03. Store 0x0 to the 'CPU selector' register, to ensure that it's in
410
+
409
- a valid state and that access to other registers won't be ignored.
411
+static bool emc_wait_irq(QTestState *qts, const EMCModule *mod, int step,
410
- 04. Store 0x0 to the 'Command field' register to make 'Command data'
412
+ bool is_tx)
411
- register return 'CPU selector' value of selected CPU
413
+{
412
- 05. Read the 'CPU device status fields' register.
414
+ uint64_t end_time =
413
- 06. If bit#0 is set, increment the present CPU count.
415
+ g_get_monotonic_time() + TIMEOUT_SECONDS * G_TIME_SPAN_SECOND;
414
- 07. Increment the iterator.
416
+
415
- 08. Store the iterator to the 'CPU selector' register.
417
+ do {
416
- 09. Read the 'Command data' register.
418
+ if (qtest_get_irq(qts, is_tx ? mod->tx_irq : mod->rx_irq)) {
417
- 10. If the value read is not zero, goto 05.
419
+ return true;
418
- 11. Otherwise store 0x0 to the 'CPU selector' register, to put it
420
+ }
419
- into a valid state and exit.
421
+ qtest_clock_step(qts, step);
420
- The iterator at this point equals "max_cpus".
422
+ } while (g_get_monotonic_time() < end_time);
421
diff --git a/docs/specs/index.rst b/docs/specs/index.rst
423
+
424
+ g_message("%s: Timeout expired", __func__);
425
+ return false;
426
+}
427
+
428
+static bool emc_wait_mista(QTestState *qts, const EMCModule *mod, int step,
429
+ uint32_t flag)
430
+{
431
+ uint64_t end_time =
432
+ g_get_monotonic_time() + TIMEOUT_SECONDS * G_TIME_SPAN_SECOND;
433
+
434
+ do {
435
+ uint32_t mista = emc_read(qts, mod, REG_MISTA);
436
+ if (mista & flag) {
437
+ return true;
438
+ }
439
+ qtest_clock_step(qts, step);
440
+ } while (g_get_monotonic_time() < end_time);
441
+
442
+ g_message("%s: Timeout expired", __func__);
443
+ return false;
444
+}
445
+
446
+static bool wait_socket_readable(int fd)
447
+{
448
+ fd_set read_fds;
449
+ struct timeval tv;
450
+ int rv;
451
+
452
+ FD_ZERO(&read_fds);
453
+ FD_SET(fd, &read_fds);
454
+ tv.tv_sec = TIMEOUT_SECONDS;
455
+ tv.tv_usec = 0;
456
+ rv = select(fd + 1, &read_fds, NULL, NULL, &tv);
457
+ if (rv == -1) {
458
+ perror("select");
459
+ } else if (rv == 0) {
460
+ g_message("%s: Timeout expired", __func__);
461
+ }
462
+ return rv == 1;
463
+}
464
+
465
+/* Initialize *desc (in host endian format). */
466
+static void init_tx_desc(NPCM7xxEMCTxDesc *desc, size_t count,
467
+ uint32_t desc_addr)
468
+{
469
+ g_assert(count >= 2);
470
+ memset(&desc[0], 0, sizeof(*desc) * count);
471
+ /* Leave the last one alone, owned by the cpu -> stops transmission. */
472
+ for (size_t i = 0; i < count - 1; ++i) {
473
+ desc[i].flags =
474
+ (TX_DESC_FLAG_OWNER_MASK | /* owner = 1: emc */
475
+ TX_DESC_FLAG_INTEN |
476
+ 0 | /* crc append = 0 */
477
+ 0 /* padding enable = 0 */);
478
+ desc[i].status_and_length =
479
+ (0 | /* collision count = 0 */
480
+ 0 | /* SQE = 0 */
481
+ 0 | /* PAU = 0 */
482
+ 0 | /* TXHA = 0 */
483
+ 0 | /* LC = 0 */
484
+ 0 | /* TXABT = 0 */
485
+ 0 | /* NCS = 0 */
486
+ 0 | /* EXDEF = 0 */
487
+ 0 | /* TXCP = 0 */
488
+ 0 | /* DEF = 0 */
489
+ 0 | /* TXINTR = 0 */
490
+ 0 /* length filled in later */);
491
+ desc[i].ntxdsa = desc_addr + (i + 1) * sizeof(*desc);
492
+ }
493
+}
494
+
495
+static void enable_tx(QTestState *qts, const EMCModule *mod,
496
+ const NPCM7xxEMCTxDesc *desc, size_t count,
497
+ uint32_t desc_addr, uint32_t mien_flags)
498
+{
499
+ /* Write the descriptors to guest memory. */
500
+ for (size_t i = 0; i < count; ++i) {
501
+ emc_write_tx_desc(qts, desc + i, desc_addr + i * sizeof(*desc));
502
+ }
503
+
504
+ /* Trigger sending the packet. */
505
+ /* The module must be reset before changing TXDLSA. */
506
+ g_assert(emc_soft_reset(qts, mod));
507
+ emc_write(qts, mod, REG_TXDLSA, desc_addr);
508
+ emc_write(qts, mod, REG_CTXDSA, ~0);
509
+ emc_write(qts, mod, REG_MIEN, REG_MIEN_ENTXCP | mien_flags);
510
+ {
511
+ uint32_t mcmdr = emc_read(qts, mod, REG_MCMDR);
512
+ mcmdr |= REG_MCMDR_TXON;
513
+ emc_write(qts, mod, REG_MCMDR, mcmdr);
514
+ }
515
+
516
+ /* Prod the device to send the packet. */
517
+ emc_write(qts, mod, REG_TSDR, 1);
518
+}
519
+
520
+static void emc_send_verify1(QTestState *qts, const EMCModule *mod, int fd,
521
+ bool with_irq, uint32_t desc_addr,
522
+ uint32_t next_desc_addr,
523
+ const char *test_data, int test_size)
524
+{
525
+ NPCM7xxEMCTxDesc result_desc;
526
+ uint32_t expected_mask, expected_value, recv_len;
527
+ int ret;
528
+ char buffer[TX_DATA_LEN];
529
+
530
+ g_assert(wait_socket_readable(fd));
531
+
532
+ /* Read the descriptor back. */
533
+ emc_read_tx_desc(qts, desc_addr, &result_desc);
534
+ /* Descriptor should be owned by cpu now. */
535
+ g_assert((result_desc.flags & TX_DESC_FLAG_OWNER_MASK) == 0);
536
+ /* Test the status bits, ignoring the length field. */
537
+ expected_mask = 0xffff << 16;
538
+ expected_value = TX_DESC_STATUS_TXCP;
539
+ if (with_irq) {
540
+ expected_value |= TX_DESC_STATUS_TXINTR;
541
+ }
542
+ g_assert_cmphex((result_desc.status_and_length & expected_mask), ==,
543
+ expected_value);
544
+
545
+ /* Check data sent to the backend. */
546
+ recv_len = ~0;
547
+ ret = qemu_recv(fd, &recv_len, sizeof(recv_len), MSG_DONTWAIT);
548
+ g_assert_cmpint(ret, == , sizeof(recv_len));
549
+
550
+ g_assert(wait_socket_readable(fd));
551
+ memset(buffer, 0xff, sizeof(buffer));
552
+ ret = qemu_recv(fd, buffer, test_size, MSG_DONTWAIT);
553
+ g_assert_cmpmem(buffer, ret, test_data, test_size);
554
+}
555
+
556
+static void emc_send_verify(QTestState *qts, const EMCModule *mod, int fd,
557
+ bool with_irq)
558
+{
559
+ NPCM7xxEMCTxDesc desc[NUM_TX_DESCRIPTORS];
560
+ uint32_t desc_addr = DESC_ADDR;
561
+ static const char test1_data[] = "TEST1";
562
+ static const char test2_data[] = "Testing 1 2 3 ...";
563
+ uint32_t data1_addr = DATA_ADDR;
564
+ uint32_t data2_addr = data1_addr + sizeof(test1_data);
565
+ bool got_tdu;
566
+ uint32_t end_desc_addr;
567
+
568
+ /* Prepare test data buffer. */
569
+ qtest_memwrite(qts, data1_addr, test1_data, sizeof(test1_data));
570
+ qtest_memwrite(qts, data2_addr, test2_data, sizeof(test2_data));
571
+
572
+ init_tx_desc(&desc[0], NUM_TX_DESCRIPTORS, desc_addr);
573
+ desc[0].txbsa = data1_addr;
574
+ desc[0].status_and_length |= sizeof(test1_data);
575
+ desc[1].txbsa = data2_addr;
576
+ desc[1].status_and_length |= sizeof(test2_data);
577
+
578
+ enable_tx(qts, mod, &desc[0], NUM_TX_DESCRIPTORS, desc_addr,
579
+ with_irq ? REG_MIEN_ENTXINTR : 0);
580
+
581
+ /*
582
+ * It's problematic to observe the interrupt for each packet.
583
+ * Instead just wait until all the packets go out.
584
+ */
585
+ got_tdu = false;
586
+ while (!got_tdu) {
587
+ if (with_irq) {
588
+ g_assert_true(emc_wait_irq(qts, mod, TX_STEP_COUNT,
589
+ /*is_tx=*/true));
590
+ } else {
591
+ g_assert_true(emc_wait_mista(qts, mod, TX_STEP_COUNT,
592
+ REG_MISTA_TXINTR));
593
+ }
594
+ got_tdu = !!(emc_read(qts, mod, REG_MISTA) & REG_MISTA_TDU);
595
+ /* If we don't have TDU yet, reset the interrupt. */
596
+ if (!got_tdu) {
597
+ emc_write(qts, mod, REG_MISTA,
598
+ emc_read(qts, mod, REG_MISTA) & 0xffff0000);
599
+ }
600
+ }
601
+
602
+ end_desc_addr = desc_addr + 2 * sizeof(desc[0]);
603
+ g_assert_cmphex(emc_read(qts, mod, REG_CTXDSA), ==, end_desc_addr);
604
+ g_assert_cmphex(emc_read(qts, mod, REG_MISTA), ==,
605
+ REG_MISTA_TXCP | REG_MISTA_TXINTR | REG_MISTA_TDU);
606
+
607
+ emc_send_verify1(qts, mod, fd, with_irq,
608
+ desc_addr, end_desc_addr,
609
+ test1_data, sizeof(test1_data));
610
+ emc_send_verify1(qts, mod, fd, with_irq,
611
+ desc_addr + sizeof(desc[0]), end_desc_addr,
612
+ test2_data, sizeof(test2_data));
613
+}
614
+
615
+/* Initialize *desc (in host endian format). */
616
+static void init_rx_desc(NPCM7xxEMCRxDesc *desc, size_t count,
617
+ uint32_t desc_addr, uint32_t data_addr)
618
+{
619
+ g_assert_true(count >= 2);
620
+ memset(desc, 0, sizeof(*desc) * count);
621
+ desc[0].rxbsa = data_addr;
622
+ desc[0].status_and_length =
623
+ (0b10 << RX_DESC_STATUS_OWNER_SHIFT | /* owner = 10: emc */
624
+ 0 | /* RP = 0 */
625
+ 0 | /* ALIE = 0 */
626
+ 0 | /* RXGD = 0 */
627
+ 0 | /* PTLE = 0 */
628
+ 0 | /* CRCE = 0 */
629
+ 0 | /* RXINTR = 0 */
630
+ 0 /* length (filled in later) */);
631
+ /* Leave the last one alone, owned by the cpu -> stops transmission. */
632
+ desc[0].nrxdsa = desc_addr + sizeof(*desc);
633
+}
634
+
635
+static void enable_rx(QTestState *qts, const EMCModule *mod,
636
+ const NPCM7xxEMCRxDesc *desc, size_t count,
637
+ uint32_t desc_addr, uint32_t mien_flags,
638
+ uint32_t mcmdr_flags)
639
+{
640
+ /*
641
+ * Write the descriptor to guest memory.
642
+ * FWIW, IWBN if the docs said the buffer needs to be at least DMARFC
643
+ * bytes.
644
+ */
645
+ for (size_t i = 0; i < count; ++i) {
646
+ emc_write_rx_desc(qts, desc + i, desc_addr + i * sizeof(*desc));
647
+ }
648
+
649
+ /* Trigger receiving the packet. */
650
+ /* The module must be reset before changing RXDLSA. */
651
+ g_assert(emc_soft_reset(qts, mod));
652
+ emc_write(qts, mod, REG_RXDLSA, desc_addr);
653
+ emc_write(qts, mod, REG_MIEN, REG_MIEN_ENRXGD | mien_flags);
654
+
655
+ /*
656
+ * We don't know what the device's macaddr is, so just accept all
657
+ * unicast packets (AUP).
658
+ */
659
+ emc_write(qts, mod, REG_CAMCMR, REG_CAMCMR_AUP);
660
+ emc_write(qts, mod, REG_CAMEN, 1 << 0);
661
+ {
662
+ uint32_t mcmdr = emc_read(qts, mod, REG_MCMDR);
663
+ mcmdr |= REG_MCMDR_RXON | mcmdr_flags;
664
+ emc_write(qts, mod, REG_MCMDR, mcmdr);
665
+ }
666
+
667
+ /* Prod the device to accept a packet. */
668
+ emc_write(qts, mod, REG_RSDR, 1);
669
+}
670
+
671
+static void emc_recv_verify(QTestState *qts, const EMCModule *mod, int fd,
672
+ bool with_irq)
673
+{
674
+ NPCM7xxEMCRxDesc desc[NUM_RX_DESCRIPTORS];
675
+ uint32_t desc_addr = DESC_ADDR;
676
+ uint32_t data_addr = DATA_ADDR;
677
+ int ret;
678
+ uint32_t expected_mask, expected_value;
679
+ NPCM7xxEMCRxDesc result_desc;
680
+
681
+ /* Prepare test data buffer. */
682
+ const char test[RX_DATA_LEN] = "TEST";
683
+ int len = htonl(sizeof(test));
684
+ const struct iovec iov[] = {
685
+ {
686
+ .iov_base = &len,
687
+ .iov_len = sizeof(len),
688
+ },{
689
+ .iov_base = (char *) test,
690
+ .iov_len = sizeof(test),
691
+ },
692
+ };
693
+
694
+ /*
695
+ * Reset the device BEFORE sending a test packet, otherwise the packet
696
+ * may get swallowed by an active device of an earlier test.
697
+ */
698
+ init_rx_desc(&desc[0], NUM_RX_DESCRIPTORS, desc_addr, data_addr);
699
+ enable_rx(qts, mod, &desc[0], NUM_RX_DESCRIPTORS, desc_addr,
700
+ with_irq ? REG_MIEN_ENRXINTR : 0, 0);
701
+
702
+ /* Send test packet to device's socket. */
703
+ ret = iov_send(fd, iov, 2, 0, sizeof(len) + sizeof(test));
704
+ g_assert_cmpint(ret, == , sizeof(test) + sizeof(len));
705
+
706
+ /* Wait for RX interrupt. */
707
+ if (with_irq) {
708
+ g_assert_true(emc_wait_irq(qts, mod, RX_STEP_COUNT, /*is_tx=*/false));
709
+ } else {
710
+ g_assert_true(emc_wait_mista(qts, mod, RX_STEP_COUNT, REG_MISTA_RXGD));
711
+ }
712
+
713
+ g_assert_cmphex(emc_read(qts, mod, REG_CRXDSA), ==,
714
+ desc_addr + sizeof(desc[0]));
715
+
716
+ expected_mask = 0xffff;
717
+ expected_value = (REG_MISTA_DENI |
718
+ REG_MISTA_RXGD |
719
+ REG_MISTA_RXINTR);
720
+ g_assert_cmphex((emc_read(qts, mod, REG_MISTA) & expected_mask),
721
+ ==, expected_value);
722
+
723
+ /* Read the descriptor back. */
724
+ emc_read_rx_desc(qts, desc_addr, &result_desc);
725
+ /* Descriptor should be owned by cpu now. */
726
+ g_assert((result_desc.status_and_length & RX_DESC_STATUS_OWNER_MASK) == 0);
727
+ /* Test the status bits, ignoring the length field. */
728
+ expected_mask = 0xffff << 16;
729
+ expected_value = RX_DESC_STATUS_RXGD;
730
+ if (with_irq) {
731
+ expected_value |= RX_DESC_STATUS_RXINTR;
732
+ }
733
+ g_assert_cmphex((result_desc.status_and_length & expected_mask), ==,
734
+ expected_value);
735
+ g_assert_cmpint(RX_DESC_PKT_LEN(result_desc.status_and_length), ==,
736
+ RX_DATA_LEN + CRC_LENGTH);
737
+
738
+ {
739
+ char buffer[RX_DATA_LEN];
740
+ qtest_memread(qts, data_addr, buffer, sizeof(buffer));
741
+ g_assert_cmpstr(buffer, == , "TEST");
742
+ }
743
+}
744
+
745
+static void emc_test_ptle(QTestState *qts, const EMCModule *mod, int fd)
746
+{
747
+ NPCM7xxEMCRxDesc desc[NUM_RX_DESCRIPTORS];
748
+ uint32_t desc_addr = DESC_ADDR;
749
+ uint32_t data_addr = DATA_ADDR;
750
+ int ret;
751
+ NPCM7xxEMCRxDesc result_desc;
752
+ uint32_t expected_mask, expected_value;
753
+
754
+ /* Prepare test data buffer. */
755
+#define PTLE_DATA_LEN 1600
756
+ char test_data[PTLE_DATA_LEN];
757
+ int len = htonl(sizeof(test_data));
758
+ const struct iovec iov[] = {
759
+ {
760
+ .iov_base = &len,
761
+ .iov_len = sizeof(len),
762
+ },{
763
+ .iov_base = (char *) test_data,
764
+ .iov_len = sizeof(test_data),
765
+ },
766
+ };
767
+ memset(test_data, 42, sizeof(test_data));
768
+
769
+ /*
770
+ * Reset the device BEFORE sending a test packet, otherwise the packet
771
+ * may get swallowed by an active device of an earlier test.
772
+ */
773
+ init_rx_desc(&desc[0], NUM_RX_DESCRIPTORS, desc_addr, data_addr);
774
+ enable_rx(qts, mod, &desc[0], NUM_RX_DESCRIPTORS, desc_addr,
775
+ REG_MIEN_ENRXINTR, REG_MCMDR_ALP);
776
+
777
+ /* Send test packet to device's socket. */
778
+ ret = iov_send(fd, iov, 2, 0, sizeof(len) + sizeof(test_data));
779
+ g_assert_cmpint(ret, == , sizeof(test_data) + sizeof(len));
780
+
781
+ /* Wait for RX interrupt. */
782
+ g_assert_true(emc_wait_irq(qts, mod, RX_STEP_COUNT, /*is_tx=*/false));
783
+
784
+ /* Read the descriptor back. */
785
+ emc_read_rx_desc(qts, desc_addr, &result_desc);
786
+ /* Descriptor should be owned by cpu now. */
787
+ g_assert((result_desc.status_and_length & RX_DESC_STATUS_OWNER_MASK) == 0);
788
+ /* Test the status bits, ignoring the length field. */
789
+ expected_mask = 0xffff << 16;
790
+ expected_value = (RX_DESC_STATUS_RXGD |
791
+ RX_DESC_STATUS_PTLE |
792
+ RX_DESC_STATUS_RXINTR);
793
+ g_assert_cmphex((result_desc.status_and_length & expected_mask), ==,
794
+ expected_value);
795
+ g_assert_cmpint(RX_DESC_PKT_LEN(result_desc.status_and_length), ==,
796
+ PTLE_DATA_LEN + CRC_LENGTH);
797
+
798
+ {
799
+ char buffer[PTLE_DATA_LEN];
800
+ qtest_memread(qts, data_addr, buffer, sizeof(buffer));
801
+ g_assert(memcmp(buffer, test_data, PTLE_DATA_LEN) == 0);
802
+ }
803
+}
804
+
805
+static void test_tx(gconstpointer test_data)
806
+{
807
+ const TestData *td = test_data;
808
+ GString *cmd_line = g_string_new("-machine quanta-gsj");
809
+ int *test_sockets = packet_test_init(emc_module_index(td->module),
810
+ cmd_line);
811
+ QTestState *qts = qtest_init(cmd_line->str);
812
+
813
+ /*
814
+ * TODO: For pedantic correctness test_sockets[0] should be closed after
815
+ * the fork and before the exec, but that will require some harness
816
+ * improvements.
817
+ */
818
+ close(test_sockets[1]);
819
+ /* Defensive programming */
820
+ test_sockets[1] = -1;
821
+
822
+ qtest_irq_intercept_in(qts, "/machine/soc/a9mpcore/gic");
823
+
824
+ emc_send_verify(qts, td->module, test_sockets[0], /*with_irq=*/false);
825
+ emc_send_verify(qts, td->module, test_sockets[0], /*with_irq=*/true);
826
+
827
+ qtest_quit(qts);
828
+}
829
+
830
+static void test_rx(gconstpointer test_data)
831
+{
832
+ const TestData *td = test_data;
833
+ GString *cmd_line = g_string_new("-machine quanta-gsj");
834
+ int *test_sockets = packet_test_init(emc_module_index(td->module),
835
+ cmd_line);
836
+ QTestState *qts = qtest_init(cmd_line->str);
837
+
838
+ /*
839
+ * TODO: For pedantic correctness test_sockets[0] should be closed after
840
+ * the fork and before the exec, but that will require some harness
841
+ * improvements.
842
+ */
843
+ close(test_sockets[1]);
844
+ /* Defensive programming */
845
+ test_sockets[1] = -1;
846
+
847
+ qtest_irq_intercept_in(qts, "/machine/soc/a9mpcore/gic");
848
+
849
+ emc_recv_verify(qts, td->module, test_sockets[0], /*with_irq=*/false);
850
+ emc_recv_verify(qts, td->module, test_sockets[0], /*with_irq=*/true);
851
+ emc_test_ptle(qts, td->module, test_sockets[0]);
852
+
853
+ qtest_quit(qts);
854
+}
855
+
856
+static void emc_add_test(const char *name, const TestData* td,
857
+ GTestDataFunc fn)
858
+{
859
+ g_autofree char *full_name = g_strdup_printf(
860
+ "npcm7xx_emc/emc[%d]/%s", emc_module_index(td->module), name);
861
+ qtest_add_data_func(full_name, td, fn);
862
+}
863
+#define add_test(name, td) emc_add_test(#name, td, test_##name)
864
+
865
+int main(int argc, char **argv)
866
+{
867
+ TestData test_data_list[ARRAY_SIZE(emc_module_list)];
868
+
869
+ g_test_init(&argc, &argv, NULL);
870
+
871
+ for (int i = 0; i < ARRAY_SIZE(emc_module_list); ++i) {
872
+ TestData *td = &test_data_list[i];
873
+
874
+ td->module = &emc_module_list[i];
875
+
876
+ add_test(init, td);
877
+ add_test(tx, td);
878
+ add_test(rx, td);
879
+ }
880
+
881
+ return g_test_run();
882
+}
883
diff --git a/tests/qtest/meson.build b/tests/qtest/meson.build
884
index XXXXXXX..XXXXXXX 100644
422
index XXXXXXX..XXXXXXX 100644
885
--- a/tests/qtest/meson.build
423
--- a/docs/specs/index.rst
886
+++ b/tests/qtest/meson.build
424
+++ b/docs/specs/index.rst
887
@@ -XXX,XX +XXX,XX @@ qtests_npcm7xx = \
425
@@ -XXX,XX +XXX,XX @@ guest hardware that is specific to QEMU.
888
'npcm7xx_rng-test',
426
acpi_hw_reduced_hotplug
889
'npcm7xx_smbus-test',
427
tpm
890
'npcm7xx_timer-test',
428
acpi_hest_ghes
891
- 'npcm7xx_watchdog_timer-test']
429
+ acpi_cpu_hotplug
892
+ 'npcm7xx_watchdog_timer-test'] + \
893
+ (slirp.found() ? ['npcm7xx_emc-test'] : [])
894
qtests_arm = \
895
(config_all_devices.has_key('CONFIG_CMSDK_APB_DUALTIMER') ? ['cmsdk-apb-dualtimer-test'] : []) + \
896
(config_all_devices.has_key('CONFIG_CMSDK_APB_TIMER') ? ['cmsdk-apb-timer-test'] : []) + \
897
--
430
--
898
2.20.1
431
2.20.1
899
432
900
433
diff view generated by jsdifflib
1
The function tc6393xb_draw_graphic32() is called in exactly one place,
1
Convert the acpi memory hotplug spec to rST.
2
so just inline the function body at its callsite. This allows us to
2
3
drop the template header entirely.
3
Note that this includes converting a lot of weird whitespace
4
4
characters to plain old spaces (the rST parser does not like
5
The code move includes a single added space after 'for' to fix
5
whatever the old ones were).
6
the coding style.
7
6
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Reviewed-by: Igor Mammedov <imammedo@redhat.com>
10
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Message-id: 20210727170414.3368-3-peter.maydell@linaro.org
11
Message-id: 20210215103215.4944-5-peter.maydell@linaro.org
12
---
10
---
13
hw/display/tc6393xb_template.h | 45 ----------------------------------
11
docs/specs/acpi_mem_hotplug.rst | 128 ++++++++++++++++++++++++++++++++
14
hw/display/tc6393xb.c | 23 ++++++++++++++---
12
docs/specs/acpi_mem_hotplug.txt | 94 -----------------------
15
2 files changed, 19 insertions(+), 49 deletions(-)
13
docs/specs/index.rst | 1 +
16
delete mode 100644 hw/display/tc6393xb_template.h
14
3 files changed, 129 insertions(+), 94 deletions(-)
17
15
create mode 100644 docs/specs/acpi_mem_hotplug.rst
18
diff --git a/hw/display/tc6393xb_template.h b/hw/display/tc6393xb_template.h
16
delete mode 100644 docs/specs/acpi_mem_hotplug.txt
17
18
diff --git a/docs/specs/acpi_mem_hotplug.rst b/docs/specs/acpi_mem_hotplug.rst
19
new file mode 100644
20
index XXXXXXX..XXXXXXX
21
--- /dev/null
22
+++ b/docs/specs/acpi_mem_hotplug.rst
23
@@ -XXX,XX +XXX,XX @@
24
+QEMU<->ACPI BIOS memory hotplug interface
25
+=========================================
26
+
27
+ACPI BIOS GPE.3 handler is dedicated for notifying OS about memory hot-add
28
+and hot-remove events.
29
+
30
+Memory hot-plug interface (IO port 0xa00-0xa17, 1-4 byte access)
31
+----------------------------------------------------------------
32
+
33
+Read access behavior
34
+^^^^^^^^^^^^^^^^^^^^
35
+
36
+[0x0-0x3]
37
+ Lo part of memory device phys address
38
+[0x4-0x7]
39
+ Hi part of memory device phys address
40
+[0x8-0xb]
41
+ Lo part of memory device size in bytes
42
+[0xc-0xf]
43
+ Hi part of memory device size in bytes
44
+[0x10-0x13]
45
+ Memory device proximity domain
46
+[0x14]
47
+ Memory device status fields
48
+
49
+ bits:
50
+
51
+ 0:
52
+ Device is enabled and may be used by guest
53
+ 1:
54
+ Device insert event, used to distinguish device for which
55
+ no device check event to OSPM was issued.
56
+ It's valid only when bit 1 is set.
57
+ 2:
58
+ Device remove event, used to distinguish device for which
59
+ no device eject request to OSPM was issued.
60
+ 3-7:
61
+ reserved and should be ignored by OSPM
62
+
63
+[0x15-0x17]
64
+ reserved
65
+
66
+Write access behavior
67
+^^^^^^^^^^^^^^^^^^^^^
68
+
69
+
70
+[0x0-0x3]
71
+ Memory device slot selector, selects active memory device.
72
+ All following accesses to other registers in 0xa00-0xa17
73
+ region will read/store data from/to selected memory device.
74
+[0x4-0x7]
75
+ OST event code reported by OSPM
76
+[0x8-0xb]
77
+ OST status code reported by OSPM
78
+[0xc-0x13]
79
+ reserved, writes into it are ignored
80
+[0x14]
81
+ Memory device control fields
82
+
83
+ bits:
84
+
85
+ 0:
86
+ reserved, OSPM must clear it before writing to register.
87
+ Due to BUG in versions prior 2.4 that field isn't cleared
88
+ when other fields are written. Keep it reserved and don't
89
+ try to reuse it.
90
+ 1:
91
+ if set to 1 clears device insert event, set by OSPM
92
+ after it has emitted device check event for the
93
+ selected memory device
94
+ 2:
95
+ if set to 1 clears device remove event, set by OSPM
96
+ after it has emitted device eject request for the
97
+ selected memory device
98
+ 3:
99
+ if set to 1 initiates device eject, set by OSPM when it
100
+ triggers memory device removal and calls _EJ0 method
101
+ 4-7:
102
+ reserved, OSPM must clear them before writing to register
103
+
104
+Selecting memory device slot beyond present range has no effect on platform:
105
+
106
+- write accesses to memory hot-plug registers not documented above are ignored
107
+- read accesses to memory hot-plug registers not documented above return
108
+ all bits set to 1.
109
+
110
+Memory hot remove process diagram
111
+---------------------------------
112
+
113
+::
114
+
115
+ +-------------+ +-----------------------+ +------------------+
116
+ | 1. QEMU | | 2. QEMU | |3. QEMU |
117
+ | device_del +---->+ device unplug request +----->+Send SCI to guest,|
118
+ | | | cb | |return control to |
119
+ | | | | |management |
120
+ +-------------+ +-----------------------+ +------------------+
121
+
122
+ +---------------------------------------------------------------------+
123
+
124
+ +---------------------+ +-------------------------+
125
+ | OSPM: | remove event | OSPM: |
126
+ | send Eject Request, | | Scan memory devices |
127
+ | clear remove event +<-------------+ for event flags |
128
+ | | | |
129
+ +---------------------+ +-------------------------+
130
+ |
131
+ |
132
+ +---------v--------+ +-----------------------+
133
+ | Guest OS: | success | OSPM: |
134
+ | process Ejection +----------->+ Execute _EJ0 method, |
135
+ | request | | set eject bit in flags|
136
+ +------------------+ +-----------------------+
137
+ |failure |
138
+ v v
139
+ +------------------------+ +-----------------------+
140
+ | OSPM: | | QEMU: |
141
+ | set OST event & status | | call device unplug cb |
142
+ | fields | | |
143
+ +------------------------+ +-----------------------+
144
+ | |
145
+ v v
146
+ +------------------+ +-------------------+
147
+ |QEMU: | |QEMU: |
148
+ |Send OST QMP event| |Send device deleted|
149
+ | | |QMP event |
150
+ +------------------+ | |
151
+ +-------------------+
152
diff --git a/docs/specs/acpi_mem_hotplug.txt b/docs/specs/acpi_mem_hotplug.txt
19
deleted file mode 100644
153
deleted file mode 100644
20
index XXXXXXX..XXXXXXX
154
index XXXXXXX..XXXXXXX
21
--- a/hw/display/tc6393xb_template.h
155
--- a/docs/specs/acpi_mem_hotplug.txt
22
+++ /dev/null
156
+++ /dev/null
23
@@ -XXX,XX +XXX,XX @@
157
@@ -XXX,XX +XXX,XX @@
24
-/*
158
-QEMU<->ACPI BIOS memory hotplug interface
25
- * Toshiba TC6393XB I/O Controller.
159
---------------------------------------
26
- * Found in Sharp Zaurus SL-6000 (tosa) or some
160
-
27
- * Toshiba e-Series PDAs.
161
-ACPI BIOS GPE.3 handler is dedicated for notifying OS about memory hot-add
28
- *
162
-and hot-remove events.
29
- * FB support code. Based on G364 fb emulator
163
-
30
- *
164
-Memory hot-plug interface (IO port 0xa00-0xa17, 1-4 byte access):
31
- * Copyright (c) 2007 Hervé Poussineau
165
----------------------------------------------------------------
32
- *
166
-0xa00:
33
- * This program is free software; you can redistribute it and/or
167
- read access:
34
- * modify it under the terms of the GNU General Public License as
168
- [0x0-0x3] Lo part of memory device phys address
35
- * published by the Free Software Foundation; either version 2 of
169
- [0x4-0x7] Hi part of memory device phys address
36
- * the License, or (at your option) any later version.
170
- [0x8-0xb] Lo part of memory device size in bytes
37
- *
171
- [0xc-0xf] Hi part of memory device size in bytes
38
- * This program is distributed in the hope that it will be useful,
172
- [0x10-0x13] Memory device proximity domain
39
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
173
- [0x14] Memory device status fields
40
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
174
- bits:
41
- * GNU General Public License for more details.
175
- 0: Device is enabled and may be used by guest
42
- *
176
- 1: Device insert event, used to distinguish device for which
43
- * You should have received a copy of the GNU General Public License along
177
- no device check event to OSPM was issued.
44
- * with this program; if not, see <http://www.gnu.org/licenses/>.
178
- It's valid only when bit 1 is set.
45
- */
179
- 2: Device remove event, used to distinguish device for which
46
-
180
- no device eject request to OSPM was issued.
47
-static void tc6393xb_draw_graphic32(TC6393xbState *s)
181
- 3-7: reserved and should be ignored by OSPM
48
-{
182
- [0x15-0x17] reserved
49
- DisplaySurface *surface = qemu_console_surface(s->con);
183
-
50
- int i;
184
- write access:
51
- uint16_t *data_buffer;
185
- [0x0-0x3] Memory device slot selector, selects active memory device.
52
- uint8_t *data_display;
186
- All following accesses to other registers in 0xa00-0xa17
53
-
187
- region will read/store data from/to selected memory device.
54
- data_buffer = s->vram_ptr;
188
- [0x4-0x7] OST event code reported by OSPM
55
- data_display = surface_data(surface);
189
- [0x8-0xb] OST status code reported by OSPM
56
- for(i = 0; i < s->scr_height; i++) {
190
- [0xc-0x13] reserved, writes into it are ignored
57
- int j;
191
- [0x14] Memory device control fields
58
- for (j = 0; j < s->scr_width; j++, data_display += 4, data_buffer++) {
192
- bits:
59
- uint16_t color = *data_buffer;
193
- 0: reserved, OSPM must clear it before writing to register.
60
- uint32_t dest_color = rgb_to_pixel32(
194
- Due to BUG in versions prior 2.4 that field isn't cleared
61
- ((color & 0xf800) * 0x108) >> 11,
195
- when other fields are written. Keep it reserved and don't
62
- ((color & 0x7e0) * 0x41) >> 9,
196
- try to reuse it.
63
- ((color & 0x1f) * 0x21) >> 2
197
- 1: if set to 1 clears device insert event, set by OSPM
64
- );
198
- after it has emitted device check event for the
65
- *(uint32_t *)data_display = dest_color;
199
- selected memory device
66
- }
200
- 2: if set to 1 clears device remove event, set by OSPM
67
- }
201
- after it has emitted device eject request for the
68
-}
202
- selected memory device
69
diff --git a/hw/display/tc6393xb.c b/hw/display/tc6393xb.c
203
- 3: if set to 1 initiates device eject, set by OSPM when it
204
- triggers memory device removal and calls _EJ0 method
205
- 4-7: reserved, OSPM must clear them before writing to register
206
-
207
-Selecting memory device slot beyond present range has no effect on platform:
208
- - write accesses to memory hot-plug registers not documented above are
209
- ignored
210
- - read accesses to memory hot-plug registers not documented above return
211
- all bits set to 1.
212
-
213
-Memory hot remove process diagram:
214
-----------------------------------
215
- +-------------+     +-----------------------+      +------------------+     
216
- |  1. QEMU    |     | 2. QEMU               |      |3. QEMU           |     
217
- |  device_del +---->+ device unplug request +----->+Send SCI to guest,|     
218
- |             |     |         cb            |      |return control to |     
219
- +-------------+     +-----------------------+      |management        |     
220
-                                                    +------------------+     
221
-                                                                             
222
- +---------------------------------------------------------------------+     
223
-                                                                             
224
- +---------------------+              +-------------------------+            
225
- | OSPM:               | remove event | OSPM:                   |            
226
- | send Eject Request, |              | Scan memory devices     |            
227
- | clear remove event  +<-------------+ for event flags         |            
228
- |                     |              |                         |            
229
- +---------------------+              +-------------------------+            
230
-           |                                                                 
231
-           |                                                                 
232
- +---------v--------+            +-----------------------+                   
233
- | Guest OS:        |  success   | OSPM:                 |                   
234
- | process Ejection +----------->+ Execute _EJ0 method,  |                   
235
- | request          |            | set eject bit in flags|                   
236
- +------------------+            +-----------------------+                   
237
-           |failure                         |                                
238
-           v                                v                                
239
- +------------------------+      +-----------------------+                   
240
- | OSPM:                  |      | QEMU:                 |                   
241
- | set OST event & status |      | call device unplug cb |                   
242
- | fields                 |      |                       |                   
243
- +------------------------+      +-----------------------+                   
244
-          |                                  |                               
245
-          v                                  v                               
246
- +------------------+              +-------------------+                     
247
- |QEMU:             |              |QEMU:              |                     
248
- |Send OST QMP event|              |Send device deleted|                     
249
- |                  |              |QMP event          |                     
250
- +------------------+              |                   |                     
251
-                                   +-------------------+
252
diff --git a/docs/specs/index.rst b/docs/specs/index.rst
70
index XXXXXXX..XXXXXXX 100644
253
index XXXXXXX..XXXXXXX 100644
71
--- a/hw/display/tc6393xb.c
254
--- a/docs/specs/index.rst
72
+++ b/hw/display/tc6393xb.c
255
+++ b/docs/specs/index.rst
73
@@ -XXX,XX +XXX,XX @@ static void tc6393xb_nand_writeb(TC6393xbState *s, hwaddr addr, uint32_t value)
256
@@ -XXX,XX +XXX,XX @@ guest hardware that is specific to QEMU.
74
(uint32_t) addr, value & 0xff);
257
tpm
75
}
258
acpi_hest_ghes
76
259
acpi_cpu_hotplug
77
-#define BITS 32
260
+ acpi_mem_hotplug
78
-#include "tc6393xb_template.h"
79
-
80
static void tc6393xb_draw_graphic(TC6393xbState *s, int full_update)
81
{
82
- tc6393xb_draw_graphic32(s);
83
+ DisplaySurface *surface = qemu_console_surface(s->con);
84
+ int i;
85
+ uint16_t *data_buffer;
86
+ uint8_t *data_display;
87
+
88
+ data_buffer = s->vram_ptr;
89
+ data_display = surface_data(surface);
90
+ for (i = 0; i < s->scr_height; i++) {
91
+ int j;
92
+ for (j = 0; j < s->scr_width; j++, data_display += 4, data_buffer++) {
93
+ uint16_t color = *data_buffer;
94
+ uint32_t dest_color = rgb_to_pixel32(
95
+ ((color & 0xf800) * 0x108) >> 11,
96
+ ((color & 0x7e0) * 0x41) >> 9,
97
+ ((color & 0x1f) * 0x21) >> 2
98
+ );
99
+ *(uint32_t *)data_display = dest_color;
100
+ }
101
+ }
102
dpy_gfx_update_full(s->con);
103
}
104
105
--
261
--
106
2.20.1
262
2.20.1
107
263
108
264
diff view generated by jsdifflib
1
Update old infocenter.arm.com URLs to the equivalent developer.arm.com
1
Convert the PCI hotplug spec document to rST.
2
ones (the old URLs should redirect, but we might as well avoid the
3
redirection notice, and the new URLs are pleasantly shorter).
4
5
This commit covers the links to the MPS2 board TRM, the various
6
Application Notes, the IoTKit and SSE-200 documents.
7
2
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
4
Reviewed-by: Igor Mammedov <imammedo@redhat.com>
10
Message-id: 20210215115138.20465-25-peter.maydell@linaro.org
11
---
5
---
12
include/hw/arm/armsse.h | 4 ++--
6
...i_pci_hotplug.txt => acpi_pci_hotplug.rst} | 37 ++++++++++---------
13
include/hw/misc/armsse-cpuid.h | 2 +-
7
docs/specs/index.rst | 1 +
14
include/hw/misc/armsse-mhu.h | 2 +-
8
2 files changed, 21 insertions(+), 17 deletions(-)
15
include/hw/misc/iotkit-secctl.h | 2 +-
9
rename docs/specs/{acpi_pci_hotplug.txt => acpi_pci_hotplug.rst} (51%)
16
include/hw/misc/iotkit-sysctl.h | 2 +-
17
include/hw/misc/iotkit-sysinfo.h | 2 +-
18
include/hw/misc/mps2-fpgaio.h | 2 +-
19
hw/arm/mps2-tz.c | 11 +++++------
20
hw/misc/armsse-cpuid.c | 2 +-
21
hw/misc/armsse-mhu.c | 2 +-
22
hw/misc/iotkit-sysctl.c | 2 +-
23
hw/misc/iotkit-sysinfo.c | 2 +-
24
hw/misc/mps2-fpgaio.c | 2 +-
25
hw/misc/mps2-scc.c | 2 +-
26
14 files changed, 19 insertions(+), 20 deletions(-)
27
10
28
diff --git a/include/hw/arm/armsse.h b/include/hw/arm/armsse.h
11
diff --git a/docs/specs/acpi_pci_hotplug.txt b/docs/specs/acpi_pci_hotplug.rst
12
similarity index 51%
13
rename from docs/specs/acpi_pci_hotplug.txt
14
rename to docs/specs/acpi_pci_hotplug.rst
29
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
30
--- a/include/hw/arm/armsse.h
16
--- a/docs/specs/acpi_pci_hotplug.txt
31
+++ b/include/hw/arm/armsse.h
17
+++ b/docs/specs/acpi_pci_hotplug.rst
32
@@ -XXX,XX +XXX,XX @@
18
@@ -XXX,XX +XXX,XX @@
33
* hardware, which include the IoT Kit and the SSE-050, SSE-100 and
19
QEMU<->ACPI BIOS PCI hotplug interface
34
* SSE-200. Currently we model:
20
---------------------------------------
35
* - the Arm IoT Kit which is documented in
21
+======================================
36
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
22
37
+ * https://developer.arm.com/documentation/ecm0601256/latest
23
QEMU supports PCI hotplug via ACPI, for PCI bus 0. This document
38
* - the SSE-200 which is documented in
24
describes the interface between QEMU and the ACPI BIOS.
39
- * http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
25
40
+ * https://developer.arm.com/documentation/101104/latest/
26
-ACPI GPE block (IO ports 0xafe0-0xafe3, byte access):
41
*
27
------------------------------------------
42
* The IoTKit contains:
28
+ACPI GPE block (IO ports 0xafe0-0xafe3, byte access)
43
* a Cortex-M33
29
+----------------------------------------------------
44
diff --git a/include/hw/misc/armsse-cpuid.h b/include/hw/misc/armsse-cpuid.h
30
31
Generic ACPI GPE block. Bit 1 (GPE.1) used to notify PCI hotplug/eject
32
event to ACPI BIOS, via SCI interrupt.
33
34
-PCI slot injection notification pending (IO port 0xae00-0xae03, 4-byte access):
35
----------------------------------------------------------------
36
+PCI slot injection notification pending (IO port 0xae00-0xae03, 4-byte access)
37
+------------------------------------------------------------------------------
38
+
39
Slot injection notification pending. One bit per slot.
40
41
Read by ACPI BIOS GPE.1 handler to notify OS of injection
42
events. Read-only.
43
44
-PCI slot removal notification (IO port 0xae04-0xae07, 4-byte access):
45
------------------------------------------------------
46
+PCI slot removal notification (IO port 0xae04-0xae07, 4-byte access)
47
+--------------------------------------------------------------------
48
+
49
Slot removal notification pending. One bit per slot.
50
51
Read by ACPI BIOS GPE.1 handler to notify OS of removal
52
events. Read-only.
53
54
-PCI device eject (IO port 0xae08-0xae0b, 4-byte access):
55
-----------------------------------------
56
+PCI device eject (IO port 0xae08-0xae0b, 4-byte access)
57
+-------------------------------------------------------
58
59
Write: Used by ACPI BIOS _EJ0 method to request device removal.
60
One bit per slot.
61
62
Read: Hotplug features register. Used by platform to identify features
63
available. Current base feature set (no bits set):
64
- - Read-only "up" register @0xae00, 4-byte access, bit per slot
65
- - Read-only "down" register @0xae04, 4-byte access, bit per slot
66
- - Read/write "eject" register @0xae08, 4-byte access,
67
- write: bit per slot eject, read: hotplug feature set
68
- - Read-only hotplug capable register @0xae0c, 4-byte access, bit per slot
69
70
-PCI removability status (IO port 0xae0c-0xae0f, 4-byte access):
71
------------------------------------------------
72
+- Read-only "up" register @0xae00, 4-byte access, bit per slot
73
+- Read-only "down" register @0xae04, 4-byte access, bit per slot
74
+- Read/write "eject" register @0xae08, 4-byte access,
75
+ write: bit per slot eject, read: hotplug feature set
76
+- Read-only hotplug capable register @0xae0c, 4-byte access, bit per slot
77
+
78
+PCI removability status (IO port 0xae0c-0xae0f, 4-byte access)
79
+--------------------------------------------------------------
80
81
Used by ACPI BIOS _RMV method to indicate removability status to OS. One
82
-bit per slot. Read-only
83
+bit per slot. Read-only.
84
diff --git a/docs/specs/index.rst b/docs/specs/index.rst
45
index XXXXXXX..XXXXXXX 100644
85
index XXXXXXX..XXXXXXX 100644
46
--- a/include/hw/misc/armsse-cpuid.h
86
--- a/docs/specs/index.rst
47
+++ b/include/hw/misc/armsse-cpuid.h
87
+++ b/docs/specs/index.rst
48
@@ -XXX,XX +XXX,XX @@
88
@@ -XXX,XX +XXX,XX @@ guest hardware that is specific to QEMU.
49
/*
89
acpi_hest_ghes
50
* This is a model of the "CPU_IDENTITY" register block which is part of the
90
acpi_cpu_hotplug
51
* Arm SSE-200 and documented in
91
acpi_mem_hotplug
52
- * http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
92
+ acpi_pci_hotplug
53
+ * https://developer.arm.com/documentation/101104/latest/
54
*
55
* QEMU interface:
56
* + QOM property "CPUID": the value to use for the CPUID register
57
diff --git a/include/hw/misc/armsse-mhu.h b/include/hw/misc/armsse-mhu.h
58
index XXXXXXX..XXXXXXX 100644
59
--- a/include/hw/misc/armsse-mhu.h
60
+++ b/include/hw/misc/armsse-mhu.h
61
@@ -XXX,XX +XXX,XX @@
62
/*
63
* This is a model of the Message Handling Unit (MHU) which is part of the
64
* Arm SSE-200 and documented in
65
- * http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
66
+ * https://developer.arm.com/documentation/101104/latest/
67
*
68
* QEMU interface:
69
* + sysbus MMIO region 0: the system information register bank
70
diff --git a/include/hw/misc/iotkit-secctl.h b/include/hw/misc/iotkit-secctl.h
71
index XXXXXXX..XXXXXXX 100644
72
--- a/include/hw/misc/iotkit-secctl.h
73
+++ b/include/hw/misc/iotkit-secctl.h
74
@@ -XXX,XX +XXX,XX @@
75
76
/* This is a model of the security controller which is part of the
77
* Arm IoT Kit and documented in
78
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
79
+ * https://developer.arm.com/documentation/ecm0601256/latest
80
*
81
* QEMU interface:
82
* + sysbus MMIO region 0 is the "secure privilege control block" registers
83
diff --git a/include/hw/misc/iotkit-sysctl.h b/include/hw/misc/iotkit-sysctl.h
84
index XXXXXXX..XXXXXXX 100644
85
--- a/include/hw/misc/iotkit-sysctl.h
86
+++ b/include/hw/misc/iotkit-sysctl.h
87
@@ -XXX,XX +XXX,XX @@
88
/*
89
* This is a model of the "system control element" which is part of the
90
* Arm IoTKit and documented in
91
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
92
+ * https://developer.arm.com/documentation/ecm0601256/latest
93
* Specifically, it implements the "system information block" and
94
* "system control register" blocks.
95
*
96
diff --git a/include/hw/misc/iotkit-sysinfo.h b/include/hw/misc/iotkit-sysinfo.h
97
index XXXXXXX..XXXXXXX 100644
98
--- a/include/hw/misc/iotkit-sysinfo.h
99
+++ b/include/hw/misc/iotkit-sysinfo.h
100
@@ -XXX,XX +XXX,XX @@
101
/*
102
* This is a model of the "system information block" which is part of the
103
* Arm IoTKit and documented in
104
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
105
+ * https://developer.arm.com/documentation/ecm0601256/latest
106
* QEMU interface:
107
* + QOM property "SYS_VERSION": value to use for SYS_VERSION register
108
* + QOM property "SYS_CONFIG": value to use for SYS_CONFIG register
109
diff --git a/include/hw/misc/mps2-fpgaio.h b/include/hw/misc/mps2-fpgaio.h
110
index XXXXXXX..XXXXXXX 100644
111
--- a/include/hw/misc/mps2-fpgaio.h
112
+++ b/include/hw/misc/mps2-fpgaio.h
113
@@ -XXX,XX +XXX,XX @@
114
/* This is a model of the FPGAIO register block in the AN505
115
* FPGA image for the MPS2 dev board; it is documented in the
116
* application note:
117
- * http://infocenter.arm.com/help/topic/com.arm.doc.dai0505b/index.html
118
+ * https://developer.arm.com/documentation/dai0505/latest/
119
*
120
* QEMU interface:
121
* + sysbus MMIO region 0: the register bank
122
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
123
index XXXXXXX..XXXXXXX 100644
124
--- a/hw/arm/mps2-tz.c
125
+++ b/hw/arm/mps2-tz.c
126
@@ -XXX,XX +XXX,XX @@
127
* https://developer.arm.com/products/system-design/development-boards/fpga-prototyping-boards/mps2
128
*
129
* Board TRM:
130
- * http://infocenter.arm.com/help/topic/com.arm.doc.100112_0200_06_en/versatile_express_cortex_m_prototyping_systems_v2m_mps2_and_v2m_mps2plus_technical_reference_100112_0200_06_en.pdf
131
+ * https://developer.arm.com/documentation/100112/latest/
132
* Application Note AN505:
133
- * http://infocenter.arm.com/help/topic/com.arm.doc.dai0505b/index.html
134
+ * https://developer.arm.com/documentation/dai0505/latest/
135
* Application Note AN521:
136
- * http://infocenter.arm.com/help/topic/com.arm.doc.dai0521c/index.html
137
+ * https://developer.arm.com/documentation/dai0521/latest/
138
* Application Note AN524:
139
* https://developer.arm.com/documentation/dai0524/latest/
140
*
141
* The AN505 defers to the Cortex-M33 processor ARMv8M IoT Kit FVP User Guide
142
* (ARM ECM0601256) for the details of some of the device layout:
143
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
144
+ * https://developer.arm.com/documentation/ecm0601256/latest
145
* Similarly, the AN521 and AN524 use the SSE-200, and the SSE-200 TRM defines
146
* most of the device layout:
147
- * http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
148
- *
149
+ * https://developer.arm.com/documentation/101104/latest/
150
*/
151
152
#include "qemu/osdep.h"
153
diff --git a/hw/misc/armsse-cpuid.c b/hw/misc/armsse-cpuid.c
154
index XXXXXXX..XXXXXXX 100644
155
--- a/hw/misc/armsse-cpuid.c
156
+++ b/hw/misc/armsse-cpuid.c
157
@@ -XXX,XX +XXX,XX @@
158
/*
159
* This is a model of the "CPU_IDENTITY" register block which is part of the
160
* Arm SSE-200 and documented in
161
- * http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
162
+ * https://developer.arm.com/documentation/101104/latest/
163
*
164
* It consists of one read-only CPUID register (set by QOM property), plus the
165
* usual ID registers.
166
diff --git a/hw/misc/armsse-mhu.c b/hw/misc/armsse-mhu.c
167
index XXXXXXX..XXXXXXX 100644
168
--- a/hw/misc/armsse-mhu.c
169
+++ b/hw/misc/armsse-mhu.c
170
@@ -XXX,XX +XXX,XX @@
171
/*
172
* This is a model of the Message Handling Unit (MHU) which is part of the
173
* Arm SSE-200 and documented in
174
- * http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
175
+ * https://developer.arm.com/documentation/101104/latest/
176
*/
177
178
#include "qemu/osdep.h"
179
diff --git a/hw/misc/iotkit-sysctl.c b/hw/misc/iotkit-sysctl.c
180
index XXXXXXX..XXXXXXX 100644
181
--- a/hw/misc/iotkit-sysctl.c
182
+++ b/hw/misc/iotkit-sysctl.c
183
@@ -XXX,XX +XXX,XX @@
184
/*
185
* This is a model of the "system control element" which is part of the
186
* Arm IoTKit and documented in
187
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
188
+ * https://developer.arm.com/documentation/ecm0601256/latest
189
* Specifically, it implements the "system control register" blocks.
190
*/
191
192
diff --git a/hw/misc/iotkit-sysinfo.c b/hw/misc/iotkit-sysinfo.c
193
index XXXXXXX..XXXXXXX 100644
194
--- a/hw/misc/iotkit-sysinfo.c
195
+++ b/hw/misc/iotkit-sysinfo.c
196
@@ -XXX,XX +XXX,XX @@
197
/*
198
* This is a model of the "system information block" which is part of the
199
* Arm IoTKit and documented in
200
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
201
+ * https://developer.arm.com/documentation/ecm0601256/latest
202
* It consists of 2 read-only version/config registers, plus the
203
* usual ID registers.
204
*/
205
diff --git a/hw/misc/mps2-fpgaio.c b/hw/misc/mps2-fpgaio.c
206
index XXXXXXX..XXXXXXX 100644
207
--- a/hw/misc/mps2-fpgaio.c
208
+++ b/hw/misc/mps2-fpgaio.c
209
@@ -XXX,XX +XXX,XX @@
210
/* This is a model of the "FPGA system control and I/O" block found
211
* in the AN505 FPGA image for the MPS2 devboard.
212
* It is documented in AN505:
213
- * http://infocenter.arm.com/help/topic/com.arm.doc.dai0505b/index.html
214
+ * https://developer.arm.com/documentation/dai0505/latest/
215
*/
216
217
#include "qemu/osdep.h"
218
diff --git a/hw/misc/mps2-scc.c b/hw/misc/mps2-scc.c
219
index XXXXXXX..XXXXXXX 100644
220
--- a/hw/misc/mps2-scc.c
221
+++ b/hw/misc/mps2-scc.c
222
@@ -XXX,XX +XXX,XX @@
223
* found in the FPGA images of MPS2 development boards.
224
*
225
* Documentation of it can be found in the MPS2 TRM:
226
- * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.100112_0100_03_en/index.html
227
+ * https://developer.arm.com/documentation/100112/latest/
228
* and also in the Application Notes documenting individual FPGA images.
229
*/
230
231
--
93
--
232
2.20.1
94
2.20.1
233
95
234
96
diff view generated by jsdifflib
1
We only include the template header once, so just inline it into the
1
Convert the ACPI NVDIMM spec document to rST.
2
source file for the device.
3
2
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
4
Reviewed-by: Igor Mammedov <imammedo@redhat.com>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
5
Message-id: 20210727170414.3368-5-peter.maydell@linaro.org
7
Message-id: 20210215103215.4944-9-peter.maydell@linaro.org
8
---
6
---
9
hw/display/omap_lcd_template.h | 154 ---------------------------------
7
docs/specs/acpi_nvdimm.rst | 228 +++++++++++++++++++++++++++++++++++++
10
hw/display/omap_lcdc.c | 127 ++++++++++++++++++++++++++-
8
docs/specs/acpi_nvdimm.txt | 188 ------------------------------
11
2 files changed, 125 insertions(+), 156 deletions(-)
9
docs/specs/index.rst | 1 +
12
delete mode 100644 hw/display/omap_lcd_template.h
10
3 files changed, 229 insertions(+), 188 deletions(-)
11
create mode 100644 docs/specs/acpi_nvdimm.rst
12
delete mode 100644 docs/specs/acpi_nvdimm.txt
13
13
14
diff --git a/hw/display/omap_lcd_template.h b/hw/display/omap_lcd_template.h
14
diff --git a/docs/specs/acpi_nvdimm.rst b/docs/specs/acpi_nvdimm.rst
15
new file mode 100644
16
index XXXXXXX..XXXXXXX
17
--- /dev/null
18
+++ b/docs/specs/acpi_nvdimm.rst
19
@@ -XXX,XX +XXX,XX @@
20
+QEMU<->ACPI BIOS NVDIMM interface
21
+=================================
22
+
23
+QEMU supports NVDIMM via ACPI. This document describes the basic concepts of
24
+NVDIMM ACPI and the interface between QEMU and the ACPI BIOS.
25
+
26
+NVDIMM ACPI Background
27
+----------------------
28
+
29
+NVDIMM is introduced in ACPI 6.0 which defines an NVDIMM root device under
30
+_SB scope with a _HID of "ACPI0012". For each NVDIMM present or intended
31
+to be supported by platform, platform firmware also exposes an ACPI
32
+Namespace Device under the root device.
33
+
34
+The NVDIMM child devices under the NVDIMM root device are defined with _ADR
35
+corresponding to the NFIT device handle. The NVDIMM root device and the
36
+NVDIMM devices can have device specific methods (_DSM) to provide additional
37
+functions specific to a particular NVDIMM implementation.
38
+
39
+This is an example from ACPI 6.0, a platform contains one NVDIMM::
40
+
41
+ Scope (\_SB){
42
+ Device (NVDR) // Root device
43
+ {
44
+ Name (_HID, "ACPI0012")
45
+ Method (_STA) {...}
46
+ Method (_FIT) {...}
47
+ Method (_DSM, ...) {...}
48
+ Device (NVD)
49
+ {
50
+ Name(_ADR, h) //where h is NFIT Device Handle for this NVDIMM
51
+ Method (_DSM, ...) {...}
52
+ }
53
+ }
54
+ }
55
+
56
+Methods supported on both NVDIMM root device and NVDIMM device
57
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
58
+
59
+_DSM (Device Specific Method)
60
+ It is a control method that enables devices to provide device specific
61
+ control functions that are consumed by the device driver.
62
+ The NVDIMM DSM specification can be found at
63
+ http://pmem.io/documents/NVDIMM_DSM_Interface_Example.pdf
64
+
65
+ Arguments:
66
+
67
+ Arg0
68
+ A Buffer containing a UUID (16 Bytes)
69
+ Arg1
70
+ An Integer containing the Revision ID (4 Bytes)
71
+ Arg2
72
+ An Integer containing the Function Index (4 Bytes)
73
+ Arg3
74
+ A package containing parameters for the function specified by the
75
+ UUID, Revision ID, and Function Index
76
+
77
+ Return Value:
78
+
79
+ If Function Index = 0, a Buffer containing a function index bitfield.
80
+ Otherwise, the return value and type depends on the UUID, revision ID
81
+ and function index which are described in the DSM specification.
82
+
83
+Methods on NVDIMM ROOT Device
84
+^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
85
+
86
+_FIT(Firmware Interface Table)
87
+ It evaluates to a buffer returning data in the format of a series of NFIT
88
+ Type Structure.
89
+
90
+ Arguments: None
91
+
92
+ Return Value:
93
+ A Buffer containing a list of NFIT Type structure entries.
94
+
95
+ The detailed definition of the structure can be found at ACPI 6.0: 5.2.25
96
+ NVDIMM Firmware Interface Table (NFIT).
97
+
98
+QEMU NVDIMM Implementation
99
+--------------------------
100
+
101
+QEMU uses 4 bytes IO Port starting from 0x0a18 and a RAM-based memory page
102
+for NVDIMM ACPI.
103
+
104
+Memory:
105
+ QEMU uses BIOS Linker/loader feature to ask BIOS to allocate a memory
106
+ page and dynamically patch its address into an int32 object named "MEMA"
107
+ in ACPI.
108
+
109
+ This page is RAM-based and it is used to transfer data between _DSM
110
+ method and QEMU. If ACPI has control, this pages is owned by ACPI which
111
+ writes _DSM input data to it, otherwise, it is owned by QEMU which
112
+ emulates _DSM access and writes the output data to it.
113
+
114
+ ACPI writes _DSM Input Data (based on the offset in the page):
115
+
116
+ [0x0 - 0x3]
117
+ 4 bytes, NVDIMM Device Handle.
118
+
119
+ The handle is completely QEMU internal thing, the values in
120
+ range [1, 0xFFFF] indicate nvdimm device. Other values are
121
+ reserved for other purposes.
122
+
123
+ Reserved handles:
124
+
125
+ - 0 is reserved for nvdimm root device named NVDR.
126
+ - 0x10000 is reserved for QEMU internal DSM function called on
127
+ the root device.
128
+
129
+ [0x4 - 0x7]
130
+ 4 bytes, Revision ID, that is the Arg1 of _DSM method.
131
+
132
+ [0x8 - 0xB]
133
+ 4 bytes. Function Index, that is the Arg2 of _DSM method.
134
+
135
+ [0xC - 0xFFF]
136
+ 4084 bytes, the Arg3 of _DSM method.
137
+
138
+ QEMU writes Output Data (based on the offset in the page):
139
+
140
+ [0x0 - 0x3]
141
+ 4 bytes, the length of result
142
+
143
+ [0x4 - 0xFFF]
144
+ 4092 bytes, the DSM result filled by QEMU
145
+
146
+IO Port 0x0a18 - 0xa1b:
147
+ ACPI writes the address of the memory page allocated by BIOS to this
148
+ port then QEMU gets the control and fills the result in the memory page.
149
+
150
+ Write Access:
151
+
152
+ [0x0a18 - 0xa1b]
153
+ 4 bytes, the address of the memory page allocated by BIOS.
154
+
155
+_DSM process diagram
156
+--------------------
157
+
158
+"MEMA" indicates the address of memory page allocated by BIOS.
159
+
160
+::
161
+
162
+ +----------------------+ +-----------------------+
163
+ | 1. OSPM | | 2. OSPM |
164
+ | save _DSM input data | | write "MEMA" to | Exit to QEMU
165
+ | to the page +----->| IO port 0x0a18 +------------+
166
+ | indicated by "MEMA" | | | |
167
+ +----------------------+ +-----------------------+ |
168
+ |
169
+ v
170
+ +--------------------+ +-----------+ +------------------+--------+
171
+ | 5 QEMU | | 4 QEMU | | 3. QEMU |
172
+ | write _DSM result | | emulate | | get _DSM input data from |
173
+ | to the page +<------+ _DSM +<-----+ the page indicated by the |
174
+ | | | | | value from the IO port |
175
+ +--------+-----------+ +-----------+ +---------------------------+
176
+ |
177
+ | Enter Guest
178
+ |
179
+ v
180
+ +--------------------------+ +--------------+
181
+ | 6 OSPM | | 7 OSPM |
182
+ | result size is returned | | _DSM return |
183
+ | by reading DSM +----->+ |
184
+ | result from the page | | |
185
+ +--------------------------+ +--------------+
186
+
187
+NVDIMM hotplug
188
+--------------
189
+
190
+ACPI BIOS GPE.4 handler is dedicated for notifying OS about nvdimm device
191
+hot-add event.
192
+
193
+QEMU internal use only _DSM functions
194
+-------------------------------------
195
+
196
+Read FIT
197
+^^^^^^^^
198
+
199
+_FIT method uses _DSM method to fetch NFIT structures blob from QEMU
200
+in 1 page sized increments which are then concatenated and returned
201
+as _FIT method result.
202
+
203
+Input parameters:
204
+
205
+Arg0
206
+ UUID {set to 648B9CF2-CDA1-4312-8AD9-49C4AF32BD62}
207
+Arg1
208
+ Revision ID (set to 1)
209
+Arg2
210
+ Function Index, 0x1
211
+Arg3
212
+ A package containing a buffer whose layout is as follows:
213
+
214
+ +----------+--------+--------+-------------------------------------------+
215
+ | Field | Length | Offset | Description |
216
+ +----------+--------+--------+-------------------------------------------+
217
+ | offset | 4 | 0 | offset in QEMU's NFIT structures blob to |
218
+ | | | | read from |
219
+ +----------+--------+--------+-------------------------------------------+
220
+
221
+Output layout in the dsm memory page:
222
+
223
+ +----------+--------+--------+-------------------------------------------+
224
+ | Field | Length | Offset | Description |
225
+ +----------+--------+--------+-------------------------------------------+
226
+ | length | 4 | 0 | length of entire returned data |
227
+ | | | | (including this header) |
228
+ +----------+--------+--------+-------------------------------------------+
229
+ | | | | return status codes |
230
+ | | | | |
231
+ | | | | - 0x0 - success |
232
+ | | | | - 0x100 - error caused by NFIT update |
233
+ | status | 4 | 4 | while read by _FIT wasn't completed |
234
+ | | | | - other codes follow Chapter 3 in |
235
+ | | | | DSM Spec Rev1 |
236
+ +----------+--------+--------+-------------------------------------------+
237
+ | fit data | Varies | 8 | contains FIT data. This field is present |
238
+ | | | | if status field is 0. |
239
+ +----------+--------+--------+-------------------------------------------+
240
+
241
+The FIT offset is maintained by the OSPM itself, current offset plus
242
+the size of the fit data returned by the function is the next offset
243
+OSPM should read. When all FIT data has been read out, zero fit data
244
+size is returned.
245
+
246
+If it returns status code 0x100, OSPM should restart to read FIT (read
247
+from offset 0 again).
248
diff --git a/docs/specs/acpi_nvdimm.txt b/docs/specs/acpi_nvdimm.txt
15
deleted file mode 100644
249
deleted file mode 100644
16
index XXXXXXX..XXXXXXX
250
index XXXXXXX..XXXXXXX
17
--- a/hw/display/omap_lcd_template.h
251
--- a/docs/specs/acpi_nvdimm.txt
18
+++ /dev/null
252
+++ /dev/null
19
@@ -XXX,XX +XXX,XX @@
253
@@ -XXX,XX +XXX,XX @@
20
-/*
254
-QEMU<->ACPI BIOS NVDIMM interface
21
- * QEMU OMAP LCD Emulator templates
255
----------------------------------
22
- *
256
-
23
- * Copyright (c) 2006 Andrzej Zaborowski <balrog@zabor.org>
257
-QEMU supports NVDIMM via ACPI. This document describes the basic concepts of
24
- *
258
-NVDIMM ACPI and the interface between QEMU and the ACPI BIOS.
25
- * Redistribution and use in source and binary forms, with or without
259
-
26
- * modification, are permitted provided that the following conditions
260
-NVDIMM ACPI Background
27
- * are met:
261
-----------------------
28
- *
262
-NVDIMM is introduced in ACPI 6.0 which defines an NVDIMM root device under
29
- * 1. Redistributions of source code must retain the above copyright
263
-_SB scope with a _HID of “ACPI0012”. For each NVDIMM present or intended
30
- * notice, this list of conditions and the following disclaimer.
264
-to be supported by platform, platform firmware also exposes an ACPI
31
- * 2. Redistributions in binary form must reproduce the above copyright
265
-Namespace Device under the root device.
32
- * notice, this list of conditions and the following disclaimer in
266
-
33
- * the documentation and/or other materials provided with the
267
-The NVDIMM child devices under the NVDIMM root device are defined with _ADR
34
- * distribution.
268
-corresponding to the NFIT device handle. The NVDIMM root device and the
35
- *
269
-NVDIMM devices can have device specific methods (_DSM) to provide additional
36
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS''
270
-functions specific to a particular NVDIMM implementation.
37
- * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
271
-
38
- * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
272
-This is an example from ACPI 6.0, a platform contains one NVDIMM:
39
- * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR
273
-
40
- * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
274
-Scope (\_SB){
41
- * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
275
- Device (NVDR) // Root device
42
- * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
276
- {
43
- * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
277
- Name (_HID, “ACPI0012”)
44
- * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
278
- Method (_STA) {...}
45
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
279
- Method (_FIT) {...}
46
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
280
- Method (_DSM, ...) {...}
47
- */
281
- Device (NVD)
48
-
282
- {
49
-/*
283
- Name(_ADR, h) //where h is NFIT Device Handle for this NVDIMM
50
- * 2-bit colour
284
- Method (_DSM, ...) {...}
51
- */
285
- }
52
-static void draw_line2_32(void *opaque, uint8_t *d, const uint8_t *s,
286
- }
53
- int width, int deststep)
54
-{
55
- uint16_t *pal = opaque;
56
- uint8_t v, r, g, b;
57
-
58
- do {
59
- v = ldub_p((void *) s);
60
- r = (pal[v & 3] >> 4) & 0xf0;
61
- g = pal[v & 3] & 0xf0;
62
- b = (pal[v & 3] << 4) & 0xf0;
63
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
64
- d += 4;
65
- v >>= 2;
66
- r = (pal[v & 3] >> 4) & 0xf0;
67
- g = pal[v & 3] & 0xf0;
68
- b = (pal[v & 3] << 4) & 0xf0;
69
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
70
- d += 4;
71
- v >>= 2;
72
- r = (pal[v & 3] >> 4) & 0xf0;
73
- g = pal[v & 3] & 0xf0;
74
- b = (pal[v & 3] << 4) & 0xf0;
75
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
76
- d += 4;
77
- v >>= 2;
78
- r = (pal[v & 3] >> 4) & 0xf0;
79
- g = pal[v & 3] & 0xf0;
80
- b = (pal[v & 3] << 4) & 0xf0;
81
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
82
- d += 4;
83
- s++;
84
- width -= 4;
85
- } while (width > 0);
86
-}
287
-}
87
-
288
-
88
-/*
289
-Method supported on both NVDIMM root device and NVDIMM device
89
- * 4-bit colour
290
-_DSM (Device Specific Method)
90
- */
291
- It is a control method that enables devices to provide device specific
91
-static void draw_line4_32(void *opaque, uint8_t *d, const uint8_t *s,
292
- control functions that are consumed by the device driver.
92
- int width, int deststep)
293
- The NVDIMM DSM specification can be found at:
93
-{
294
- http://pmem.io/documents/NVDIMM_DSM_Interface_Example.pdf
94
- uint16_t *pal = opaque;
295
-
95
- uint8_t v, r, g, b;
296
- Arguments:
96
-
297
- Arg0 – A Buffer containing a UUID (16 Bytes)
97
- do {
298
- Arg1 – An Integer containing the Revision ID (4 Bytes)
98
- v = ldub_p((void *) s);
299
- Arg2 – An Integer containing the Function Index (4 Bytes)
99
- r = (pal[v & 0xf] >> 4) & 0xf0;
300
- Arg3 – A package containing parameters for the function specified by the
100
- g = pal[v & 0xf] & 0xf0;
301
- UUID, Revision ID, and Function Index
101
- b = (pal[v & 0xf] << 4) & 0xf0;
302
-
102
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
303
- Return Value:
103
- d += 4;
304
- If Function Index = 0, a Buffer containing a function index bitfield.
104
- v >>= 4;
305
- Otherwise, the return value and type depends on the UUID, revision ID
105
- r = (pal[v & 0xf] >> 4) & 0xf0;
306
- and function index which are described in the DSM specification.
106
- g = pal[v & 0xf] & 0xf0;
307
-
107
- b = (pal[v & 0xf] << 4) & 0xf0;
308
-Methods on NVDIMM ROOT Device
108
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
309
-_FIT(Firmware Interface Table)
109
- d += 4;
310
- It evaluates to a buffer returning data in the format of a series of NFIT
110
- s++;
311
- Type Structure.
111
- width -= 2;
312
-
112
- } while (width > 0);
313
- Arguments: None
113
-}
314
-
114
-
315
- Return Value:
115
-/*
316
- A Buffer containing a list of NFIT Type structure entries.
116
- * 8-bit colour
317
-
117
- */
318
- The detailed definition of the structure can be found at ACPI 6.0: 5.2.25
118
-static void draw_line8_32(void *opaque, uint8_t *d, const uint8_t *s,
319
- NVDIMM Firmware Interface Table (NFIT).
119
- int width, int deststep)
320
-
120
-{
321
-QEMU NVDIMM Implementation
121
- uint16_t *pal = opaque;
322
-==========================
122
- uint8_t v, r, g, b;
323
-QEMU uses 4 bytes IO Port starting from 0x0a18 and a RAM-based memory page
123
-
324
-for NVDIMM ACPI.
124
- do {
325
-
125
- v = ldub_p((void *) s);
326
-Memory:
126
- r = (pal[v] >> 4) & 0xf0;
327
- QEMU uses BIOS Linker/loader feature to ask BIOS to allocate a memory
127
- g = pal[v] & 0xf0;
328
- page and dynamically patch its address into an int32 object named "MEMA"
128
- b = (pal[v] << 4) & 0xf0;
329
- in ACPI.
129
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
330
-
130
- s++;
331
- This page is RAM-based and it is used to transfer data between _DSM
131
- d += 4;
332
- method and QEMU. If ACPI has control, this pages is owned by ACPI which
132
- } while (-- width != 0);
333
- writes _DSM input data to it, otherwise, it is owned by QEMU which
133
-}
334
- emulates _DSM access and writes the output data to it.
134
-
335
-
135
-/*
336
- ACPI writes _DSM Input Data (based on the offset in the page):
136
- * 12-bit colour
337
- [0x0 - 0x3]: 4 bytes, NVDIMM Device Handle.
137
- */
338
-
138
-static void draw_line12_32(void *opaque, uint8_t *d, const uint8_t *s,
339
- The handle is completely QEMU internal thing, the values in
139
- int width, int deststep)
340
- range [1, 0xFFFF] indicate nvdimm device. Other values are
140
-{
341
- reserved for other purposes.
141
- uint16_t v;
342
-
142
- uint8_t r, g, b;
343
- Reserved handles:
143
-
344
- 0 is reserved for nvdimm root device named NVDR.
144
- do {
345
- 0x10000 is reserved for QEMU internal DSM function called on
145
- v = lduw_le_p((void *) s);
346
- the root device.
146
- r = (v >> 4) & 0xf0;
347
-
147
- g = v & 0xf0;
348
- [0x4 - 0x7]: 4 bytes, Revision ID, that is the Arg1 of _DSM method.
148
- b = (v << 4) & 0xf0;
349
- [0x8 - 0xB]: 4 bytes. Function Index, that is the Arg2 of _DSM method.
149
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
350
- [0xC - 0xFFF]: 4084 bytes, the Arg3 of _DSM method.
150
- s += 2;
351
-
151
- d += 4;
352
- QEMU Writes Output Data (based on the offset in the page):
152
- } while (-- width != 0);
353
- [0x0 - 0x3]: 4 bytes, the length of result
153
-}
354
- [0x4 - 0xFFF]: 4092 bytes, the DSM result filled by QEMU
154
-
355
-
155
-/*
356
-IO Port 0x0a18 - 0xa1b:
156
- * 16-bit colour
357
- ACPI writes the address of the memory page allocated by BIOS to this
157
- */
358
- port then QEMU gets the control and fills the result in the memory page.
158
-static void draw_line16_32(void *opaque, uint8_t *d, const uint8_t *s,
359
-
159
- int width, int deststep)
360
- write Access:
160
-{
361
- [0x0a18 - 0xa1b]: 4 bytes, the address of the memory page allocated
161
- uint16_t v;
362
- by BIOS.
162
- uint8_t r, g, b;
363
-
163
-
364
-_DSM process diagram:
164
- do {
365
----------------------
165
- v = lduw_le_p((void *) s);
366
-"MEMA" indicates the address of memory page allocated by BIOS.
166
- r = (v >> 8) & 0xf8;
367
-
167
- g = (v >> 3) & 0xfc;
368
- +----------------------+   +-----------------------+
168
- b = (v << 3) & 0xf8;
369
- |   1. OSPM   |      | 2. OSPM |
169
- ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
370
- | save _DSM input data | | write "MEMA" to | Exit to QEMU
170
- s += 2;
371
- | to the page +----->| IO port 0x0a18 +------------+
171
- d += 4;
372
- | indicated by "MEMA" | | | |
172
- } while (-- width != 0);
373
- +----------------------+ +-----------------------+ |
173
-}
374
-  |
174
diff --git a/hw/display/omap_lcdc.c b/hw/display/omap_lcdc.c
375
-  v
376
- +------------- ----+ +-----------+ +------------------+--------+
377
- | 5 QEMU | | 4 QEMU | | 3. QEMU |
378
- | write _DSM result | | emulate | | get _DSM input data from |
379
- | to the page +<------+ _DSM +<-----+ the page indicated by the |
380
- | | | | | value from the IO port |
381
- +--------+-----------+ +-----------+ +---------------------------+
382
- |
383
- | Enter Guest
384
- |
385
- v
386
- +--------------------------+ +--------------+
387
- | 6 OSPM | | 7 OSPM |
388
- | result size is returned | | _DSM return |
389
- | by reading DSM +----->+ |
390
- | result from the page | | |
391
- +--------------------------+ +--------------+
392
-
393
-NVDIMM hotplug
394
---------------
395
-ACPI BIOS GPE.4 handler is dedicated for notifying OS about nvdimm device
396
-hot-add event.
397
-
398
-QEMU internal use only _DSM function
399
-------------------------------------
400
-1) Read FIT
401
- _FIT method uses _DSM method to fetch NFIT structures blob from QEMU
402
- in 1 page sized increments which are then concatenated and returned
403
- as _FIT method result.
404
-
405
- Input parameters:
406
- Arg0 – UUID {set to 648B9CF2-CDA1-4312-8AD9-49C4AF32BD62}
407
- Arg1 – Revision ID (set to 1)
408
- Arg2 - Function Index, 0x1
409
- Arg3 - A package containing a buffer whose layout is as follows:
410
-
411
- +----------+--------+--------+-------------------------------------------+
412
- | Field | Length | Offset | Description |
413
- +----------+--------+--------+-------------------------------------------+
414
- | offset | 4 | 0 | offset in QEMU's NFIT structures blob to |
415
- | | | | read from |
416
- +----------+--------+--------+-------------------------------------------+
417
-
418
- Output layout in the dsm memory page:
419
- +----------+--------+--------+-------------------------------------------+
420
- | Field | Length | Offset | Description |
421
- +----------+--------+--------+-------------------------------------------+
422
- | length | 4 | 0 | length of entire returned data |
423
- | | | | (including this header) |
424
- +----------+-----------------+-------------------------------------------+
425
- | | | | return status codes |
426
- | | | | 0x0 - success |
427
- | | | | 0x100 - error caused by NFIT update while |
428
- | status | 4 | 4 | read by _FIT wasn't completed, other |
429
- | | | | codes follow Chapter 3 in DSM Spec Rev1 |
430
- +----------+-----------------+-------------------------------------------+
431
- | fit data | Varies | 8 | contains FIT data, this field is present |
432
- | | | | if status field is 0; |
433
- +----------+--------+--------+-------------------------------------------+
434
-
435
- The FIT offset is maintained by the OSPM itself, current offset plus
436
- the size of the fit data returned by the function is the next offset
437
- OSPM should read. When all FIT data has been read out, zero fit data
438
- size is returned.
439
-
440
- If it returns status code 0x100, OSPM should restart to read FIT (read
441
- from offset 0 again).
442
diff --git a/docs/specs/index.rst b/docs/specs/index.rst
175
index XXXXXXX..XXXXXXX 100644
443
index XXXXXXX..XXXXXXX 100644
176
--- a/hw/display/omap_lcdc.c
444
--- a/docs/specs/index.rst
177
+++ b/hw/display/omap_lcdc.c
445
+++ b/docs/specs/index.rst
178
@@ -XXX,XX +XXX,XX @@ static void omap_lcd_interrupts(struct omap_lcd_panel_s *s)
446
@@ -XXX,XX +XXX,XX @@ guest hardware that is specific to QEMU.
179
447
acpi_cpu_hotplug
180
#define draw_line_func drawfn
448
acpi_mem_hotplug
181
449
acpi_pci_hotplug
182
-#define DEPTH 32
450
+ acpi_nvdimm
183
-#include "omap_lcd_template.h"
184
+/*
185
+ * 2-bit colour
186
+ */
187
+static void draw_line2_32(void *opaque, uint8_t *d, const uint8_t *s,
188
+ int width, int deststep)
189
+{
190
+ uint16_t *pal = opaque;
191
+ uint8_t v, r, g, b;
192
+
193
+ do {
194
+ v = ldub_p((void *) s);
195
+ r = (pal[v & 3] >> 4) & 0xf0;
196
+ g = pal[v & 3] & 0xf0;
197
+ b = (pal[v & 3] << 4) & 0xf0;
198
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
199
+ d += 4;
200
+ v >>= 2;
201
+ r = (pal[v & 3] >> 4) & 0xf0;
202
+ g = pal[v & 3] & 0xf0;
203
+ b = (pal[v & 3] << 4) & 0xf0;
204
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
205
+ d += 4;
206
+ v >>= 2;
207
+ r = (pal[v & 3] >> 4) & 0xf0;
208
+ g = pal[v & 3] & 0xf0;
209
+ b = (pal[v & 3] << 4) & 0xf0;
210
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
211
+ d += 4;
212
+ v >>= 2;
213
+ r = (pal[v & 3] >> 4) & 0xf0;
214
+ g = pal[v & 3] & 0xf0;
215
+ b = (pal[v & 3] << 4) & 0xf0;
216
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
217
+ d += 4;
218
+ s++;
219
+ width -= 4;
220
+ } while (width > 0);
221
+}
222
+
223
+/*
224
+ * 4-bit colour
225
+ */
226
+static void draw_line4_32(void *opaque, uint8_t *d, const uint8_t *s,
227
+ int width, int deststep)
228
+{
229
+ uint16_t *pal = opaque;
230
+ uint8_t v, r, g, b;
231
+
232
+ do {
233
+ v = ldub_p((void *) s);
234
+ r = (pal[v & 0xf] >> 4) & 0xf0;
235
+ g = pal[v & 0xf] & 0xf0;
236
+ b = (pal[v & 0xf] << 4) & 0xf0;
237
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
238
+ d += 4;
239
+ v >>= 4;
240
+ r = (pal[v & 0xf] >> 4) & 0xf0;
241
+ g = pal[v & 0xf] & 0xf0;
242
+ b = (pal[v & 0xf] << 4) & 0xf0;
243
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
244
+ d += 4;
245
+ s++;
246
+ width -= 2;
247
+ } while (width > 0);
248
+}
249
+
250
+/*
251
+ * 8-bit colour
252
+ */
253
+static void draw_line8_32(void *opaque, uint8_t *d, const uint8_t *s,
254
+ int width, int deststep)
255
+{
256
+ uint16_t *pal = opaque;
257
+ uint8_t v, r, g, b;
258
+
259
+ do {
260
+ v = ldub_p((void *) s);
261
+ r = (pal[v] >> 4) & 0xf0;
262
+ g = pal[v] & 0xf0;
263
+ b = (pal[v] << 4) & 0xf0;
264
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
265
+ s++;
266
+ d += 4;
267
+ } while (-- width != 0);
268
+}
269
+
270
+/*
271
+ * 12-bit colour
272
+ */
273
+static void draw_line12_32(void *opaque, uint8_t *d, const uint8_t *s,
274
+ int width, int deststep)
275
+{
276
+ uint16_t v;
277
+ uint8_t r, g, b;
278
+
279
+ do {
280
+ v = lduw_le_p((void *) s);
281
+ r = (v >> 4) & 0xf0;
282
+ g = v & 0xf0;
283
+ b = (v << 4) & 0xf0;
284
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
285
+ s += 2;
286
+ d += 4;
287
+ } while (-- width != 0);
288
+}
289
+
290
+/*
291
+ * 16-bit colour
292
+ */
293
+static void draw_line16_32(void *opaque, uint8_t *d, const uint8_t *s,
294
+ int width, int deststep)
295
+{
296
+ uint16_t v;
297
+ uint8_t r, g, b;
298
+
299
+ do {
300
+ v = lduw_le_p((void *) s);
301
+ r = (v >> 8) & 0xf8;
302
+ g = (v >> 3) & 0xfc;
303
+ b = (v << 3) & 0xf8;
304
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
305
+ s += 2;
306
+ d += 4;
307
+ } while (-- width != 0);
308
+}
309
310
static void omap_update_display(void *opaque)
311
{
312
--
451
--
313
2.20.1
452
2.20.1
314
453
315
454
diff view generated by jsdifflib
1
Add brief documentation of the new mps3-an524 board.
1
Add entries for the ACPI specs documents in docs/specs to
2
appropriate sections of MAINTAINERS.
2
3
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
5
Reviewed-by: Igor Mammedov <imammedo@redhat.com>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20210727170414.3368-6-peter.maydell@linaro.org
6
Message-id: 20210215115138.20465-24-peter.maydell@linaro.org
7
---
7
---
8
docs/system/arm/mps2.rst | 24 ++++++++++++++++++------
8
MAINTAINERS | 5 +++++
9
1 file changed, 18 insertions(+), 6 deletions(-)
9
1 file changed, 5 insertions(+)
10
10
11
diff --git a/docs/system/arm/mps2.rst b/docs/system/arm/mps2.rst
11
diff --git a/MAINTAINERS b/MAINTAINERS
12
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
13
--- a/docs/system/arm/mps2.rst
13
--- a/MAINTAINERS
14
+++ b/docs/system/arm/mps2.rst
14
+++ b/MAINTAINERS
15
@@ -XXX,XX +XXX,XX @@
15
@@ -XXX,XX +XXX,XX @@ F: qapi/acpi.json
16
-Arm MPS2 boards (``mps2-an385``, ``mps2-an386``, ``mps2-an500``, ``mps2-an505``, ``mps2-an511``, ``mps2-an521``)
16
F: tests/qtest/bios-tables-test*
17
-================================================================================================================
17
F: tests/qtest/acpi-utils.[hc]
18
+Arm MPS2 and MPS3 boards (``mps2-an385``, ``mps2-an386``, ``mps2-an500``, ``mps2-an505``, ``mps2-an511``, ``mps2-an521``, ``mps3-an524``)
18
F: tests/data/acpi/
19
+=========================================================================================================================================
19
+F: docs/specs/acpi_cpu_hotplug.rst
20
20
+F: docs/specs/acpi_mem_hotplug.rst
21
These board models all use Arm M-profile CPUs.
21
+F: docs/specs/acpi_pci_hotplug.rst
22
22
+F: docs/specs/acpi_hw_reduced_hotplug.rst
23
-The Arm MPS2 and MPS2+ dev boards are FPGA based (the 2+ has a bigger
23
24
-FPGA but is otherwise the same as the 2). Since the CPU itself
24
ACPI/HEST/GHES
25
-and most of the devices are in the FPGA, the details of the board
25
R: Dongjiu Geng <gengdongjiu1@gmail.com>
26
-as seen by the guest depend significantly on the FPGA image.
26
@@ -XXX,XX +XXX,XX @@ F: hw/acpi/nvdimm.c
27
+The Arm MPS2, MPS2+ and MPS3 dev boards are FPGA based (the 2+ has a
27
F: hw/mem/nvdimm.c
28
+bigger FPGA but is otherwise the same as the 2; the 3 has a bigger
28
F: include/hw/mem/nvdimm.h
29
+FPGA again, can handle 4GB of RAM and has a USB controller and QSPI flash).
29
F: docs/nvdimm.txt
30
+
30
+F: docs/specs/acpi_nvdimm.rst
31
+Since the CPU itself and most of the devices are in the FPGA, the
31
32
+details of the board as seen by the guest depend significantly on the
32
e1000x
33
+FPGA image.
33
M: Dmitry Fleytman <dmitry.fleytman@gmail.com>
34
35
QEMU models the following FPGA images:
36
37
@@ -XXX,XX +XXX,XX @@ QEMU models the following FPGA images:
38
Cortex-M3 'DesignStart' as documented in Arm Application Note AN511
39
``mps2-an521``
40
Dual Cortex-M33 as documented in Arm Application Note AN521
41
+``mps3-an524``
42
+ Dual Cortex-M33 on an MPS3, as documented in Arm Application Note AN524
43
44
Differences between QEMU and real hardware:
45
46
- AN385/AN386 remapping of low 16K of memory to either ZBT SSRAM1 or to
47
block RAM is unimplemented (QEMU always maps this to ZBT SSRAM1, as
48
if zbt_boot_ctrl is always zero)
49
+- AN524 remapping of low memory to either BRAM or to QSPI flash is
50
+ unimplemented (QEMU always maps this to BRAM, ignoring the
51
+ SCC CFG_REG0 memory-remap bit)
52
- QEMU provides a LAN9118 ethernet rather than LAN9220; the only guest
53
visible difference is that the LAN9118 doesn't support checksum
54
offloading
55
+- QEMU does not model the QSPI flash in MPS3 boards as real QSPI
56
+ flash, but only as simple ROM, so attempting to rewrite the flash
57
+ from the guest will fail
58
+- QEMU does not model the USB controller in MPS3 boards
59
--
34
--
60
2.20.1
35
2.20.1
61
36
62
37
diff view generated by jsdifflib
1
The omap_lcdc template header is already only included once, for
1
The xen_available() function is used only to produce an error
2
DEPTH==32, but it still has all the macro-driven parameterization
2
for some Xen-specific command line options in QEMU binaries where
3
for other depths. Expand out all the macros in the header.
3
Xen support was not compiled in: it just returns the value of
4
the CONFIG_XEN define.
5
6
Now that accelerators are QOM classes, we can check for
7
"does this binary have Xen compiled in" with accel_find("xen"),
8
and drop the xen_available() function.
4
9
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
12
Message-id: 20210730105947.28215-2-peter.maydell@linaro.org
8
Message-id: 20210215103215.4944-6-peter.maydell@linaro.org
9
---
13
---
10
hw/display/omap_lcd_template.h | 67 ++++++++++++++--------------------
14
include/sysemu/arch_init.h | 1 -
11
1 file changed, 28 insertions(+), 39 deletions(-)
15
softmmu/arch_init.c | 9 ---------
16
softmmu/vl.c | 6 +++---
17
3 files changed, 3 insertions(+), 13 deletions(-)
12
18
13
diff --git a/hw/display/omap_lcd_template.h b/hw/display/omap_lcd_template.h
19
diff --git a/include/sysemu/arch_init.h b/include/sysemu/arch_init.h
14
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/display/omap_lcd_template.h
21
--- a/include/sysemu/arch_init.h
16
+++ b/hw/display/omap_lcd_template.h
22
+++ b/include/sysemu/arch_init.h
17
@@ -XXX,XX +XXX,XX @@
23
@@ -XXX,XX +XXX,XX @@ enum {
18
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24
extern const uint32_t arch_type;
19
*/
25
20
26
int kvm_available(void);
21
-#if DEPTH == 32
27
-int xen_available(void);
22
-# define BPP 4
28
23
-# define PIXEL_TYPE uint32_t
29
/* default virtio transport per architecture */
24
-#else
30
#define QEMU_ARCH_VIRTIO_PCI (QEMU_ARCH_ALPHA | QEMU_ARCH_ARM | \
25
-# error unsupport depth
31
diff --git a/softmmu/arch_init.c b/softmmu/arch_init.c
26
-#endif
32
index XXXXXXX..XXXXXXX 100644
27
-
33
--- a/softmmu/arch_init.c
28
/*
34
+++ b/softmmu/arch_init.c
29
* 2-bit colour
35
@@ -XXX,XX +XXX,XX @@ int kvm_available(void)
30
*/
36
return 0;
31
-static void glue(draw_line2_, DEPTH)(void *opaque,
32
- uint8_t *d, const uint8_t *s, int width, int deststep)
33
+static void draw_line2_32(void *opaque, uint8_t *d, const uint8_t *s,
34
+ int width, int deststep)
35
{
36
uint16_t *pal = opaque;
37
uint8_t v, r, g, b;
38
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line2_, DEPTH)(void *opaque,
39
r = (pal[v & 3] >> 4) & 0xf0;
40
g = pal[v & 3] & 0xf0;
41
b = (pal[v & 3] << 4) & 0xf0;
42
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
43
- d += BPP;
44
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
45
+ d += 4;
46
v >>= 2;
47
r = (pal[v & 3] >> 4) & 0xf0;
48
g = pal[v & 3] & 0xf0;
49
b = (pal[v & 3] << 4) & 0xf0;
50
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
51
- d += BPP;
52
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
53
+ d += 4;
54
v >>= 2;
55
r = (pal[v & 3] >> 4) & 0xf0;
56
g = pal[v & 3] & 0xf0;
57
b = (pal[v & 3] << 4) & 0xf0;
58
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
59
- d += BPP;
60
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
61
+ d += 4;
62
v >>= 2;
63
r = (pal[v & 3] >> 4) & 0xf0;
64
g = pal[v & 3] & 0xf0;
65
b = (pal[v & 3] << 4) & 0xf0;
66
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
67
- d += BPP;
68
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
69
+ d += 4;
70
s ++;
71
width -= 4;
72
} while (width > 0);
73
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line2_, DEPTH)(void *opaque,
74
/*
75
* 4-bit colour
76
*/
77
-static void glue(draw_line4_, DEPTH)(void *opaque,
78
- uint8_t *d, const uint8_t *s, int width, int deststep)
79
+static void draw_line4_32(void *opaque, uint8_t *d, const uint8_t *s,
80
+ int width, int deststep)
81
{
82
uint16_t *pal = opaque;
83
uint8_t v, r, g, b;
84
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line4_, DEPTH)(void *opaque,
85
r = (pal[v & 0xf] >> 4) & 0xf0;
86
g = pal[v & 0xf] & 0xf0;
87
b = (pal[v & 0xf] << 4) & 0xf0;
88
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
89
- d += BPP;
90
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
91
+ d += 4;
92
v >>= 4;
93
r = (pal[v & 0xf] >> 4) & 0xf0;
94
g = pal[v & 0xf] & 0xf0;
95
b = (pal[v & 0xf] << 4) & 0xf0;
96
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
97
- d += BPP;
98
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
99
+ d += 4;
100
s ++;
101
width -= 2;
102
} while (width > 0);
103
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line4_, DEPTH)(void *opaque,
104
/*
105
* 8-bit colour
106
*/
107
-static void glue(draw_line8_, DEPTH)(void *opaque,
108
- uint8_t *d, const uint8_t *s, int width, int deststep)
109
+static void draw_line8_32(void *opaque, uint8_t *d, const uint8_t *s,
110
+ int width, int deststep)
111
{
112
uint16_t *pal = opaque;
113
uint8_t v, r, g, b;
114
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line8_, DEPTH)(void *opaque,
115
r = (pal[v] >> 4) & 0xf0;
116
g = pal[v] & 0xf0;
117
b = (pal[v] << 4) & 0xf0;
118
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
119
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
120
s ++;
121
- d += BPP;
122
+ d += 4;
123
} while (-- width != 0);
124
}
125
126
/*
127
* 12-bit colour
128
*/
129
-static void glue(draw_line12_, DEPTH)(void *opaque,
130
- uint8_t *d, const uint8_t *s, int width, int deststep)
131
+static void draw_line12_32(void *opaque, uint8_t *d, const uint8_t *s,
132
+ int width, int deststep)
133
{
134
uint16_t v;
135
uint8_t r, g, b;
136
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line12_, DEPTH)(void *opaque,
137
r = (v >> 4) & 0xf0;
138
g = v & 0xf0;
139
b = (v << 4) & 0xf0;
140
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
141
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
142
s += 2;
143
- d += BPP;
144
+ d += 4;
145
} while (-- width != 0);
146
}
147
148
/*
149
* 16-bit colour
150
*/
151
-static void glue(draw_line16_, DEPTH)(void *opaque,
152
- uint8_t *d, const uint8_t *s, int width, int deststep)
153
+static void draw_line16_32(void *opaque, uint8_t *d, const uint8_t *s,
154
+ int width, int deststep)
155
{
156
#if defined(HOST_WORDS_BIGENDIAN) == defined(TARGET_WORDS_BIGENDIAN)
157
memcpy(d, s, width * 2);
158
@@ -XXX,XX +XXX,XX @@ static void glue(draw_line16_, DEPTH)(void *opaque,
159
r = (v >> 8) & 0xf8;
160
g = (v >> 3) & 0xfc;
161
b = (v << 3) & 0xf8;
162
- ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b);
163
+ ((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
164
s += 2;
165
- d += BPP;
166
+ d += 4;
167
} while (-- width != 0);
168
#endif
37
#endif
169
}
38
}
170
-
39
-
171
-#undef DEPTH
40
-int xen_available(void)
172
-#undef BPP
41
-{
173
-#undef PIXEL_TYPE
42
-#ifdef CONFIG_XEN
43
- return 1;
44
-#else
45
- return 0;
46
-#endif
47
-}
48
diff --git a/softmmu/vl.c b/softmmu/vl.c
49
index XXXXXXX..XXXXXXX 100644
50
--- a/softmmu/vl.c
51
+++ b/softmmu/vl.c
52
@@ -XXX,XX +XXX,XX @@ void qemu_init(int argc, char **argv, char **envp)
53
has_defaults = 0;
54
break;
55
case QEMU_OPTION_xen_domid:
56
- if (!(xen_available())) {
57
+ if (!(accel_find("xen"))) {
58
error_report("Option not supported for this target");
59
exit(1);
60
}
61
xen_domid = atoi(optarg);
62
break;
63
case QEMU_OPTION_xen_attach:
64
- if (!(xen_available())) {
65
+ if (!(accel_find("xen"))) {
66
error_report("Option not supported for this target");
67
exit(1);
68
}
69
xen_mode = XEN_ATTACH;
70
break;
71
case QEMU_OPTION_xen_domid_restrict:
72
- if (!(xen_available())) {
73
+ if (!(accel_find("xen"))) {
74
error_report("Option not supported for this target");
75
exit(1);
76
}
174
--
77
--
175
2.20.1
78
2.20.1
176
79
177
80
diff view generated by jsdifflib
1
Instead of hardcoding the MachineClass default_ram_size and
1
The kvm_available() function reports whether KVM support was
2
default_ram_id fields, set them on class creation by finding the
2
compiled into the QEMU binary; it returns the value of the
3
entry in the RAMInfo array which is marked as being the QEMU system
3
CONFIG_KVM define.
4
RAM.
4
5
The only place in the codebase where we use this function is
6
in qmp_query_kvm(). Now that accelerators are based on QOM
7
classes we can instead use accel_find("kvm") and remove the
8
kvm_available() function.
5
9
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210215115138.20465-18-peter.maydell@linaro.org
12
Message-id: 20210730105947.28215-3-peter.maydell@linaro.org
9
---
13
---
10
hw/arm/mps2-tz.c | 24 ++++++++++++++++++++++--
14
include/sysemu/arch_init.h | 2 --
11
1 file changed, 22 insertions(+), 2 deletions(-)
15
monitor/qmp-cmds.c | 2 +-
16
softmmu/arch_init.c | 9 ---------
17
3 files changed, 1 insertion(+), 12 deletions(-)
12
18
13
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
19
diff --git a/include/sysemu/arch_init.h b/include/sysemu/arch_init.h
14
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/mps2-tz.c
21
--- a/include/sysemu/arch_init.h
16
+++ b/hw/arm/mps2-tz.c
22
+++ b/include/sysemu/arch_init.h
17
@@ -XXX,XX +XXX,XX @@ static void mps2tz_class_init(ObjectClass *oc, void *data)
23
@@ -XXX,XX +XXX,XX @@ enum {
18
24
19
mc->init = mps2tz_common_init;
25
extern const uint32_t arch_type;
20
iic->check = mps2_tz_idau_check;
26
21
- mc->default_ram_size = 16 * MiB;
27
-int kvm_available(void);
22
- mc->default_ram_id = "mps.ram";
28
-
23
+}
29
/* default virtio transport per architecture */
24
+
30
#define QEMU_ARCH_VIRTIO_PCI (QEMU_ARCH_ALPHA | QEMU_ARCH_ARM | \
25
+static void mps2tz_set_default_ram_info(MPS2TZMachineClass *mmc)
31
QEMU_ARCH_HPPA | QEMU_ARCH_I386 | \
26
+{
32
diff --git a/monitor/qmp-cmds.c b/monitor/qmp-cmds.c
27
+ /*
33
index XXXXXXX..XXXXXXX 100644
28
+ * Set mc->default_ram_size and default_ram_id from the
34
--- a/monitor/qmp-cmds.c
29
+ * information in mmc->raminfo.
35
+++ b/monitor/qmp-cmds.c
30
+ */
36
@@ -XXX,XX +XXX,XX @@ KvmInfo *qmp_query_kvm(Error **errp)
31
+ MachineClass *mc = MACHINE_CLASS(mmc);
37
KvmInfo *info = g_malloc0(sizeof(*info));
32
+ const RAMInfo *p;
38
33
+
39
info->enabled = kvm_enabled();
34
+ for (p = mmc->raminfo; p->name; p++) {
40
- info->present = kvm_available();
35
+ if (p->mrindex < 0) {
41
+ info->present = accel_find("kvm");
36
+ /* Found the entry for "system memory" */
42
37
+ mc->default_ram_size = p->size;
43
return info;
38
+ mc->default_ram_id = p->name;
39
+ return;
40
+ }
41
+ }
42
+ g_assert_not_reached();
43
}
44
}
44
45
diff --git a/softmmu/arch_init.c b/softmmu/arch_init.c
45
static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
46
index XXXXXXX..XXXXXXX 100644
46
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
47
--- a/softmmu/arch_init.c
47
mmc->numirq = 92;
48
+++ b/softmmu/arch_init.c
48
mmc->raminfo = an505_raminfo;
49
@@ -XXX,XX +XXX,XX @@ int graphic_depth = 32;
49
mmc->armsse_type = TYPE_IOTKIT;
50
#endif
50
+ mps2tz_set_default_ram_info(mmc);
51
51
}
52
const uint32_t arch_type = QEMU_ARCH;
52
53
-
53
static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
54
-int kvm_available(void)
54
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
55
-{
55
mmc->numirq = 92;
56
-#ifdef CONFIG_KVM
56
mmc->raminfo = an505_raminfo; /* AN521 is the same as AN505 here */
57
- return 1;
57
mmc->armsse_type = TYPE_SSE200;
58
-#else
58
+ mps2tz_set_default_ram_info(mmc);
59
- return 0;
59
}
60
-#endif
60
61
-}
61
static const TypeInfo mps2tz_info = {
62
--
62
--
63
2.20.1
63
2.20.1
64
64
65
65
diff view generated by jsdifflib
1
The AN505 and AN521 have the same device layout, but the AN524 is
1
arch_init.c does very little but has a long list of #include lines.
2
somewhat different. Allow for more than one PPCInfo array, which can
2
Remove all the unnecessary ones.
3
be selected based on the board type.
4
3
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20210215115138.20465-16-peter.maydell@linaro.org
6
Message-id: 20210730105947.28215-4-peter.maydell@linaro.org
8
---
7
---
9
hw/arm/mps2-tz.c | 16 ++++++++++++++--
8
softmmu/arch_init.c | 7 -------
10
1 file changed, 14 insertions(+), 2 deletions(-)
9
1 file changed, 7 deletions(-)
11
10
12
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
11
diff --git a/softmmu/arch_init.c b/softmmu/arch_init.c
13
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/arm/mps2-tz.c
13
--- a/softmmu/arch_init.c
15
+++ b/hw/arm/mps2-tz.c
14
+++ b/softmmu/arch_init.c
16
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
15
@@ -XXX,XX +XXX,XX @@
17
MemoryRegion *system_memory = get_system_memory();
16
*/
18
DeviceState *iotkitdev;
17
#include "qemu/osdep.h"
19
DeviceState *dev_splitter;
18
#include "sysemu/arch_init.h"
20
+ const PPCInfo *ppcs;
19
-#include "hw/pci/pci.h"
21
+ int num_ppcs;
20
-#include "hw/audio/soundhw.h"
22
int i;
21
-#include "qapi/error.h"
23
22
-#include "qemu/config-file.h"
24
if (strcmp(machine->cpu_type, mc->default_cpu_type) != 0) {
23
-#include "qemu/error-report.h"
25
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
24
-#include "hw/acpi/acpi.h"
26
* + wire up the PPC's control lines to the IoTKit object
25
-#include "qemu/help_option.h"
27
*/
26
28
27
#ifdef TARGET_SPARC
29
- const PPCInfo ppcs[] = { {
28
int graphic_width = 1024;
30
+ const PPCInfo an505_ppcs[] = { {
31
.name = "apb_ppcexp0",
32
.ports = {
33
{ "ssram-0", make_mpc, &mms->ssram_mpc[0], 0x58007000, 0x1000 },
34
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
35
},
36
};
37
38
- for (i = 0; i < ARRAY_SIZE(ppcs); i++) {
39
+ switch (mmc->fpga_type) {
40
+ case FPGA_AN505:
41
+ case FPGA_AN521:
42
+ ppcs = an505_ppcs;
43
+ num_ppcs = ARRAY_SIZE(an505_ppcs);
44
+ break;
45
+ default:
46
+ g_assert_not_reached();
47
+ }
48
+
49
+ for (i = 0; i < num_ppcs; i++) {
50
const PPCInfo *ppcinfo = &ppcs[i];
51
TZPPC *ppc = &mms->ppc[i];
52
DeviceState *ppcdev;
53
--
29
--
54
2.20.1
30
2.20.1
55
31
56
32
diff view generated by jsdifflib
1
Now the template header is included only for BITS==32, expand
1
Instead of using an ifdef ladder in arch_init.c (which we then have
2
out all the macros that depended on the BITS setting.
2
to manually update every time we add or remove a target
3
architecture), have meson.build put "#define QEMU_ARCH QEMU_ARCH_FOO"
4
in the config-target.h file.
3
5
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20210215103215.4944-4-peter.maydell@linaro.org
9
Message-id: 20210730105947.28215-5-peter.maydell@linaro.org
7
---
10
---
8
hw/display/tc6393xb_template.h | 35 ++++------------------------------
11
meson.build | 2 ++
9
1 file changed, 4 insertions(+), 31 deletions(-)
12
softmmu/arch_init.c | 41 -----------------------------------------
13
2 files changed, 2 insertions(+), 41 deletions(-)
10
14
11
diff --git a/hw/display/tc6393xb_template.h b/hw/display/tc6393xb_template.h
15
diff --git a/meson.build b/meson.build
12
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
13
--- a/hw/display/tc6393xb_template.h
17
--- a/meson.build
14
+++ b/hw/display/tc6393xb_template.h
18
+++ b/meson.build
15
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ foreach target : target_dirs
16
* with this program; if not, see <http://www.gnu.org/licenses/>.
20
config_target_data.set(k, v)
17
*/
21
endif
18
22
endforeach
19
-#if BITS == 8
23
+ config_target_data.set('QEMU_ARCH',
20
-# define SET_PIXEL(addr, color) (*(uint8_t *)addr = color)
24
+ 'QEMU_ARCH_' + config_target['TARGET_BASE_ARCH'].to_upper())
21
-#elif BITS == 15 || BITS == 16
25
config_target_h += {target: configure_file(output: target + '-config-target.h',
22
-# define SET_PIXEL(addr, color) (*(uint16_t *)addr = color)
26
configuration: config_target_data)}
23
-#elif BITS == 24
27
24
-# define SET_PIXEL(addr, color) \
28
diff --git a/softmmu/arch_init.c b/softmmu/arch_init.c
25
- do { \
29
index XXXXXXX..XXXXXXX 100644
26
- addr[0] = color; \
30
--- a/softmmu/arch_init.c
27
- addr[1] = (color) >> 8; \
31
+++ b/softmmu/arch_init.c
28
- addr[2] = (color) >> 16; \
32
@@ -XXX,XX +XXX,XX @@ int graphic_height = 600;
29
- } while (0)
33
int graphic_depth = 32;
30
-#elif BITS == 32
34
#endif
31
-# define SET_PIXEL(addr, color) (*(uint32_t *)addr = color)
35
32
-#else
36
-
33
-# error unknown bit depth
37
-#if defined(TARGET_ALPHA)
38
-#define QEMU_ARCH QEMU_ARCH_ALPHA
39
-#elif defined(TARGET_ARM)
40
-#define QEMU_ARCH QEMU_ARCH_ARM
41
-#elif defined(TARGET_CRIS)
42
-#define QEMU_ARCH QEMU_ARCH_CRIS
43
-#elif defined(TARGET_HPPA)
44
-#define QEMU_ARCH QEMU_ARCH_HPPA
45
-#elif defined(TARGET_I386)
46
-#define QEMU_ARCH QEMU_ARCH_I386
47
-#elif defined(TARGET_M68K)
48
-#define QEMU_ARCH QEMU_ARCH_M68K
49
-#elif defined(TARGET_MICROBLAZE)
50
-#define QEMU_ARCH QEMU_ARCH_MICROBLAZE
51
-#elif defined(TARGET_MIPS)
52
-#define QEMU_ARCH QEMU_ARCH_MIPS
53
-#elif defined(TARGET_NIOS2)
54
-#define QEMU_ARCH QEMU_ARCH_NIOS2
55
-#elif defined(TARGET_OPENRISC)
56
-#define QEMU_ARCH QEMU_ARCH_OPENRISC
57
-#elif defined(TARGET_PPC)
58
-#define QEMU_ARCH QEMU_ARCH_PPC
59
-#elif defined(TARGET_RISCV)
60
-#define QEMU_ARCH QEMU_ARCH_RISCV
61
-#elif defined(TARGET_RX)
62
-#define QEMU_ARCH QEMU_ARCH_RX
63
-#elif defined(TARGET_S390X)
64
-#define QEMU_ARCH QEMU_ARCH_S390X
65
-#elif defined(TARGET_SH4)
66
-#define QEMU_ARCH QEMU_ARCH_SH4
67
-#elif defined(TARGET_SPARC)
68
-#define QEMU_ARCH QEMU_ARCH_SPARC
69
-#elif defined(TARGET_TRICORE)
70
-#define QEMU_ARCH QEMU_ARCH_TRICORE
71
-#elif defined(TARGET_XTENSA)
72
-#define QEMU_ARCH QEMU_ARCH_XTENSA
73
-#elif defined(TARGET_AVR)
74
-#define QEMU_ARCH QEMU_ARCH_AVR
34
-#endif
75
-#endif
35
-
76
-
36
-
77
const uint32_t arch_type = QEMU_ARCH;
37
-static void glue(tc6393xb_draw_graphic, BITS)(TC6393xbState *s)
38
+static void tc6393xb_draw_graphic32(TC6393xbState *s)
39
{
40
DisplaySurface *surface = qemu_console_surface(s->con);
41
int i;
42
@@ -XXX,XX +XXX,XX @@ static void glue(tc6393xb_draw_graphic, BITS)(TC6393xbState *s)
43
data_buffer = s->vram_ptr;
44
data_display = surface_data(surface);
45
for(i = 0; i < s->scr_height; i++) {
46
-#if (BITS == 16)
47
- memcpy(data_display, data_buffer, s->scr_width * 2);
48
- data_buffer += s->scr_width;
49
- data_display += surface_stride(surface);
50
-#else
51
int j;
52
- for (j = 0; j < s->scr_width; j++, data_display += BITS / 8, data_buffer++) {
53
+ for (j = 0; j < s->scr_width; j++, data_display += 4, data_buffer++) {
54
uint16_t color = *data_buffer;
55
- uint32_t dest_color = glue(rgb_to_pixel, BITS)(
56
+ uint32_t dest_color = rgb_to_pixel32(
57
((color & 0xf800) * 0x108) >> 11,
58
((color & 0x7e0) * 0x41) >> 9,
59
((color & 0x1f) * 0x21) >> 2
60
);
61
- SET_PIXEL(data_display, dest_color);
62
+ *(uint32_t *)data_display = dest_color;
63
}
64
-#endif
65
}
66
}
67
-
68
-#undef BITS
69
-#undef SET_PIXEL
70
--
78
--
71
2.20.1
79
2.20.1
72
80
73
81
diff view generated by jsdifflib
1
The AN524 has a PL031 RTC, which we have a model of; provide it
1
When Hexagon was added we forgot to add it to the QEMU_ARCH_*
2
rather than an unimplemented-device stub.
2
enumeration. This doesn't cause a visible effect because at the
3
moment Hexagon is linux-user only and the QEMU_ARCH_* constants are
4
only used in softmmu, but we might as well add it in, since it's the
5
only architecture currently missing from the list.
3
6
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20210215115138.20465-23-peter.maydell@linaro.org
10
Reviewed-by: Taylor Simpson <tsimpson@quicinc.com>
11
Message-id: 20210730105947.28215-6-peter.maydell@linaro.org
8
---
12
---
9
hw/arm/mps2-tz.c | 22 ++++++++++++++++++++--
13
include/sysemu/arch_init.h | 1 +
10
1 file changed, 20 insertions(+), 2 deletions(-)
14
1 file changed, 1 insertion(+)
11
15
12
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
16
diff --git a/include/sysemu/arch_init.h b/include/sysemu/arch_init.h
13
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/arm/mps2-tz.c
18
--- a/include/sysemu/arch_init.h
15
+++ b/hw/arm/mps2-tz.c
19
+++ b/include/sysemu/arch_init.h
16
@@ -XXX,XX +XXX,XX @@
20
@@ -XXX,XX +XXX,XX @@ enum {
17
#include "hw/misc/tz-msc.h"
21
QEMU_ARCH_RISCV = (1 << 19),
18
#include "hw/arm/armsse.h"
22
QEMU_ARCH_RX = (1 << 20),
19
#include "hw/dma/pl080.h"
23
QEMU_ARCH_AVR = (1 << 21),
20
+#include "hw/rtc/pl031.h"
24
+ QEMU_ARCH_HEXAGON = (1 << 22),
21
#include "hw/ssi/pl022.h"
25
22
#include "hw/i2c/arm_sbcon_i2c.h"
26
QEMU_ARCH_NONE = (1 << 31),
23
#include "hw/net/lan9118.h"
27
};
24
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
25
UnimplementedDeviceState gpio[4];
26
UnimplementedDeviceState gfx;
27
UnimplementedDeviceState cldc;
28
- UnimplementedDeviceState rtc;
29
UnimplementedDeviceState usb;
30
+ PL031State rtc;
31
PL080State dma[4];
32
TZMSC msc[4];
33
CMSDKAPBUART uart[6];
34
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_i2c(MPS2TZMachineState *mms, void *opaque,
35
return sysbus_mmio_get_region(s, 0);
36
}
37
38
+static MemoryRegion *make_rtc(MPS2TZMachineState *mms, void *opaque,
39
+ const char *name, hwaddr size,
40
+ const int *irqs)
41
+{
42
+ PL031State *pl031 = opaque;
43
+ SysBusDevice *s;
44
+
45
+ object_initialize_child(OBJECT(mms), name, pl031, TYPE_PL031);
46
+ s = SYS_BUS_DEVICE(pl031);
47
+ sysbus_realize(s, &error_fatal);
48
+ /*
49
+ * The board docs don't give an IRQ number for the PL031, so
50
+ * presumably it is not connected.
51
+ */
52
+ return sysbus_mmio_get_region(s, 0);
53
+}
54
+
55
static void create_non_mpc_ram(MPS2TZMachineState *mms)
56
{
57
/*
58
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
59
60
{ /* port 9 reserved */ },
61
{ "clcd", make_unimp_dev, &mms->cldc, 0x4130a000, 0x1000 },
62
- { "rtc", make_unimp_dev, &mms->rtc, 0x4130b000, 0x1000 },
63
+ { "rtc", make_rtc, &mms->rtc, 0x4130b000, 0x1000 },
64
},
65
}, {
66
.name = "ahb_ppcexp0",
67
--
28
--
68
2.20.1
29
2.20.1
69
30
70
31
diff view generated by jsdifflib
1
We create an OR gate to wire together the overflow IRQs for all the
1
The QEMU_ARCH_VIRTIO_* defines are used only in one file,
2
UARTs on the board; this has to have twice the number of inputs as
2
qdev-monitor.c. Move them to that file.
3
there are UARTs, since each UART feeds it a TX overflow and an RX
4
overflow interrupt line. Replace the hardcoded '10' with a
5
calculation based on the size of the uart[] array in the
6
MPS2TZMachineState. (We rely on OR gate inputs that are never wired
7
up or asserted being treated as always-zero.)
8
3
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
Message-id: 20210215115138.20465-15-peter.maydell@linaro.org
6
Reviewed-by: Markus Armbruster <armbru@redhat.com>
7
Message-id: 20210730105947.28215-7-peter.maydell@linaro.org
12
---
8
---
13
hw/arm/mps2-tz.c | 11 ++++++++---
9
include/sysemu/arch_init.h | 9 ---------
14
1 file changed, 8 insertions(+), 3 deletions(-)
10
softmmu/qdev-monitor.c | 9 +++++++++
11
2 files changed, 9 insertions(+), 9 deletions(-)
15
12
16
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
13
diff --git a/include/sysemu/arch_init.h b/include/sysemu/arch_init.h
17
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/arm/mps2-tz.c
15
--- a/include/sysemu/arch_init.h
19
+++ b/hw/arm/mps2-tz.c
16
+++ b/include/sysemu/arch_init.h
20
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
17
@@ -XXX,XX +XXX,XX @@ enum {
21
*/
18
22
memory_region_add_subregion(system_memory, 0x80000000, machine->ram);
19
extern const uint32_t arch_type;
23
20
24
- /* The overflow IRQs for all UARTs are ORed together.
21
-/* default virtio transport per architecture */
25
+ /*
22
-#define QEMU_ARCH_VIRTIO_PCI (QEMU_ARCH_ALPHA | QEMU_ARCH_ARM | \
26
+ * The overflow IRQs for all UARTs are ORed together.
23
- QEMU_ARCH_HPPA | QEMU_ARCH_I386 | \
27
* Tx, Rx and "combined" IRQs are sent to the NVIC separately.
24
- QEMU_ARCH_MIPS | QEMU_ARCH_PPC | \
28
- * Create the OR gate for this.
25
- QEMU_ARCH_RISCV | QEMU_ARCH_SH4 | \
29
+ * Create the OR gate for this: it has one input for the TX overflow
26
- QEMU_ARCH_SPARC | QEMU_ARCH_XTENSA)
30
+ * and one for the RX overflow for each UART we might have.
27
-#define QEMU_ARCH_VIRTIO_CCW (QEMU_ARCH_S390X)
31
+ * (If the board has fewer than the maximum possible number of UARTs
28
-#define QEMU_ARCH_VIRTIO_MMIO (QEMU_ARCH_M68K)
32
+ * those inputs are never wired up and are treated as always-zero.)
29
-
33
*/
30
#endif
34
object_initialize_child(OBJECT(mms), "uart-irq-orgate",
31
diff --git a/softmmu/qdev-monitor.c b/softmmu/qdev-monitor.c
35
&mms->uart_irq_orgate, TYPE_OR_IRQ);
32
index XXXXXXX..XXXXXXX 100644
36
- object_property_set_int(OBJECT(&mms->uart_irq_orgate), "num-lines", 10,
33
--- a/softmmu/qdev-monitor.c
37
+ object_property_set_int(OBJECT(&mms->uart_irq_orgate), "num-lines",
34
+++ b/softmmu/qdev-monitor.c
38
+ 2 * ARRAY_SIZE(mms->uart),
35
@@ -XXX,XX +XXX,XX @@ typedef struct QDevAlias
39
&error_fatal);
36
uint32_t arch_mask;
40
qdev_realize(DEVICE(&mms->uart_irq_orgate), NULL, &error_fatal);
37
} QDevAlias;
41
qdev_connect_gpio_out(DEVICE(&mms->uart_irq_orgate), 0,
38
39
+/* default virtio transport per architecture */
40
+#define QEMU_ARCH_VIRTIO_PCI (QEMU_ARCH_ALPHA | QEMU_ARCH_ARM | \
41
+ QEMU_ARCH_HPPA | QEMU_ARCH_I386 | \
42
+ QEMU_ARCH_MIPS | QEMU_ARCH_PPC | \
43
+ QEMU_ARCH_RISCV | QEMU_ARCH_SH4 | \
44
+ QEMU_ARCH_SPARC | QEMU_ARCH_XTENSA)
45
+#define QEMU_ARCH_VIRTIO_CCW (QEMU_ARCH_S390X)
46
+#define QEMU_ARCH_VIRTIO_MMIO (QEMU_ARCH_M68K)
47
+
48
/* Please keep this table sorted by typename. */
49
static const QDevAlias qdev_alias_table[] = {
50
{ "AC97", "ac97" }, /* -soundhw name */
42
--
51
--
43
2.20.1
52
2.20.1
44
53
45
54
diff view generated by jsdifflib
1
The AN524 has a USB controller (an ISP1763); we don't have a model of
1
arch_init.h only defines the QEMU_ARCH_* enumeration and the
2
it but we should provide a stub "unimplemented-device" for it. This
2
arch_type global. Don't include it in files that don't use those.
3
is slightly complicated because the USB controller shares a PPC port
4
with the ethernet controller.
5
6
Implement a make_* function which provides creates a container
7
MemoryRegion with both the ethernet controller and an
8
unimplemented-device stub for the USB controller.
9
3
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
5
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
12
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
13
Message-id: 20210215115138.20465-22-peter.maydell@linaro.org
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
8
Message-id: 20210730105947.28215-8-peter.maydell@linaro.org
14
---
9
---
15
hw/arm/mps2-tz.c | 48 +++++++++++++++++++++++++++++++++++++++++++++++-
10
blockdev.c | 1 -
16
1 file changed, 47 insertions(+), 1 deletion(-)
11
hw/i386/pc.c | 1 -
12
hw/i386/pc_piix.c | 1 -
13
hw/i386/pc_q35.c | 1 -
14
hw/mips/jazz.c | 1 -
15
hw/mips/malta.c | 1 -
16
hw/ppc/prep.c | 1 -
17
hw/riscv/sifive_e.c | 1 -
18
hw/riscv/sifive_u.c | 1 -
19
hw/riscv/spike.c | 1 -
20
hw/riscv/virt.c | 1 -
21
monitor/qmp-cmds.c | 1 -
22
target/ppc/cpu_init.c | 1 -
23
target/s390x/cpu-sysemu.c | 1 -
24
14 files changed, 14 deletions(-)
17
25
18
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
26
diff --git a/blockdev.c b/blockdev.c
19
index XXXXXXX..XXXXXXX 100644
27
index XXXXXXX..XXXXXXX 100644
20
--- a/hw/arm/mps2-tz.c
28
--- a/blockdev.c
21
+++ b/hw/arm/mps2-tz.c
29
+++ b/blockdev.c
22
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
30
@@ -XXX,XX +XXX,XX @@
23
31
#include "sysemu/iothread.h"
24
ARMSSE iotkit;
32
#include "block/block_int.h"
25
MemoryRegion ram[MPS2TZ_RAM_MAX];
33
#include "block/trace.h"
26
+ MemoryRegion eth_usb_container;
34
-#include "sysemu/arch_init.h"
27
+
35
#include "sysemu/runstate.h"
28
MPS2SCC scc;
36
#include "sysemu/replay.h"
29
MPS2FPGAIO fpgaio;
37
#include "qemu/cutils.h"
30
TZPPC ppc[5];
38
diff --git a/hw/i386/pc.c b/hw/i386/pc.c
31
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
39
index XXXXXXX..XXXXXXX 100644
32
UnimplementedDeviceState gfx;
40
--- a/hw/i386/pc.c
33
UnimplementedDeviceState cldc;
41
+++ b/hw/i386/pc.c
34
UnimplementedDeviceState rtc;
42
@@ -XXX,XX +XXX,XX @@
35
+ UnimplementedDeviceState usb;
43
#include "hw/xen/start_info.h"
36
PL080State dma[4];
44
#include "ui/qemu-spice.h"
37
TZMSC msc[4];
45
#include "exec/memory.h"
38
CMSDKAPBUART uart[6];
46
-#include "sysemu/arch_init.h"
39
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_eth_dev(MPS2TZMachineState *mms, void *opaque,
47
#include "qemu/bitmap.h"
40
return sysbus_mmio_get_region(s, 0);
48
#include "qemu/config-file.h"
41
}
49
#include "qemu/error-report.h"
42
50
diff --git a/hw/i386/pc_piix.c b/hw/i386/pc_piix.c
43
+static MemoryRegion *make_eth_usb(MPS2TZMachineState *mms, void *opaque,
51
index XXXXXXX..XXXXXXX 100644
44
+ const char *name, hwaddr size,
52
--- a/hw/i386/pc_piix.c
45
+ const int *irqs)
53
+++ b/hw/i386/pc_piix.c
46
+{
54
@@ -XXX,XX +XXX,XX @@
47
+ /*
55
#include "sysemu/kvm.h"
48
+ * The AN524 makes the ethernet and USB share a PPC port.
56
#include "hw/kvm/clock.h"
49
+ * irqs[] is the ethernet IRQ.
57
#include "hw/sysbus.h"
50
+ */
58
-#include "sysemu/arch_init.h"
51
+ SysBusDevice *s;
59
#include "hw/i2c/smbus_eeprom.h"
52
+ NICInfo *nd = &nd_table[0];
60
#include "hw/xen/xen-x86.h"
53
+
61
#include "exec/memory.h"
54
+ memory_region_init(&mms->eth_usb_container, OBJECT(mms),
62
diff --git a/hw/i386/pc_q35.c b/hw/i386/pc_q35.c
55
+ "mps2-tz-eth-usb-container", 0x200000);
63
index XXXXXXX..XXXXXXX 100644
56
+
64
--- a/hw/i386/pc_q35.c
57
+ /*
65
+++ b/hw/i386/pc_q35.c
58
+ * In hardware this is a LAN9220; the LAN9118 is software compatible
66
@@ -XXX,XX +XXX,XX @@
59
+ * except that it doesn't support the checksum-offload feature.
67
#include "qemu/osdep.h"
60
+ */
68
#include "qemu/units.h"
61
+ qemu_check_nic_model(nd, "lan9118");
69
#include "hw/loader.h"
62
+ mms->lan9118 = qdev_new(TYPE_LAN9118);
70
-#include "sysemu/arch_init.h"
63
+ qdev_set_nic_properties(mms->lan9118, nd);
71
#include "hw/i2c/smbus_eeprom.h"
64
+
72
#include "hw/rtc/mc146818rtc.h"
65
+ s = SYS_BUS_DEVICE(mms->lan9118);
73
#include "sysemu/kvm.h"
66
+ sysbus_realize_and_unref(s, &error_fatal);
74
diff --git a/hw/mips/jazz.c b/hw/mips/jazz.c
67
+ sysbus_connect_irq(s, 0, get_sse_irq_in(mms, irqs[0]));
75
index XXXXXXX..XXXXXXX 100644
68
+
76
--- a/hw/mips/jazz.c
69
+ memory_region_add_subregion(&mms->eth_usb_container,
77
+++ b/hw/mips/jazz.c
70
+ 0, sysbus_mmio_get_region(s, 0));
78
@@ -XXX,XX +XXX,XX @@
71
+
79
#include "hw/isa/isa.h"
72
+ /* The USB OTG controller is an ISP1763; we don't have a model of it. */
80
#include "hw/block/fdc.h"
73
+ object_initialize_child(OBJECT(mms), "usb-otg",
81
#include "sysemu/sysemu.h"
74
+ &mms->usb, TYPE_UNIMPLEMENTED_DEVICE);
82
-#include "sysemu/arch_init.h"
75
+ qdev_prop_set_string(DEVICE(&mms->usb), "name", "usb-otg");
83
#include "hw/boards.h"
76
+ qdev_prop_set_uint64(DEVICE(&mms->usb), "size", 0x100000);
84
#include "net/net.h"
77
+ s = SYS_BUS_DEVICE(&mms->usb);
85
#include "hw/scsi/esp.h"
78
+ sysbus_realize(s, &error_fatal);
86
diff --git a/hw/mips/malta.c b/hw/mips/malta.c
79
+
87
index XXXXXXX..XXXXXXX 100644
80
+ memory_region_add_subregion(&mms->eth_usb_container,
88
--- a/hw/mips/malta.c
81
+ 0x100000, sysbus_mmio_get_region(s, 0));
89
+++ b/hw/mips/malta.c
82
+
90
@@ -XXX,XX +XXX,XX @@
83
+ return &mms->eth_usb_container;
91
#include "hw/mips/mips.h"
84
+}
92
#include "hw/mips/cpudevs.h"
85
+
93
#include "hw/pci/pci.h"
86
static MemoryRegion *make_mpc(MPS2TZMachineState *mms, void *opaque,
94
-#include "sysemu/arch_init.h"
87
const char *name, hwaddr size,
95
#include "qemu/log.h"
88
const int *irqs)
96
#include "hw/mips/bios.h"
89
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
97
#include "hw/ide.h"
90
{ "gpio1", make_unimp_dev, &mms->gpio[1], 0x41101000, 0x1000 },
98
diff --git a/hw/ppc/prep.c b/hw/ppc/prep.c
91
{ "gpio2", make_unimp_dev, &mms->gpio[2], 0x41102000, 0x1000 },
99
index XXXXXXX..XXXXXXX 100644
92
{ "gpio3", make_unimp_dev, &mms->gpio[3], 0x41103000, 0x1000 },
100
--- a/hw/ppc/prep.c
93
- { "eth", make_eth_dev, NULL, 0x41400000, 0x100000, { 48 } },
101
+++ b/hw/ppc/prep.c
94
+ { "eth-usb", make_eth_usb, NULL, 0x41400000, 0x200000, { 48 } },
102
@@ -XXX,XX +XXX,XX @@
95
},
103
#include "hw/rtc/mc146818rtc.h"
96
},
104
#include "hw/isa/pc87312.h"
97
};
105
#include "hw/qdev-properties.h"
106
-#include "sysemu/arch_init.h"
107
#include "sysemu/kvm.h"
108
#include "sysemu/reset.h"
109
#include "trace.h"
110
diff --git a/hw/riscv/sifive_e.c b/hw/riscv/sifive_e.c
111
index XXXXXXX..XXXXXXX 100644
112
--- a/hw/riscv/sifive_e.c
113
+++ b/hw/riscv/sifive_e.c
114
@@ -XXX,XX +XXX,XX @@
115
#include "hw/intc/sifive_plic.h"
116
#include "hw/misc/sifive_e_prci.h"
117
#include "chardev/char.h"
118
-#include "sysemu/arch_init.h"
119
#include "sysemu/sysemu.h"
120
121
static const MemMapEntry sifive_e_memmap[] = {
122
diff --git a/hw/riscv/sifive_u.c b/hw/riscv/sifive_u.c
123
index XXXXXXX..XXXXXXX 100644
124
--- a/hw/riscv/sifive_u.c
125
+++ b/hw/riscv/sifive_u.c
126
@@ -XXX,XX +XXX,XX @@
127
#include "hw/intc/sifive_plic.h"
128
#include "chardev/char.h"
129
#include "net/eth.h"
130
-#include "sysemu/arch_init.h"
131
#include "sysemu/device_tree.h"
132
#include "sysemu/runstate.h"
133
#include "sysemu/sysemu.h"
134
diff --git a/hw/riscv/spike.c b/hw/riscv/spike.c
135
index XXXXXXX..XXXXXXX 100644
136
--- a/hw/riscv/spike.c
137
+++ b/hw/riscv/spike.c
138
@@ -XXX,XX +XXX,XX @@
139
#include "hw/char/riscv_htif.h"
140
#include "hw/intc/sifive_clint.h"
141
#include "chardev/char.h"
142
-#include "sysemu/arch_init.h"
143
#include "sysemu/device_tree.h"
144
#include "sysemu/sysemu.h"
145
146
diff --git a/hw/riscv/virt.c b/hw/riscv/virt.c
147
index XXXXXXX..XXXXXXX 100644
148
--- a/hw/riscv/virt.c
149
+++ b/hw/riscv/virt.c
150
@@ -XXX,XX +XXX,XX @@
151
#include "hw/intc/sifive_plic.h"
152
#include "hw/misc/sifive_test.h"
153
#include "chardev/char.h"
154
-#include "sysemu/arch_init.h"
155
#include "sysemu/device_tree.h"
156
#include "sysemu/sysemu.h"
157
#include "hw/pci/pci.h"
158
diff --git a/monitor/qmp-cmds.c b/monitor/qmp-cmds.c
159
index XXXXXXX..XXXXXXX 100644
160
--- a/monitor/qmp-cmds.c
161
+++ b/monitor/qmp-cmds.c
162
@@ -XXX,XX +XXX,XX @@
163
#include "sysemu/kvm.h"
164
#include "sysemu/runstate.h"
165
#include "sysemu/runstate-action.h"
166
-#include "sysemu/arch_init.h"
167
#include "sysemu/blockdev.h"
168
#include "sysemu/block-backend.h"
169
#include "qapi/error.h"
170
diff --git a/target/ppc/cpu_init.c b/target/ppc/cpu_init.c
171
index XXXXXXX..XXXXXXX 100644
172
--- a/target/ppc/cpu_init.c
173
+++ b/target/ppc/cpu_init.c
174
@@ -XXX,XX +XXX,XX @@
175
#include "disas/dis-asm.h"
176
#include "exec/gdbstub.h"
177
#include "kvm_ppc.h"
178
-#include "sysemu/arch_init.h"
179
#include "sysemu/cpus.h"
180
#include "sysemu/hw_accel.h"
181
#include "sysemu/tcg.h"
182
diff --git a/target/s390x/cpu-sysemu.c b/target/s390x/cpu-sysemu.c
183
index XXXXXXX..XXXXXXX 100644
184
--- a/target/s390x/cpu-sysemu.c
185
+++ b/target/s390x/cpu-sysemu.c
186
@@ -XXX,XX +XXX,XX @@
187
188
#include "hw/s390x/pv.h"
189
#include "hw/boards.h"
190
-#include "sysemu/arch_init.h"
191
#include "sysemu/sysemu.h"
192
#include "sysemu/tcg.h"
193
#include "hw/core/sysemu-cpu-ops.h"
98
--
194
--
99
2.20.1
195
2.20.1
100
196
101
197
diff view generated by jsdifflib
1
Set the FPGAIO num-leds and have-switches properties explicitly
1
We added a stub for the arch_type global in commit 5964ed56d9a1 so
2
per-board, rather than relying on the defaults. The AN505 and AN521
2
that we could compile blockdev.c into the tools. However, in commit
3
both have the same settings as the default values, but the AN524 will
3
9db1d3a2be9bf we removed the only use of arch_type from blockdev.c.
4
be different.
4
The stub is therefore no longer needed, and we can delete it again,
5
together with the QEMU_ARCH_NONE value that only the stub was using.
5
6
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20210215115138.20465-8-peter.maydell@linaro.org
10
Message-id: 20210730105947.28215-9-peter.maydell@linaro.org
10
---
11
---
11
hw/arm/mps2-tz.c | 9 +++++++++
12
include/sysemu/arch_init.h | 2 --
12
1 file changed, 9 insertions(+)
13
stubs/arch_type.c | 4 ----
14
stubs/meson.build | 1 -
15
3 files changed, 7 deletions(-)
16
delete mode 100644 stubs/arch_type.c
13
17
14
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
18
diff --git a/include/sysemu/arch_init.h b/include/sysemu/arch_init.h
15
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/arm/mps2-tz.c
20
--- a/include/sysemu/arch_init.h
17
+++ b/hw/arm/mps2-tz.c
21
+++ b/include/sysemu/arch_init.h
18
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineClass {
22
@@ -XXX,XX +XXX,XX @@ enum {
19
uint32_t sysclk_frq; /* Main SYSCLK frequency in Hz */
23
QEMU_ARCH_RX = (1 << 20),
20
uint32_t len_oscclk;
24
QEMU_ARCH_AVR = (1 << 21),
21
const uint32_t *oscclk;
25
QEMU_ARCH_HEXAGON = (1 << 22),
22
+ uint32_t fpgaio_num_leds; /* Number of LEDs in FPGAIO LED0 register */
26
-
23
+ bool fpgaio_has_switches; /* Does FPGAIO have SWITCH register? */
27
- QEMU_ARCH_NONE = (1 << 31),
24
const char *armsse_type;
25
};
28
};
26
29
27
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_fpgaio(MPS2TZMachineState *mms, void *opaque,
30
extern const uint32_t arch_type;
28
const char *name, hwaddr size)
31
diff --git a/stubs/arch_type.c b/stubs/arch_type.c
29
{
32
deleted file mode 100644
30
MPS2FPGAIO *fpgaio = opaque;
33
index XXXXXXX..XXXXXXX
31
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
34
--- a/stubs/arch_type.c
32
35
+++ /dev/null
33
object_initialize_child(OBJECT(mms), "fpgaio", fpgaio, TYPE_MPS2_FPGAIO);
36
@@ -XXX,XX +XXX,XX @@
34
+ qdev_prop_set_uint32(DEVICE(fpgaio), "num-leds", mmc->fpgaio_num_leds);
37
-#include "qemu/osdep.h"
35
+ qdev_prop_set_bit(DEVICE(fpgaio), "has-switches", mmc->fpgaio_has_switches);
38
-#include "sysemu/arch_init.h"
36
sysbus_realize(SYS_BUS_DEVICE(fpgaio), &error_fatal);
39
-
37
return sysbus_mmio_get_region(SYS_BUS_DEVICE(fpgaio), 0);
40
-const uint32_t arch_type = QEMU_ARCH_NONE;
38
}
41
diff --git a/stubs/meson.build b/stubs/meson.build
39
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
42
index XXXXXXX..XXXXXXX 100644
40
mmc->sysclk_frq = 20 * 1000 * 1000; /* 20MHz */
43
--- a/stubs/meson.build
41
mmc->oscclk = an505_oscclk;
44
+++ b/stubs/meson.build
42
mmc->len_oscclk = ARRAY_SIZE(an505_oscclk);
45
@@ -XXX,XX +XXX,XX @@
43
+ mmc->fpgaio_num_leds = 2;
46
-stub_ss.add(files('arch_type.c'))
44
+ mmc->fpgaio_has_switches = false;
47
stub_ss.add(files('bdrv-next-monitor-owned.c'))
45
mmc->armsse_type = TYPE_IOTKIT;
48
stub_ss.add(files('blk-commit-all.c'))
46
}
49
stub_ss.add(files('blk-exp-close-all.c'))
47
48
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
49
mmc->sysclk_frq = 20 * 1000 * 1000; /* 20MHz */
50
mmc->oscclk = an505_oscclk; /* AN521 is the same as AN505 here */
51
mmc->len_oscclk = ARRAY_SIZE(an505_oscclk);
52
+ mmc->fpgaio_num_leds = 2;
53
+ mmc->fpgaio_has_switches = false;
54
mmc->armsse_type = TYPE_SSE200;
55
}
56
57
--
50
--
58
2.20.1
51
2.20.1
59
52
60
53
diff view generated by jsdifflib
1
The armv7m_load_kernel() function takes a mem_size argument which it
1
The gunzip() function reads various fields from a passed in source
2
expects to be the size of the memory region at guest address 0. (It
2
buffer in order to skip a header before passing the actual compressed
3
uses this argument only as a limit on how large a raw image file it
3
data to the zlib inflate() function. It does check whether the
4
can load at address zero).
4
passed in buffer is too small, but unfortunately it checks that only
5
after reading bytes from the src buffer, so it could read off the end
6
of the buffer.
5
7
6
Instead of hardcoding this value, find the RAMInfo corresponding to
8
You can see this with valgrind:
7
the 0 address and extract its size.
8
9
10
$ printf "%b" '\x1f\x8b' > /tmp/image
11
$ valgrind qemu-system-aarch64 -display none -M virt -cpu max -kernel /tmp/image
12
[...]
13
==19224== Invalid read of size 1
14
==19224== at 0x67302E: gunzip (loader.c:558)
15
==19224== by 0x673907: load_image_gzipped_buffer (loader.c:788)
16
==19224== by 0xA18032: load_aarch64_image (boot.c:932)
17
==19224== by 0xA18489: arm_setup_direct_kernel_boot (boot.c:1063)
18
==19224== by 0xA18D90: arm_load_kernel (boot.c:1317)
19
==19224== by 0x9F3651: machvirt_init (virt.c:2114)
20
==19224== by 0x794B7A: machine_run_board_init (machine.c:1272)
21
==19224== by 0xD5CAD3: qemu_init_board (vl.c:2618)
22
==19224== by 0xD5CCA6: qmp_x_exit_preconfig (vl.c:2692)
23
==19224== by 0xD5F32E: qemu_init (vl.c:3713)
24
==19224== by 0x5ADDB1: main (main.c:49)
25
==19224== Address 0x3802a873 is 0 bytes after a block of size 3 alloc'd
26
==19224== at 0x4C31B0F: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
27
==19224== by 0x61E7657: g_file_get_contents (in /usr/lib/x86_64-linux-gnu/libglib-2.0.so.0.5600.4)
28
==19224== by 0x673895: load_image_gzipped_buffer (loader.c:771)
29
==19224== by 0xA18032: load_aarch64_image (boot.c:932)
30
==19224== by 0xA18489: arm_setup_direct_kernel_boot (boot.c:1063)
31
==19224== by 0xA18D90: arm_load_kernel (boot.c:1317)
32
==19224== by 0x9F3651: machvirt_init (virt.c:2114)
33
==19224== by 0x794B7A: machine_run_board_init (machine.c:1272)
34
==19224== by 0xD5CAD3: qemu_init_board (vl.c:2618)
35
==19224== by 0xD5CCA6: qmp_x_exit_preconfig (vl.c:2692)
36
==19224== by 0xD5F32E: qemu_init (vl.c:3713)
37
==19224== by 0x5ADDB1: main (main.c:49)
38
39
Check that we have enough bytes of data to read the header bytes that
40
we read before we read them.
41
42
Fixes: Coverity 1458997
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
43
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
44
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
45
Message-id: 20210812141803.20913-1-peter.maydell@linaro.org
12
Message-id: 20210215115138.20465-20-peter.maydell@linaro.org
13
---
46
---
14
hw/arm/mps2-tz.c | 17 ++++++++++++++++-
47
hw/core/loader.c | 35 +++++++++++++++++++++++++----------
15
1 file changed, 16 insertions(+), 1 deletion(-)
48
1 file changed, 25 insertions(+), 10 deletions(-)
16
49
17
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
50
diff --git a/hw/core/loader.c b/hw/core/loader.c
18
index XXXXXXX..XXXXXXX 100644
51
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/arm/mps2-tz.c
52
--- a/hw/core/loader.c
20
+++ b/hw/arm/mps2-tz.c
53
+++ b/hw/core/loader.c
21
@@ -XXX,XX +XXX,XX @@ static void create_non_mpc_ram(MPS2TZMachineState *mms)
54
@@ -XXX,XX +XXX,XX @@ ssize_t gunzip(void *dst, size_t dstlen, uint8_t *src, size_t srclen)
55
56
/* skip header */
57
i = 10;
58
+ if (srclen < 4) {
59
+ goto toosmall;
60
+ }
61
flags = src[3];
62
if (src[2] != DEFLATED || (flags & RESERVED) != 0) {
63
puts ("Error: Bad gzipped data\n");
64
return -1;
22
}
65
}
23
}
66
- if ((flags & EXTRA_FIELD) != 0)
24
67
+ if ((flags & EXTRA_FIELD) != 0) {
25
+static uint32_t boot_ram_size(MPS2TZMachineState *mms)
68
+ if (srclen < 12) {
26
+{
69
+ goto toosmall;
27
+ /* Return the size of the RAM block at guest address zero */
70
+ }
28
+ const RAMInfo *p;
71
i = 12 + src[10] + (src[11] << 8);
29
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
72
- if ((flags & ORIG_NAME) != 0)
30
+
73
- while (src[i++] != 0)
31
+ for (p = mmc->raminfo; p->name; p++) {
74
- ;
32
+ if (p->base == 0) {
75
- if ((flags & COMMENT) != 0)
33
+ return p->size;
76
- while (src[i++] != 0)
77
- ;
78
- if ((flags & HEAD_CRC) != 0)
79
+ }
80
+ if ((flags & ORIG_NAME) != 0) {
81
+ while (i < srclen && src[i++] != 0) {
82
+ /* do nothing */
34
+ }
83
+ }
35
+ }
84
+ }
36
+ g_assert_not_reached();
85
+ if ((flags & COMMENT) != 0) {
37
+}
86
+ while (i < srclen && src[i++] != 0) {
87
+ /* do nothing */
88
+ }
89
+ }
90
+ if ((flags & HEAD_CRC) != 0) {
91
i += 2;
92
+ }
93
if (i >= srclen) {
94
- puts ("Error: gunzip out of data in header\n");
95
- return -1;
96
+ goto toosmall;
97
}
98
99
s.zalloc = zalloc;
100
@@ -XXX,XX +XXX,XX @@ ssize_t gunzip(void *dst, size_t dstlen, uint8_t *src, size_t srclen)
101
inflateEnd(&s);
102
103
return dstbytes;
38
+
104
+
39
static void mps2tz_common_init(MachineState *machine)
105
+toosmall:
40
{
106
+ puts("Error: gunzip out of data in header\n");
41
MPS2TZMachineState *mms = MPS2TZ_MACHINE(machine);
107
+ return -1;
42
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
43
44
create_non_mpc_ram(mms);
45
46
- armv7m_load_kernel(ARM_CPU(first_cpu), machine->kernel_filename, 0x400000);
47
+ armv7m_load_kernel(ARM_CPU(first_cpu), machine->kernel_filename,
48
+ boot_ram_size(mms));
49
}
108
}
50
109
51
static void mps2_tz_idau_check(IDAUInterface *ii, uint32_t address,
110
/* Load a U-Boot image. */
52
--
111
--
53
2.20.1
112
2.20.1
54
113
55
114
diff view generated by jsdifflib
1
From: Peter Collingbourne <pcc@google.com>
1
In the alignment check added to qemu_ram_alloc_from_fd() in commit
2
ce317be98db0dfdfa, the condition includes a check that 'mr' is not
3
NULL. This check is unnecessary because we can assume that the
4
caller always passes us a valid MemoryRegion, and indeed later in the
5
function we assume mr is not NULL when we pass it to file_ram_alloc()
6
as new_block->mr. Remove it.
2
7
3
Section D6.7 of the ARM ARM states:
8
Fixes: Coverity 1459867
9
Fixes: ce317be98d ("exec: fetch the alignment of Linux devdax pmem character device nodes")
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Reviewed-by: Jingqi Liu <jingqi.liu@intel.com>
12
Message-id: 20210812150624.29139-1-peter.maydell@linaro.org
13
---
14
softmmu/physmem.c | 2 +-
15
1 file changed, 1 insertion(+), 1 deletion(-)
4
16
5
For the purpose of determining Tag Check Fault handling, unprivileged
17
diff --git a/softmmu/physmem.c b/softmmu/physmem.c
6
load and store instructions are treated as if executed at EL0 when
7
executed at either:
8
- EL1, when the Effective value of PSTATE.UAO is 0.
9
- EL2, when both the Effective value of HCR_EL2.{E2H, TGE} is {1, 1}
10
and the Effective value of PSTATE.UAO is 0.
11
12
ARM has confirmed a defect in the pseudocode function
13
AArch64.TagCheckFault that makes it inconsistent with the above
14
wording. The remedy is to adjust references to PSTATE.EL in that
15
function to instead refer to AArch64.AccessUsesEL(acctype), so
16
that unprivileged instructions use SCTLR_EL1.TCF0 and TFSRE0_EL1.
17
The exception type for synchronous tag check faults remains unchanged.
18
19
This patch implements the described change by partially reverting
20
commits 50244cc76abc and cc97b0019bb5.
21
22
Signed-off-by: Peter Collingbourne <pcc@google.com>
23
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
24
Message-id: 20210219201820.2672077-1-pcc@google.com
25
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
26
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
27
---
28
target/arm/helper.c | 2 +-
29
target/arm/mte_helper.c | 13 +++++++++----
30
2 files changed, 10 insertions(+), 5 deletions(-)
31
32
diff --git a/target/arm/helper.c b/target/arm/helper.c
33
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
34
--- a/target/arm/helper.c
19
--- a/softmmu/physmem.c
35
+++ b/target/arm/helper.c
20
+++ b/softmmu/physmem.c
36
@@ -XXX,XX +XXX,XX @@ static uint32_t rebuild_hflags_a64(CPUARMState *env, int el, int fp_el,
21
@@ -XXX,XX +XXX,XX @@ RAMBlock *qemu_ram_alloc_from_fd(ram_addr_t size, MemoryRegion *mr,
37
if (FIELD_EX32(flags, TBFLAG_A64, UNPRIV)
38
&& tbid
39
&& !(env->pstate & PSTATE_TCO)
40
- && (sctlr & SCTLR_TCF)
41
+ && (sctlr & SCTLR_TCF0)
42
&& allocation_tag_access_enabled(env, 0, sctlr)) {
43
flags = FIELD_DP32(flags, TBFLAG_A64, MTE0_ACTIVE, 1);
44
}
45
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
46
index XXXXXXX..XXXXXXX 100644
47
--- a/target/arm/mte_helper.c
48
+++ b/target/arm/mte_helper.c
49
@@ -XXX,XX +XXX,XX @@ static void mte_check_fail(CPUARMState *env, uint32_t desc,
50
reg_el = regime_el(env, arm_mmu_idx);
51
sctlr = env->cp15.sctlr_el[reg_el];
52
53
- el = arm_current_el(env);
54
- if (el == 0) {
55
+ switch (arm_mmu_idx) {
56
+ case ARMMMUIdx_E10_0:
57
+ case ARMMMUIdx_E20_0:
58
+ el = 0;
59
tcf = extract64(sctlr, 38, 2);
60
- } else {
61
+ break;
62
+ default:
63
+ el = reg_el;
64
tcf = extract64(sctlr, 40, 2);
65
}
22
}
66
23
67
@@ -XXX,XX +XXX,XX @@ static void mte_check_fail(CPUARMState *env, uint32_t desc,
24
file_align = get_file_align(fd);
68
env->exception.vaddress = dirty_ptr;
25
- if (file_align > 0 && mr && file_align > mr->align) {
69
26
+ if (file_align > 0 && file_align > mr->align) {
70
is_write = FIELD_EX32(desc, MTEDESC, WRITE);
27
error_setg(errp, "backing store align 0x%" PRIx64
71
- syn = syn_data_abort_no_iss(el != 0, 0, 0, 0, 0, is_write, 0x11);
28
" is larger than 'align' option 0x%" PRIx64,
72
+ syn = syn_data_abort_no_iss(arm_current_el(env) != 0, 0, 0, 0, 0,
29
file_align, mr->align);
73
+ is_write, 0x11);
74
raise_exception(env, EXCP_DATA_ABORT, syn, exception_target_el(env));
75
/* noreturn, but fall through to the assert anyway */
76
77
--
30
--
78
2.20.1
31
2.20.1
79
32
80
33
diff view generated by jsdifflib
1
Move the specification of the IRQ information for the uart, ethernet,
1
The realpath() function can return NULL on error, so we need to check
2
dma and spi devices to the data structures. (The other devices
2
for it to avoid crashing when we try to strstr() into it.
3
handled by the PPCPortInfo structures don't have any interrupt lines
3
This can happen if we run out of memory, or if /sys/ is not mounted,
4
we need to wire up.)
4
among other situations.
5
5
6
Fixes: Coverity 1459913, 1460474
7
Fixes: ce317be98db0 ("exec: fetch the alignment of Linux devdax pmem character device nodes")
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Jingqi Liu <jingqi.liu@intel.com>
8
Message-id: 20210215115138.20465-14-peter.maydell@linaro.org
10
Message-id: 20210812151525.31456-1-peter.maydell@linaro.org
9
---
11
---
10
hw/arm/mps2-tz.c | 52 +++++++++++++++++++++++-------------------------
12
softmmu/physmem.c | 3 +++
11
1 file changed, 25 insertions(+), 27 deletions(-)
13
1 file changed, 3 insertions(+)
12
14
13
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
15
diff --git a/softmmu/physmem.c b/softmmu/physmem.c
14
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/mps2-tz.c
17
--- a/softmmu/physmem.c
16
+++ b/hw/arm/mps2-tz.c
18
+++ b/softmmu/physmem.c
17
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
19
@@ -XXX,XX +XXX,XX @@ static int64_t get_file_align(int fd)
18
const char *name, hwaddr size,
20
path = g_strdup_printf("/sys/dev/char/%d:%d",
19
const int *irqs)
21
major(st.st_rdev), minor(st.st_rdev));
20
{
22
rpath = realpath(path, NULL);
21
+ /* The irq[] array is tx, rx, combined, in that order */
23
+ if (!rpath) {
22
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
24
+ return -errno;
23
CMSDKAPBUART *uart = opaque;
25
+ }
24
int i = uart - &mms->uart[0];
26
25
- int rxirqno = i * 2 + 32;
27
rc = daxctl_new(&ctx);
26
- int txirqno = i * 2 + 33;
28
if (rc) {
27
- int combirqno = i + 42;
28
SysBusDevice *s;
29
DeviceState *orgate_dev = DEVICE(&mms->uart_irq_orgate);
30
31
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
32
qdev_prop_set_uint32(DEVICE(uart), "pclk-frq", mmc->sysclk_frq);
33
sysbus_realize(SYS_BUS_DEVICE(uart), &error_fatal);
34
s = SYS_BUS_DEVICE(uart);
35
- sysbus_connect_irq(s, 0, get_sse_irq_in(mms, txirqno));
36
- sysbus_connect_irq(s, 1, get_sse_irq_in(mms, rxirqno));
37
+ sysbus_connect_irq(s, 0, get_sse_irq_in(mms, irqs[0]));
38
+ sysbus_connect_irq(s, 1, get_sse_irq_in(mms, irqs[1]));
39
sysbus_connect_irq(s, 2, qdev_get_gpio_in(orgate_dev, i * 2));
40
sysbus_connect_irq(s, 3, qdev_get_gpio_in(orgate_dev, i * 2 + 1));
41
- sysbus_connect_irq(s, 4, get_sse_irq_in(mms, combirqno));
42
+ sysbus_connect_irq(s, 4, get_sse_irq_in(mms, irqs[2]));
43
return sysbus_mmio_get_region(SYS_BUS_DEVICE(uart), 0);
44
}
45
46
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_eth_dev(MPS2TZMachineState *mms, void *opaque,
47
48
s = SYS_BUS_DEVICE(mms->lan9118);
49
sysbus_realize_and_unref(s, &error_fatal);
50
- sysbus_connect_irq(s, 0, get_sse_irq_in(mms, 48));
51
+ sysbus_connect_irq(s, 0, get_sse_irq_in(mms, irqs[0]));
52
return sysbus_mmio_get_region(s, 0);
53
}
54
55
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_dma(MPS2TZMachineState *mms, void *opaque,
56
const char *name, hwaddr size,
57
const int *irqs)
58
{
59
+ /* The irq[] array is DMACINTR, DMACINTERR, DMACINTTC, in that order */
60
PL080State *dma = opaque;
61
int i = dma - &mms->dma[0];
62
SysBusDevice *s;
63
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_dma(MPS2TZMachineState *mms, void *opaque,
64
65
s = SYS_BUS_DEVICE(dma);
66
/* Wire up DMACINTR, DMACINTERR, DMACINTTC */
67
- sysbus_connect_irq(s, 0, get_sse_irq_in(mms, 58 + i * 3));
68
- sysbus_connect_irq(s, 1, get_sse_irq_in(mms, 56 + i * 3));
69
- sysbus_connect_irq(s, 2, get_sse_irq_in(mms, 57 + i * 3));
70
+ sysbus_connect_irq(s, 0, get_sse_irq_in(mms, irqs[0]));
71
+ sysbus_connect_irq(s, 1, get_sse_irq_in(mms, irqs[1]));
72
+ sysbus_connect_irq(s, 2, get_sse_irq_in(mms, irqs[2]));
73
74
g_free(mscname);
75
return sysbus_mmio_get_region(s, 0);
76
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_spi(MPS2TZMachineState *mms, void *opaque,
77
* lines are set via the "MISC" register in the MPS2 FPGAIO device.
78
*/
79
PL022State *spi = opaque;
80
- int i = spi - &mms->spi[0];
81
SysBusDevice *s;
82
83
object_initialize_child(OBJECT(mms), name, spi, TYPE_PL022);
84
sysbus_realize(SYS_BUS_DEVICE(spi), &error_fatal);
85
s = SYS_BUS_DEVICE(spi);
86
- sysbus_connect_irq(s, 0, get_sse_irq_in(mms, 51 + i));
87
+ sysbus_connect_irq(s, 0, get_sse_irq_in(mms, irqs[0]));
88
return sysbus_mmio_get_region(s, 0);
89
}
90
91
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
92
}, {
93
.name = "apb_ppcexp1",
94
.ports = {
95
- { "spi0", make_spi, &mms->spi[0], 0x40205000, 0x1000 },
96
- { "spi1", make_spi, &mms->spi[1], 0x40206000, 0x1000 },
97
- { "spi2", make_spi, &mms->spi[2], 0x40209000, 0x1000 },
98
- { "spi3", make_spi, &mms->spi[3], 0x4020a000, 0x1000 },
99
- { "spi4", make_spi, &mms->spi[4], 0x4020b000, 0x1000 },
100
- { "uart0", make_uart, &mms->uart[0], 0x40200000, 0x1000 },
101
- { "uart1", make_uart, &mms->uart[1], 0x40201000, 0x1000 },
102
- { "uart2", make_uart, &mms->uart[2], 0x40202000, 0x1000 },
103
- { "uart3", make_uart, &mms->uart[3], 0x40203000, 0x1000 },
104
- { "uart4", make_uart, &mms->uart[4], 0x40204000, 0x1000 },
105
+ { "spi0", make_spi, &mms->spi[0], 0x40205000, 0x1000, { 51 } },
106
+ { "spi1", make_spi, &mms->spi[1], 0x40206000, 0x1000, { 52 } },
107
+ { "spi2", make_spi, &mms->spi[2], 0x40209000, 0x1000, { 53 } },
108
+ { "spi3", make_spi, &mms->spi[3], 0x4020a000, 0x1000, { 54 } },
109
+ { "spi4", make_spi, &mms->spi[4], 0x4020b000, 0x1000, { 55 } },
110
+ { "uart0", make_uart, &mms->uart[0], 0x40200000, 0x1000, { 32, 33, 42 } },
111
+ { "uart1", make_uart, &mms->uart[1], 0x40201000, 0x1000, { 34, 35, 43 } },
112
+ { "uart2", make_uart, &mms->uart[2], 0x40202000, 0x1000, { 36, 37, 44 } },
113
+ { "uart3", make_uart, &mms->uart[3], 0x40203000, 0x1000, { 38, 39, 45 } },
114
+ { "uart4", make_uart, &mms->uart[4], 0x40204000, 0x1000, { 40, 41, 46 } },
115
{ "i2c0", make_i2c, &mms->i2c[0], 0x40207000, 0x1000 },
116
{ "i2c1", make_i2c, &mms->i2c[1], 0x40208000, 0x1000 },
117
{ "i2c2", make_i2c, &mms->i2c[2], 0x4020c000, 0x1000 },
118
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
119
{ "gpio1", make_unimp_dev, &mms->gpio[1], 0x40101000, 0x1000 },
120
{ "gpio2", make_unimp_dev, &mms->gpio[2], 0x40102000, 0x1000 },
121
{ "gpio3", make_unimp_dev, &mms->gpio[3], 0x40103000, 0x1000 },
122
- { "eth", make_eth_dev, NULL, 0x42000000, 0x100000 },
123
+ { "eth", make_eth_dev, NULL, 0x42000000, 0x100000, { 48 } },
124
},
125
}, {
126
.name = "ahb_ppcexp1",
127
.ports = {
128
- { "dma0", make_dma, &mms->dma[0], 0x40110000, 0x1000 },
129
- { "dma1", make_dma, &mms->dma[1], 0x40111000, 0x1000 },
130
- { "dma2", make_dma, &mms->dma[2], 0x40112000, 0x1000 },
131
- { "dma3", make_dma, &mms->dma[3], 0x40113000, 0x1000 },
132
+ { "dma0", make_dma, &mms->dma[0], 0x40110000, 0x1000, { 58, 56, 57 } },
133
+ { "dma1", make_dma, &mms->dma[1], 0x40111000, 0x1000, { 61, 59, 60 } },
134
+ { "dma2", make_dma, &mms->dma[2], 0x40112000, 0x1000, { 64, 62, 63 } },
135
+ { "dma3", make_dma, &mms->dma[3], 0x40113000, 0x1000, { 67, 65, 66 } },
136
},
137
},
138
};
139
--
29
--
140
2.20.1
30
2.20.1
141
31
142
32
diff view generated by jsdifflib
1
The mps2-tz code uses PPCPortInfo data structures to define what
1
We don't currently zero-initialize the 'struct sockaddr_in' that
2
devices are present and how they are wired up. Currently we use
2
parse_host_port() fills in, so any fields we don't explicitly
3
these to specify device types and addresses, but hard-code the
3
initialize might be left as random garbage. POSIX states that
4
interrupt line wiring in each make_* helper function. This works for
4
implementations may define extensions in sockaddr_in, and that those
5
the two boards we have at the moment, but the AN524 has some devices
5
extensions must not trigger if zero-initialized. So not zero
6
with different interrupt assignments.
6
initializing might result in inadvertently triggering an impdef
7
extension.
7
8
8
This commit adds the framework to allow PPCPortInfo structures to
9
memset() the sockaddr_in before we start to fill it in.
9
specify interrupt numbers. We add an array of interrupt numbers to
10
the PPCPortInfo struct, and pass it through to the make_* helpers.
11
The following commit will change the make_* helpers over to using the
12
framework.
13
10
11
Fixes: Coverity CID 1005338
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
13
Reviewed-by: Eric Blake <eblake@redhat.com>
16
Message-id: 20210215115138.20465-13-peter.maydell@linaro.org
14
Message-id: 20210813150506.7768-2-peter.maydell@linaro.org
17
---
15
---
18
hw/arm/mps2-tz.c | 36 ++++++++++++++++++++++++------------
16
net/net.c | 2 ++
19
1 file changed, 24 insertions(+), 12 deletions(-)
17
1 file changed, 2 insertions(+)
20
18
21
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
19
diff --git a/net/net.c b/net/net.c
22
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
23
--- a/hw/arm/mps2-tz.c
21
--- a/net/net.c
24
+++ b/hw/arm/mps2-tz.c
22
+++ b/net/net.c
25
@@ -XXX,XX +XXX,XX @@ static qemu_irq get_sse_irq_in(MPS2TZMachineState *mms, int irqno)
23
@@ -XXX,XX +XXX,XX @@ int parse_host_port(struct sockaddr_in *saddr, const char *str,
26
* needs to be plugged into the downstream end of the PPC port.
24
const char *addr, *p, *r;
27
*/
25
int port, ret = 0;
28
typedef MemoryRegion *MakeDevFn(MPS2TZMachineState *mms, void *opaque,
26
29
- const char *name, hwaddr size);
27
+ memset(saddr, 0, sizeof(*saddr));
30
+ const char *name, hwaddr size,
28
+
31
+ const int *irqs);
29
substrings = g_strsplit(str, ":", 2);
32
30
if (!substrings || !substrings[0] || !substrings[1]) {
33
typedef struct PPCPortInfo {
31
error_setg(errp, "host address '%s' doesn't contain ':' "
34
const char *name;
35
@@ -XXX,XX +XXX,XX @@ typedef struct PPCPortInfo {
36
void *opaque;
37
hwaddr addr;
38
hwaddr size;
39
+ int irqs[3]; /* currently no device needs more IRQ lines than this */
40
} PPCPortInfo;
41
42
typedef struct PPCInfo {
43
@@ -XXX,XX +XXX,XX @@ typedef struct PPCInfo {
44
} PPCInfo;
45
46
static MemoryRegion *make_unimp_dev(MPS2TZMachineState *mms,
47
- void *opaque,
48
- const char *name, hwaddr size)
49
+ void *opaque,
50
+ const char *name, hwaddr size,
51
+ const int *irqs)
52
{
53
/* Initialize, configure and realize a TYPE_UNIMPLEMENTED_DEVICE,
54
* and return a pointer to its MemoryRegion.
55
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_unimp_dev(MPS2TZMachineState *mms,
56
}
57
58
static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
59
- const char *name, hwaddr size)
60
+ const char *name, hwaddr size,
61
+ const int *irqs)
62
{
63
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
64
CMSDKAPBUART *uart = opaque;
65
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
66
}
67
68
static MemoryRegion *make_scc(MPS2TZMachineState *mms, void *opaque,
69
- const char *name, hwaddr size)
70
+ const char *name, hwaddr size,
71
+ const int *irqs)
72
{
73
MPS2SCC *scc = opaque;
74
DeviceState *sccdev;
75
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_scc(MPS2TZMachineState *mms, void *opaque,
76
}
77
78
static MemoryRegion *make_fpgaio(MPS2TZMachineState *mms, void *opaque,
79
- const char *name, hwaddr size)
80
+ const char *name, hwaddr size,
81
+ const int *irqs)
82
{
83
MPS2FPGAIO *fpgaio = opaque;
84
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
85
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_fpgaio(MPS2TZMachineState *mms, void *opaque,
86
}
87
88
static MemoryRegion *make_eth_dev(MPS2TZMachineState *mms, void *opaque,
89
- const char *name, hwaddr size)
90
+ const char *name, hwaddr size,
91
+ const int *irqs)
92
{
93
SysBusDevice *s;
94
NICInfo *nd = &nd_table[0];
95
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_eth_dev(MPS2TZMachineState *mms, void *opaque,
96
}
97
98
static MemoryRegion *make_mpc(MPS2TZMachineState *mms, void *opaque,
99
- const char *name, hwaddr size)
100
+ const char *name, hwaddr size,
101
+ const int *irqs)
102
{
103
TZMPC *mpc = opaque;
104
int i = mpc - &mms->ssram_mpc[0];
105
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_mpc(MPS2TZMachineState *mms, void *opaque,
106
}
107
108
static MemoryRegion *make_dma(MPS2TZMachineState *mms, void *opaque,
109
- const char *name, hwaddr size)
110
+ const char *name, hwaddr size,
111
+ const int *irqs)
112
{
113
PL080State *dma = opaque;
114
int i = dma - &mms->dma[0];
115
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_dma(MPS2TZMachineState *mms, void *opaque,
116
}
117
118
static MemoryRegion *make_spi(MPS2TZMachineState *mms, void *opaque,
119
- const char *name, hwaddr size)
120
+ const char *name, hwaddr size,
121
+ const int *irqs)
122
{
123
/*
124
* The AN505 has five PL022 SPI controllers.
125
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_spi(MPS2TZMachineState *mms, void *opaque,
126
}
127
128
static MemoryRegion *make_i2c(MPS2TZMachineState *mms, void *opaque,
129
- const char *name, hwaddr size)
130
+ const char *name, hwaddr size,
131
+ const int *irqs)
132
{
133
ArmSbconI2CState *i2c = opaque;
134
SysBusDevice *s;
135
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
136
continue;
137
}
138
139
- mr = pinfo->devfn(mms, pinfo->opaque, pinfo->name, pinfo->size);
140
+ mr = pinfo->devfn(mms, pinfo->opaque, pinfo->name, pinfo->size,
141
+ pinfo->irqs);
142
portname = g_strdup_printf("port[%d]", port);
143
object_property_set_link(OBJECT(ppc), portname, OBJECT(mr),
144
&error_fatal);
145
--
32
--
146
2.20.1
33
2.20.1
147
34
148
35
diff view generated by jsdifflib
1
The AN524 has more interrupt lines than the AN505 and AN521; make
1
Zero-initialize sockaddr_in and sockaddr_un structs that we're about
2
numirq board-specific rather than a compile-time constant.
2
to fill in and pass to bind() or connect(), to ensure we don't leave
3
3
possible implementation-defined extension fields as uninitialized
4
Since the difference is small (92 on the current boards and 95 on the
4
garbage.
5
new one) we don't dynamically allocate the cpu_irq_splitter[] array
6
but leave it as a fixed length array whose size is the maximum needed
7
for any of the boards.
8
5
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Eric Blake <eblake@redhat.com>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210813150506.7768-3-peter.maydell@linaro.org
12
Message-id: 20210215115138.20465-10-peter.maydell@linaro.org
13
---
9
---
14
hw/arm/mps2-tz.c | 15 ++++++++++-----
10
gdbstub.c | 4 ++--
15
1 file changed, 10 insertions(+), 5 deletions(-)
11
1 file changed, 2 insertions(+), 2 deletions(-)
16
12
17
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
13
diff --git a/gdbstub.c b/gdbstub.c
18
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/arm/mps2-tz.c
15
--- a/gdbstub.c
20
+++ b/hw/arm/mps2-tz.c
16
+++ b/gdbstub.c
21
@@ -XXX,XX +XXX,XX @@
17
@@ -XXX,XX +XXX,XX @@ static bool gdb_accept_socket(int gdb_fd)
22
#include "hw/qdev-clock.h"
18
23
#include "qom/object.h"
19
static int gdbserver_open_socket(const char *path)
24
25
-#define MPS2TZ_NUMIRQ 92
26
+#define MPS2TZ_NUMIRQ_MAX 92
27
28
typedef enum MPS2TZFPGAType {
29
FPGA_AN505,
30
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineClass {
31
const uint32_t *oscclk;
32
uint32_t fpgaio_num_leds; /* Number of LEDs in FPGAIO LED0 register */
33
bool fpgaio_has_switches; /* Does FPGAIO have SWITCH register? */
34
+ int numirq; /* Number of external interrupts */
35
const char *armsse_type;
36
};
37
38
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
39
SplitIRQ sec_resp_splitter;
40
qemu_or_irq uart_irq_orgate;
41
DeviceState *lan9118;
42
- SplitIRQ cpu_irq_splitter[MPS2TZ_NUMIRQ];
43
+ SplitIRQ cpu_irq_splitter[MPS2TZ_NUMIRQ_MAX];
44
Clock *sysclk;
45
Clock *s32kclk;
46
};
47
@@ -XXX,XX +XXX,XX @@ static qemu_irq get_sse_irq_in(MPS2TZMachineState *mms, int irqno)
48
{
20
{
49
/* Return a qemu_irq which will signal IRQ n to all CPUs in the SSE. */
21
- struct sockaddr_un sockaddr;
50
MachineClass *mc = MACHINE_GET_CLASS(mms);
22
+ struct sockaddr_un sockaddr = {};
51
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
23
int fd, ret;
52
24
53
- assert(irqno < MPS2TZ_NUMIRQ);
25
fd = socket(AF_UNIX, SOCK_STREAM, 0);
54
+ assert(irqno < mmc->numirq);
26
@@ -XXX,XX +XXX,XX @@ static int gdbserver_open_socket(const char *path)
55
27
56
if (mc->max_cpus > 1) {
28
static bool gdb_accept_tcp(int gdb_fd)
57
return qdev_get_gpio_in(DEVICE(&mms->cpu_irq_splitter[irqno]), 0);
29
{
58
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
30
- struct sockaddr_in sockaddr;
59
iotkitdev = DEVICE(&mms->iotkit);
31
+ struct sockaddr_in sockaddr = {};
60
object_property_set_link(OBJECT(&mms->iotkit), "memory",
32
socklen_t len;
61
OBJECT(system_memory), &error_abort);
33
int fd;
62
- qdev_prop_set_uint32(iotkitdev, "EXP_NUMIRQ", MPS2TZ_NUMIRQ);
63
+ qdev_prop_set_uint32(iotkitdev, "EXP_NUMIRQ", mmc->numirq);
64
qdev_connect_clock_in(iotkitdev, "MAINCLK", mms->sysclk);
65
qdev_connect_clock_in(iotkitdev, "S32KCLK", mms->s32kclk);
66
sysbus_realize(SYS_BUS_DEVICE(&mms->iotkit), &error_fatal);
67
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
68
* board. If there is only one CPU, we can just wire the device IRQ
69
* directly to the SSE's IRQ input.
70
*/
71
+ assert(mmc->numirq <= MPS2TZ_NUMIRQ_MAX);
72
if (mc->max_cpus > 1) {
73
- for (i = 0; i < MPS2TZ_NUMIRQ; i++) {
74
+ for (i = 0; i < mmc->numirq; i++) {
75
char *name = g_strdup_printf("mps2-irq-splitter%d", i);
76
SplitIRQ *splitter = &mms->cpu_irq_splitter[i];
77
78
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
79
mmc->len_oscclk = ARRAY_SIZE(an505_oscclk);
80
mmc->fpgaio_num_leds = 2;
81
mmc->fpgaio_has_switches = false;
82
+ mmc->numirq = 92;
83
mmc->armsse_type = TYPE_IOTKIT;
84
}
85
86
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
87
mmc->len_oscclk = ARRAY_SIZE(an505_oscclk);
88
mmc->fpgaio_num_leds = 2;
89
mmc->fpgaio_has_switches = false;
90
+ mmc->numirq = 92;
91
mmc->armsse_type = TYPE_SSE200;
92
}
93
34
94
--
35
--
95
2.20.1
36
2.20.1
96
37
97
38
diff view generated by jsdifflib
1
On the MPS2 boards, the first 32 interrupt lines are entirely
1
Zero-initialize the sockaddr_in struct that we're about to fill in
2
internal to the SSE; interrupt lines for devices outside the SSE
2
and pass to bind(), to ensure we don't leave possible
3
start at 32. In the application notes that document each FPGA image,
3
implementation-defined extension fields as uninitialized garbage.
4
the interrupt wiring is documented from the point of view of the CPU,
5
so '0' is the first of the SSE's interrupts and the devices in the
6
FPGA image itself are '32' and up: so the UART 0 Receive interrupt is
7
32, the SPI #0 interrupt is 51, and so on.
8
9
Within our implementation, because the external interrupts must be
10
connected to the EXP_IRQ[0...n] lines of the SSE object, we made the
11
get_sse_irq_in() function take an irqno whose values start at 0 for
12
the first FPGA device interrupt. In this numbering scheme the UART 0
13
Receive interrupt is 0, the SPI #0 interrupt is 19, and so on.
14
15
The result of these two different numbering schemes has been that
16
half of the devices were wired up to the wrong IRQs: the UART IRQs
17
are wired up correctly, but the DMA and SPI devices were passing
18
start-at-32 values to get_sse_irq_in() and so being mis-connected.
19
20
Fix the bug by making get_sse_irq_in() take values specified with the
21
same scheme that the hardware manuals use, to avoid confusion.
22
4
23
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
24
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Eric Blake <eblake@redhat.com>
25
Message-id: 20210215115138.20465-12-peter.maydell@linaro.org
7
Reviewed-by: Corey Minyard <cminyard@mvista.com>
8
Acked-by: Thomas Huth <thuth@redhat.com>
9
Message-id: 20210813150506.7768-4-peter.maydell@linaro.org
26
---
10
---
27
hw/arm/mps2-tz.c | 24 +++++++++++++++++-------
11
tests/qtest/ipmi-bt-test.c | 2 +-
28
1 file changed, 17 insertions(+), 7 deletions(-)
12
1 file changed, 1 insertion(+), 1 deletion(-)
29
13
30
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
14
diff --git a/tests/qtest/ipmi-bt-test.c b/tests/qtest/ipmi-bt-test.c
31
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
32
--- a/hw/arm/mps2-tz.c
16
--- a/tests/qtest/ipmi-bt-test.c
33
+++ b/hw/arm/mps2-tz.c
17
+++ b/tests/qtest/ipmi-bt-test.c
34
@@ -XXX,XX +XXX,XX @@ static void make_ram_alias(MemoryRegion *mr, const char *name,
18
@@ -XXX,XX +XXX,XX @@ static void test_enable_irq(void)
35
19
*/
36
static qemu_irq get_sse_irq_in(MPS2TZMachineState *mms, int irqno)
20
static void open_socket(void)
37
{
21
{
38
- /* Return a qemu_irq which will signal IRQ n to all CPUs in the SSE. */
22
- struct sockaddr_in myaddr;
39
+ /*
23
+ struct sockaddr_in myaddr = {};
40
+ * Return a qemu_irq which will signal IRQ n to all CPUs in the
24
socklen_t addrlen;
41
+ * SSE. The irqno should be as the CPU sees it, so the first
25
42
+ * external-to-the-SSE interrupt is 32.
26
myaddr.sin_family = AF_INET;
43
+ */
44
MachineClass *mc = MACHINE_GET_CLASS(mms);
45
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
46
47
- assert(irqno < mmc->numirq);
48
+ assert(irqno >= 32 && irqno < (mmc->numirq + 32));
49
+
50
+ /*
51
+ * Convert from "CPU irq number" (as listed in the FPGA image
52
+ * documentation) to the SSE external-interrupt number.
53
+ */
54
+ irqno -= 32;
55
56
if (mc->max_cpus > 1) {
57
return qdev_get_gpio_in(DEVICE(&mms->cpu_irq_splitter[irqno]), 0);
58
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
59
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
60
CMSDKAPBUART *uart = opaque;
61
int i = uart - &mms->uart[0];
62
- int rxirqno = i * 2;
63
- int txirqno = i * 2 + 1;
64
- int combirqno = i + 10;
65
+ int rxirqno = i * 2 + 32;
66
+ int txirqno = i * 2 + 33;
67
+ int combirqno = i + 42;
68
SysBusDevice *s;
69
DeviceState *orgate_dev = DEVICE(&mms->uart_irq_orgate);
70
71
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_eth_dev(MPS2TZMachineState *mms, void *opaque,
72
73
s = SYS_BUS_DEVICE(mms->lan9118);
74
sysbus_realize_and_unref(s, &error_fatal);
75
- sysbus_connect_irq(s, 0, get_sse_irq_in(mms, 16));
76
+ sysbus_connect_irq(s, 0, get_sse_irq_in(mms, 48));
77
return sysbus_mmio_get_region(s, 0);
78
}
79
80
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
81
&error_fatal);
82
qdev_realize(DEVICE(&mms->uart_irq_orgate), NULL, &error_fatal);
83
qdev_connect_gpio_out(DEVICE(&mms->uart_irq_orgate), 0,
84
- get_sse_irq_in(mms, 15));
85
+ get_sse_irq_in(mms, 47));
86
87
/* Most of the devices in the FPGA are behind Peripheral Protection
88
* Controllers. The required order for initializing things is:
89
--
27
--
90
2.20.1
28
2.20.1
91
29
92
30
diff view generated by jsdifflib
1
We were previously using the default OSCCLK settings, which are
1
Zero-initialize sockaddr_in and sockaddr_un structs that we're about
2
correct for the older MPS2 boards (mps2-an385, mps2-an386,
2
to fill in and pass to bind() or connect(), to ensure we don't leave
3
mps2-an500, mps2-an511), but wrong for the mps2-an505 and mps2-511
3
possible implementation-defined extension fields as uninitialized
4
implemented in mps2-tz.c. Now we're setting the values explicitly we
4
garbage.
5
can fix them to be correct.
6
5
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Eric Blake <eblake@redhat.com>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210813150506.7768-5-peter.maydell@linaro.org
10
Message-id: 20210215115138.20465-4-peter.maydell@linaro.org
11
---
9
---
12
hw/arm/mps2-tz.c | 4 ++--
10
tests/tcg/multiarch/linux-test.c | 4 ++--
13
1 file changed, 2 insertions(+), 2 deletions(-)
11
1 file changed, 2 insertions(+), 2 deletions(-)
14
12
15
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
13
diff --git a/tests/tcg/multiarch/linux-test.c b/tests/tcg/multiarch/linux-test.c
16
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/mps2-tz.c
15
--- a/tests/tcg/multiarch/linux-test.c
18
+++ b/hw/arm/mps2-tz.c
16
+++ b/tests/tcg/multiarch/linux-test.c
19
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_scc(MPS2TZMachineState *mms, void *opaque,
17
@@ -XXX,XX +XXX,XX @@ static void test_time(void)
20
qdev_prop_set_uint32(sccdev, "scc-id", mmc->scc_id);
18
static int server_socket(void)
21
/* This will need to be per-FPGA image eventually */
19
{
22
qdev_prop_set_uint32(sccdev, "len-oscclk", 3);
20
int val, fd;
23
- qdev_prop_set_uint32(sccdev, "oscclk[0]", 50000000);
21
- struct sockaddr_in sockaddr;
24
- qdev_prop_set_uint32(sccdev, "oscclk[1]", 24576000);
22
+ struct sockaddr_in sockaddr = {};
25
+ qdev_prop_set_uint32(sccdev, "oscclk[0]", 40000000);
23
26
+ qdev_prop_set_uint32(sccdev, "oscclk[1]", 24580000);
24
/* server socket */
27
qdev_prop_set_uint32(sccdev, "oscclk[2]", 25000000);
25
fd = chk_error(socket(PF_INET, SOCK_STREAM, 0));
28
sysbus_realize(SYS_BUS_DEVICE(scc), &error_fatal);
26
@@ -XXX,XX +XXX,XX @@ static int server_socket(void)
29
return sysbus_mmio_get_region(SYS_BUS_DEVICE(sccdev), 0);
27
static int client_socket(uint16_t port)
28
{
29
int fd;
30
- struct sockaddr_in sockaddr;
31
+ struct sockaddr_in sockaddr = {};
32
33
/* server socket */
34
fd = chk_error(socket(PF_INET, SOCK_STREAM, 0));
30
--
35
--
31
2.20.1
36
2.20.1
32
37
33
38
diff view generated by jsdifflib
1
The macro draw_line_func is used only once; just expand it.
1
The SoC realize can fail for legitimate reasons, because it propagates
2
errors up from CPU realize, which in turn can be provoked by user
3
error in setting commandline options. Use error_fatal so we report
4
the error message to the user and exit, rather than asserting
5
via error_abort.
2
6
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Message-id: 20210215103215.4944-10-peter.maydell@linaro.org
10
Message-id: 20210816135842.25302-2-peter.maydell@linaro.org
7
---
11
---
8
hw/display/omap_lcdc.c | 4 +---
12
hw/arm/raspi.c | 2 +-
9
1 file changed, 1 insertion(+), 3 deletions(-)
13
1 file changed, 1 insertion(+), 1 deletion(-)
10
14
11
diff --git a/hw/display/omap_lcdc.c b/hw/display/omap_lcdc.c
15
diff --git a/hw/arm/raspi.c b/hw/arm/raspi.c
12
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
13
--- a/hw/display/omap_lcdc.c
17
--- a/hw/arm/raspi.c
14
+++ b/hw/display/omap_lcdc.c
18
+++ b/hw/arm/raspi.c
15
@@ -XXX,XX +XXX,XX @@ static void omap_lcd_interrupts(struct omap_lcd_panel_s *s)
19
@@ -XXX,XX +XXX,XX @@ static void raspi_machine_init(MachineState *machine)
16
qemu_irq_lower(s->irq);
20
object_property_add_const_link(OBJECT(&s->soc), "ram", OBJECT(machine->ram));
17
}
21
object_property_set_int(OBJECT(&s->soc), "board-rev", board_rev,
18
22
&error_abort);
19
-#define draw_line_func drawfn
23
- qdev_realize(DEVICE(&s->soc), NULL, &error_abort);
20
-
24
+ qdev_realize(DEVICE(&s->soc), NULL, &error_fatal);
21
/*
25
22
* 2-bit colour
26
/* Create and plug in the SD cards */
23
*/
27
di = drive_get_next(IF_SD);
24
@@ -XXX,XX +XXX,XX @@ static void omap_update_display(void *opaque)
25
{
26
struct omap_lcd_panel_s *omap_lcd = (struct omap_lcd_panel_s *) opaque;
27
DisplaySurface *surface;
28
- draw_line_func draw_line;
29
+ drawfn draw_line;
30
int size, height, first, last;
31
int width, linesize, step, bpp, frame_offset;
32
hwaddr frame_base;
33
--
28
--
34
2.20.1
29
2.20.1
35
30
36
31
diff view generated by jsdifflib
1
From: Rebecca Cran <rebecca@nuviainc.com>
1
KVM cannot support multiple address spaces per CPU; if you try to
2
create more than one then cpu_address_space_init() will assert.
2
3
3
Enable FEAT_SSBS for the "max" 32-bit CPU.
4
In the Arm CPU realize function, detect the configurations which
5
would cause us to need more than one AS, and cleanly fail the
6
realize rather than blundering on into the assertion. This
7
turns this:
8
$ qemu-system-aarch64 -enable-kvm -display none -cpu max -machine raspi3b
9
qemu-system-aarch64: ../../softmmu/physmem.c:747: cpu_address_space_init: Assertion `asidx == 0 || !kvm_enabled()' failed.
10
Aborted
4
11
5
Signed-off-by: Rebecca Cran <rebecca@nuviainc.com>
12
into:
13
$ qemu-system-aarch64 -enable-kvm -display none -machine raspi3b
14
qemu-system-aarch64: Cannot enable KVM when guest CPU has EL3 enabled
15
16
and this:
17
$ qemu-system-aarch64 -enable-kvm -display none -machine mps3-an524
18
qemu-system-aarch64: ../../softmmu/physmem.c:747: cpu_address_space_init: Assertion `asidx == 0 || !kvm_enabled()' failed.
19
Aborted
20
21
into:
22
$ qemu-system-aarch64 -enable-kvm -display none -machine mps3-an524
23
qemu-system-aarch64: Cannot enable KVM when using an M-profile guest CPU
24
25
Fixes: https://gitlab.com/qemu-project/qemu/-/issues/528
26
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
27
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20210216224543.16142-4-rebecca@nuviainc.com
28
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
[PMM: fix typo causing compilation failure]
29
Message-id: 20210816135842.25302-3-peter.maydell@linaro.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
30
---
11
target/arm/cpu.c | 4 ++++
31
target/arm/cpu.c | 23 +++++++++++++++++++++++
12
1 file changed, 4 insertions(+)
32
1 file changed, 23 insertions(+)
13
33
14
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
34
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
15
index XXXXXXX..XXXXXXX 100644
35
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/cpu.c
36
--- a/target/arm/cpu.c
17
+++ b/target/arm/cpu.c
37
+++ b/target/arm/cpu.c
18
@@ -XXX,XX +XXX,XX @@ static void arm_max_initfn(Object *obj)
38
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
19
t = cpu->isar.id_pfr0;
39
}
20
t = FIELD_DP32(t, ID_PFR0, DIT, 1);
40
}
21
cpu->isar.id_pfr0 = t;
41
42
+ if (kvm_enabled()) {
43
+ /*
44
+ * Catch all the cases which might cause us to create more than one
45
+ * address space for the CPU (otherwise we will assert() later in
46
+ * cpu_address_space_init()).
47
+ */
48
+ if (arm_feature(env, ARM_FEATURE_M)) {
49
+ error_setg(errp,
50
+ "Cannot enable KVM when using an M-profile guest CPU");
51
+ return;
52
+ }
53
+ if (cpu->has_el3) {
54
+ error_setg(errp,
55
+ "Cannot enable KVM when guest CPU has EL3 enabled");
56
+ return;
57
+ }
58
+ if (cpu->tag_memory) {
59
+ error_setg(errp,
60
+ "Cannot enable KVM when guest CPUs has MTE enabled");
61
+ return;
62
+ }
63
+ }
22
+
64
+
23
+ t = cpu->isar.id_pfr2;
65
{
24
+ t = FIELD_DP32(t, ID_PFR2, SSBS, 1);
66
uint64_t scale;
25
+ cpu->isar.id_pfr2 = t;
67
26
}
27
#endif
28
}
29
--
68
--
30
2.20.1
69
2.20.1
31
70
32
71
diff view generated by jsdifflib
1
Fix some minor coding style issues in the template header,
1
Now that the CPU realize function will fail cleanly if we ask for EL3
2
so checkpatch doesn't complain when we move the code.
2
when KVM is enabled, we don't need to check for errors explicitly in
3
the virt board code. The reported message is slightly different;
4
it is now:
5
qemu-system-aarch64: Cannot enable KVM when guest CPU has EL3 enabled
6
instead of:
7
qemu-system-aarch64: mach-virt: KVM does not support Security extensions
8
9
We don't delete the MTE check because there the logic is more
10
complex; deleting the check would work but makes the error message
11
less helpful, as it would read:
12
qemu-system-aarch64: MTE requested, but not supported by the guest CPU
13
instead of:
14
qemu-system-aarch64: mach-virt: KVM does not support providing MTE to the guest CPU
3
15
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
16
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
17
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
18
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Message-id: 20210215103215.4944-8-peter.maydell@linaro.org
19
Message-id: 20210816135842.25302-4-peter.maydell@linaro.org
8
---
20
---
9
hw/display/omap_lcd_template.h | 6 +++---
21
hw/arm/virt.c | 5 -----
10
1 file changed, 3 insertions(+), 3 deletions(-)
22
1 file changed, 5 deletions(-)
11
23
12
diff --git a/hw/display/omap_lcd_template.h b/hw/display/omap_lcd_template.h
24
diff --git a/hw/arm/virt.c b/hw/arm/virt.c
13
index XXXXXXX..XXXXXXX 100644
25
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/display/omap_lcd_template.h
26
--- a/hw/arm/virt.c
15
+++ b/hw/display/omap_lcd_template.h
27
+++ b/hw/arm/virt.c
16
@@ -XXX,XX +XXX,XX @@ static void draw_line2_32(void *opaque, uint8_t *d, const uint8_t *s,
28
@@ -XXX,XX +XXX,XX @@ static void machvirt_init(MachineState *machine)
17
b = (pal[v & 3] << 4) & 0xf0;
29
}
18
((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
30
19
d += 4;
31
if (vms->secure) {
20
- s ++;
32
- if (kvm_enabled()) {
21
+ s++;
33
- error_report("mach-virt: KVM does not support Security extensions");
22
width -= 4;
34
- exit(1);
23
} while (width > 0);
35
- }
24
}
36
-
25
@@ -XXX,XX +XXX,XX @@ static void draw_line4_32(void *opaque, uint8_t *d, const uint8_t *s,
37
/*
26
b = (pal[v & 0xf] << 4) & 0xf0;
38
* The Secure view of the world is the same as the NonSecure,
27
((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
39
* but with a few extra devices. Create it as a container region
28
d += 4;
29
- s ++;
30
+ s++;
31
width -= 2;
32
} while (width > 0);
33
}
34
@@ -XXX,XX +XXX,XX @@ static void draw_line8_32(void *opaque, uint8_t *d, const uint8_t *s,
35
g = pal[v] & 0xf0;
36
b = (pal[v] << 4) & 0xf0;
37
((uint32_t *) d)[0] = rgb_to_pixel32(r, g, b);
38
- s ++;
39
+ s++;
40
d += 4;
41
} while (-- width != 0);
42
}
43
--
40
--
44
2.20.1
41
2.20.1
45
42
46
43
diff view generated by jsdifflib
1
Add support for the mps3-an524 board; this is an SSE-200 based FPGA
1
In v7, the HSTR register has a TTEE bit which allows EL0/EL1 accesses
2
image, like the existing mps2-an521. It has a usefully larger amount
2
to the Thumb2EE TEECR and TEEHBR registers to be trapped to the
3
of RAM, and a PL031 RTC, as well as some more minor differences.
3
hypervisor. Implement these traps.
4
5
In real hardware this image runs on a newer generation of the FPGA
6
board, the MPS3 rather than the older MPS2. Architecturally the two
7
boards are similar, so we implement the MPS3 boards in the mps2-tz.c
8
file as variations of the existing MPS2 boards.
9
4
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
12
Message-id: 20210215115138.20465-21-peter.maydell@linaro.org
7
Message-id: 20210816180305.20137-2-peter.maydell@linaro.org
13
---
8
---
14
hw/arm/mps2-tz.c | 139 +++++++++++++++++++++++++++++++++++++++++++++--
9
target/arm/cpu.h | 2 ++
15
1 file changed, 135 insertions(+), 4 deletions(-)
10
target/arm/helper.c | 18 ++++++++++++++++--
11
2 files changed, 18 insertions(+), 2 deletions(-)
16
12
17
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
13
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
18
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/arm/mps2-tz.c
15
--- a/target/arm/cpu.h
20
+++ b/hw/arm/mps2-tz.c
16
+++ b/target/arm/cpu.h
21
@@ -XXX,XX +XXX,XX @@
17
@@ -XXX,XX +XXX,XX @@ static inline void xpsr_write(CPUARMState *env, uint32_t val, uint32_t mask)
22
* This source file covers the following FPGA images, for TrustZone cores:
18
#define SCR_ENSCXT (1U << 25)
23
* "mps2-an505" -- Cortex-M33 as documented in ARM Application Note AN505
19
#define SCR_ATA (1U << 26)
24
* "mps2-an521" -- Dual Cortex-M33 as documented in Application Note AN521
20
25
+ * "mps2-an524" -- Dual Cortex-M33 as documented in Application Note AN524
21
+#define HSTR_TTEE (1 << 16)
26
*
27
* Links to the TRM for the board itself and to the various Application
28
* Notes which document the FPGA images can be found here:
29
@@ -XXX,XX +XXX,XX @@
30
* http://infocenter.arm.com/help/topic/com.arm.doc.dai0505b/index.html
31
* Application Note AN521:
32
* http://infocenter.arm.com/help/topic/com.arm.doc.dai0521c/index.html
33
+ * Application Note AN524:
34
+ * https://developer.arm.com/documentation/dai0524/latest/
35
*
36
* The AN505 defers to the Cortex-M33 processor ARMv8M IoT Kit FVP User Guide
37
* (ARM ECM0601256) for the details of some of the device layout:
38
* http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
39
- * Similarly, the AN521 uses the SSE-200, and the SSE-200 TRM defines
40
+ * Similarly, the AN521 and AN524 use the SSE-200, and the SSE-200 TRM defines
41
* most of the device layout:
42
* http://infocenter.arm.com/help/topic/com.arm.doc.101104_0100_00_en/corelink_sse200_subsystem_for_embedded_technical_reference_manual_101104_0100_00_en.pdf
43
*
44
@@ -XXX,XX +XXX,XX @@
45
#include "hw/qdev-clock.h"
46
#include "qom/object.h"
47
48
-#define MPS2TZ_NUMIRQ_MAX 92
49
+#define MPS2TZ_NUMIRQ_MAX 95
50
#define MPS2TZ_RAM_MAX 4
51
52
typedef enum MPS2TZFPGAType {
53
FPGA_AN505,
54
FPGA_AN521,
55
+ FPGA_AN524,
56
} MPS2TZFPGAType;
57
58
/*
59
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
60
TZPPC ppc[5];
61
TZMPC mpc[3];
62
PL022State spi[5];
63
- ArmSbconI2CState i2c[4];
64
+ ArmSbconI2CState i2c[5];
65
UnimplementedDeviceState i2s_audio;
66
UnimplementedDeviceState gpio[4];
67
UnimplementedDeviceState gfx;
68
+ UnimplementedDeviceState cldc;
69
+ UnimplementedDeviceState rtc;
70
PL080State dma[4];
71
TZMSC msc[4];
72
- CMSDKAPBUART uart[5];
73
+ CMSDKAPBUART uart[6];
74
SplitIRQ sec_resp_splitter;
75
qemu_or_irq uart_irq_orgate;
76
DeviceState *lan9118;
77
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
78
#define TYPE_MPS2TZ_MACHINE "mps2tz"
79
#define TYPE_MPS2TZ_AN505_MACHINE MACHINE_TYPE_NAME("mps2-an505")
80
#define TYPE_MPS2TZ_AN521_MACHINE MACHINE_TYPE_NAME("mps2-an521")
81
+#define TYPE_MPS3TZ_AN524_MACHINE MACHINE_TYPE_NAME("mps3-an524")
82
83
OBJECT_DECLARE_TYPE(MPS2TZMachineState, MPS2TZMachineClass, MPS2TZ_MACHINE)
84
85
@@ -XXX,XX +XXX,XX @@ static const uint32_t an505_oscclk[] = {
86
25000000,
87
};
88
89
+static const uint32_t an524_oscclk[] = {
90
+ 24000000,
91
+ 32000000,
92
+ 50000000,
93
+ 50000000,
94
+ 24576000,
95
+ 23750000,
96
+};
97
+
22
+
98
static const RAMInfo an505_raminfo[] = { {
23
/* Return the current FPSCR value. */
99
.name = "ssram-0",
24
uint32_t vfp_get_fpscr(CPUARMState *env);
100
.base = 0x00000000,
25
void vfp_set_fpscr(CPUARMState *env, uint32_t val);
101
@@ -XXX,XX +XXX,XX @@ static const RAMInfo an505_raminfo[] = { {
26
diff --git a/target/arm/helper.c b/target/arm/helper.c
102
},
27
index XXXXXXX..XXXXXXX 100644
103
};
28
--- a/target/arm/helper.c
104
29
+++ b/target/arm/helper.c
105
+static const RAMInfo an524_raminfo[] = { {
30
@@ -XXX,XX +XXX,XX @@ static void teecr_write(CPUARMState *env, const ARMCPRegInfo *ri,
106
+ .name = "bram",
31
env->teecr = value;
107
+ .base = 0x00000000,
108
+ .size = 512 * KiB,
109
+ .mpc = 0,
110
+ .mrindex = 0,
111
+ }, {
112
+ .name = "sram",
113
+ .base = 0x20000000,
114
+ .size = 32 * 4 * KiB,
115
+ .mpc = 1,
116
+ .mrindex = 1,
117
+ }, {
118
+ /* We don't model QSPI flash yet; for now expose it as simple ROM */
119
+ .name = "QSPI",
120
+ .base = 0x28000000,
121
+ .size = 8 * MiB,
122
+ .mpc = 1,
123
+ .mrindex = 2,
124
+ .flags = IS_ROM,
125
+ }, {
126
+ .name = "DDR",
127
+ .base = 0x60000000,
128
+ .size = 2 * GiB,
129
+ .mpc = 2,
130
+ .mrindex = -1,
131
+ }, {
132
+ .name = NULL,
133
+ },
134
+};
135
+
136
static const RAMInfo *find_raminfo_for_mpc(MPS2TZMachineState *mms, int mpc)
137
{
138
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
139
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
140
},
141
};
142
143
+ const PPCInfo an524_ppcs[] = { {
144
+ .name = "apb_ppcexp0",
145
+ .ports = {
146
+ { "bram-mpc", make_mpc, &mms->mpc[0], 0x58007000, 0x1000 },
147
+ { "qspi-mpc", make_mpc, &mms->mpc[1], 0x58008000, 0x1000 },
148
+ { "ddr-mpc", make_mpc, &mms->mpc[2], 0x58009000, 0x1000 },
149
+ },
150
+ }, {
151
+ .name = "apb_ppcexp1",
152
+ .ports = {
153
+ { "i2c0", make_i2c, &mms->i2c[0], 0x41200000, 0x1000 },
154
+ { "i2c1", make_i2c, &mms->i2c[1], 0x41201000, 0x1000 },
155
+ { "spi0", make_spi, &mms->spi[0], 0x41202000, 0x1000, { 52 } },
156
+ { "spi1", make_spi, &mms->spi[1], 0x41203000, 0x1000, { 53 } },
157
+ { "spi2", make_spi, &mms->spi[2], 0x41204000, 0x1000, { 54 } },
158
+ { "i2c2", make_i2c, &mms->i2c[2], 0x41205000, 0x1000 },
159
+ { "i2c3", make_i2c, &mms->i2c[3], 0x41206000, 0x1000 },
160
+ { /* port 7 reserved */ },
161
+ { "i2c4", make_i2c, &mms->i2c[4], 0x41208000, 0x1000 },
162
+ },
163
+ }, {
164
+ .name = "apb_ppcexp2",
165
+ .ports = {
166
+ { "scc", make_scc, &mms->scc, 0x41300000, 0x1000 },
167
+ { "i2s-audio", make_unimp_dev, &mms->i2s_audio,
168
+ 0x41301000, 0x1000 },
169
+ { "fpgaio", make_fpgaio, &mms->fpgaio, 0x41302000, 0x1000 },
170
+ { "uart0", make_uart, &mms->uart[0], 0x41303000, 0x1000, { 32, 33, 42 } },
171
+ { "uart1", make_uart, &mms->uart[1], 0x41304000, 0x1000, { 34, 35, 43 } },
172
+ { "uart2", make_uart, &mms->uart[2], 0x41305000, 0x1000, { 36, 37, 44 } },
173
+ { "uart3", make_uart, &mms->uart[3], 0x41306000, 0x1000, { 38, 39, 45 } },
174
+ { "uart4", make_uart, &mms->uart[4], 0x41307000, 0x1000, { 40, 41, 46 } },
175
+ { "uart5", make_uart, &mms->uart[5], 0x41308000, 0x1000, { 124, 125, 126 } },
176
+
177
+ { /* port 9 reserved */ },
178
+ { "clcd", make_unimp_dev, &mms->cldc, 0x4130a000, 0x1000 },
179
+ { "rtc", make_unimp_dev, &mms->rtc, 0x4130b000, 0x1000 },
180
+ },
181
+ }, {
182
+ .name = "ahb_ppcexp0",
183
+ .ports = {
184
+ { "gpio0", make_unimp_dev, &mms->gpio[0], 0x41100000, 0x1000 },
185
+ { "gpio1", make_unimp_dev, &mms->gpio[1], 0x41101000, 0x1000 },
186
+ { "gpio2", make_unimp_dev, &mms->gpio[2], 0x41102000, 0x1000 },
187
+ { "gpio3", make_unimp_dev, &mms->gpio[3], 0x41103000, 0x1000 },
188
+ { "eth", make_eth_dev, NULL, 0x41400000, 0x100000, { 48 } },
189
+ },
190
+ },
191
+ };
192
+
193
switch (mmc->fpga_type) {
194
case FPGA_AN505:
195
case FPGA_AN521:
196
ppcs = an505_ppcs;
197
num_ppcs = ARRAY_SIZE(an505_ppcs);
198
break;
199
+ case FPGA_AN524:
200
+ ppcs = an524_ppcs;
201
+ num_ppcs = ARRAY_SIZE(an524_ppcs);
202
+ break;
203
default:
204
g_assert_not_reached();
205
}
206
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
207
mps2tz_set_default_ram_info(mmc);
208
}
32
}
209
33
210
+static void mps3tz_an524_class_init(ObjectClass *oc, void *data)
34
+static CPAccessResult teecr_access(CPUARMState *env, const ARMCPRegInfo *ri,
35
+ bool isread)
211
+{
36
+{
212
+ MachineClass *mc = MACHINE_CLASS(oc);
37
+ /*
213
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_CLASS(oc);
38
+ * HSTR.TTEE only exists in v7A, not v8A, but v8A doesn't have T2EE
214
+
39
+ * at all, so we don't need to check whether we're v8A.
215
+ mc->desc = "ARM MPS3 with AN524 FPGA image for dual Cortex-M33";
40
+ */
216
+ mc->default_cpus = 2;
41
+ if (arm_current_el(env) < 2 && !arm_is_secure_below_el3(env) &&
217
+ mc->min_cpus = mc->default_cpus;
42
+ (env->cp15.hstr_el2 & HSTR_TTEE)) {
218
+ mc->max_cpus = mc->default_cpus;
43
+ return CP_ACCESS_TRAP_EL2;
219
+ mmc->fpga_type = FPGA_AN524;
44
+ }
220
+ mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m33");
45
+ return CP_ACCESS_OK;
221
+ mmc->scc_id = 0x41045240;
222
+ mmc->sysclk_frq = 32 * 1000 * 1000; /* 32MHz */
223
+ mmc->oscclk = an524_oscclk;
224
+ mmc->len_oscclk = ARRAY_SIZE(an524_oscclk);
225
+ mmc->fpgaio_num_leds = 10;
226
+ mmc->fpgaio_has_switches = true;
227
+ mmc->numirq = 95;
228
+ mmc->raminfo = an524_raminfo;
229
+ mmc->armsse_type = TYPE_SSE200;
230
+ mps2tz_set_default_ram_info(mmc);
231
+}
46
+}
232
+
47
+
233
static const TypeInfo mps2tz_info = {
48
static CPAccessResult teehbr_access(CPUARMState *env, const ARMCPRegInfo *ri,
234
.name = TYPE_MPS2TZ_MACHINE,
49
bool isread)
235
.parent = TYPE_MACHINE,
236
@@ -XXX,XX +XXX,XX @@ static const TypeInfo mps2tz_an521_info = {
237
.class_init = mps2tz_an521_class_init,
238
};
239
240
+static const TypeInfo mps3tz_an524_info = {
241
+ .name = TYPE_MPS3TZ_AN524_MACHINE,
242
+ .parent = TYPE_MPS2TZ_MACHINE,
243
+ .class_init = mps3tz_an524_class_init,
244
+};
245
+
246
static void mps2tz_machine_init(void)
247
{
50
{
248
type_register_static(&mps2tz_info);
51
if (arm_current_el(env) == 0 && (env->teecr & 1)) {
249
type_register_static(&mps2tz_an505_info);
52
return CP_ACCESS_TRAP;
250
type_register_static(&mps2tz_an521_info);
53
}
251
+ type_register_static(&mps3tz_an524_info);
54
- return CP_ACCESS_OK;
55
+ return teecr_access(env, ri, isread);
252
}
56
}
253
57
254
type_init(mps2tz_machine_init);
58
static const ARMCPRegInfo t2ee_cp_reginfo[] = {
59
{ .name = "TEECR", .cp = 14, .crn = 0, .crm = 0, .opc1 = 6, .opc2 = 0,
60
.access = PL1_RW, .fieldoffset = offsetof(CPUARMState, teecr),
61
.resetvalue = 0,
62
- .writefn = teecr_write },
63
+ .writefn = teecr_write, .accessfn = teecr_access },
64
{ .name = "TEEHBR", .cp = 14, .crn = 1, .crm = 0, .opc1 = 6, .opc2 = 0,
65
.access = PL0_RW, .fieldoffset = offsetof(CPUARMState, teehbr),
66
.accessfn = teehbr_access, .resetvalue = 0 },
255
--
67
--
256
2.20.1
68
2.20.1
257
69
258
70
diff view generated by jsdifflib
1
From: Rebecca Cran <rebecca@nuviainc.com>
1
In v7A, the HSTR register has a TJDBX bit which traps NS EL0/EL1
2
access to the JOSCR and JMCR trivial Jazelle registers, and also BXJ.
3
Implement these traps. In v8A this HSTR bit doesn't exist, so don't
4
trap for v8A CPUs.
2
5
3
Add support for FEAT_SSBS. SSBS (Speculative Store Bypass Safe) is an
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
optional feature in ARMv8.0, and mandatory in ARMv8.5.
5
6
Signed-off-by: Rebecca Cran <rebecca@nuviainc.com>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210216224543.16142-2-rebecca@nuviainc.com
8
Message-id: 20210816180305.20137-3-peter.maydell@linaro.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
9
---
11
target/arm/cpu.h | 15 ++++++++++++++-
10
target/arm/cpu.h | 1 +
12
target/arm/internals.h | 6 ++++++
11
target/arm/helper.h | 2 ++
13
target/arm/helper.c | 37 +++++++++++++++++++++++++++++++++++++
12
target/arm/syndrome.h | 7 +++++++
14
target/arm/translate-a64.c | 12 ++++++++++++
13
target/arm/helper.c | 17 +++++++++++++++++
15
4 files changed, 69 insertions(+), 1 deletion(-)
14
target/arm/op_helper.c | 16 ++++++++++++++++
15
target/arm/translate.c | 12 ++++++++++++
16
6 files changed, 55 insertions(+)
16
17
17
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
18
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
18
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
19
--- a/target/arm/cpu.h
20
--- a/target/arm/cpu.h
20
+++ b/target/arm/cpu.h
21
+++ b/target/arm/cpu.h
21
@@ -XXX,XX +XXX,XX @@ void pmu_init(ARMCPU *cpu);
22
@@ -XXX,XX +XXX,XX @@ static inline void xpsr_write(CPUARMState *env, uint32_t val, uint32_t mask)
22
#define SCTLR_TE (1U << 30) /* AArch32 only */
23
#define SCR_ATA (1U << 26)
23
#define SCTLR_EnIB (1U << 30) /* v8.3, AArch64 only */
24
24
#define SCTLR_EnIA (1U << 31) /* v8.3, AArch64 only */
25
#define HSTR_TTEE (1 << 16)
25
+#define SCTLR_DSSBS_32 (1U << 31) /* v8.5, AArch32 only */
26
+#define HSTR_TJDBX (1 << 17)
26
#define SCTLR_BT0 (1ULL << 35) /* v8.5-BTI */
27
27
#define SCTLR_BT1 (1ULL << 36) /* v8.5-BTI */
28
/* Return the current FPSCR value. */
28
#define SCTLR_ITFSB (1ULL << 37) /* v8.5-MemTag */
29
uint32_t vfp_get_fpscr(CPUARMState *env);
29
@@ -XXX,XX +XXX,XX @@ void pmu_init(ARMCPU *cpu);
30
diff --git a/target/arm/helper.h b/target/arm/helper.h
30
#define SCTLR_TCF (3ULL << 40) /* v8.5-MemTag */
31
index XXXXXXX..XXXXXXX 100644
31
#define SCTLR_ATA0 (1ULL << 42) /* v8.5-MemTag */
32
--- a/target/arm/helper.h
32
#define SCTLR_ATA (1ULL << 43) /* v8.5-MemTag */
33
+++ b/target/arm/helper.h
33
-#define SCTLR_DSSBS (1ULL << 44) /* v8.5 */
34
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_2(v7m_vlldm, void, env, i32)
34
+#define SCTLR_DSSBS_64 (1ULL << 44) /* v8.5, AArch64 only */
35
35
36
DEF_HELPER_2(v8m_stackcheck, void, env, i32)
36
#define CPTR_TCPAC (1U << 31)
37
37
#define CPTR_TTA (1U << 20)
38
+DEF_HELPER_FLAGS_2(check_bxj_trap, TCG_CALL_NO_WG, void, env, i32)
38
@@ -XXX,XX +XXX,XX @@ void pmu_init(ARMCPU *cpu);
39
+
39
#define CPSR_IL (1U << 20)
40
DEF_HELPER_4(access_check_cp_reg, void, env, ptr, i32, i32)
40
#define CPSR_DIT (1U << 21)
41
DEF_HELPER_3(set_cp_reg, void, env, ptr, i32)
41
#define CPSR_PAN (1U << 22)
42
DEF_HELPER_2(get_cp_reg, i32, env, ptr)
42
+#define CPSR_SSBS (1U << 23)
43
diff --git a/target/arm/syndrome.h b/target/arm/syndrome.h
43
#define CPSR_J (1U << 24)
44
index XXXXXXX..XXXXXXX 100644
44
#define CPSR_IT_0_1 (3U << 25)
45
--- a/target/arm/syndrome.h
45
#define CPSR_Q (1U << 27)
46
+++ b/target/arm/syndrome.h
46
@@ -XXX,XX +XXX,XX @@ void pmu_init(ARMCPU *cpu);
47
@@ -XXX,XX +XXX,XX @@ enum arm_exception_class {
47
#define PSTATE_A (1U << 8)
48
EC_ADVSIMDFPACCESSTRAP = 0x07,
48
#define PSTATE_D (1U << 9)
49
EC_FPIDTRAP = 0x08,
49
#define PSTATE_BTYPE (3U << 10)
50
EC_PACTRAP = 0x09,
50
+#define PSTATE_SSBS (1U << 12)
51
+ EC_BXJTRAP = 0x0a,
51
#define PSTATE_IL (1U << 20)
52
EC_CP14RRTTRAP = 0x0c,
52
#define PSTATE_SS (1U << 21)
53
EC_BTITRAP = 0x0d,
53
#define PSTATE_PAN (1U << 22)
54
EC_ILLEGALSTATE = 0x0e,
54
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_aa32_dit(const ARMISARegisters *id)
55
@@ -XXX,XX +XXX,XX @@ static inline uint32_t syn_btitrap(int btype)
55
return FIELD_EX32(id->id_pfr0, ID_PFR0, DIT) != 0;
56
return (EC_BTITRAP << ARM_EL_EC_SHIFT) | btype;
56
}
57
}
57
58
58
+static inline bool isar_feature_aa32_ssbs(const ARMISARegisters *id)
59
+static inline uint32_t syn_bxjtrap(int cv, int cond, int rm)
59
+{
60
+{
60
+ return FIELD_EX32(id->id_pfr2, ID_PFR2, SSBS) != 0;
61
+ return (EC_BXJTRAP << ARM_EL_EC_SHIFT) | ARM_EL_IL |
62
+ (cv << 24) | (cond << 20) | rm;
61
+}
63
+}
62
+
64
+
63
/*
65
static inline uint32_t syn_insn_abort(int same_el, int ea, int s1ptw, int fsc)
64
* 64-bit feature tests via id registers.
66
{
65
*/
67
return (EC_INSNABORT << ARM_EL_EC_SHIFT) | (same_el << ARM_EL_EC_SHIFT)
66
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_aa64_dit(const ARMISARegisters *id)
67
return FIELD_EX64(id->id_aa64pfr0, ID_AA64PFR0, DIT) != 0;
68
}
69
70
+static inline bool isar_feature_aa64_ssbs(const ARMISARegisters *id)
71
+{
72
+ return FIELD_EX64(id->id_aa64pfr1, ID_AA64PFR1, SSBS) != 0;
73
+}
74
+
75
/*
76
* Feature tests for "does this exist in either 32-bit or 64-bit?"
77
*/
78
diff --git a/target/arm/internals.h b/target/arm/internals.h
79
index XXXXXXX..XXXXXXX 100644
80
--- a/target/arm/internals.h
81
+++ b/target/arm/internals.h
82
@@ -XXX,XX +XXX,XX @@ static inline uint32_t aarch32_cpsr_valid_mask(uint64_t features,
83
if (isar_feature_aa32_dit(id)) {
84
valid |= CPSR_DIT;
85
}
86
+ if (isar_feature_aa32_ssbs(id)) {
87
+ valid |= CPSR_SSBS;
88
+ }
89
90
return valid;
91
}
92
@@ -XXX,XX +XXX,XX @@ static inline uint32_t aarch64_pstate_valid_mask(const ARMISARegisters *id)
93
if (isar_feature_aa64_dit(id)) {
94
valid |= PSTATE_DIT;
95
}
96
+ if (isar_feature_aa64_ssbs(id)) {
97
+ valid |= PSTATE_SSBS;
98
+ }
99
if (isar_feature_aa64_mte(id)) {
100
valid |= PSTATE_TCO;
101
}
102
diff --git a/target/arm/helper.c b/target/arm/helper.c
68
diff --git a/target/arm/helper.c b/target/arm/helper.c
103
index XXXXXXX..XXXXXXX 100644
69
index XXXXXXX..XXXXXXX 100644
104
--- a/target/arm/helper.c
70
--- a/target/arm/helper.c
105
+++ b/target/arm/helper.c
71
+++ b/target/arm/helper.c
106
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo dit_reginfo = {
72
@@ -XXX,XX +XXX,XX @@ static CPAccessResult access_jazelle(CPUARMState *env, const ARMCPRegInfo *ri,
107
.readfn = aa64_dit_read, .writefn = aa64_dit_write
73
return CP_ACCESS_OK;
108
};
74
}
109
75
110
+static uint64_t aa64_ssbs_read(CPUARMState *env, const ARMCPRegInfo *ri)
76
+static CPAccessResult access_joscr_jmcr(CPUARMState *env,
77
+ const ARMCPRegInfo *ri, bool isread)
111
+{
78
+{
112
+ return env->pstate & PSTATE_SSBS;
79
+ /*
80
+ * HSTR.TJDBX traps JOSCR and JMCR accesses, but it exists only
81
+ * in v7A, not in v8A.
82
+ */
83
+ if (!arm_feature(env, ARM_FEATURE_V8) &&
84
+ arm_current_el(env) < 2 && !arm_is_secure_below_el3(env) &&
85
+ (env->cp15.hstr_el2 & HSTR_TJDBX)) {
86
+ return CP_ACCESS_TRAP_EL2;
87
+ }
88
+ return CP_ACCESS_OK;
113
+}
89
+}
114
+
90
+
115
+static void aa64_ssbs_write(CPUARMState *env, const ARMCPRegInfo *ri,
91
static const ARMCPRegInfo jazelle_regs[] = {
116
+ uint64_t value)
92
{ .name = "JIDR",
93
.cp = 14, .crn = 0, .crm = 0, .opc1 = 7, .opc2 = 0,
94
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo jazelle_regs[] = {
95
.type = ARM_CP_CONST, .resetvalue = 0 },
96
{ .name = "JOSCR",
97
.cp = 14, .crn = 1, .crm = 0, .opc1 = 7, .opc2 = 0,
98
+ .accessfn = access_joscr_jmcr,
99
.access = PL1_RW, .type = ARM_CP_CONST, .resetvalue = 0 },
100
{ .name = "JMCR",
101
.cp = 14, .crn = 2, .crm = 0, .opc1 = 7, .opc2 = 0,
102
+ .accessfn = access_joscr_jmcr,
103
.access = PL1_RW, .type = ARM_CP_CONST, .resetvalue = 0 },
104
REGINFO_SENTINEL
105
};
106
diff --git a/target/arm/op_helper.c b/target/arm/op_helper.c
107
index XXXXXXX..XXXXXXX 100644
108
--- a/target/arm/op_helper.c
109
+++ b/target/arm/op_helper.c
110
@@ -XXX,XX +XXX,XX @@ void HELPER(setend)(CPUARMState *env)
111
arm_rebuild_hflags(env);
112
}
113
114
+void HELPER(check_bxj_trap)(CPUARMState *env, uint32_t rm)
117
+{
115
+{
118
+ env->pstate = (env->pstate & ~PSTATE_SSBS) | (value & PSTATE_SSBS);
116
+ /*
117
+ * Only called if in NS EL0 or EL1 for a BXJ for a v7A CPU;
118
+ * check if HSTR.TJDBX means we need to trap to EL2.
119
+ */
120
+ if (env->cp15.hstr_el2 & HSTR_TJDBX) {
121
+ /*
122
+ * We know the condition code check passed, so take the IMPDEF
123
+ * choice to always report CV=1 COND 0xe
124
+ */
125
+ uint32_t syn = syn_bxjtrap(1, 0xe, rm);
126
+ raise_exception_ra(env, EXCP_HYP_TRAP, syn, 2, GETPC());
127
+ }
119
+}
128
+}
120
+
129
+
121
+static const ARMCPRegInfo ssbs_reginfo = {
130
#ifndef CONFIG_USER_ONLY
122
+ .name = "SSBS", .state = ARM_CP_STATE_AA64,
131
/* Function checks whether WFx (WFI/WFE) instructions are set up to be trapped.
123
+ .opc0 = 3, .opc1 = 3, .crn = 4, .crm = 2, .opc2 = 6,
132
* The function returns the target EL (1-3) if the instruction is to be trapped;
124
+ .type = ARM_CP_NO_RAW, .access = PL0_RW,
133
diff --git a/target/arm/translate.c b/target/arm/translate.c
125
+ .readfn = aa64_ssbs_read, .writefn = aa64_ssbs_write
134
index XXXXXXX..XXXXXXX 100644
126
+};
135
--- a/target/arm/translate.c
127
+
136
+++ b/target/arm/translate.c
128
static CPAccessResult aa64_cacheop_poc_access(CPUARMState *env,
137
@@ -XXX,XX +XXX,XX @@ static bool trans_BXJ(DisasContext *s, arg_BXJ *a)
129
const ARMCPRegInfo *ri,
138
if (!ENABLE_ARCH_5J || arm_dc_feature(s, ARM_FEATURE_M)) {
130
bool isread)
139
return false;
131
@@ -XXX,XX +XXX,XX @@ void register_cp_regs_for_features(ARMCPU *cpu)
132
if (cpu_isar_feature(aa64_dit, cpu)) {
133
define_one_arm_cp_reg(cpu, &dit_reginfo);
134
}
140
}
135
+ if (cpu_isar_feature(aa64_ssbs, cpu)) {
141
+ /*
136
+ define_one_arm_cp_reg(cpu, &ssbs_reginfo);
142
+ * v7A allows BXJ to be trapped via HSTR.TJDBX. We don't waste a
143
+ * TBFLAGS bit on a basically-never-happens case, so call a helper
144
+ * function to check for the trap and raise the exception if needed
145
+ * (passing it the register number for the syndrome value).
146
+ * v8A doesn't have this HSTR bit.
147
+ */
148
+ if (!arm_dc_feature(s, ARM_FEATURE_V8) &&
149
+ arm_dc_feature(s, ARM_FEATURE_EL2) &&
150
+ s->current_el < 2 && s->ns) {
151
+ gen_helper_check_bxj_trap(cpu_env, tcg_constant_i32(a->rm));
137
+ }
152
+ }
138
153
/* Trivial implementation equivalent to bx. */
139
if (arm_feature(env, ARM_FEATURE_EL2) && cpu_isar_feature(aa64_vh, cpu)) {
154
gen_bx(s, load_reg(s, a->rm));
140
define_arm_cp_regs(cpu, vhe_reginfo);
155
return true;
141
@@ -XXX,XX +XXX,XX @@ static void take_aarch32_exception(CPUARMState *env, int new_mode,
142
env->uncached_cpsr &= ~(CPSR_IL | CPSR_J);
143
env->daif |= mask;
144
145
+ if (cpu_isar_feature(aa32_ssbs, env_archcpu(env))) {
146
+ if (env->cp15.sctlr_el[new_el] & SCTLR_DSSBS_32) {
147
+ env->uncached_cpsr |= CPSR_SSBS;
148
+ } else {
149
+ env->uncached_cpsr &= ~CPSR_SSBS;
150
+ }
151
+ }
152
+
153
if (new_mode == ARM_CPU_MODE_HYP) {
154
env->thumb = (env->cp15.sctlr_el[2] & SCTLR_TE) != 0;
155
env->elr_el[2] = env->regs[15];
156
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_do_interrupt_aarch64(CPUState *cs)
157
new_mode |= PSTATE_TCO;
158
}
159
160
+ if (cpu_isar_feature(aa64_ssbs, cpu)) {
161
+ if (env->cp15.sctlr_el[new_el] & SCTLR_DSSBS_64) {
162
+ new_mode |= PSTATE_SSBS;
163
+ } else {
164
+ new_mode &= ~PSTATE_SSBS;
165
+ }
166
+ }
167
+
168
pstate_write(env, PSTATE_DAIF | new_mode);
169
env->aarch64 = 1;
170
aarch64_restore_sp(env, new_el);
171
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
172
index XXXXXXX..XXXXXXX 100644
173
--- a/target/arm/translate-a64.c
174
+++ b/target/arm/translate-a64.c
175
@@ -XXX,XX +XXX,XX @@ static void handle_msr_i(DisasContext *s, uint32_t insn,
176
tcg_temp_free_i32(t1);
177
break;
178
179
+ case 0x19: /* SSBS */
180
+ if (!dc_isar_feature(aa64_ssbs, s)) {
181
+ goto do_unallocated;
182
+ }
183
+ if (crm & 1) {
184
+ set_pstate_bits(PSTATE_SSBS);
185
+ } else {
186
+ clear_pstate_bits(PSTATE_SSBS);
187
+ }
188
+ /* Don't need to rebuild hflags since SSBS is a nop */
189
+ break;
190
+
191
case 0x1a: /* DIT */
192
if (!dc_isar_feature(aa64_dit, s)) {
193
goto do_unallocated;
194
--
156
--
195
2.20.1
157
2.20.1
196
158
197
159
diff view generated by jsdifflib
1
For a long time now the UI layer has guaranteed that the console
1
Currently we rely on all the callsites of cpsr_write() to rebuild the
2
surface is always 32 bits per pixel RGB. Remove the legacy dead
2
cached hflags if they change one of the CPSR bits which we use as a
3
code from the tc6393xb display device which was handling the
3
TB flag and cache in hflags. This is a bit awkward when we want to
4
possibility that the console surface was some other format.
4
change the set of CPSR bits that we cache, because it means we need
5
to re-audit all the cpsr_write() callsites to see which flags they
6
are writing and whether they now need to rebuild the hflags.
7
8
Switch instead to making cpsr_write() call arm_rebuild_hflags()
9
itself if one of the bits being changed is a cached bit.
10
11
We don't do the rebuild for the CPSRWriteRaw write type, because that
12
kind of write is generally doing something special anyway. For the
13
CPSRWriteRaw callsites in the KVM code and inbound migration we
14
definitely don't want to recalculate the hflags; the callsites in
15
boot.c and arm-powerctl.c have to do a rebuild-hflags call themselves
16
anyway because of other CPU state changes they make.
17
18
This allows us to drop explicit arm_rebuild_hflags() calls in a
19
couple of places where the only reason we needed to call it was the
20
CPSR write.
21
22
This fixes a bug where we were incorrectly failing to rebuild hflags
23
in the code path for a gdbstub write to CPSR, which meant that you
24
could make QEMU assert by breaking into a running guest, altering the
25
CPSR to change the value of, for example, CPSR.E, and then
26
continuing.
5
27
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
28
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
29
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210215103215.4944-3-peter.maydell@linaro.org
30
Message-id: 20210817201843.3829-1-peter.maydell@linaro.org
9
---
31
---
10
include/ui/console.h | 10 ----------
32
target/arm/cpu.h | 10 ++++++++--
11
hw/display/tc6393xb.c | 33 +--------------------------------
33
linux-user/arm/signal.c | 2 --
12
2 files changed, 1 insertion(+), 42 deletions(-)
34
target/arm/helper.c | 5 +++++
35
3 files changed, 13 insertions(+), 4 deletions(-)
13
36
14
diff --git a/include/ui/console.h b/include/ui/console.h
37
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
15
index XXXXXXX..XXXXXXX 100644
38
index XXXXXXX..XXXXXXX 100644
16
--- a/include/ui/console.h
39
--- a/target/arm/cpu.h
17
+++ b/include/ui/console.h
40
+++ b/target/arm/cpu.h
18
@@ -XXX,XX +XXX,XX @@ PixelFormat qemu_default_pixelformat(int bpp);
41
@@ -XXX,XX +XXX,XX @@ uint32_t cpsr_read(CPUARMState *env);
19
DisplaySurface *qemu_create_displaysurface(int width, int height);
42
typedef enum CPSRWriteType {
20
void qemu_free_displaysurface(DisplaySurface *surface);
43
CPSRWriteByInstr = 0, /* from guest MSR or CPS */
21
44
CPSRWriteExceptionReturn = 1, /* from guest exception return insn */
22
-static inline int is_surface_bgr(DisplaySurface *surface)
45
- CPSRWriteRaw = 2, /* trust values, do not switch reg banks */
23
-{
46
+ CPSRWriteRaw = 2,
24
- if (PIXMAN_FORMAT_BPP(surface->format) == 32 &&
47
+ /* trust values, no reg bank switch, no hflags rebuild */
25
- PIXMAN_FORMAT_TYPE(surface->format) == PIXMAN_TYPE_ABGR) {
48
CPSRWriteByGDBStub = 3, /* from the GDB stub */
26
- return 1;
49
} CPSRWriteType;
27
- } else {
50
28
- return 0;
51
-/* Set the CPSR. Note that some bits of mask must be all-set or all-clear.*/
29
- }
52
+/*
30
-}
53
+ * Set the CPSR. Note that some bits of mask must be all-set or all-clear.
31
-
54
+ * This will do an arm_rebuild_hflags() if any of the bits in @mask
32
static inline int is_buffer_shared(DisplaySurface *surface)
55
+ * correspond to TB flags bits cached in the hflags, unless @write_type
56
+ * is CPSRWriteRaw.
57
+ */
58
void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
59
CPSRWriteType write_type);
60
61
diff --git a/linux-user/arm/signal.c b/linux-user/arm/signal.c
62
index XXXXXXX..XXXXXXX 100644
63
--- a/linux-user/arm/signal.c
64
+++ b/linux-user/arm/signal.c
65
@@ -XXX,XX +XXX,XX @@ setup_return(CPUARMState *env, struct target_sigaction *ka,
66
env->regs[14] = retcode;
67
env->regs[15] = handler & (thumb ? ~1 : ~3);
68
cpsr_write(env, cpsr, CPSR_IT | CPSR_T | CPSR_E, CPSRWriteByInstr);
69
- arm_rebuild_hflags(env);
70
71
return 0;
72
}
73
@@ -XXX,XX +XXX,XX @@ restore_sigcontext(CPUARMState *env, struct target_sigcontext *sc)
74
__get_user(env->regs[15], &sc->arm_pc);
75
__get_user(cpsr, &sc->arm_cpsr);
76
cpsr_write(env, cpsr, CPSR_USER | CPSR_EXEC, CPSRWriteByInstr);
77
- arm_rebuild_hflags(env);
78
79
err |= !valid_user_regs(env);
80
81
diff --git a/target/arm/helper.c b/target/arm/helper.c
82
index XXXXXXX..XXXXXXX 100644
83
--- a/target/arm/helper.c
84
+++ b/target/arm/helper.c
85
@@ -XXX,XX +XXX,XX @@ void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
86
CPSRWriteType write_type)
33
{
87
{
34
return !(surface->flags & QEMU_ALLOCATED_FLAG);
88
uint32_t changed_daif;
35
diff --git a/hw/display/tc6393xb.c b/hw/display/tc6393xb.c
89
+ bool rebuild_hflags = (write_type != CPSRWriteRaw) &&
36
index XXXXXXX..XXXXXXX 100644
90
+ (mask & (CPSR_M | CPSR_E | CPSR_IL));
37
--- a/hw/display/tc6393xb.c
91
38
+++ b/hw/display/tc6393xb.c
92
if (mask & CPSR_NZCV) {
39
@@ -XXX,XX +XXX,XX @@ static void tc6393xb_nand_writeb(TC6393xbState *s, hwaddr addr, uint32_t value)
93
env->ZF = (~val) & CPSR_Z;
40
(uint32_t) addr, value & 0xff);
94
@@ -XXX,XX +XXX,XX @@ void cpsr_write(CPUARMState *env, uint32_t val, uint32_t mask,
95
}
96
mask &= ~CACHED_CPSR_BITS;
97
env->uncached_cpsr = (env->uncached_cpsr & ~mask) | (val & mask);
98
+ if (rebuild_hflags) {
99
+ arm_rebuild_hflags(env);
100
+ }
41
}
101
}
42
102
43
-#define BITS 8
103
/* Sign/zero extend */
44
-#include "tc6393xb_template.h"
45
-#define BITS 15
46
-#include "tc6393xb_template.h"
47
-#define BITS 16
48
-#include "tc6393xb_template.h"
49
-#define BITS 24
50
-#include "tc6393xb_template.h"
51
#define BITS 32
52
#include "tc6393xb_template.h"
53
54
static void tc6393xb_draw_graphic(TC6393xbState *s, int full_update)
55
{
56
- DisplaySurface *surface = qemu_console_surface(s->con);
57
-
58
- switch (surface_bits_per_pixel(surface)) {
59
- case 8:
60
- tc6393xb_draw_graphic8(s);
61
- break;
62
- case 15:
63
- tc6393xb_draw_graphic15(s);
64
- break;
65
- case 16:
66
- tc6393xb_draw_graphic16(s);
67
- break;
68
- case 24:
69
- tc6393xb_draw_graphic24(s);
70
- break;
71
- case 32:
72
- tc6393xb_draw_graphic32(s);
73
- break;
74
- default:
75
- printf("tc6393xb: unknown depth %d\n",
76
- surface_bits_per_pixel(surface));
77
- return;
78
- }
79
-
80
+ tc6393xb_draw_graphic32(s);
81
dpy_gfx_update_full(s->con);
82
}
83
84
--
104
--
85
2.20.1
105
2.20.1
86
106
87
107
diff view generated by jsdifflib
1
From: Doug Evans <dje@google.com>
1
From: Tong Ho <tong.ho@xilinx.com>
2
2
3
This is a 10/100 ethernet device that has several features.
3
Add unimplemented APU mmio region to xlnx-versal for booting
4
Only the ones needed by the Linux driver have been implemented.
4
bare-metal guests built with standalone bsp, which access the
5
See npcm7xx_emc.c for a list of unimplemented features.
5
region from one of the following places:
6
https://github.com/Xilinx/embeddedsw/blob/release-2020.2/lib/bsp/standalone/src/arm/ARMv8/64bit/armclang/boot.S#L139
7
https://github.com/Xilinx/embeddedsw/blob/release-2020.2/lib/bsp/standalone/src/arm/ARMv8/64bit/gcc/boot.S#L183
6
8
7
Reviewed-by: Hao Wu <wuhaotsh@google.com>
9
Acked-by: Alistair Francis <alistair.francis@wdc.com>
8
Reviewed-by: Avi Fishman <avi.fishman@nuvoton.com>
10
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Tong Ho <tong.ho@xilinx.com>
10
Signed-off-by: Doug Evans <dje@google.com>
12
Message-id: 20210823173818.201259-2-tong.ho@xilinx.com
11
Message-id: 20210218212453.831406-3-dje@google.com
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
---
14
---
14
docs/system/arm/nuvoton.rst | 3 ++-
15
include/hw/arm/xlnx-versal.h | 2 ++
15
include/hw/arm/npcm7xx.h | 2 ++
16
hw/arm/xlnx-versal.c | 2 ++
16
hw/arm/npcm7xx.c | 50 +++++++++++++++++++++++++++++++++++--
17
2 files changed, 4 insertions(+)
17
3 files changed, 52 insertions(+), 3 deletions(-)
18
18
19
diff --git a/docs/system/arm/nuvoton.rst b/docs/system/arm/nuvoton.rst
19
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
20
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
21
--- a/docs/system/arm/nuvoton.rst
21
--- a/include/hw/arm/xlnx-versal.h
22
+++ b/docs/system/arm/nuvoton.rst
22
+++ b/include/hw/arm/xlnx-versal.h
23
@@ -XXX,XX +XXX,XX @@ Supported devices
23
@@ -XXX,XX +XXX,XX @@ struct Versal {
24
* Analog to Digital Converter (ADC)
24
#define MM_IOU_SCNTRS_SIZE 0x10000
25
* Pulse Width Modulation (PWM)
25
#define MM_FPD_CRF 0xfd1a0000U
26
* SMBus controller (SMBF)
26
#define MM_FPD_CRF_SIZE 0x140000
27
+ * Ethernet controller (EMC)
27
+#define MM_FPD_FPD_APU 0xfd5c0000
28
28
+#define MM_FPD_FPD_APU_SIZE 0x100
29
Missing devices
29
30
---------------
30
#define MM_PMC_SD0 0xf1040000U
31
@@ -XXX,XX +XXX,XX @@ Missing devices
31
#define MM_PMC_SD0_SIZE 0x10000
32
* Shared memory (SHM)
32
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
33
* eSPI slave interface
34
35
- * Ethernet controllers (GMAC and EMC)
36
+ * Ethernet controller (GMAC)
37
* USB device (USBD)
38
* Peripheral SPI controller (PSPI)
39
* SD/MMC host
40
diff --git a/include/hw/arm/npcm7xx.h b/include/hw/arm/npcm7xx.h
41
index XXXXXXX..XXXXXXX 100644
33
index XXXXXXX..XXXXXXX 100644
42
--- a/include/hw/arm/npcm7xx.h
34
--- a/hw/arm/xlnx-versal.c
43
+++ b/include/hw/arm/npcm7xx.h
35
+++ b/hw/arm/xlnx-versal.c
44
@@ -XXX,XX +XXX,XX @@
36
@@ -XXX,XX +XXX,XX @@ static void versal_unimp(Versal *s)
45
#include "hw/misc/npcm7xx_gcr.h"
37
MM_CRL, MM_CRL_SIZE);
46
#include "hw/misc/npcm7xx_pwm.h"
38
versal_unimp_area(s, "crf", &s->mr_ps,
47
#include "hw/misc/npcm7xx_rng.h"
39
MM_FPD_CRF, MM_FPD_CRF_SIZE);
48
+#include "hw/net/npcm7xx_emc.h"
40
+ versal_unimp_area(s, "apu", &s->mr_ps,
49
#include "hw/nvram/npcm7xx_otp.h"
41
+ MM_FPD_FPD_APU, MM_FPD_FPD_APU_SIZE);
50
#include "hw/timer/npcm7xx_timer.h"
42
versal_unimp_area(s, "crp", &s->mr_ps,
51
#include "hw/ssi/npcm7xx_fiu.h"
43
MM_PMC_CRP, MM_PMC_CRP_SIZE);
52
@@ -XXX,XX +XXX,XX @@ typedef struct NPCM7xxState {
44
versal_unimp_area(s, "iou-scntr", &s->mr_ps,
53
EHCISysBusState ehci;
54
OHCISysBusState ohci;
55
NPCM7xxFIUState fiu[2];
56
+ NPCM7xxEMCState emc[2];
57
} NPCM7xxState;
58
59
#define TYPE_NPCM7XX "npcm7xx"
60
diff --git a/hw/arm/npcm7xx.c b/hw/arm/npcm7xx.c
61
index XXXXXXX..XXXXXXX 100644
62
--- a/hw/arm/npcm7xx.c
63
+++ b/hw/arm/npcm7xx.c
64
@@ -XXX,XX +XXX,XX @@ enum NPCM7xxInterrupt {
65
NPCM7XX_UART1_IRQ,
66
NPCM7XX_UART2_IRQ,
67
NPCM7XX_UART3_IRQ,
68
+ NPCM7XX_EMC1RX_IRQ = 15,
69
+ NPCM7XX_EMC1TX_IRQ,
70
NPCM7XX_TIMER0_IRQ = 32, /* Timer Module 0 */
71
NPCM7XX_TIMER1_IRQ,
72
NPCM7XX_TIMER2_IRQ,
73
@@ -XXX,XX +XXX,XX @@ enum NPCM7xxInterrupt {
74
NPCM7XX_SMBUS15_IRQ,
75
NPCM7XX_PWM0_IRQ = 93, /* PWM module 0 */
76
NPCM7XX_PWM1_IRQ, /* PWM module 1 */
77
+ NPCM7XX_EMC2RX_IRQ = 114,
78
+ NPCM7XX_EMC2TX_IRQ,
79
NPCM7XX_GPIO0_IRQ = 116,
80
NPCM7XX_GPIO1_IRQ,
81
NPCM7XX_GPIO2_IRQ,
82
@@ -XXX,XX +XXX,XX @@ static const hwaddr npcm7xx_smbus_addr[] = {
83
0xf008f000,
84
};
85
86
+/* Register base address for each EMC Module */
87
+static const hwaddr npcm7xx_emc_addr[] = {
88
+ 0xf0825000,
89
+ 0xf0826000,
90
+};
91
+
92
static const struct {
93
hwaddr regs_addr;
94
uint32_t unconnected_pins;
95
@@ -XXX,XX +XXX,XX @@ static void npcm7xx_init(Object *obj)
96
for (i = 0; i < ARRAY_SIZE(s->pwm); i++) {
97
object_initialize_child(obj, "pwm[*]", &s->pwm[i], TYPE_NPCM7XX_PWM);
98
}
99
+
100
+ for (i = 0; i < ARRAY_SIZE(s->emc); i++) {
101
+ object_initialize_child(obj, "emc[*]", &s->emc[i], TYPE_NPCM7XX_EMC);
102
+ }
103
}
104
105
static void npcm7xx_realize(DeviceState *dev, Error **errp)
106
@@ -XXX,XX +XXX,XX @@ static void npcm7xx_realize(DeviceState *dev, Error **errp)
107
sysbus_connect_irq(sbd, i, npcm7xx_irq(s, NPCM7XX_PWM0_IRQ + i));
108
}
109
110
+ /*
111
+ * EMC Modules. Cannot fail.
112
+ * The mapping of the device to its netdev backend works as follows:
113
+ * emc[i] = nd_table[i]
114
+ * This works around the inability to specify the netdev property for the
115
+ * emc device: it's not pluggable and thus the -device option can't be
116
+ * used.
117
+ */
118
+ QEMU_BUILD_BUG_ON(ARRAY_SIZE(npcm7xx_emc_addr) != ARRAY_SIZE(s->emc));
119
+ QEMU_BUILD_BUG_ON(ARRAY_SIZE(s->emc) != 2);
120
+ for (i = 0; i < ARRAY_SIZE(s->emc); i++) {
121
+ s->emc[i].emc_num = i;
122
+ SysBusDevice *sbd = SYS_BUS_DEVICE(&s->emc[i]);
123
+ if (nd_table[i].used) {
124
+ qemu_check_nic_model(&nd_table[i], TYPE_NPCM7XX_EMC);
125
+ qdev_set_nic_properties(DEVICE(sbd), &nd_table[i]);
126
+ }
127
+ /*
128
+ * The device exists regardless of whether it's connected to a QEMU
129
+ * netdev backend. So always instantiate it even if there is no
130
+ * backend.
131
+ */
132
+ sysbus_realize(sbd, &error_abort);
133
+ sysbus_mmio_map(sbd, 0, npcm7xx_emc_addr[i]);
134
+ int tx_irq = i == 0 ? NPCM7XX_EMC1TX_IRQ : NPCM7XX_EMC2TX_IRQ;
135
+ int rx_irq = i == 0 ? NPCM7XX_EMC1RX_IRQ : NPCM7XX_EMC2RX_IRQ;
136
+ /*
137
+ * N.B. The values for the second argument sysbus_connect_irq are
138
+ * chosen to match the registration order in npcm7xx_emc_realize.
139
+ */
140
+ sysbus_connect_irq(sbd, 0, npcm7xx_irq(s, tx_irq));
141
+ sysbus_connect_irq(sbd, 1, npcm7xx_irq(s, rx_irq));
142
+ }
143
+
144
/*
145
* Flash Interface Unit (FIU). Can fail if incorrect number of chip selects
146
* specified, but this is a programming error.
147
@@ -XXX,XX +XXX,XX @@ static void npcm7xx_realize(DeviceState *dev, Error **errp)
148
create_unimplemented_device("npcm7xx.vcd", 0xf0810000, 64 * KiB);
149
create_unimplemented_device("npcm7xx.ece", 0xf0820000, 8 * KiB);
150
create_unimplemented_device("npcm7xx.vdma", 0xf0822000, 8 * KiB);
151
- create_unimplemented_device("npcm7xx.emc1", 0xf0825000, 4 * KiB);
152
- create_unimplemented_device("npcm7xx.emc2", 0xf0826000, 4 * KiB);
153
create_unimplemented_device("npcm7xx.usbd[0]", 0xf0830000, 4 * KiB);
154
create_unimplemented_device("npcm7xx.usbd[1]", 0xf0831000, 4 * KiB);
155
create_unimplemented_device("npcm7xx.usbd[2]", 0xf0832000, 4 * KiB);
156
--
45
--
157
2.20.1
46
2.20.1
158
47
159
48
diff view generated by jsdifflib
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
1
From: Tong Ho <tong.ho@xilinx.com>
2
2
3
We hint the 'has_rpu' property is no longer required since commit
3
Add unimplemented APU mmio region to xlnx-zynqmp for booting
4
6908ec448b4 ("xlnx-zynqmp: Properly support the smp command line
4
bare-metal guests built with standalone bsp, which access the
5
option") which was released in QEMU v2.11.0.
5
region from one of the following places:
6
https://github.com/Xilinx/embeddedsw/blob/release-2020.2/lib/bsp/standalone/src/arm/ARMv8/64bit/armclang/boot.S#L139
7
https://github.com/Xilinx/embeddedsw/blob/release-2020.2/lib/bsp/standalone/src/arm/ARMv8/64bit/gcc/boot.S#L183
6
8
7
Beside, this device is marked 'user_creatable = false', so the
9
Acked-by: Alistair Francis <alistair.francis@wdc.com>
8
only thing that could be setting the property is the board code
10
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
that creates the device.
11
Signed-off-by: Tong Ho <tong.ho@xilinx.com>
10
12
Message-id: 20210823173818.201259-3-tong.ho@xilinx.com
11
Since the property is not user-facing, we can remove it without
12
going through the deprecation process.
13
14
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
16
Message-id: 20210219144350.1979905-1-f4bug@amsat.org
17
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
18
---
14
---
19
include/hw/arm/xlnx-zynqmp.h | 2 --
15
include/hw/arm/xlnx-zynqmp.h | 7 +++++++
20
hw/arm/xlnx-zynqmp.c | 6 ------
16
hw/arm/xlnx-zynqmp.c | 32 ++++++++++++++++++++++++++++++++
21
2 files changed, 8 deletions(-)
17
2 files changed, 39 insertions(+)
22
18
23
diff --git a/include/hw/arm/xlnx-zynqmp.h b/include/hw/arm/xlnx-zynqmp.h
19
diff --git a/include/hw/arm/xlnx-zynqmp.h b/include/hw/arm/xlnx-zynqmp.h
24
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
25
--- a/include/hw/arm/xlnx-zynqmp.h
21
--- a/include/hw/arm/xlnx-zynqmp.h
26
+++ b/include/hw/arm/xlnx-zynqmp.h
22
+++ b/include/hw/arm/xlnx-zynqmp.h
23
@@ -XXX,XX +XXX,XX @@ OBJECT_DECLARE_SIMPLE_TYPE(XlnxZynqMPState, XLNX_ZYNQMP)
24
#define XLNX_ZYNQMP_MAX_RAM_SIZE (XLNX_ZYNQMP_MAX_LOW_RAM_SIZE + \
25
XLNX_ZYNQMP_MAX_HIGH_RAM_SIZE)
26
27
+/*
28
+ * Unimplemented mmio regions needed to boot some images.
29
+ */
30
+#define XLNX_ZYNQMP_NUM_UNIMP_AREAS 1
31
+
32
struct XlnxZynqMPState {
33
/*< private >*/
34
DeviceState parent_obj;
27
@@ -XXX,XX +XXX,XX @@ struct XlnxZynqMPState {
35
@@ -XXX,XX +XXX,XX @@ struct XlnxZynqMPState {
28
bool secure;
36
MemoryRegion *ddr_ram;
29
/* Has the ARM Virtualization extensions? */
37
MemoryRegion ddr_ram_low, ddr_ram_high;
30
bool virt;
38
31
- /* Has the RPU subsystem? */
39
+ MemoryRegion mr_unimp[XLNX_ZYNQMP_NUM_UNIMP_AREAS];
32
- bool has_rpu;
40
+
33
41
CadenceGEMState gem[XLNX_ZYNQMP_NUM_GEMS];
34
/* CAN bus. */
42
CadenceUARTState uart[XLNX_ZYNQMP_NUM_UARTS];
35
CanBusState *canbus[XLNX_ZYNQMP_NUM_CAN];
43
XlnxZynqMPCANState can[XLNX_ZYNQMP_NUM_CAN];
36
diff --git a/hw/arm/xlnx-zynqmp.c b/hw/arm/xlnx-zynqmp.c
44
diff --git a/hw/arm/xlnx-zynqmp.c b/hw/arm/xlnx-zynqmp.c
37
index XXXXXXX..XXXXXXX 100644
45
index XXXXXXX..XXXXXXX 100644
38
--- a/hw/arm/xlnx-zynqmp.c
46
--- a/hw/arm/xlnx-zynqmp.c
39
+++ b/hw/arm/xlnx-zynqmp.c
47
+++ b/hw/arm/xlnx-zynqmp.c
48
@@ -XXX,XX +XXX,XX @@
49
#include "qemu/module.h"
50
#include "hw/arm/xlnx-zynqmp.h"
51
#include "hw/intc/arm_gic_common.h"
52
+#include "hw/misc/unimp.h"
53
#include "hw/boards.h"
54
#include "sysemu/kvm.h"
55
#include "sysemu/sysemu.h"
56
@@ -XXX,XX +XXX,XX @@
57
#define DPDMA_ADDR 0xfd4c0000
58
#define DPDMA_IRQ 116
59
60
+#define APU_ADDR 0xfd5c0000
61
+#define APU_SIZE 0x100
62
+
63
#define IPI_ADDR 0xFF300000
64
#define IPI_IRQ 64
65
66
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_create_rpu(MachineState *ms, XlnxZynqMPState *s,
67
qdev_realize(DEVICE(&s->rpu_cluster), NULL, &error_fatal);
68
}
69
70
+static void xlnx_zynqmp_create_unimp_mmio(XlnxZynqMPState *s)
71
+{
72
+ static const struct UnimpInfo {
73
+ const char *name;
74
+ hwaddr base;
75
+ hwaddr size;
76
+ } unimp_areas[ARRAY_SIZE(s->mr_unimp)] = {
77
+ { .name = "apu", APU_ADDR, APU_SIZE },
78
+ };
79
+ unsigned int nr;
80
+
81
+ for (nr = 0; nr < ARRAY_SIZE(unimp_areas); nr++) {
82
+ const struct UnimpInfo *info = &unimp_areas[nr];
83
+ DeviceState *dev = qdev_new(TYPE_UNIMPLEMENTED_DEVICE);
84
+ SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
85
+
86
+ assert(info->name && info->base && info->size > 0);
87
+ qdev_prop_set_string(dev, "name", info->name);
88
+ qdev_prop_set_uint64(dev, "size", info->size);
89
+ object_property_add_child(OBJECT(s), info->name, OBJECT(dev));
90
+
91
+ sysbus_realize_and_unref(sbd, &error_fatal);
92
+ sysbus_mmio_map(sbd, 0, info->base);
93
+ }
94
+}
95
+
96
static void xlnx_zynqmp_init(Object *obj)
97
{
98
MachineState *ms = MACHINE(qdev_get_machine());
40
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
99
@@ -XXX,XX +XXX,XX @@ static void xlnx_zynqmp_realize(DeviceState *dev, Error **errp)
41
}
100
sysbus_mmio_map(SYS_BUS_DEVICE(&s->rtc), 0, RTC_ADDR);
42
}
101
sysbus_connect_irq(SYS_BUS_DEVICE(&s->rtc), 0, gic_spi[RTC_IRQ]);
43
102
44
- if (s->has_rpu) {
103
+ xlnx_zynqmp_create_unimp_mmio(s);
45
- info_report("The 'has_rpu' property is no longer required, to use the "
104
+
46
- "RPUs just use -smp 6.");
105
for (i = 0; i < XLNX_ZYNQMP_NUM_GDMA_CH; i++) {
47
- }
106
if (!object_property_set_uint(OBJECT(&s->gdma[i]), "bus-width", 128,
48
-
107
errp)) {
49
xlnx_zynqmp_create_rpu(ms, s, boot_cpu, &err);
50
if (err) {
51
error_propagate(errp, err);
52
@@ -XXX,XX +XXX,XX @@ static Property xlnx_zynqmp_props[] = {
53
DEFINE_PROP_STRING("boot-cpu", XlnxZynqMPState, boot_cpu),
54
DEFINE_PROP_BOOL("secure", XlnxZynqMPState, secure, false),
55
DEFINE_PROP_BOOL("virtualization", XlnxZynqMPState, virt, false),
56
- DEFINE_PROP_BOOL("has_rpu", XlnxZynqMPState, has_rpu, false),
57
DEFINE_PROP_LINK("ddr-ram", XlnxZynqMPState, ddr_ram, TYPE_MEMORY_REGION,
58
MemoryRegion *),
59
DEFINE_PROP_LINK("canbus0", XlnxZynqMPState, canbus[0], TYPE_CAN_BUS,
60
--
108
--
61
2.20.1
109
2.20.1
62
110
63
111
diff view generated by jsdifflib
Deleted patch
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
1
3
The STATUS register will be reset to IDLE in
4
cnpcm7xx_smbus_enter_reset(), no need to preset
5
it in instance_init().
6
7
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Hao Wu <wuhaotsh@google.com>
9
Message-id: 20210228224813.312532-1-f4bug@amsat.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
hw/i2c/npcm7xx_smbus.c | 1 -
13
1 file changed, 1 deletion(-)
14
15
diff --git a/hw/i2c/npcm7xx_smbus.c b/hw/i2c/npcm7xx_smbus.c
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/i2c/npcm7xx_smbus.c
18
+++ b/hw/i2c/npcm7xx_smbus.c
19
@@ -XXX,XX +XXX,XX @@ static void npcm7xx_smbus_init(Object *obj)
20
sysbus_init_mmio(sbd, &s->iomem);
21
22
s->bus = i2c_init_bus(DEVICE(s), "i2c-bus");
23
- s->status = NPCM7XX_SMBUS_STATUS_IDLE;
24
}
25
26
static const VMStateDescription vmstate_npcm7xx_smbus = {
27
--
28
2.20.1
29
30
diff view generated by jsdifflib
Deleted patch
1
From: schspa <schspa@gmail.com>
2
1
3
At the moment the following QEMU command line triggers an assertion
4
failure On xlnx-versal SOC:
5
qemu-system-aarch64 \
6
-machine xlnx-versal-virt -nographic -smp 2 -m 128 \
7
-fsdev local,id=shareid,path=${HOME}/work,security_model=none \
8
-device virtio-9p-device,fsdev=shareid,mount_tag=share \
9
-fsdev local,id=shareid1,path=${HOME}/Music,security_model=none \
10
-device virtio-9p-device,fsdev=shareid1,mount_tag=share1
11
12
qemu-system-aarch64: ../migration/savevm.c:860:
13
vmstate_register_with_alias_id:
14
Assertion `!se->compat || se->instance_id == 0' failed.
15
16
This problem was fixed on arm virt platform in commit f58b39d2d5b
17
("virtio-mmio: format transport base address in BusClass.get_dev_path")
18
19
It works perfectly on arm virt platform. but there is still there on
20
xlnx-versal SOC.
21
22
The main difference between arm virt and xlnx-versal is they use
23
different way to create virtio-mmio qdev. on arm virt, it calls
24
sysbus_create_simple("virtio-mmio", base, pic[irq]); which will call
25
sysbus_mmio_map internally and assign base address to subsys device
26
mmio correctly. but xlnx-versal's implements won't do this.
27
28
However, xlnx-versal can't switch to sysbus_create_simple() to create
29
virtio-mmio device. It's because xlnx-versal's cpu use
30
VersalVirt.soc.fpd.apu.mr as it's memory. which is subregion of
31
system_memory. sysbus_create_simple will add virtio to system_memory,
32
which can't be accessed by cpu.
33
34
Besides, xlnx-versal can't add sysbus_mmio_map api call too, because
35
this will add memory region to system_memory, and it can't be added
36
to VersalVirt.soc.fpd.apu.mr again.
37
38
We can solve this by assign correct base address offset on dev_path.
39
40
This path was test on aarch64 virt & xlnx-versal platform.
41
42
Signed-off-by: schspa <schspa@gmail.com>
43
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
44
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
45
---
46
hw/virtio/virtio-mmio.c | 13 +++++++------
47
1 file changed, 7 insertions(+), 6 deletions(-)
48
49
diff --git a/hw/virtio/virtio-mmio.c b/hw/virtio/virtio-mmio.c
50
index XXXXXXX..XXXXXXX 100644
51
--- a/hw/virtio/virtio-mmio.c
52
+++ b/hw/virtio/virtio-mmio.c
53
@@ -XXX,XX +XXX,XX @@ static char *virtio_mmio_bus_get_dev_path(DeviceState *dev)
54
BusState *virtio_mmio_bus;
55
VirtIOMMIOProxy *virtio_mmio_proxy;
56
char *proxy_path;
57
- SysBusDevice *proxy_sbd;
58
char *path;
59
+ MemoryRegionSection section;
60
61
virtio_mmio_bus = qdev_get_parent_bus(dev);
62
virtio_mmio_proxy = VIRTIO_MMIO(virtio_mmio_bus->parent);
63
@@ -XXX,XX +XXX,XX @@ static char *virtio_mmio_bus_get_dev_path(DeviceState *dev)
64
}
65
66
/* Otherwise, we append the base address of the transport. */
67
- proxy_sbd = SYS_BUS_DEVICE(virtio_mmio_proxy);
68
- assert(proxy_sbd->num_mmio == 1);
69
- assert(proxy_sbd->mmio[0].memory == &virtio_mmio_proxy->iomem);
70
+ section = memory_region_find(&virtio_mmio_proxy->iomem, 0, 0x200);
71
+ assert(section.mr);
72
73
if (proxy_path) {
74
path = g_strdup_printf("%s/virtio-mmio@" TARGET_FMT_plx, proxy_path,
75
- proxy_sbd->mmio[0].addr);
76
+ section.offset_within_address_space);
77
} else {
78
path = g_strdup_printf("virtio-mmio@" TARGET_FMT_plx,
79
- proxy_sbd->mmio[0].addr);
80
+ section.offset_within_address_space);
81
}
82
+ memory_region_unref(section.mr);
83
+
84
g_free(proxy_path);
85
return path;
86
}
87
--
88
2.20.1
89
90
diff view generated by jsdifflib
Deleted patch
1
For a long time now the UI layer has guaranteed that the console
2
surface is always 32 bits per pixel RGB. Remove the legacy dead
3
code from the milkymist display device which was handling the
4
possibility that the console surface was some other format.
5
1
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210215103215.4944-2-peter.maydell@linaro.org
9
---
10
hw/arm/musicpal.c | 64 ++++++++++++++++++-----------------------------
11
1 file changed, 24 insertions(+), 40 deletions(-)
12
13
diff --git a/hw/arm/musicpal.c b/hw/arm/musicpal.c
14
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/musicpal.c
16
+++ b/hw/arm/musicpal.c
17
@@ -XXX,XX +XXX,XX @@ static uint8_t scale_lcd_color(musicpal_lcd_state *s, uint8_t col)
18
}
19
}
20
21
-#define SET_LCD_PIXEL(depth, type) \
22
-static inline void glue(set_lcd_pixel, depth) \
23
- (musicpal_lcd_state *s, int x, int y, type col) \
24
-{ \
25
- int dx, dy; \
26
- DisplaySurface *surface = qemu_console_surface(s->con); \
27
- type *pixel = &((type *) surface_data(surface))[(y * 128 * 3 + x) * 3]; \
28
-\
29
- for (dy = 0; dy < 3; dy++, pixel += 127 * 3) \
30
- for (dx = 0; dx < 3; dx++, pixel++) \
31
- *pixel = col; \
32
+static inline void set_lcd_pixel32(musicpal_lcd_state *s,
33
+ int x, int y, uint32_t col)
34
+{
35
+ int dx, dy;
36
+ DisplaySurface *surface = qemu_console_surface(s->con);
37
+ uint32_t *pixel =
38
+ &((uint32_t *) surface_data(surface))[(y * 128 * 3 + x) * 3];
39
+
40
+ for (dy = 0; dy < 3; dy++, pixel += 127 * 3) {
41
+ for (dx = 0; dx < 3; dx++, pixel++) {
42
+ *pixel = col;
43
+ }
44
+ }
45
}
46
-SET_LCD_PIXEL(8, uint8_t)
47
-SET_LCD_PIXEL(16, uint16_t)
48
-SET_LCD_PIXEL(32, uint32_t)
49
50
static void lcd_refresh(void *opaque)
51
{
52
musicpal_lcd_state *s = opaque;
53
- DisplaySurface *surface = qemu_console_surface(s->con);
54
int x, y, col;
55
56
- switch (surface_bits_per_pixel(surface)) {
57
- case 0:
58
- return;
59
-#define LCD_REFRESH(depth, func) \
60
- case depth: \
61
- col = func(scale_lcd_color(s, (MP_LCD_TEXTCOLOR >> 16) & 0xff), \
62
- scale_lcd_color(s, (MP_LCD_TEXTCOLOR >> 8) & 0xff), \
63
- scale_lcd_color(s, MP_LCD_TEXTCOLOR & 0xff)); \
64
- for (x = 0; x < 128; x++) { \
65
- for (y = 0; y < 64; y++) { \
66
- if (s->video_ram[x + (y/8)*128] & (1 << (y % 8))) { \
67
- glue(set_lcd_pixel, depth)(s, x, y, col); \
68
- } else { \
69
- glue(set_lcd_pixel, depth)(s, x, y, 0); \
70
- } \
71
- } \
72
- } \
73
- break;
74
- LCD_REFRESH(8, rgb_to_pixel8)
75
- LCD_REFRESH(16, rgb_to_pixel16)
76
- LCD_REFRESH(32, (is_surface_bgr(surface) ?
77
- rgb_to_pixel32bgr : rgb_to_pixel32))
78
- default:
79
- hw_error("unsupported colour depth %i\n",
80
- surface_bits_per_pixel(surface));
81
+ col = rgb_to_pixel32(scale_lcd_color(s, (MP_LCD_TEXTCOLOR >> 16) & 0xff),
82
+ scale_lcd_color(s, (MP_LCD_TEXTCOLOR >> 8) & 0xff),
83
+ scale_lcd_color(s, MP_LCD_TEXTCOLOR & 0xff));
84
+ for (x = 0; x < 128; x++) {
85
+ for (y = 0; y < 64; y++) {
86
+ if (s->video_ram[x + (y / 8) * 128] & (1 << (y % 8))) {
87
+ set_lcd_pixel32(s, x, y, col);
88
+ } else {
89
+ set_lcd_pixel32(s, x, y, 0);
90
+ }
91
+ }
92
}
93
94
dpy_gfx_update(s->con, 0, 0, 128*3, 64*3);
95
--
96
2.20.1
97
98
diff view generated by jsdifflib
Deleted patch
1
The draw_line16_32() function in the omap_lcdc template header
2
includes an ifdef for the case where HOST_WORDS_BIGENDIAN matches
3
TARGET_WORDS_BIGENDIAN. This is trying to optimise for "source
4
bitmap and destination bitmap format match", but it is broken,
5
because in this function the formats don't match: the source is
6
16-bit colour and the destination is 32-bit colour, so a memcpy()
7
will produce corrupted graphics output. Drop the bogus ifdef.
8
1
9
This bug was introduced in commit ea644cf343129, when we dropped
10
support for DEPTH values other than 32 from the template header.
11
The old #if line was
12
#if DEPTH == 16 && defined(HOST_WORDS_BIGENDIAN) == defined(TARGET_WORDS_BIGENDIAN)
13
and this was mistakenly changed to
14
#if defined(HOST_WORDS_BIGENDIAN) == defined(TARGET_WORDS_BIGENDIAN)
15
rather than deleting the #if as now having an always-false condition.
16
17
Fixes: ea644cf343129
18
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
19
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
20
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
21
Message-id: 20210215103215.4944-7-peter.maydell@linaro.org
22
---
23
hw/display/omap_lcd_template.h | 4 ----
24
1 file changed, 4 deletions(-)
25
26
diff --git a/hw/display/omap_lcd_template.h b/hw/display/omap_lcd_template.h
27
index XXXXXXX..XXXXXXX 100644
28
--- a/hw/display/omap_lcd_template.h
29
+++ b/hw/display/omap_lcd_template.h
30
@@ -XXX,XX +XXX,XX @@ static void draw_line12_32(void *opaque, uint8_t *d, const uint8_t *s,
31
static void draw_line16_32(void *opaque, uint8_t *d, const uint8_t *s,
32
int width, int deststep)
33
{
34
-#if defined(HOST_WORDS_BIGENDIAN) == defined(TARGET_WORDS_BIGENDIAN)
35
- memcpy(d, s, width * 2);
36
-#else
37
uint16_t v;
38
uint8_t r, g, b;
39
40
@@ -XXX,XX +XXX,XX @@ static void draw_line16_32(void *opaque, uint8_t *d, const uint8_t *s,
41
s += 2;
42
d += 4;
43
} while (-- width != 0);
44
-#endif
45
}
46
--
47
2.20.1
48
49
diff view generated by jsdifflib
Deleted patch
1
For a long time now the UI layer has guaranteed that the console
2
surface is always 32 bits per pixel, RGB. The TCX code already
3
assumes 32bpp, but it still has some checks of is_surface_bgr()
4
in an attempt to support 32bpp BGR. is_surface_bgr() will always
5
return false for the qemu_console_surface(), unless the display
6
device itself has deliberately created an alternate-format
7
surface via a function like qemu_create_displaysurface_from().
8
1
9
Drop the never-used BGR-handling code, and assert that we have
10
a 32-bit surface rather than just doing nothing if it isn't.
11
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Reviewed-by: Mark Cave-Ayland <mark.cave-ayland@ilande.co.uk>
14
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
15
Message-id: 20210215102149.20513-1-peter.maydell@linaro.org
16
---
17
hw/display/tcx.c | 31 ++++++++-----------------------
18
1 file changed, 8 insertions(+), 23 deletions(-)
19
20
diff --git a/hw/display/tcx.c b/hw/display/tcx.c
21
index XXXXXXX..XXXXXXX 100644
22
--- a/hw/display/tcx.c
23
+++ b/hw/display/tcx.c
24
@@ -XXX,XX +XXX,XX @@ static int tcx_check_dirty(TCXState *s, DirtyBitmapSnapshot *snap,
25
26
static void update_palette_entries(TCXState *s, int start, int end)
27
{
28
- DisplaySurface *surface = qemu_console_surface(s->con);
29
int i;
30
31
for (i = start; i < end; i++) {
32
- if (is_surface_bgr(surface)) {
33
- s->palette[i] = rgb_to_pixel32bgr(s->r[i], s->g[i], s->b[i]);
34
- } else {
35
- s->palette[i] = rgb_to_pixel32(s->r[i], s->g[i], s->b[i]);
36
- }
37
+ s->palette[i] = rgb_to_pixel32(s->r[i], s->g[i], s->b[i]);
38
}
39
tcx_set_dirty(s, 0, memory_region_size(&s->vram_mem));
40
}
41
@@ -XXX,XX +XXX,XX @@ static void tcx_draw_cursor32(TCXState *s1, uint8_t *d,
42
}
43
44
/*
45
- XXX Could be much more optimal:
46
- * detect if line/page/whole screen is in 24 bit mode
47
- * if destination is also BGR, use memcpy
48
- */
49
+ * XXX Could be much more optimal:
50
+ * detect if line/page/whole screen is in 24 bit mode
51
+ */
52
static inline void tcx24_draw_line32(TCXState *s1, uint8_t *d,
53
const uint8_t *s, int width,
54
const uint32_t *cplane,
55
const uint32_t *s24)
56
{
57
- DisplaySurface *surface = qemu_console_surface(s1->con);
58
- int x, bgr, r, g, b;
59
+ int x, r, g, b;
60
uint8_t val, *p8;
61
uint32_t *p = (uint32_t *)d;
62
uint32_t dval;
63
- bgr = is_surface_bgr(surface);
64
for(x = 0; x < width; x++, s++, s24++) {
65
if (be32_to_cpu(*cplane) & 0x03000000) {
66
/* 24-bit direct, BGR order */
67
@@ -XXX,XX +XXX,XX @@ static inline void tcx24_draw_line32(TCXState *s1, uint8_t *d,
68
b = *p8++;
69
g = *p8++;
70
r = *p8;
71
- if (bgr)
72
- dval = rgb_to_pixel32bgr(r, g, b);
73
- else
74
- dval = rgb_to_pixel32(r, g, b);
75
+ dval = rgb_to_pixel32(r, g, b);
76
} else {
77
/* 8-bit pseudocolor */
78
val = *s;
79
@@ -XXX,XX +XXX,XX @@ static void tcx_update_display(void *opaque)
80
int y, y_start, dd, ds;
81
uint8_t *d, *s;
82
83
- if (surface_bits_per_pixel(surface) != 32) {
84
- return;
85
- }
86
+ assert(surface_bits_per_pixel(surface) == 32);
87
88
page = 0;
89
y_start = -1;
90
@@ -XXX,XX +XXX,XX @@ static void tcx24_update_display(void *opaque)
91
uint8_t *d, *s;
92
uint32_t *cptr, *s24;
93
94
- if (surface_bits_per_pixel(surface) != 32) {
95
- return;
96
- }
97
+ assert(surface_bits_per_pixel(surface) == 32);
98
99
page = 0;
100
y_start = -1;
101
--
102
2.20.1
103
104
diff view generated by jsdifflib
Deleted patch
1
The AN524 has a different SYSCLK frequency from the AN505 and AN521;
2
make the SYSCLK frequency a field in the MPS2TZMachineClass rather
3
than a compile-time constant so we can support the AN524.
4
1
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210215115138.20465-2-peter.maydell@linaro.org
9
---
10
hw/arm/mps2-tz.c | 10 ++++++----
11
1 file changed, 6 insertions(+), 4 deletions(-)
12
13
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
14
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/mps2-tz.c
16
+++ b/hw/arm/mps2-tz.c
17
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineClass {
18
MachineClass parent;
19
MPS2TZFPGAType fpga_type;
20
uint32_t scc_id;
21
+ uint32_t sysclk_frq; /* Main SYSCLK frequency in Hz */
22
const char *armsse_type;
23
};
24
25
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineState {
26
27
OBJECT_DECLARE_TYPE(MPS2TZMachineState, MPS2TZMachineClass, MPS2TZ_MACHINE)
28
29
-/* Main SYSCLK frequency in Hz */
30
-#define SYSCLK_FRQ 20000000
31
/* Slow 32Khz S32KCLK frequency in Hz */
32
#define S32KCLK_FRQ (32 * 1000)
33
34
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_unimp_dev(MPS2TZMachineState *mms,
35
static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
36
const char *name, hwaddr size)
37
{
38
+ MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
39
CMSDKAPBUART *uart = opaque;
40
int i = uart - &mms->uart[0];
41
int rxirqno = i * 2;
42
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_uart(MPS2TZMachineState *mms, void *opaque,
43
44
object_initialize_child(OBJECT(mms), name, uart, TYPE_CMSDK_APB_UART);
45
qdev_prop_set_chr(DEVICE(uart), "chardev", serial_hd(i));
46
- qdev_prop_set_uint32(DEVICE(uart), "pclk-frq", SYSCLK_FRQ);
47
+ qdev_prop_set_uint32(DEVICE(uart), "pclk-frq", mmc->sysclk_frq);
48
sysbus_realize(SYS_BUS_DEVICE(uart), &error_fatal);
49
s = SYS_BUS_DEVICE(uart);
50
sysbus_connect_irq(s, 0, get_sse_irq_in(mms, txirqno));
51
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
52
53
/* These clocks don't need migration because they are fixed-frequency */
54
mms->sysclk = clock_new(OBJECT(machine), "SYSCLK");
55
- clock_set_hz(mms->sysclk, SYSCLK_FRQ);
56
+ clock_set_hz(mms->sysclk, mmc->sysclk_frq);
57
mms->s32kclk = clock_new(OBJECT(machine), "S32KCLK");
58
clock_set_hz(mms->s32kclk, S32KCLK_FRQ);
59
60
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
61
mmc->fpga_type = FPGA_AN505;
62
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m33");
63
mmc->scc_id = 0x41045050;
64
+ mmc->sysclk_frq = 20 * 1000 * 1000; /* 20MHz */
65
mmc->armsse_type = TYPE_IOTKIT;
66
}
67
68
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
69
mmc->fpga_type = FPGA_AN521;
70
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m33");
71
mmc->scc_id = 0x41045210;
72
+ mmc->sysclk_frq = 20 * 1000 * 1000; /* 20MHz */
73
mmc->armsse_type = TYPE_SSE200;
74
}
75
76
--
77
2.20.1
78
79
diff view generated by jsdifflib
Deleted patch
1
Currently the MPS2 SCC device implements a fixed number of OSCCLK
2
values (3). The variant of this device in the MPS3 AN524 board has 6
3
OSCCLK values. Switch to using a PROP_ARRAY, which allows board code
4
to specify how large the OSCCLK array should be as well as its
5
values.
6
1
7
With a variable-length property array, the SCC no longer specifies
8
default values for the OSCCLKs, so we must set them explicitly in the
9
board code. This defaults are actually incorrect for the an521 and
10
an505; we will correct this bug in a following patch.
11
12
This is a migration compatibility break for all the mps boards.
13
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
16
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
17
Message-id: 20210215115138.20465-3-peter.maydell@linaro.org
18
---
19
include/hw/misc/mps2-scc.h | 7 +++----
20
hw/arm/mps2-tz.c | 5 +++++
21
hw/arm/mps2.c | 5 +++++
22
hw/misc/mps2-scc.c | 24 +++++++++++++-----------
23
4 files changed, 26 insertions(+), 15 deletions(-)
24
25
diff --git a/include/hw/misc/mps2-scc.h b/include/hw/misc/mps2-scc.h
26
index XXXXXXX..XXXXXXX 100644
27
--- a/include/hw/misc/mps2-scc.h
28
+++ b/include/hw/misc/mps2-scc.h
29
@@ -XXX,XX +XXX,XX @@
30
#define TYPE_MPS2_SCC "mps2-scc"
31
OBJECT_DECLARE_SIMPLE_TYPE(MPS2SCC, MPS2_SCC)
32
33
-#define NUM_OSCCLK 3
34
-
35
struct MPS2SCC {
36
/*< private >*/
37
SysBusDevice parent_obj;
38
@@ -XXX,XX +XXX,XX @@ struct MPS2SCC {
39
uint32_t dll;
40
uint32_t aid;
41
uint32_t id;
42
- uint32_t oscclk[NUM_OSCCLK];
43
- uint32_t oscclk_reset[NUM_OSCCLK];
44
+ uint32_t num_oscclk;
45
+ uint32_t *oscclk;
46
+ uint32_t *oscclk_reset;
47
};
48
49
#endif
50
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
51
index XXXXXXX..XXXXXXX 100644
52
--- a/hw/arm/mps2-tz.c
53
+++ b/hw/arm/mps2-tz.c
54
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_scc(MPS2TZMachineState *mms, void *opaque,
55
qdev_prop_set_uint32(sccdev, "scc-cfg4", 0x2);
56
qdev_prop_set_uint32(sccdev, "scc-aid", 0x00200008);
57
qdev_prop_set_uint32(sccdev, "scc-id", mmc->scc_id);
58
+ /* This will need to be per-FPGA image eventually */
59
+ qdev_prop_set_uint32(sccdev, "len-oscclk", 3);
60
+ qdev_prop_set_uint32(sccdev, "oscclk[0]", 50000000);
61
+ qdev_prop_set_uint32(sccdev, "oscclk[1]", 24576000);
62
+ qdev_prop_set_uint32(sccdev, "oscclk[2]", 25000000);
63
sysbus_realize(SYS_BUS_DEVICE(scc), &error_fatal);
64
return sysbus_mmio_get_region(SYS_BUS_DEVICE(sccdev), 0);
65
}
66
diff --git a/hw/arm/mps2.c b/hw/arm/mps2.c
67
index XXXXXXX..XXXXXXX 100644
68
--- a/hw/arm/mps2.c
69
+++ b/hw/arm/mps2.c
70
@@ -XXX,XX +XXX,XX @@ static void mps2_common_init(MachineState *machine)
71
qdev_prop_set_uint32(sccdev, "scc-cfg4", 0x2);
72
qdev_prop_set_uint32(sccdev, "scc-aid", 0x00200008);
73
qdev_prop_set_uint32(sccdev, "scc-id", mmc->scc_id);
74
+ /* All these FPGA images have the same OSCCLK configuration */
75
+ qdev_prop_set_uint32(sccdev, "len-oscclk", 3);
76
+ qdev_prop_set_uint32(sccdev, "oscclk[0]", 50000000);
77
+ qdev_prop_set_uint32(sccdev, "oscclk[1]", 24576000);
78
+ qdev_prop_set_uint32(sccdev, "oscclk[2]", 25000000);
79
sysbus_realize(SYS_BUS_DEVICE(&mms->scc), &error_fatal);
80
sysbus_mmio_map(SYS_BUS_DEVICE(sccdev), 0, 0x4002f000);
81
object_initialize_child(OBJECT(mms), "fpgaio",
82
diff --git a/hw/misc/mps2-scc.c b/hw/misc/mps2-scc.c
83
index XXXXXXX..XXXXXXX 100644
84
--- a/hw/misc/mps2-scc.c
85
+++ b/hw/misc/mps2-scc.c
86
@@ -XXX,XX +XXX,XX @@ static bool scc_cfg_write(MPS2SCC *s, unsigned function,
87
{
88
trace_mps2_scc_cfg_write(function, device, value);
89
90
- if (function != 1 || device >= NUM_OSCCLK) {
91
+ if (function != 1 || device >= s->num_oscclk) {
92
qemu_log_mask(LOG_GUEST_ERROR,
93
"MPS2 SCC config write: bad function %d device %d\n",
94
function, device);
95
@@ -XXX,XX +XXX,XX @@ static bool scc_cfg_write(MPS2SCC *s, unsigned function,
96
static bool scc_cfg_read(MPS2SCC *s, unsigned function,
97
unsigned device, uint32_t *value)
98
{
99
- if (function != 1 || device >= NUM_OSCCLK) {
100
+ if (function != 1 || device >= s->num_oscclk) {
101
qemu_log_mask(LOG_GUEST_ERROR,
102
"MPS2 SCC config read: bad function %d device %d\n",
103
function, device);
104
@@ -XXX,XX +XXX,XX @@ static void mps2_scc_reset(DeviceState *dev)
105
s->cfgctrl = 0x100000;
106
s->cfgstat = 0;
107
s->dll = 0xffff0001;
108
- for (i = 0; i < NUM_OSCCLK; i++) {
109
+ for (i = 0; i < s->num_oscclk; i++) {
110
s->oscclk[i] = s->oscclk_reset[i];
111
}
112
for (i = 0; i < ARRAY_SIZE(s->led); i++) {
113
@@ -XXX,XX +XXX,XX @@ static void mps2_scc_realize(DeviceState *dev, Error **errp)
114
LED_COLOR_GREEN, name);
115
g_free(name);
116
}
117
+
118
+ s->oscclk = g_new0(uint32_t, s->num_oscclk);
119
}
120
121
static const VMStateDescription mps2_scc_vmstate = {
122
.name = "mps2-scc",
123
- .version_id = 1,
124
- .minimum_version_id = 1,
125
+ .version_id = 2,
126
+ .minimum_version_id = 2,
127
.fields = (VMStateField[]) {
128
VMSTATE_UINT32(cfg0, MPS2SCC),
129
VMSTATE_UINT32(cfg1, MPS2SCC),
130
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription mps2_scc_vmstate = {
131
VMSTATE_UINT32(cfgctrl, MPS2SCC),
132
VMSTATE_UINT32(cfgstat, MPS2SCC),
133
VMSTATE_UINT32(dll, MPS2SCC),
134
- VMSTATE_UINT32_ARRAY(oscclk, MPS2SCC, NUM_OSCCLK),
135
+ VMSTATE_VARRAY_UINT32(oscclk, MPS2SCC, num_oscclk,
136
+ 0, vmstate_info_uint32, uint32_t),
137
VMSTATE_END_OF_LIST()
138
}
139
};
140
@@ -XXX,XX +XXX,XX @@ static Property mps2_scc_properties[] = {
141
DEFINE_PROP_UINT32("scc-cfg4", MPS2SCC, cfg4, 0),
142
DEFINE_PROP_UINT32("scc-aid", MPS2SCC, aid, 0),
143
DEFINE_PROP_UINT32("scc-id", MPS2SCC, id, 0),
144
- /* These are the initial settings for the source clocks on the board.
145
+ /*
146
+ * These are the initial settings for the source clocks on the board.
147
* In hardware they can be configured via a config file read by the
148
* motherboard configuration controller to suit the FPGA image.
149
- * These default values are used by most of the standard FPGA images.
150
*/
151
- DEFINE_PROP_UINT32("oscclk0", MPS2SCC, oscclk_reset[0], 50000000),
152
- DEFINE_PROP_UINT32("oscclk1", MPS2SCC, oscclk_reset[1], 24576000),
153
- DEFINE_PROP_UINT32("oscclk2", MPS2SCC, oscclk_reset[2], 25000000),
154
+ DEFINE_PROP_ARRAY("oscclk", MPS2SCC, num_oscclk, oscclk_reset,
155
+ qdev_prop_uint32, uint32_t),
156
DEFINE_PROP_END_OF_LIST(),
157
};
158
159
--
160
2.20.1
161
162
diff view generated by jsdifflib
Deleted patch
1
The AN505 and AN511 happen to share the same OSCCLK values, but the
2
AN524 will have a different set (and more of them), so split the
3
settings out to be per-board.
4
1
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210215115138.20465-5-peter.maydell@linaro.org
9
---
10
hw/arm/mps2-tz.c | 23 ++++++++++++++++++-----
11
1 file changed, 18 insertions(+), 5 deletions(-)
12
13
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
14
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/mps2-tz.c
16
+++ b/hw/arm/mps2-tz.c
17
@@ -XXX,XX +XXX,XX @@ struct MPS2TZMachineClass {
18
MPS2TZFPGAType fpga_type;
19
uint32_t scc_id;
20
uint32_t sysclk_frq; /* Main SYSCLK frequency in Hz */
21
+ uint32_t len_oscclk;
22
+ const uint32_t *oscclk;
23
const char *armsse_type;
24
};
25
26
@@ -XXX,XX +XXX,XX @@ OBJECT_DECLARE_TYPE(MPS2TZMachineState, MPS2TZMachineClass, MPS2TZ_MACHINE)
27
/* Slow 32Khz S32KCLK frequency in Hz */
28
#define S32KCLK_FRQ (32 * 1000)
29
30
+static const uint32_t an505_oscclk[] = {
31
+ 40000000,
32
+ 24580000,
33
+ 25000000,
34
+};
35
+
36
/* Create an alias of an entire original MemoryRegion @orig
37
* located at @base in the memory map.
38
*/
39
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_scc(MPS2TZMachineState *mms, void *opaque,
40
MPS2SCC *scc = opaque;
41
DeviceState *sccdev;
42
MPS2TZMachineClass *mmc = MPS2TZ_MACHINE_GET_CLASS(mms);
43
+ uint32_t i;
44
45
object_initialize_child(OBJECT(mms), "scc", scc, TYPE_MPS2_SCC);
46
sccdev = DEVICE(scc);
47
qdev_prop_set_uint32(sccdev, "scc-cfg4", 0x2);
48
qdev_prop_set_uint32(sccdev, "scc-aid", 0x00200008);
49
qdev_prop_set_uint32(sccdev, "scc-id", mmc->scc_id);
50
- /* This will need to be per-FPGA image eventually */
51
- qdev_prop_set_uint32(sccdev, "len-oscclk", 3);
52
- qdev_prop_set_uint32(sccdev, "oscclk[0]", 40000000);
53
- qdev_prop_set_uint32(sccdev, "oscclk[1]", 24580000);
54
- qdev_prop_set_uint32(sccdev, "oscclk[2]", 25000000);
55
+ qdev_prop_set_uint32(sccdev, "len-oscclk", mmc->len_oscclk);
56
+ for (i = 0; i < mmc->len_oscclk; i++) {
57
+ g_autofree char *propname = g_strdup_printf("oscclk[%u]", i);
58
+ qdev_prop_set_uint32(sccdev, propname, mmc->oscclk[i]);
59
+ }
60
sysbus_realize(SYS_BUS_DEVICE(scc), &error_fatal);
61
return sysbus_mmio_get_region(SYS_BUS_DEVICE(sccdev), 0);
62
}
63
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
64
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m33");
65
mmc->scc_id = 0x41045050;
66
mmc->sysclk_frq = 20 * 1000 * 1000; /* 20MHz */
67
+ mmc->oscclk = an505_oscclk;
68
+ mmc->len_oscclk = ARRAY_SIZE(an505_oscclk);
69
mmc->armsse_type = TYPE_IOTKIT;
70
}
71
72
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an521_class_init(ObjectClass *oc, void *data)
73
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m33");
74
mmc->scc_id = 0x41045210;
75
mmc->sysclk_frq = 20 * 1000 * 1000; /* 20MHz */
76
+ mmc->oscclk = an505_oscclk; /* AN521 is the same as AN505 here */
77
+ mmc->len_oscclk = ARRAY_SIZE(an505_oscclk);
78
mmc->armsse_type = TYPE_SSE200;
79
}
80
81
--
82
2.20.1
83
84
diff view generated by jsdifflib
Deleted patch
1
MPS3 boards have an extra SWITCH register in the FPGAIO block which
2
reports the value of some switches. Implement this, governed by a
3
property the board code can use to specify whether whether it exists.
4
1
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20210215115138.20465-7-peter.maydell@linaro.org
9
---
10
include/hw/misc/mps2-fpgaio.h | 1 +
11
hw/misc/mps2-fpgaio.c | 10 ++++++++++
12
2 files changed, 11 insertions(+)
13
14
diff --git a/include/hw/misc/mps2-fpgaio.h b/include/hw/misc/mps2-fpgaio.h
15
index XXXXXXX..XXXXXXX 100644
16
--- a/include/hw/misc/mps2-fpgaio.h
17
+++ b/include/hw/misc/mps2-fpgaio.h
18
@@ -XXX,XX +XXX,XX @@ struct MPS2FPGAIO {
19
MemoryRegion iomem;
20
LEDState *led[MPS2FPGAIO_MAX_LEDS];
21
uint32_t num_leds;
22
+ bool has_switches;
23
24
uint32_t led0;
25
uint32_t prescale;
26
diff --git a/hw/misc/mps2-fpgaio.c b/hw/misc/mps2-fpgaio.c
27
index XXXXXXX..XXXXXXX 100644
28
--- a/hw/misc/mps2-fpgaio.c
29
+++ b/hw/misc/mps2-fpgaio.c
30
@@ -XXX,XX +XXX,XX @@ REG32(CLK100HZ, 0x14)
31
REG32(COUNTER, 0x18)
32
REG32(PRESCALE, 0x1c)
33
REG32(PSCNTR, 0x20)
34
+REG32(SWITCH, 0x28)
35
REG32(MISC, 0x4c)
36
37
static uint32_t counter_from_tickoff(int64_t now, int64_t tick_offset, int frq)
38
@@ -XXX,XX +XXX,XX @@ static uint64_t mps2_fpgaio_read(void *opaque, hwaddr offset, unsigned size)
39
resync_counter(s);
40
r = s->pscntr;
41
break;
42
+ case A_SWITCH:
43
+ if (!s->has_switches) {
44
+ goto bad_offset;
45
+ }
46
+ /* User-togglable board switches. We don't model that, so report 0. */
47
+ r = 0;
48
+ break;
49
default:
50
+ bad_offset:
51
qemu_log_mask(LOG_GUEST_ERROR,
52
"MPS2 FPGAIO read: bad offset %x\n", (int) offset);
53
r = 0;
54
@@ -XXX,XX +XXX,XX @@ static Property mps2_fpgaio_properties[] = {
55
DEFINE_PROP_UINT32("prescale-clk", MPS2FPGAIO, prescale_clk, 20000000),
56
/* Number of LEDs controlled by LED0 register */
57
DEFINE_PROP_UINT32("num-leds", MPS2FPGAIO, num_leds, 2),
58
+ DEFINE_PROP_BOOL("has-switches", MPS2FPGAIO, has_switches, false),
59
DEFINE_PROP_END_OF_LIST(),
60
};
61
62
--
63
2.20.1
64
65
diff view generated by jsdifflib
Deleted patch
1
The AN524 version of the SCC interface has different behaviour for
2
some of the CFG registers; implement it.
3
1
4
Each board in this family can have minor differences in the meaning
5
of the CFG registers, so rather than trying to specify all the
6
possible semantics via individual device properties, we make the
7
behaviour conditional on the part-number field of the SCC_ID register
8
which the board code already passes us.
9
10
For the AN524, the differences are:
11
* CFG3 is reserved rather than being board switches
12
* CFG5 is a new register ("ACLK Frequency in Hz")
13
* CFG6 is a new register ("Clock divider for BRAM")
14
15
We implement both of the new registers as reads-as-written.
16
17
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
18
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
19
Message-id: 20210215115138.20465-11-peter.maydell@linaro.org
20
---
21
include/hw/misc/mps2-scc.h | 3 ++
22
hw/misc/mps2-scc.c | 71 ++++++++++++++++++++++++++++++++++++--
23
2 files changed, 72 insertions(+), 2 deletions(-)
24
25
diff --git a/include/hw/misc/mps2-scc.h b/include/hw/misc/mps2-scc.h
26
index XXXXXXX..XXXXXXX 100644
27
--- a/include/hw/misc/mps2-scc.h
28
+++ b/include/hw/misc/mps2-scc.h
29
@@ -XXX,XX +XXX,XX @@ struct MPS2SCC {
30
31
uint32_t cfg0;
32
uint32_t cfg1;
33
+ uint32_t cfg2;
34
uint32_t cfg4;
35
+ uint32_t cfg5;
36
+ uint32_t cfg6;
37
uint32_t cfgdata_rtn;
38
uint32_t cfgdata_out;
39
uint32_t cfgctrl;
40
diff --git a/hw/misc/mps2-scc.c b/hw/misc/mps2-scc.c
41
index XXXXXXX..XXXXXXX 100644
42
--- a/hw/misc/mps2-scc.c
43
+++ b/hw/misc/mps2-scc.c
44
@@ -XXX,XX +XXX,XX @@
45
46
REG32(CFG0, 0)
47
REG32(CFG1, 4)
48
+REG32(CFG2, 8)
49
REG32(CFG3, 0xc)
50
REG32(CFG4, 0x10)
51
+REG32(CFG5, 0x14)
52
+REG32(CFG6, 0x18)
53
REG32(CFGDATA_RTN, 0xa0)
54
REG32(CFGDATA_OUT, 0xa4)
55
REG32(CFGCTRL, 0xa8)
56
@@ -XXX,XX +XXX,XX @@ REG32(DLL, 0x100)
57
REG32(AID, 0xFF8)
58
REG32(ID, 0xFFC)
59
60
+static int scc_partno(MPS2SCC *s)
61
+{
62
+ /* Return the partno field of the SCC_ID (0x524, 0x511, etc) */
63
+ return extract32(s->id, 4, 8);
64
+}
65
+
66
/* Handle a write via the SYS_CFG channel to the specified function/device.
67
* Return false on error (reported to guest via SYS_CFGCTRL ERROR bit).
68
*/
69
@@ -XXX,XX +XXX,XX @@ static uint64_t mps2_scc_read(void *opaque, hwaddr offset, unsigned size)
70
case A_CFG1:
71
r = s->cfg1;
72
break;
73
+ case A_CFG2:
74
+ if (scc_partno(s) != 0x524) {
75
+ /* CFG2 reserved on other boards */
76
+ goto bad_offset;
77
+ }
78
+ r = s->cfg2;
79
+ break;
80
case A_CFG3:
81
+ if (scc_partno(s) == 0x524) {
82
+ /* CFG3 reserved on AN524 */
83
+ goto bad_offset;
84
+ }
85
/* These are user-settable DIP switches on the board. We don't
86
* model that, so just return zeroes.
87
*/
88
@@ -XXX,XX +XXX,XX @@ static uint64_t mps2_scc_read(void *opaque, hwaddr offset, unsigned size)
89
case A_CFG4:
90
r = s->cfg4;
91
break;
92
+ case A_CFG5:
93
+ if (scc_partno(s) != 0x524) {
94
+ /* CFG5 reserved on other boards */
95
+ goto bad_offset;
96
+ }
97
+ r = s->cfg5;
98
+ break;
99
+ case A_CFG6:
100
+ if (scc_partno(s) != 0x524) {
101
+ /* CFG6 reserved on other boards */
102
+ goto bad_offset;
103
+ }
104
+ r = s->cfg6;
105
+ break;
106
case A_CFGDATA_RTN:
107
r = s->cfgdata_rtn;
108
break;
109
@@ -XXX,XX +XXX,XX @@ static uint64_t mps2_scc_read(void *opaque, hwaddr offset, unsigned size)
110
r = s->id;
111
break;
112
default:
113
+ bad_offset:
114
qemu_log_mask(LOG_GUEST_ERROR,
115
"MPS2 SCC read: bad offset %x\n", (int) offset);
116
r = 0;
117
@@ -XXX,XX +XXX,XX @@ static void mps2_scc_write(void *opaque, hwaddr offset, uint64_t value,
118
led_set_state(s->led[i], extract32(value, i, 1));
119
}
120
break;
121
+ case A_CFG2:
122
+ if (scc_partno(s) != 0x524) {
123
+ /* CFG2 reserved on other boards */
124
+ goto bad_offset;
125
+ }
126
+ /* AN524: QSPI Select signal */
127
+ s->cfg2 = value;
128
+ break;
129
+ case A_CFG5:
130
+ if (scc_partno(s) != 0x524) {
131
+ /* CFG5 reserved on other boards */
132
+ goto bad_offset;
133
+ }
134
+ /* AN524: ACLK frequency in Hz */
135
+ s->cfg5 = value;
136
+ break;
137
+ case A_CFG6:
138
+ if (scc_partno(s) != 0x524) {
139
+ /* CFG6 reserved on other boards */
140
+ goto bad_offset;
141
+ }
142
+ /* AN524: Clock divider for BRAM */
143
+ s->cfg6 = value;
144
+ break;
145
case A_CFGDATA_OUT:
146
s->cfgdata_out = value;
147
break;
148
@@ -XXX,XX +XXX,XX @@ static void mps2_scc_write(void *opaque, hwaddr offset, uint64_t value,
149
s->dll = deposit32(s->dll, 24, 8, extract32(value, 24, 8));
150
break;
151
default:
152
+ bad_offset:
153
qemu_log_mask(LOG_GUEST_ERROR,
154
"MPS2 SCC write: bad offset 0x%x\n", (int) offset);
155
break;
156
@@ -XXX,XX +XXX,XX @@ static void mps2_scc_reset(DeviceState *dev)
157
trace_mps2_scc_reset();
158
s->cfg0 = 0;
159
s->cfg1 = 0;
160
+ s->cfg2 = 0;
161
+ s->cfg5 = 0;
162
+ s->cfg6 = 0;
163
s->cfgdata_rtn = 0;
164
s->cfgdata_out = 0;
165
s->cfgctrl = 0x100000;
166
@@ -XXX,XX +XXX,XX @@ static void mps2_scc_realize(DeviceState *dev, Error **errp)
167
168
static const VMStateDescription mps2_scc_vmstate = {
169
.name = "mps2-scc",
170
- .version_id = 2,
171
- .minimum_version_id = 2,
172
+ .version_id = 3,
173
+ .minimum_version_id = 3,
174
.fields = (VMStateField[]) {
175
VMSTATE_UINT32(cfg0, MPS2SCC),
176
VMSTATE_UINT32(cfg1, MPS2SCC),
177
+ VMSTATE_UINT32(cfg2, MPS2SCC),
178
+ /* cfg3, cfg4 are read-only so need not be migrated */
179
+ VMSTATE_UINT32(cfg5, MPS2SCC),
180
+ VMSTATE_UINT32(cfg6, MPS2SCC),
181
VMSTATE_UINT32(cfgdata_rtn, MPS2SCC),
182
VMSTATE_UINT32(cfgdata_out, MPS2SCC),
183
VMSTATE_UINT32(cfgctrl, MPS2SCC),
184
--
185
2.20.1
186
187
diff view generated by jsdifflib