1
The following changes since commit 922582ace2df59572a671f5c0c5c6c5c706995e5:
1
The following changes since commit 661c2e1ab29cd9c4d268ae3f44712e8d421c0e56:
2
2
3
Merge tag 'pull-hppa-20240515' of https://gitlab.com/rth7680/qemu into staging (2024-05-15 11:46:58 +0200)
3
scripts/checkpatch: Fix a typo (2025-03-04 09:30:26 +0800)
4
4
5
are available in the Git repository at:
5
are available in the Git repository at:
6
6
7
https://gitlab.com/gaosong/qemu.git tags/pull-loongarch-20240516
7
https://gitlab.com/bibo-mao/qemu.git tags/pull-loongarch-20250305
8
8
9
for you to fetch changes up to d55d16700a2e2b36c7e34724d4d77f4a75c5243a:
9
for you to fetch changes up to 0a629c827300d514cc1f61806414d214fcf75051:
10
10
11
target/loongarch/kvm: fpu save the vreg registers high 192bit (2024-05-16 16:32:35 +0800)
11
target/loongarch: Adjust the cpu reset action to a proper position (2025-03-05 09:45:52 +0800)
12
12
13
----------------------------------------------------------------
13
----------------------------------------------------------------
14
pull-loongarch-20240516
14
pull-loongarch-20250305 queue
15
15
16
----------------------------------------------------------------
16
----------------------------------------------------------------
17
Bibo Mao (3):
17
Bibo Mao (14):
18
hw/loongarch: Add compat machine for 9.1
18
hw/intc/loongarch_ipi: Add basic hotplug framework
19
hw/loongarch: Remove minimum and default memory size
19
hw/intc/loongarch_ipi: Implment cpu hotplug interface
20
tests: Add migration test for loongarch64
20
hw/intc/loongarch_ipi: Notify ipi object when cpu is plugged
21
hw/intc/loongarch_extioi: Move gpio irq initial to common code
22
hw/intc/loongarch_extioi: Add basic hotplug framework
23
hw/intc/loongarch_extioi: Implment cpu hotplug interface
24
hw/intc/loongarch_extioi: Use cpu plug notification
25
hw/loongarch/virt: Add CPU topology support
26
hw/loongarch/virt: Add topo properties on CPU object
27
hw/loongarch/virt: Add basic cpu plug interface framework
28
hw/loongarch/virt: Implement cpu unplug interface
29
hw/loongarch/virt: Implement cpu plug interface
30
hw/loongarch/virt: Update the ACPI table for hotplug cpu
31
hw/loongarch/virt: Enable cpu hotplug feature on virt machine
21
32
22
Song Gao (2):
33
Xianglai Li (1):
23
target/loongarch/kvm: Fix VM recovery from disk failures
34
target/loongarch: Adjust the cpu reset action to a proper position
24
target/loongarch/kvm: fpu save the vreg registers high 192bit
25
35
26
hw/loongarch/virt.c | 66 +++++++++++++++++++++++---------
36
hw/intc/loongarch_extioi.c | 8 +-
27
target/loongarch/kvm/kvm.c | 6 +++
37
hw/intc/loongarch_extioi_common.c | 84 ++++++++++-
28
target/loongarch/machine.c | 6 ++-
38
hw/intc/loongarch_ipi.c | 71 +++++++++
29
tests/migration/Makefile | 2 +-
39
hw/loongarch/Kconfig | 1 +
30
tests/migration/loongarch64/Makefile | 18 +++++++++
40
hw/loongarch/virt-acpi-build.c | 35 ++++-
31
tests/migration/loongarch64/a-b-kernel.S | 49 ++++++++++++++++++++++++
41
hw/loongarch/virt.c | 301 ++++++++++++++++++++++++++++++++++----
32
tests/migration/loongarch64/a-b-kernel.h | 16 ++++++++
42
include/hw/loongarch/virt.h | 1 +
33
tests/migration/migration-test.h | 3 ++
43
target/loongarch/cpu.c | 25 +++-
34
tests/qtest/meson.build | 2 +-
44
target/loongarch/cpu.h | 11 ++
35
tests/qtest/migration-test.c | 10 +++++
45
9 files changed, 493 insertions(+), 44 deletions(-)
36
10 files changed, 156 insertions(+), 22 deletions(-)
37
create mode 100644 tests/migration/loongarch64/Makefile
38
create mode 100644 tests/migration/loongarch64/a-b-kernel.S
39
create mode 100644 tests/migration/loongarch64/a-b-kernel.h
diff view generated by jsdifflib
1
From: Bibo Mao <maobibo@loongson.cn>
1
LoongArch ipi can send interrupt to multiple CPUs, interrupt routing
2
to CPU comes from destination physical cpu id. Here hotplug interface
3
is added for IPI object, so that parent irq line can be connected, and
4
routing table can be added for new created cpu.
2
5
3
This patch adds migration test support for loongarch64. The test code
6
Here only basic hotplug framework is added, it is stub function.
4
comes from aarch64 mostly, only that it booted as bios in qemu since
5
kernel requires elf format and bios uses binary format.
6
7
In addition to providing the binary, this patch also includes the source
8
code and the build script in tests/migration/loongarch64. So users can
9
change the source and/or re-compile the binary as they wish.
10
7
11
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
8
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
12
Acked-by: Thomas Huth <thuth@redhat.com>
13
Acked-by: Peter Xu <peterx@redhat.com>
14
Reviewed-by: Fabiano Rosas <farosas@suse.de>
15
Reviewed-by: Song Gao <gaosong@loongson.cn>
16
Tested-by: Song Gao <gaosong@loongson.cn>
17
Message-Id: <20240511034220.3030560-4-maobibo@loongson.cn>
18
Signed-off-by: Song Gao <gaosong@loongson.cn>
19
---
9
---
20
tests/migration/Makefile | 2 +-
10
hw/intc/loongarch_ipi.c | 32 ++++++++++++++++++++++++++++++++
21
tests/migration/loongarch64/Makefile | 18 +++++++++
11
1 file changed, 32 insertions(+)
22
tests/migration/loongarch64/a-b-kernel.S | 49 ++++++++++++++++++++++++
23
tests/migration/loongarch64/a-b-kernel.h | 16 ++++++++
24
tests/migration/migration-test.h | 3 ++
25
tests/qtest/meson.build | 2 +-
26
tests/qtest/migration-test.c | 10 +++++
27
7 files changed, 98 insertions(+), 2 deletions(-)
28
create mode 100644 tests/migration/loongarch64/Makefile
29
create mode 100644 tests/migration/loongarch64/a-b-kernel.S
30
create mode 100644 tests/migration/loongarch64/a-b-kernel.h
31
12
32
diff --git a/tests/migration/Makefile b/tests/migration/Makefile
13
diff --git a/hw/intc/loongarch_ipi.c b/hw/intc/loongarch_ipi.c
33
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
34
--- a/tests/migration/Makefile
15
--- a/hw/intc/loongarch_ipi.c
35
+++ b/tests/migration/Makefile
16
+++ b/hw/intc/loongarch_ipi.c
36
@@ -XXX,XX +XXX,XX @@
37
# See the COPYING file in the top-level directory.
38
#
39
40
-TARGET_LIST = i386 aarch64 s390x
41
+TARGET_LIST = i386 aarch64 s390x loongarch64
42
43
SRC_PATH = ../..
44
45
diff --git a/tests/migration/loongarch64/Makefile b/tests/migration/loongarch64/Makefile
46
new file mode 100644
47
index XXXXXXX..XXXXXXX
48
--- /dev/null
49
+++ b/tests/migration/loongarch64/Makefile
50
@@ -XXX,XX +XXX,XX @@
51
+# To specify cross compiler prefix, use CROSS_PREFIX=
52
+# $ make CROSS_PREFIX=loongarch64-linux-gnu-
53
+
54
+.PHONY: all clean
55
+all: a-b-kernel.h
56
+
57
+a-b-kernel.h: loongarch64.kernel
58
+    echo "$$__note" > $@
59
+    xxd -i $< | sed -e 's/.*int.*//' >> $@
60
+
61
+loongarch64.kernel: loongarch64.elf
62
+    $(CROSS_PREFIX)objcopy -j .text -O binary $< $@
63
+
64
+loongarch64.elf: a-b-kernel.S
65
+    $(CROSS_PREFIX)gcc -o $@ -nostdlib -Wl,--build-id=none $<
66
+
67
+clean:
68
+    $(RM) *.kernel *.elf
69
diff --git a/tests/migration/loongarch64/a-b-kernel.S b/tests/migration/loongarch64/a-b-kernel.S
70
new file mode 100644
71
index XXXXXXX..XXXXXXX
72
--- /dev/null
73
+++ b/tests/migration/loongarch64/a-b-kernel.S
74
@@ -XXX,XX +XXX,XX @@
75
+/* SPDX-License-Identifier: GPL-2.0-or-later */
76
+/*
77
+ * Copyright (c) 2024 Loongson Technology Corporation Limited
78
+ */
79
+#include "../migration-test.h"
80
+
81
+#define LOONGARCH_CSR_CRMD 0
82
+#define LOONGARCH_VIRT_UART 0x1FE001E0
83
+.section .text
84
+
85
+ .globl _start
86
+_start:
87
+ /* output char 'A' to UART16550 */
88
+ li.d $t0, LOONGARCH_VIRT_UART
89
+ li.w $t1, 'A'
90
+ st.b $t1, $t0, 0
91
+
92
+ /* traverse test memory region */
93
+ li.d $t0, LOONGARCH_TEST_MEM_START
94
+ li.d $t1, LOONGARCH_TEST_MEM_END
95
+ li.d $t2, TEST_MEM_PAGE_SIZE
96
+ li.d $t4, LOONGARCH_VIRT_UART
97
+ li.w $t5, 'B'
98
+
99
+clean:
100
+ st.b $zero, $t0, 0
101
+ add.d $t0, $t0, $t2
102
+ bne $t0, $t1, clean
103
+ /* keeps a counter so we can limit the output speed */
104
+ addi.d $t6, $zero, 0
105
+
106
+mainloop:
107
+ li.d $t0, LOONGARCH_TEST_MEM_START
108
+
109
+innerloop:
110
+ ld.bu $t3, $t0, 0
111
+ addi.w $t3, $t3, 1
112
+ ext.w.b $t3, $t3
113
+ st.b $t3, $t0, 0
114
+ add.d $t0, $t0, $t2
115
+ bne $t0, $t1, innerloop
116
+
117
+ addi.d $t6, $t6, 1
118
+ andi $t6, $t6, 31
119
+ bnez $t6, mainloop
120
+
121
+ st.b $t5, $t4, 0
122
+ b mainloop
123
+ nop
124
diff --git a/tests/migration/loongarch64/a-b-kernel.h b/tests/migration/loongarch64/a-b-kernel.h
125
new file mode 100644
126
index XXXXXXX..XXXXXXX
127
--- /dev/null
128
+++ b/tests/migration/loongarch64/a-b-kernel.h
129
@@ -XXX,XX +XXX,XX @@
130
+/* This file is automatically generated from the assembly file in
131
+* tests/migration/loongarch64. Edit that file and then run "make all"
132
+* inside tests/migration to update, and then remember to send both
133
+* the header and the assembler differences in your patch submission.
134
+*/
135
+unsigned char loongarch64_kernel[] = {
136
+ 0x0c, 0xc0, 0x3f, 0x14, 0x8c, 0x81, 0x87, 0x03, 0x0d, 0x04, 0x81, 0x03,
137
+ 0x8d, 0x01, 0x00, 0x29, 0x0c, 0x00, 0x04, 0x14, 0x0d, 0x80, 0x0c, 0x14,
138
+ 0x2e, 0x00, 0x00, 0x14, 0x10, 0xc0, 0x3f, 0x14, 0x10, 0x82, 0x87, 0x03,
139
+ 0x11, 0x08, 0x81, 0x03, 0x80, 0x01, 0x00, 0x29, 0x8c, 0xb9, 0x10, 0x00,
140
+ 0x8d, 0xf9, 0xff, 0x5f, 0x12, 0x00, 0xc0, 0x02, 0x0c, 0x00, 0x04, 0x14,
141
+ 0x8f, 0x01, 0x00, 0x2a, 0xef, 0x05, 0x80, 0x02, 0xef, 0x5d, 0x00, 0x00,
142
+ 0x8f, 0x01, 0x00, 0x29, 0x8c, 0xb9, 0x10, 0x00, 0x8d, 0xed, 0xff, 0x5f,
143
+ 0x52, 0x06, 0xc0, 0x02, 0x52, 0x7e, 0x40, 0x03, 0x5f, 0xde, 0xff, 0x47,
144
+ 0x11, 0x02, 0x00, 0x29, 0xff, 0xd7, 0xff, 0x53, 0x00, 0x00, 0x40, 0x03
145
+};
146
diff --git a/tests/migration/migration-test.h b/tests/migration/migration-test.h
147
index XXXXXXX..XXXXXXX 100644
148
--- a/tests/migration/migration-test.h
149
+++ b/tests/migration/migration-test.h
150
@@ -XXX,XX +XXX,XX @@
17
@@ -XXX,XX +XXX,XX @@
151
*/
18
*/
152
#define ARM_TEST_MAX_KERNEL_SIZE (512 * 1024)
19
153
20
#include "qemu/osdep.h"
154
+/* LoongArch64 */
21
+#include "qemu/error-report.h"
155
+#define LOONGARCH_TEST_MEM_START (32 * 1024 * 1024)
22
#include "hw/boards.h"
156
+#define LOONGARCH_TEST_MEM_END (100 * 1024 * 1024)
23
#include "qapi/error.h"
157
#endif /* MIGRATION_TEST_H */
24
#include "hw/intc/loongarch_ipi.h"
158
diff --git a/tests/qtest/meson.build b/tests/qtest/meson.build
25
@@ -XXX,XX +XXX,XX @@ static void loongarch_ipi_realize(DeviceState *dev, Error **errp)
159
index XXXXXXX..XXXXXXX 100644
26
}
160
--- a/tests/qtest/meson.build
27
}
161
+++ b/tests/qtest/meson.build
28
162
@@ -XXX,XX +XXX,XX @@ qtests_hppa = ['boot-serial-test'] + \
29
+static void loongarch_ipi_cpu_plug(HotplugHandler *hotplug_dev,
163
(config_all_devices.has_key('CONFIG_VGA') ? ['display-vga-test'] : [])
30
+ DeviceState *dev, Error **errp)
164
31
+{
165
qtests_loongarch64 = qtests_filter + \
32
+ Object *obj = OBJECT(dev);
166
- ['boot-serial-test']
33
+
167
+ ['boot-serial-test', 'migration-test']
34
+ if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
168
35
+ warn_report("LoongArch extioi: Invalid %s device type",
169
qtests_m68k = ['boot-serial-test'] + \
36
+ object_get_typename(obj));
170
qtests_filter
37
+ return;
171
diff --git a/tests/qtest/migration-test.c b/tests/qtest/migration-test.c
38
+ }
172
index XXXXXXX..XXXXXXX 100644
39
+}
173
--- a/tests/qtest/migration-test.c
40
+
174
+++ b/tests/qtest/migration-test.c
41
+static void loongarch_ipi_cpu_unplug(HotplugHandler *hotplug_dev,
175
@@ -XXX,XX +XXX,XX @@ static char *bootpath;
42
+ DeviceState *dev, Error **errp)
176
#include "tests/migration/i386/a-b-bootblock.h"
43
+{
177
#include "tests/migration/aarch64/a-b-kernel.h"
44
+ Object *obj = OBJECT(dev);
178
#include "tests/migration/s390x/a-b-bios.h"
45
+
179
+#include "tests/migration/loongarch64/a-b-kernel.h"
46
+ if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
180
47
+ warn_report("LoongArch extioi: Invalid %s device type",
181
static void bootfile_create(char *dir, bool suspend_me)
48
+ object_get_typename(obj));
49
+ return;
50
+ }
51
+}
52
+
53
static void loongarch_ipi_class_init(ObjectClass *klass, void *data)
182
{
54
{
183
@@ -XXX,XX +XXX,XX @@ static void bootfile_create(char *dir, bool suspend_me)
55
LoongsonIPICommonClass *licc = LOONGSON_IPI_COMMON_CLASS(klass);
184
content = aarch64_kernel;
56
+ HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(klass);
185
len = sizeof(aarch64_kernel);
57
LoongarchIPIClass *lic = LOONGARCH_IPI_CLASS(klass);
186
g_assert(sizeof(aarch64_kernel) <= ARM_TEST_MAX_KERNEL_SIZE);
58
DeviceClass *dc = DEVICE_CLASS(klass);
187
+ } else if (strcmp(arch, "loongarch64") == 0) {
59
188
+ content = loongarch64_kernel;
60
@@ -XXX,XX +XXX,XX @@ static void loongarch_ipi_class_init(ObjectClass *klass, void *data)
189
+ len = sizeof(loongarch64_kernel);
61
&lic->parent_realize);
190
} else {
62
licc->get_iocsr_as = get_iocsr_as;
191
g_assert_not_reached();
63
licc->cpu_by_arch_id = loongarch_cpu_by_arch_id;
64
+ hc->plug = loongarch_ipi_cpu_plug;
65
+ hc->unplug = loongarch_ipi_cpu_unplug;
66
}
67
68
static const TypeInfo loongarch_ipi_types[] = {
69
@@ -XXX,XX +XXX,XX @@ static const TypeInfo loongarch_ipi_types[] = {
70
.instance_size = sizeof(LoongarchIPIState),
71
.class_size = sizeof(LoongarchIPIClass),
72
.class_init = loongarch_ipi_class_init,
73
+ .interfaces = (InterfaceInfo[]) {
74
+ { TYPE_HOTPLUG_HANDLER },
75
+ { }
76
+ },
192
}
77
}
193
@@ -XXX,XX +XXX,XX @@ static int test_migrate_start(QTestState **from, QTestState **to,
78
};
194
arch_opts = g_strdup_printf("-cpu max -kernel %s", bootpath);
79
195
start_address = ARM_TEST_MEM_START;
196
end_address = ARM_TEST_MEM_END;
197
+ } else if (strcmp(arch, "loongarch64") == 0) {
198
+ memory_size = "256M";
199
+ machine_alias = "virt";
200
+ arch_opts = g_strdup_printf("-cpu la464 -bios %s", bootpath);
201
+ start_address = LOONGARCH_TEST_MEM_START;
202
+ end_address = LOONGARCH_TEST_MEM_END;
203
} else {
204
g_assert_not_reached();
205
}
206
--
80
--
207
2.34.1
81
2.43.5
diff view generated by jsdifflib
New patch
1
Add logic cpu allocation and cpu mapping with cpu hotplug interface.
2
When cpu is added, connect ipi gpio irq to CPU IRQ_IPI irq pin.
1
3
4
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
5
---
6
hw/intc/loongarch_ipi.c | 39 +++++++++++++++++++++++++++++++++++++++
7
1 file changed, 39 insertions(+)
8
9
diff --git a/hw/intc/loongarch_ipi.c b/hw/intc/loongarch_ipi.c
10
index XXXXXXX..XXXXXXX 100644
11
--- a/hw/intc/loongarch_ipi.c
12
+++ b/hw/intc/loongarch_ipi.c
13
@@ -XXX,XX +XXX,XX @@ static int loongarch_cpu_by_arch_id(LoongsonIPICommonState *lics,
14
return MEMTX_ERROR;
15
}
16
17
+static IPICore *loongarch_ipi_get_cpu(LoongsonIPICommonState *lics,
18
+ DeviceState *dev)
19
+{
20
+ CPUClass *k = CPU_GET_CLASS(dev);
21
+ uint64_t arch_id = k->get_arch_id(CPU(dev));
22
+ int i;
23
+
24
+ for (i = 0; i < lics->num_cpu; i++) {
25
+ if (lics->cpu[i].arch_id == arch_id) {
26
+ return &lics->cpu[i];
27
+ }
28
+ }
29
+
30
+ return NULL;
31
+}
32
+
33
static void loongarch_ipi_realize(DeviceState *dev, Error **errp)
34
{
35
LoongsonIPICommonState *lics = LOONGSON_IPI_COMMON(dev);
36
@@ -XXX,XX +XXX,XX @@ static void loongarch_ipi_realize(DeviceState *dev, Error **errp)
37
static void loongarch_ipi_cpu_plug(HotplugHandler *hotplug_dev,
38
DeviceState *dev, Error **errp)
39
{
40
+ LoongsonIPICommonState *lics = LOONGSON_IPI_COMMON(hotplug_dev);
41
Object *obj = OBJECT(dev);
42
+ IPICore *core;
43
+ int index;
44
45
if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
46
warn_report("LoongArch extioi: Invalid %s device type",
47
object_get_typename(obj));
48
return;
49
}
50
+
51
+ core = loongarch_ipi_get_cpu(lics, dev);
52
+ if (!core) {
53
+ return;
54
+ }
55
+
56
+ core->cpu = CPU(dev);
57
+ index = core - lics->cpu;
58
+
59
+ /* connect ipi irq to cpu irq */
60
+ qdev_connect_gpio_out(DEVICE(lics), index, qdev_get_gpio_in(dev, IRQ_IPI));
61
}
62
63
static void loongarch_ipi_cpu_unplug(HotplugHandler *hotplug_dev,
64
DeviceState *dev, Error **errp)
65
{
66
+ LoongsonIPICommonState *lics = LOONGSON_IPI_COMMON(hotplug_dev);
67
Object *obj = OBJECT(dev);
68
+ IPICore *core;
69
70
if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
71
warn_report("LoongArch extioi: Invalid %s device type",
72
object_get_typename(obj));
73
return;
74
}
75
+
76
+ core = loongarch_ipi_get_cpu(lics, dev);
77
+ if (!core) {
78
+ return;
79
+ }
80
+
81
+ core->cpu = NULL;
82
}
83
84
static void loongarch_ipi_class_init(ObjectClass *klass, void *data)
85
--
86
2.43.5
diff view generated by jsdifflib
New patch
1
Use hotplug_handler_plug() to nofity ipi object when cold-plug
2
cpu is created, so that ipi can set and configure irq routing
3
to new cpu.
1
4
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
6
---
7
hw/loongarch/virt.c | 5 ++---
8
1 file changed, 2 insertions(+), 3 deletions(-)
9
10
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
11
index XXXXXXX..XXXXXXX 100644
12
--- a/hw/loongarch/virt.c
13
+++ b/hw/loongarch/virt.c
14
@@ -XXX,XX +XXX,XX @@ static void virt_cpu_irq_init(LoongArchVirtMachineState *lvms)
15
MachineClass *mc = MACHINE_GET_CLASS(ms);
16
const CPUArchIdList *possible_cpus;
17
CPUState *cs;
18
+ Error *err = NULL;
19
20
/* cpu nodes */
21
possible_cpus = mc->possible_cpu_arch_ids(ms);
22
@@ -XXX,XX +XXX,XX @@ static void virt_cpu_irq_init(LoongArchVirtMachineState *lvms)
23
continue;
24
}
25
26
- /* connect ipi irq to cpu irq */
27
- qdev_connect_gpio_out(lvms->ipi, num,
28
- qdev_get_gpio_in(DEVICE(cs), IRQ_IPI));
29
+ hotplug_handler_plug(HOTPLUG_HANDLER(lvms->ipi), DEVICE(cs), &err);
30
31
/*
32
* connect ext irq to the cpu irq
33
--
34
2.43.5
diff view generated by jsdifflib
New patch
1
When cpu is added, it will connect gpio irq line to cpu irq.
2
And cpu hot-add is put in common code, move gpio irq initial
3
part into common code.
1
4
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
6
---
7
hw/intc/loongarch_extioi.c | 8 +-------
8
hw/intc/loongarch_extioi_common.c | 6 +++++-
9
2 files changed, 6 insertions(+), 8 deletions(-)
10
11
diff --git a/hw/intc/loongarch_extioi.c b/hw/intc/loongarch_extioi.c
12
index XXXXXXX..XXXXXXX 100644
13
--- a/hw/intc/loongarch_extioi.c
14
+++ b/hw/intc/loongarch_extioi.c
15
@@ -XXX,XX +XXX,XX @@ static void loongarch_extioi_realize(DeviceState *dev, Error **errp)
16
LoongArchExtIOIClass *lec = LOONGARCH_EXTIOI_GET_CLASS(dev);
17
SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
18
Error *local_err = NULL;
19
- int i, pin;
20
+ int i;
21
22
lec->parent_realize(dev, &local_err);
23
if (local_err) {
24
@@ -XXX,XX +XXX,XX @@ static void loongarch_extioi_realize(DeviceState *dev, Error **errp)
25
} else {
26
s->status |= BIT(EXTIOI_ENABLE);
27
}
28
-
29
- for (i = 0; i < s->num_cpu; i++) {
30
- for (pin = 0; pin < LS3A_INTC_IP; pin++) {
31
- qdev_init_gpio_out(dev, &s->cpu[i].parent_irq[pin], 1);
32
- }
33
- }
34
}
35
36
static void loongarch_extioi_unrealize(DeviceState *dev)
37
diff --git a/hw/intc/loongarch_extioi_common.c b/hw/intc/loongarch_extioi_common.c
38
index XXXXXXX..XXXXXXX 100644
39
--- a/hw/intc/loongarch_extioi_common.c
40
+++ b/hw/intc/loongarch_extioi_common.c
41
@@ -XXX,XX +XXX,XX @@ static void loongarch_extioi_common_realize(DeviceState *dev, Error **errp)
42
MachineState *machine = MACHINE(qdev_get_machine());
43
MachineClass *mc = MACHINE_GET_CLASS(machine);
44
const CPUArchIdList *id_list;
45
- int i;
46
+ int i, pin;
47
48
assert(mc->possible_cpu_arch_ids);
49
id_list = mc->possible_cpu_arch_ids(machine);
50
@@ -XXX,XX +XXX,XX @@ static void loongarch_extioi_common_realize(DeviceState *dev, Error **errp)
51
for (i = 0; i < s->num_cpu; i++) {
52
s->cpu[i].arch_id = id_list->cpus[i].arch_id;
53
s->cpu[i].cpu = CPU(id_list->cpus[i].cpu);
54
+
55
+ for (pin = 0; pin < LS3A_INTC_IP; pin++) {
56
+ qdev_init_gpio_out(dev, &s->cpu[i].parent_irq[pin], 1);
57
+ }
58
}
59
}
60
61
--
62
2.43.5
diff view generated by jsdifflib
New patch
1
LoongArch extioi interrupt controller routes peripheral interrupt
2
to multiple CPUs, physical cpu id is used in interrupt routing table.
3
Here hotplug interface is added for extioi object, so that parent irq
4
line can be connected, and routing table can be added for new created
5
cpu.
1
6
7
Here only basic hotplug framework is added, it is stub function.
8
9
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
10
---
11
hw/intc/loongarch_extioi_common.c | 33 +++++++++++++++++++++++++++++++
12
1 file changed, 33 insertions(+)
13
14
diff --git a/hw/intc/loongarch_extioi_common.c b/hw/intc/loongarch_extioi_common.c
15
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/intc/loongarch_extioi_common.c
17
+++ b/hw/intc/loongarch_extioi_common.c
18
@@ -XXX,XX +XXX,XX @@
19
* Copyright (C) 2024 Loongson Technology Corporation Limited
20
*/
21
#include "qemu/osdep.h"
22
+#include "qemu/error-report.h"
23
#include "qemu/module.h"
24
#include "qapi/error.h"
25
#include "hw/qdev-properties.h"
26
#include "hw/intc/loongarch_extioi_common.h"
27
#include "migration/vmstate.h"
28
+#include "target/loongarch/cpu.h"
29
+
30
+static void loongarch_extioi_cpu_plug(HotplugHandler *hotplug_dev,
31
+ DeviceState *dev, Error **errp)
32
+{
33
+ Object *obj = OBJECT(dev);
34
+
35
+ if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
36
+ warn_report("LoongArch extioi: Invalid %s device type",
37
+ object_get_typename(obj));
38
+ return;
39
+ }
40
+}
41
+
42
+static void loongarch_extioi_cpu_unplug(HotplugHandler *hotplug_dev,
43
+ DeviceState *dev, Error **errp)
44
+{
45
+ Object *obj = OBJECT(dev);
46
+
47
+ if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
48
+ warn_report("LoongArch extioi: Invalid %s device type",
49
+ object_get_typename(obj));
50
+ return;
51
+ }
52
+}
53
54
static void loongarch_extioi_common_realize(DeviceState *dev, Error **errp)
55
{
56
@@ -XXX,XX +XXX,XX @@ static void loongarch_extioi_common_class_init(ObjectClass *klass, void *data)
57
{
58
DeviceClass *dc = DEVICE_CLASS(klass);
59
LoongArchExtIOICommonClass *lecc = LOONGARCH_EXTIOI_COMMON_CLASS(klass);
60
+ HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(klass);
61
62
device_class_set_parent_realize(dc, loongarch_extioi_common_realize,
63
&lecc->parent_realize);
64
device_class_set_props(dc, extioi_properties);
65
dc->vmsd = &vmstate_loongarch_extioi;
66
+ hc->plug = loongarch_extioi_cpu_plug;
67
+ hc->unplug = loongarch_extioi_cpu_unplug;
68
}
69
70
static const TypeInfo loongarch_extioi_common_types[] = {
71
@@ -XXX,XX +XXX,XX @@ static const TypeInfo loongarch_extioi_common_types[] = {
72
.instance_size = sizeof(LoongArchExtIOICommonState),
73
.class_size = sizeof(LoongArchExtIOICommonClass),
74
.class_init = loongarch_extioi_common_class_init,
75
+ .interfaces = (InterfaceInfo[]) {
76
+ { TYPE_HOTPLUG_HANDLER },
77
+ { }
78
+ },
79
.abstract = true,
80
}
81
};
82
--
83
2.43.5
diff view generated by jsdifflib
New patch
1
When cpu is added, connect extioi gpio irq to CPU irq pin.
1
2
3
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
4
---
5
hw/intc/loongarch_extioi_common.c | 45 +++++++++++++++++++++++++++++++
6
1 file changed, 45 insertions(+)
7
8
diff --git a/hw/intc/loongarch_extioi_common.c b/hw/intc/loongarch_extioi_common.c
9
index XXXXXXX..XXXXXXX 100644
10
--- a/hw/intc/loongarch_extioi_common.c
11
+++ b/hw/intc/loongarch_extioi_common.c
12
@@ -XXX,XX +XXX,XX @@
13
#include "migration/vmstate.h"
14
#include "target/loongarch/cpu.h"
15
16
+static ExtIOICore *loongarch_extioi_get_cpu(LoongArchExtIOICommonState *s,
17
+ DeviceState *dev)
18
+{
19
+ CPUClass *k = CPU_GET_CLASS(dev);
20
+ uint64_t arch_id = k->get_arch_id(CPU(dev));
21
+ int i;
22
+
23
+ for (i = 0; i < s->num_cpu; i++) {
24
+ if (s->cpu[i].arch_id == arch_id) {
25
+ return &s->cpu[i];
26
+ }
27
+ }
28
+
29
+ return NULL;
30
+}
31
+
32
static void loongarch_extioi_cpu_plug(HotplugHandler *hotplug_dev,
33
DeviceState *dev, Error **errp)
34
{
35
+ LoongArchExtIOICommonState *s = LOONGARCH_EXTIOI_COMMON(hotplug_dev);
36
Object *obj = OBJECT(dev);
37
+ ExtIOICore *core;
38
+ int pin, index;
39
40
if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
41
warn_report("LoongArch extioi: Invalid %s device type",
42
object_get_typename(obj));
43
return;
44
}
45
+
46
+ core = loongarch_extioi_get_cpu(s, dev);
47
+ if (!core) {
48
+ return;
49
+ }
50
+
51
+ core->cpu = CPU(dev);
52
+ index = core - s->cpu;
53
+
54
+ /*
55
+ * connect extioi irq to the cpu irq
56
+ * cpu_pin[LS3A_INTC_IP + 2 : 2] <= intc_pin[LS3A_INTC_IP : 0]
57
+ */
58
+ for (pin = 0; pin < LS3A_INTC_IP; pin++) {
59
+ qdev_connect_gpio_out(DEVICE(s), index * LS3A_INTC_IP + pin,
60
+ qdev_get_gpio_in(dev, pin + 2));
61
+ }
62
}
63
64
static void loongarch_extioi_cpu_unplug(HotplugHandler *hotplug_dev,
65
DeviceState *dev, Error **errp)
66
{
67
+ LoongArchExtIOICommonState *s = LOONGARCH_EXTIOI_COMMON(hotplug_dev);
68
Object *obj = OBJECT(dev);
69
+ ExtIOICore *core;
70
71
if (!object_dynamic_cast(obj, TYPE_LOONGARCH_CPU)) {
72
warn_report("LoongArch extioi: Invalid %s device type",
73
object_get_typename(obj));
74
return;
75
}
76
+
77
+ core = loongarch_extioi_get_cpu(s, dev);
78
+ if (!core) {
79
+ return;
80
+ }
81
+
82
+ core->cpu = NULL;
83
}
84
85
static void loongarch_extioi_common_realize(DeviceState *dev, Error **errp)
86
--
87
2.43.5
diff view generated by jsdifflib
1
From: Bibo Mao <maobibo@loongson.cn>
1
Use hotplug_handler_plug() to nofity extioi object when cold-plug
2
2
cpu is created, so that extioi can set and configure irq routing
3
Some qtest test cases such as numa use default memory size of generic
3
to new cpu.
4
machine class, which is 128M by fault.
5
6
Here generic default memory size is used, and also remove minimum memory
7
size which is 1G originally.
8
4
9
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
10
Reviewed-by: Song Gao <gaosong@loongson.cn>
11
Message-Id: <20240511034220.3030560-3-maobibo@loongson.cn>
12
Signed-off-by: Song Gao <gaosong@loongson.cn>
13
---
6
---
14
hw/loongarch/virt.c | 5 -----
7
hw/loongarch/virt.c | 12 ++----------
15
1 file changed, 5 deletions(-)
8
1 file changed, 2 insertions(+), 10 deletions(-)
16
9
17
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
10
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
18
index XXXXXXX..XXXXXXX 100644
11
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/loongarch/virt.c
12
--- a/hw/loongarch/virt.c
20
+++ b/hw/loongarch/virt.c
13
+++ b/hw/loongarch/virt.c
21
@@ -XXX,XX +XXX,XX @@ static void virt_init(MachineState *machine)
14
@@ -XXX,XX +XXX,XX @@ static void virt_devices_init(DeviceState *pch_pic,
22
cpu_model = LOONGARCH_CPU_TYPE_NAME("la464");
15
16
static void virt_cpu_irq_init(LoongArchVirtMachineState *lvms)
17
{
18
- int num, pin;
19
+ int num;
20
MachineState *ms = MACHINE(lvms);
21
MachineClass *mc = MACHINE_GET_CLASS(ms);
22
const CPUArchIdList *possible_cpus;
23
@@ -XXX,XX +XXX,XX @@ static void virt_cpu_irq_init(LoongArchVirtMachineState *lvms)
24
}
25
26
hotplug_handler_plug(HOTPLUG_HANDLER(lvms->ipi), DEVICE(cs), &err);
27
-
28
- /*
29
- * connect ext irq to the cpu irq
30
- * cpu_pin[9:2] <= intc_pin[7:0]
31
- */
32
- for (pin = 0; pin < LS3A_INTC_IP; pin++) {
33
- qdev_connect_gpio_out(lvms->extioi, (num * LS3A_INTC_IP + pin),
34
- qdev_get_gpio_in(DEVICE(cs), pin + 2));
35
- }
36
+ hotplug_handler_plug(HOTPLUG_HANDLER(lvms->extioi), DEVICE(cs), &err);
23
}
37
}
24
38
}
25
- if (ram_size < 1 * GiB) {
39
26
- error_report("ram_size must be greater than 1G.");
27
- exit(1);
28
- }
29
create_fdt(lvms);
30
31
/* Create IOCSR space */
32
@@ -XXX,XX +XXX,XX @@ static void virt_class_init(ObjectClass *oc, void *data)
33
HotplugHandlerClass *hc = HOTPLUG_HANDLER_CLASS(oc);
34
35
mc->init = virt_init;
36
- mc->default_ram_size = 1 * GiB;
37
mc->default_cpu_type = LOONGARCH_CPU_TYPE_NAME("la464");
38
mc->default_ram_id = "loongarch.ram";
39
mc->max_cpus = LOONGARCH_MAX_CPUS;
40
--
40
--
41
2.34.1
41
2.43.5
diff view generated by jsdifflib
New patch
1
Add topological relationships for Loongarch VCPU and initialize
2
topology member variables.
1
3
4
On LoongArch system there is socket/core/thread topo information,
5
physical CPU id is calculated from CPU topo, every topo sub-field is
6
aligned by power of 2. So it is different from logical cpu index.
7
8
Co-developed-by: Xianglai Li <lixianglai@loongson.cn>
9
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
10
---
11
hw/loongarch/virt.c | 59 ++++++++++++++++++++++++++++++++++++------
12
target/loongarch/cpu.h | 6 +++++
13
2 files changed, 57 insertions(+), 8 deletions(-)
14
15
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/loongarch/virt.c
18
+++ b/hw/loongarch/virt.c
19
@@ -XXX,XX +XXX,XX @@ static void virt_initfn(Object *obj)
20
virt_flash_create(lvms);
21
}
22
23
+static void virt_get_topo_from_index(MachineState *ms,
24
+ LoongArchCPUTopo *topo, int index)
25
+{
26
+ topo->socket_id = index / (ms->smp.cores * ms->smp.threads);
27
+ topo->core_id = index / ms->smp.threads % ms->smp.cores;
28
+ topo->thread_id = index % ms->smp.threads;
29
+}
30
+
31
+static unsigned int topo_align_up(unsigned int count)
32
+{
33
+ g_assert(count >= 1);
34
+ count -= 1;
35
+ return BIT(count ? 32 - clz32(count) : 0);
36
+}
37
+
38
+/*
39
+ * LoongArch Reference Manual Vol1, Chapter 7.4.12 CPU Identity
40
+ * For CPU architecture, bit0 .. bit8 is valid for CPU id, max cpuid is 512
41
+ * However for IPI/Eiointc interrupt controller, max supported cpu id for
42
+ * irq routingis 256
43
+ *
44
+ * Here max cpu id is 256 for virt machine
45
+ */
46
+static int virt_get_arch_id_from_topo(MachineState *ms, LoongArchCPUTopo *topo)
47
+{
48
+ int arch_id, threads, cores, sockets;
49
+
50
+ threads = topo_align_up(ms->smp.threads);
51
+ cores = topo_align_up(ms->smp.cores);
52
+ sockets = topo_align_up(ms->smp.sockets);
53
+ if ((threads * cores * sockets) > 256) {
54
+ error_report("Exceeding max cpuid 256 with sockets[%d] cores[%d]"
55
+ " threads[%d]", ms->smp.sockets, ms->smp.cores,
56
+ ms->smp.threads);
57
+ exit(1);
58
+ }
59
+
60
+ arch_id = topo->thread_id + topo->core_id * threads;
61
+ arch_id += topo->socket_id * threads * cores;
62
+ return arch_id;
63
+}
64
+
65
static bool memhp_type_supported(DeviceState *dev)
66
{
67
/* we only support pc dimm now */
68
@@ -XXX,XX +XXX,XX @@ static HotplugHandler *virt_get_hotplug_handler(MachineState *machine,
69
70
static const CPUArchIdList *virt_possible_cpu_arch_ids(MachineState *ms)
71
{
72
- int n;
73
+ int n, arch_id;
74
unsigned int max_cpus = ms->smp.max_cpus;
75
+ LoongArchCPUTopo topo;
76
77
if (ms->possible_cpus) {
78
assert(ms->possible_cpus->len == max_cpus);
79
@@ -XXX,XX +XXX,XX @@ static const CPUArchIdList *virt_possible_cpu_arch_ids(MachineState *ms)
80
sizeof(CPUArchId) * max_cpus);
81
ms->possible_cpus->len = max_cpus;
82
for (n = 0; n < ms->possible_cpus->len; n++) {
83
+ virt_get_topo_from_index(ms, &topo, n);
84
+ arch_id = virt_get_arch_id_from_topo(ms, &topo);
85
ms->possible_cpus->cpus[n].type = ms->cpu_type;
86
- ms->possible_cpus->cpus[n].arch_id = n;
87
-
88
+ ms->possible_cpus->cpus[n].arch_id = arch_id;
89
+ ms->possible_cpus->cpus[n].vcpus_count = 1;
90
ms->possible_cpus->cpus[n].props.has_socket_id = true;
91
- ms->possible_cpus->cpus[n].props.socket_id =
92
- n / (ms->smp.cores * ms->smp.threads);
93
+ ms->possible_cpus->cpus[n].props.socket_id = topo.socket_id;
94
ms->possible_cpus->cpus[n].props.has_core_id = true;
95
- ms->possible_cpus->cpus[n].props.core_id =
96
- n / ms->smp.threads % ms->smp.cores;
97
+ ms->possible_cpus->cpus[n].props.core_id = topo.core_id;
98
ms->possible_cpus->cpus[n].props.has_thread_id = true;
99
- ms->possible_cpus->cpus[n].props.thread_id = n % ms->smp.threads;
100
+ ms->possible_cpus->cpus[n].props.thread_id = topo.thread_id;
101
}
102
return ms->possible_cpus;
103
}
104
diff --git a/target/loongarch/cpu.h b/target/loongarch/cpu.h
105
index XXXXXXX..XXXXXXX 100644
106
--- a/target/loongarch/cpu.h
107
+++ b/target/loongarch/cpu.h
108
@@ -XXX,XX +XXX,XX @@ typedef struct CPUArchState {
109
#endif
110
} CPULoongArchState;
111
112
+typedef struct LoongArchCPUTopo {
113
+ int32_t socket_id; /* socket-id of this VCPU */
114
+ int32_t core_id; /* core-id of this VCPU */
115
+ int32_t thread_id; /* thread-id of this VCPU */
116
+} LoongArchCPUTopo;
117
+
118
/**
119
* LoongArchCPU:
120
* @env: #CPULoongArchState
121
--
122
2.43.5
diff view generated by jsdifflib
New patch
1
Add some properties such as socket_id, core_id, thread_id and node_id
2
on LoongArch CPU object.
1
3
4
Co-developed-by: Xianglai Li <lixianglai@loongson.cn>
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
6
---
7
target/loongarch/cpu.c | 9 +++++++++
8
target/loongarch/cpu.h | 4 ++++
9
2 files changed, 13 insertions(+)
10
11
diff --git a/target/loongarch/cpu.c b/target/loongarch/cpu.c
12
index XXXXXXX..XXXXXXX 100644
13
--- a/target/loongarch/cpu.c
14
+++ b/target/loongarch/cpu.c
15
@@ -XXX,XX +XXX,XX @@
16
#include "system/tcg.h"
17
#include "system/kvm.h"
18
#include "kvm/kvm_loongarch.h"
19
+#include "hw/qdev-properties.h"
20
#include "exec/exec-all.h"
21
#include "exec/translation-block.h"
22
#include "cpu.h"
23
@@ -XXX,XX +XXX,XX @@ static int64_t loongarch_cpu_get_arch_id(CPUState *cs)
24
}
25
#endif
26
27
+static const Property loongarch_cpu_properties[] = {
28
+ DEFINE_PROP_INT32("socket-id", LoongArchCPU, socket_id, 0),
29
+ DEFINE_PROP_INT32("core-id", LoongArchCPU, core_id, 0),
30
+ DEFINE_PROP_INT32("thread-id", LoongArchCPU, thread_id, 0),
31
+ DEFINE_PROP_INT32("node-id", LoongArchCPU, node_id, CPU_UNSET_NUMA_NODE_ID),
32
+};
33
+
34
static void loongarch_cpu_class_init(ObjectClass *c, void *data)
35
{
36
LoongArchCPUClass *lacc = LOONGARCH_CPU_CLASS(c);
37
@@ -XXX,XX +XXX,XX @@ static void loongarch_cpu_class_init(ObjectClass *c, void *data)
38
DeviceClass *dc = DEVICE_CLASS(c);
39
ResettableClass *rc = RESETTABLE_CLASS(c);
40
41
+ device_class_set_props(dc, loongarch_cpu_properties);
42
device_class_set_parent_realize(dc, loongarch_cpu_realizefn,
43
&lacc->parent_realize);
44
resettable_class_set_parent_phases(rc, NULL, loongarch_cpu_reset_hold, NULL,
45
diff --git a/target/loongarch/cpu.h b/target/loongarch/cpu.h
46
index XXXXXXX..XXXXXXX 100644
47
--- a/target/loongarch/cpu.h
48
+++ b/target/loongarch/cpu.h
49
@@ -XXX,XX +XXX,XX @@ struct ArchCPU {
50
OnOffAuto lasx;
51
OnOffAuto kvm_pv_ipi;
52
OnOffAuto kvm_steal_time;
53
+ int32_t socket_id; /* socket-id of this CPU */
54
+ int32_t core_id; /* core-id of this CPU */
55
+ int32_t thread_id; /* thread-id of this CPU */
56
+ int32_t node_id; /* NUMA node of this CPU */
57
58
/* 'compatible' string for this CPU for Linux device trees */
59
const char *dtb_compatible;
60
--
61
2.43.5
diff view generated by jsdifflib
1
From: Bibo Mao <maobibo@loongson.cn>
1
Add basic cpu hotplug interface framework, cpu hotplug interface is
2
stub function and only framework is added here.
2
3
3
Since migration test case requires compat machine type support,
4
Co-developed-by: Xianglai Li <lixianglai@loongson.cn>
4
compat machine is added for qemu 9.1 here.
5
6
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
7
Reviewed-by: Song Gao <gaosong@loongson.cn>
8
Message-Id: <20240511034220.3030560-2-maobibo@loongson.cn>
9
Signed-off-by: Song Gao <gaosong@loongson.cn>
10
---
6
---
11
hw/loongarch/virt.c | 61 +++++++++++++++++++++++++++++++++++----------
7
hw/loongarch/virt.c | 29 +++++++++++++++++++++++++++++
12
1 file changed, 48 insertions(+), 13 deletions(-)
8
target/loongarch/cpu.c | 13 +++++++++++++
9
target/loongarch/cpu.h | 1 +
10
3 files changed, 43 insertions(+)
13
11
14
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
12
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
15
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/loongarch/virt.c
14
--- a/hw/loongarch/virt.c
17
+++ b/hw/loongarch/virt.c
15
+++ b/hw/loongarch/virt.c
18
@@ -XXX,XX +XXX,XX @@ static void virt_class_init(ObjectClass *oc, void *data)
16
@@ -XXX,XX +XXX,XX @@ static int virt_get_arch_id_from_topo(MachineState *ms, LoongArchCPUTopo *topo)
19
#endif
17
return arch_id;
20
}
18
}
21
19
22
-static const TypeInfo virt_machine_types[] = {
20
+static void virt_cpu_pre_plug(HotplugHandler *hotplug_dev,
23
- {
21
+ DeviceState *dev, Error **errp)
24
- .name = TYPE_LOONGARCH_VIRT_MACHINE,
25
- .parent = TYPE_MACHINE,
26
- .instance_size = sizeof(LoongArchVirtMachineState),
27
- .class_init = virt_class_init,
28
- .instance_init = virt_initfn,
29
- .interfaces = (InterfaceInfo[]) {
30
- { TYPE_HOTPLUG_HANDLER },
31
- { }
32
- },
33
- }
34
+#define DEFINE_VIRT_MACHINE_LATEST(major, minor, latest) \
35
+ static void virt_##major##_##minor##_class_init(ObjectClass *oc, \
36
+ void *data) \
37
+ { \
38
+ MachineClass *mc = MACHINE_CLASS(oc); \
39
+ virt_machine_##major##_##minor##_options(mc); \
40
+ mc->desc = "QEMU " # major "." # minor " LoongArch Virtual Machine"; \
41
+ if (latest) { \
42
+ mc->alias = "virt"; \
43
+ } \
44
+ } \
45
+ static const TypeInfo machvirt_##major##_##minor##_info = { \
46
+ .name = MACHINE_TYPE_NAME("virt-" # major "." # minor), \
47
+ .parent = TYPE_LOONGARCH_VIRT_MACHINE, \
48
+ .class_init = virt_##major##_##minor##_class_init, \
49
+ }; \
50
+ static void machvirt_machine_##major##_##minor##_init(void) \
51
+ { \
52
+ type_register_static(&machvirt_##major##_##minor##_info); \
53
+ } \
54
+ type_init(machvirt_machine_##major##_##minor##_init);
55
+
56
+#define DEFINE_VIRT_MACHINE_AS_LATEST(major, minor) \
57
+ DEFINE_VIRT_MACHINE_LATEST(major, minor, true)
58
+#define DEFINE_VIRT_MACHINE(major, minor) \
59
+ DEFINE_VIRT_MACHINE_LATEST(major, minor, false)
60
+
61
+static const TypeInfo virt_machine_info = {
62
+ .name = TYPE_LOONGARCH_VIRT_MACHINE,
63
+ .parent = TYPE_MACHINE,
64
+ .abstract = true,
65
+ .instance_size = sizeof(LoongArchVirtMachineState),
66
+ .class_init = virt_class_init,
67
+ .instance_init = virt_initfn,
68
+ .interfaces = (InterfaceInfo[]) {
69
+ { TYPE_HOTPLUG_HANDLER },
70
+ { }
71
+ },
72
};
73
74
-DEFINE_TYPES(virt_machine_types)
75
+static void machvirt_machine_init(void)
76
+{
77
+ type_register_static(&virt_machine_info);
78
+}
79
+type_init(machvirt_machine_init);
80
+
81
+static void virt_machine_9_1_options(MachineClass *mc)
82
+{
22
+{
83
+}
23
+}
84
+DEFINE_VIRT_MACHINE_AS_LATEST(9, 1)
24
+
25
+static void virt_cpu_unplug_request(HotplugHandler *hotplug_dev,
26
+ DeviceState *dev, Error **errp)
27
+{
28
+}
29
+
30
+static void virt_cpu_unplug(HotplugHandler *hotplug_dev,
31
+ DeviceState *dev, Error **errp)
32
+{
33
+}
34
+
35
+static void virt_cpu_plug(HotplugHandler *hotplug_dev,
36
+ DeviceState *dev, Error **errp)
37
+{
38
+}
39
+
40
static bool memhp_type_supported(DeviceState *dev)
41
{
42
/* we only support pc dimm now */
43
@@ -XXX,XX +XXX,XX @@ static void virt_device_pre_plug(HotplugHandler *hotplug_dev,
44
{
45
if (memhp_type_supported(dev)) {
46
virt_mem_pre_plug(hotplug_dev, dev, errp);
47
+ } else if (object_dynamic_cast(OBJECT(dev), TYPE_LOONGARCH_CPU)) {
48
+ virt_cpu_pre_plug(hotplug_dev, dev, errp);
49
}
50
}
51
52
@@ -XXX,XX +XXX,XX @@ static void virt_device_unplug_request(HotplugHandler *hotplug_dev,
53
{
54
if (memhp_type_supported(dev)) {
55
virt_mem_unplug_request(hotplug_dev, dev, errp);
56
+ } else if (object_dynamic_cast(OBJECT(dev), TYPE_LOONGARCH_CPU)) {
57
+ virt_cpu_unplug_request(hotplug_dev, dev, errp);
58
}
59
}
60
61
@@ -XXX,XX +XXX,XX @@ static void virt_device_unplug(HotplugHandler *hotplug_dev,
62
{
63
if (memhp_type_supported(dev)) {
64
virt_mem_unplug(hotplug_dev, dev, errp);
65
+ } else if (object_dynamic_cast(OBJECT(dev), TYPE_LOONGARCH_CPU)) {
66
+ virt_cpu_unplug(hotplug_dev, dev, errp);
67
}
68
}
69
70
@@ -XXX,XX +XXX,XX @@ static void virt_device_plug_cb(HotplugHandler *hotplug_dev,
71
}
72
} else if (memhp_type_supported(dev)) {
73
virt_mem_plug(hotplug_dev, dev, errp);
74
+ } else if (object_dynamic_cast(OBJECT(dev), TYPE_LOONGARCH_CPU)) {
75
+ virt_cpu_plug(hotplug_dev, dev, errp);
76
}
77
}
78
79
@@ -XXX,XX +XXX,XX @@ static HotplugHandler *virt_get_hotplug_handler(MachineState *machine,
80
MachineClass *mc = MACHINE_GET_CLASS(machine);
81
82
if (device_is_dynamic_sysbus(mc, dev) ||
83
+ object_dynamic_cast(OBJECT(dev), TYPE_LOONGARCH_CPU) ||
84
object_dynamic_cast(OBJECT(dev), TYPE_VIRTIO_IOMMU_PCI) ||
85
memhp_type_supported(dev)) {
86
return HOTPLUG_HANDLER(machine);
87
diff --git a/target/loongarch/cpu.c b/target/loongarch/cpu.c
88
index XXXXXXX..XXXXXXX 100644
89
--- a/target/loongarch/cpu.c
90
+++ b/target/loongarch/cpu.c
91
@@ -XXX,XX +XXX,XX @@ static void loongarch_cpu_realizefn(DeviceState *dev, Error **errp)
92
lacc->parent_realize(dev, errp);
93
}
94
95
+static void loongarch_cpu_unrealizefn(DeviceState *dev)
96
+{
97
+ LoongArchCPUClass *lacc = LOONGARCH_CPU_GET_CLASS(dev);
98
+
99
+#ifndef CONFIG_USER_ONLY
100
+ cpu_remove_sync(CPU(dev));
101
+#endif
102
+
103
+ lacc->parent_unrealize(dev);
104
+}
105
+
106
static bool loongarch_get_lsx(Object *obj, Error **errp)
107
{
108
return LOONGARCH_CPU(obj)->lsx != ON_OFF_AUTO_OFF;
109
@@ -XXX,XX +XXX,XX @@ static void loongarch_cpu_class_init(ObjectClass *c, void *data)
110
device_class_set_props(dc, loongarch_cpu_properties);
111
device_class_set_parent_realize(dc, loongarch_cpu_realizefn,
112
&lacc->parent_realize);
113
+ device_class_set_parent_unrealize(dc, loongarch_cpu_unrealizefn,
114
+ &lacc->parent_unrealize);
115
resettable_class_set_parent_phases(rc, NULL, loongarch_cpu_reset_hold, NULL,
116
&lacc->parent_phases);
117
118
diff --git a/target/loongarch/cpu.h b/target/loongarch/cpu.h
119
index XXXXXXX..XXXXXXX 100644
120
--- a/target/loongarch/cpu.h
121
+++ b/target/loongarch/cpu.h
122
@@ -XXX,XX +XXX,XX @@ struct LoongArchCPUClass {
123
CPUClass parent_class;
124
125
DeviceRealize parent_realize;
126
+ DeviceUnrealize parent_unrealize;
127
ResettablePhases parent_phases;
128
};
129
85
--
130
--
86
2.34.1
131
2.43.5
diff view generated by jsdifflib
New patch
1
Implement cpu unplug interfaces including virt_cpu_unplug_request()
2
and virt_cpu_unplug().
1
3
4
Co-developed-by: Xianglai Li <lixianglai@loongson.cn>
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
6
---
7
hw/loongarch/virt.c | 58 +++++++++++++++++++++++++++++++++++++++++++++
8
1 file changed, 58 insertions(+)
9
10
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
11
index XXXXXXX..XXXXXXX 100644
12
--- a/hw/loongarch/virt.c
13
+++ b/hw/loongarch/virt.c
14
@@ -XXX,XX +XXX,XX @@ static int virt_get_arch_id_from_topo(MachineState *ms, LoongArchCPUTopo *topo)
15
return arch_id;
16
}
17
18
+/* Find cpu slot in machine->possible_cpus by arch_id */
19
+static CPUArchId *virt_find_cpu_slot(MachineState *ms, int arch_id)
20
+{
21
+ int n;
22
+ for (n = 0; n < ms->possible_cpus->len; n++) {
23
+ if (ms->possible_cpus->cpus[n].arch_id == arch_id) {
24
+ return &ms->possible_cpus->cpus[n];
25
+ }
26
+ }
27
+
28
+ return NULL;
29
+}
30
+
31
static void virt_cpu_pre_plug(HotplugHandler *hotplug_dev,
32
DeviceState *dev, Error **errp)
33
{
34
@@ -XXX,XX +XXX,XX @@ static void virt_cpu_pre_plug(HotplugHandler *hotplug_dev,
35
static void virt_cpu_unplug_request(HotplugHandler *hotplug_dev,
36
DeviceState *dev, Error **errp)
37
{
38
+ LoongArchVirtMachineState *lvms = LOONGARCH_VIRT_MACHINE(hotplug_dev);
39
+ Error *err = NULL;
40
+ LoongArchCPU *cpu = LOONGARCH_CPU(dev);
41
+ CPUState *cs = CPU(dev);
42
+
43
+ if (cs->cpu_index == 0) {
44
+ error_setg(&err, "hot-unplug of boot cpu(id%d=%d:%d:%d) not supported",
45
+ cs->cpu_index, cpu->socket_id,
46
+ cpu->core_id, cpu->thread_id);
47
+ error_propagate(errp, err);
48
+ return;
49
+ }
50
+
51
+ hotplug_handler_unplug_request(HOTPLUG_HANDLER(lvms->acpi_ged), dev, &err);
52
+ if (err) {
53
+ error_propagate(errp, err);
54
+ }
55
}
56
57
static void virt_cpu_unplug(HotplugHandler *hotplug_dev,
58
DeviceState *dev, Error **errp)
59
{
60
+ CPUArchId *cpu_slot;
61
+ Error *err = NULL;
62
+ LoongArchCPU *cpu = LOONGARCH_CPU(dev);
63
+ LoongArchVirtMachineState *lvms = LOONGARCH_VIRT_MACHINE(hotplug_dev);
64
+
65
+ /* Notify ipi and extioi irqchip to remove interrupt routing to CPU */
66
+ hotplug_handler_unplug(HOTPLUG_HANDLER(lvms->ipi), dev, &err);
67
+ if (err) {
68
+ error_propagate(errp, err);
69
+ return;
70
+ }
71
+
72
+ hotplug_handler_unplug(HOTPLUG_HANDLER(lvms->extioi), dev, &err);
73
+ if (err) {
74
+ error_propagate(errp, err);
75
+ return;
76
+ }
77
+
78
+ /* Notify acpi ged CPU removed */
79
+ hotplug_handler_unplug(HOTPLUG_HANDLER(lvms->acpi_ged), dev, &err);
80
+ if (err) {
81
+ error_propagate(errp, err);
82
+ return;
83
+ }
84
+
85
+ cpu_slot = virt_find_cpu_slot(MACHINE(lvms), cpu->phy_id);
86
+ cpu_slot->cpu = NULL;
87
+ return;
88
}
89
90
static void virt_cpu_plug(HotplugHandler *hotplug_dev,
91
--
92
2.43.5
diff view generated by jsdifflib
1
On kvm side, get_fpu/set_fpu save the vreg registers high 192bits,
1
Implement cpu plug interface, and cold-plug cpu uses plug interface
2
but QEMU missing.
2
when cpu object is created.
3
3
4
Signed-off-by: Song Gao <gaosong@loongson.cn>
4
Co-developed-by: Xianglai Li <lixianglai@loongson.cn>
5
Reviewed-by: Bibo Mao <maobibo@loongson.cn>
5
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
6
Message-Id: <20240514110752.989572-1-gaosong@loongson.cn>
7
---
6
---
8
target/loongarch/kvm/kvm.c | 6 ++++++
7
hw/loongarch/virt.c | 88 ++++++++++++++++++++++++++++++++++++------
9
1 file changed, 6 insertions(+)
8
target/loongarch/cpu.c | 1 +
9
2 files changed, 78 insertions(+), 11 deletions(-)
10
10
11
diff --git a/target/loongarch/kvm/kvm.c b/target/loongarch/kvm/kvm.c
11
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
12
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
13
--- a/target/loongarch/kvm/kvm.c
13
--- a/hw/loongarch/virt.c
14
+++ b/target/loongarch/kvm/kvm.c
14
+++ b/hw/loongarch/virt.c
15
@@ -XXX,XX +XXX,XX @@ static int kvm_loongarch_get_regs_fp(CPUState *cs)
15
@@ -XXX,XX +XXX,XX @@ static void fw_cfg_add_memory(MachineState *ms)
16
env->fcsr0 = fpu.fcsr;
16
17
for (i = 0; i < 32; i++) {
17
static void virt_init(MachineState *machine)
18
env->fpr[i].vreg.UD[0] = fpu.fpr[i].val64[0];
18
{
19
+ env->fpr[i].vreg.UD[1] = fpu.fpr[i].val64[1];
19
- LoongArchCPU *lacpu;
20
+ env->fpr[i].vreg.UD[2] = fpu.fpr[i].val64[2];
20
const char *cpu_model = machine->cpu_type;
21
+ env->fpr[i].vreg.UD[3] = fpu.fpr[i].val64[3];
21
MemoryRegion *address_space_mem = get_system_memory();
22
LoongArchVirtMachineState *lvms = LOONGARCH_VIRT_MACHINE(machine);
23
int i;
24
hwaddr base, size, ram_size = machine->ram_size;
25
- const CPUArchIdList *possible_cpus;
26
MachineClass *mc = MACHINE_GET_CLASS(machine);
27
- CPUState *cpu;
28
+ Object *cpuobj;
29
30
if (!cpu_model) {
31
cpu_model = LOONGARCH_CPU_TYPE_NAME("la464");
32
@@ -XXX,XX +XXX,XX @@ static void virt_init(MachineState *machine)
33
memory_region_add_subregion(&lvms->system_iocsr, 0, &lvms->iocsr_mem);
34
35
/* Init CPUs */
36
- possible_cpus = mc->possible_cpu_arch_ids(machine);
37
- for (i = 0; i < possible_cpus->len; i++) {
38
- cpu = cpu_create(machine->cpu_type);
39
- cpu->cpu_index = i;
40
- machine->possible_cpus->cpus[i].cpu = cpu;
41
- lacpu = LOONGARCH_CPU(cpu);
42
- lacpu->phy_id = machine->possible_cpus->cpus[i].arch_id;
43
- lacpu->env.address_space_iocsr = &lvms->as_iocsr;
44
+ mc->possible_cpu_arch_ids(machine);
45
+ for (i = 0; i < machine->smp.cpus; i++) {
46
+ cpuobj = object_new(machine->cpu_type);
47
+ if (cpuobj == NULL) {
48
+ error_report("Fail to create object with type %s ",
49
+ machine->cpu_type);
50
+ exit(EXIT_FAILURE);
51
+ }
52
+ qdev_realize_and_unref(DEVICE(cpuobj), NULL, &error_fatal);
22
}
53
}
23
for (i = 0; i < 8; i++) {
54
fw_cfg_add_memory(machine);
24
env->cf[i] = fpu.fcc & 0xFF;
55
25
@@ -XXX,XX +XXX,XX @@ static int kvm_loongarch_put_regs_fp(CPUState *cs)
56
@@ -XXX,XX +XXX,XX @@ static CPUArchId *virt_find_cpu_slot(MachineState *ms, int arch_id)
26
fpu.fcc = 0;
57
return NULL;
27
for (i = 0; i < 32; i++) {
58
}
28
fpu.fpr[i].val64[0] = env->fpr[i].vreg.UD[0];
59
29
+ fpu.fpr[i].val64[1] = env->fpr[i].vreg.UD[1];
60
+/* Find cpu slot for cold-plut CPU object where cpu is NULL */
30
+ fpu.fpr[i].val64[2] = env->fpr[i].vreg.UD[2];
61
+static CPUArchId *virt_find_empty_cpu_slot(MachineState *ms)
31
+ fpu.fpr[i].val64[3] = env->fpr[i].vreg.UD[3];
62
+{
32
}
63
+ int n;
33
64
+ for (n = 0; n < ms->possible_cpus->len; n++) {
34
for (i = 0; i < 8; i++) {
65
+ if (ms->possible_cpus->cpus[n].cpu == NULL) {
66
+ return &ms->possible_cpus->cpus[n];
67
+ }
68
+ }
69
+
70
+ return NULL;
71
+}
72
+
73
static void virt_cpu_pre_plug(HotplugHandler *hotplug_dev,
74
DeviceState *dev, Error **errp)
75
{
76
+ LoongArchVirtMachineState *lvms = LOONGARCH_VIRT_MACHINE(hotplug_dev);
77
+ MachineState *ms = MACHINE(OBJECT(hotplug_dev));
78
+ LoongArchCPU *cpu = LOONGARCH_CPU(dev);
79
+ CPUState *cs = CPU(dev);
80
+ CPUArchId *cpu_slot;
81
+ Error *err = NULL;
82
+ LoongArchCPUTopo topo;
83
+
84
+ if (lvms->acpi_ged) {
85
+ error_setg(&err, "CPU hotplug not supported");
86
+ goto out;
87
+ } else {
88
+ /* For cold-add cpu, find empty cpu slot */
89
+ cpu_slot = virt_find_empty_cpu_slot(ms);
90
+ topo.socket_id = cpu_slot->props.socket_id;
91
+ topo.core_id = cpu_slot->props.core_id;
92
+ topo.thread_id = cpu_slot->props.thread_id;
93
+ object_property_set_int(OBJECT(dev), "socket-id", topo.socket_id, NULL);
94
+ object_property_set_int(OBJECT(dev), "core-id", topo.core_id, NULL);
95
+ object_property_set_int(OBJECT(dev), "thread-id", topo.thread_id, NULL);
96
+ }
97
+
98
+ cpu->env.address_space_iocsr = &lvms->as_iocsr;
99
+ cpu->phy_id = cpu_slot->arch_id;
100
+ cs->cpu_index = cpu_slot - ms->possible_cpus->cpus;
101
+ numa_cpu_pre_plug(cpu_slot, dev, &err);
102
+out:
103
+ if (err) {
104
+ error_propagate(errp, err);
105
+ }
106
}
107
108
static void virt_cpu_unplug_request(HotplugHandler *hotplug_dev,
109
@@ -XXX,XX +XXX,XX @@ static void virt_cpu_unplug(HotplugHandler *hotplug_dev,
110
static void virt_cpu_plug(HotplugHandler *hotplug_dev,
111
DeviceState *dev, Error **errp)
112
{
113
+ CPUArchId *cpu_slot;
114
+ LoongArchCPU *cpu = LOONGARCH_CPU(dev);
115
+ LoongArchVirtMachineState *lvms = LOONGARCH_VIRT_MACHINE(hotplug_dev);
116
+ Error *err = NULL;
117
+
118
+ cpu_slot = virt_find_cpu_slot(MACHINE(lvms), cpu->phy_id);
119
+ cpu_slot->cpu = CPU(dev);
120
+ if (lvms->ipi) {
121
+ hotplug_handler_plug(HOTPLUG_HANDLER(lvms->ipi), dev, &err);
122
+ if (err) {
123
+ error_propagate(errp, err);
124
+ return;
125
+ }
126
+ }
127
+
128
+ if (lvms->extioi) {
129
+ hotplug_handler_plug(HOTPLUG_HANDLER(lvms->extioi), dev, &err);
130
+ if (err) {
131
+ error_propagate(errp, err);
132
+ return;
133
+ }
134
+ }
135
+
136
+ return;
137
}
138
139
static bool memhp_type_supported(DeviceState *dev)
140
diff --git a/target/loongarch/cpu.c b/target/loongarch/cpu.c
141
index XXXXXXX..XXXXXXX 100644
142
--- a/target/loongarch/cpu.c
143
+++ b/target/loongarch/cpu.c
144
@@ -XXX,XX +XXX,XX @@ static void loongarch_cpu_class_init(ObjectClass *c, void *data)
145
#ifdef CONFIG_TCG
146
cc->tcg_ops = &loongarch_tcg_ops;
147
#endif
148
+ dc->user_creatable = true;
149
}
150
151
static const gchar *loongarch32_gdb_arch_name(CPUState *cs)
35
--
152
--
36
2.34.1
153
2.43.5
diff view generated by jsdifflib
1
vmstate does not save kvm_state_conter,
1
On LoongArch virt machine, ACPI GED hardware is used for CPU hotplug
2
which can cause VM recovery from disk to fail.
2
handler, here CPU hotplug support feature is added based on GED handler,
3
also CPU scan and reject method is added about CPU device in DSDT table.
3
4
4
Cc: qemu-stable@nongnu.org
5
Co-developed-by: Xianglai Li <lixianglai@loongson.cn>
5
Signed-off-by: Song Gao <gaosong@loongson.cn>
6
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
6
Acked-by: Peter Xu <peterx@redhat.com>
7
Message-Id: <20240508024732.3127792-1-gaosong@loongson.cn>
8
---
7
---
9
target/loongarch/machine.c | 6 ++++--
8
hw/loongarch/Kconfig | 1 +
10
1 file changed, 4 insertions(+), 2 deletions(-)
9
hw/loongarch/virt-acpi-build.c | 35 +++++++++++++++++++++++++++++++---
10
hw/loongarch/virt.c | 10 ++++++++++
11
include/hw/loongarch/virt.h | 1 +
12
4 files changed, 44 insertions(+), 3 deletions(-)
11
13
12
diff --git a/target/loongarch/machine.c b/target/loongarch/machine.c
14
diff --git a/hw/loongarch/Kconfig b/hw/loongarch/Kconfig
13
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
14
--- a/target/loongarch/machine.c
16
--- a/hw/loongarch/Kconfig
15
+++ b/target/loongarch/machine.c
17
+++ b/hw/loongarch/Kconfig
16
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_tlb = {
18
@@ -XXX,XX +XXX,XX @@ config LOONGARCH_VIRT
17
/* LoongArch CPU state */
19
select LOONGARCH_EXTIOI
18
const VMStateDescription vmstate_loongarch_cpu = {
20
select LS7A_RTC
19
.name = "cpu",
21
select SMBIOS
20
- .version_id = 1,
22
+ select ACPI_CPU_HOTPLUG
21
- .minimum_version_id = 1,
23
select ACPI_PCI
22
+ .version_id = 2,
24
select ACPI_HW_REDUCED
23
+ .minimum_version_id = 2,
25
select FW_CFG_DMA
24
.fields = (const VMStateField[]) {
26
diff --git a/hw/loongarch/virt-acpi-build.c b/hw/loongarch/virt-acpi-build.c
25
VMSTATE_UINTTL_ARRAY(env.gpr, LoongArchCPU, 32),
27
index XXXXXXX..XXXXXXX 100644
26
VMSTATE_UINTTL(env.pc, LoongArchCPU),
28
--- a/hw/loongarch/virt-acpi-build.c
27
@@ -XXX,XX +XXX,XX @@ const VMStateDescription vmstate_loongarch_cpu = {
29
+++ b/hw/loongarch/virt-acpi-build.c
28
VMSTATE_UINT64(env.CSR_DERA, LoongArchCPU),
30
@@ -XXX,XX +XXX,XX @@
29
VMSTATE_UINT64(env.CSR_DSAVE, LoongArchCPU),
31
#define ACPI_BUILD_DPRINTF(fmt, ...)
30
32
#endif
31
+ VMSTATE_UINT64(kvm_state_counter, LoongArchCPU),
33
34
+static void virt_madt_cpu_entry(int uid,
35
+ const CPUArchIdList *apic_ids,
36
+ GArray *entry, bool force_enabled)
37
+{
38
+ uint32_t flags, apic_id = apic_ids->cpus[uid].arch_id;
32
+
39
+
33
VMSTATE_END_OF_LIST()
40
+ flags = apic_ids->cpus[uid].cpu || force_enabled ? 1 /* Enabled */ : 0;
34
},
41
+
35
.subsections = (const VMStateDescription * const []) {
42
+ /* Rev 1.0b, Table 5-13 Processor Local APIC Structure */
43
+ build_append_int_noprefix(entry, 0, 1); /* Type */
44
+ build_append_int_noprefix(entry, 8, 1); /* Length */
45
+ build_append_int_noprefix(entry, uid, 1); /* ACPI Processor ID */
46
+ build_append_int_noprefix(entry, apic_id, 1); /* APIC ID */
47
+ build_append_int_noprefix(entry, flags, 4); /* Flags */
48
+}
49
+
50
/* build FADT */
51
static void init_common_fadt_data(AcpiFadtData *data)
52
{
53
@@ -XXX,XX +XXX,XX @@ build_madt(GArray *table_data, BIOSLinker *linker,
54
MachineState *ms = MACHINE(lvms);
55
MachineClass *mc = MACHINE_GET_CLASS(ms);
56
const CPUArchIdList *arch_ids = mc->possible_cpu_arch_ids(ms);
57
- int i, arch_id;
58
+ int i, arch_id, flags;
59
AcpiTable table = { .sig = "APIC", .rev = 1, .oem_id = lvms->oem_id,
60
.oem_table_id = lvms->oem_table_id };
61
62
@@ -XXX,XX +XXX,XX @@ build_madt(GArray *table_data, BIOSLinker *linker,
63
for (i = 0; i < arch_ids->len; i++) {
64
/* Processor Core Interrupt Controller Structure */
65
arch_id = arch_ids->cpus[i].arch_id;
66
-
67
+ flags = arch_ids->cpus[i].cpu ? 1 : 0;
68
build_append_int_noprefix(table_data, 17, 1); /* Type */
69
build_append_int_noprefix(table_data, 15, 1); /* Length */
70
build_append_int_noprefix(table_data, 1, 1); /* Version */
71
build_append_int_noprefix(table_data, i, 4); /* ACPI Processor ID */
72
build_append_int_noprefix(table_data, arch_id, 4); /* Core ID */
73
- build_append_int_noprefix(table_data, 1, 4); /* Flags */
74
+ build_append_int_noprefix(table_data, flags, 4); /* Flags */
75
}
76
77
/* Extend I/O Interrupt Controller Structure */
78
@@ -XXX,XX +XXX,XX @@ build_la_ged_aml(Aml *dsdt, MachineState *machine)
79
{
80
uint32_t event;
81
LoongArchVirtMachineState *lvms = LOONGARCH_VIRT_MACHINE(machine);
82
+ CPUHotplugFeatures opts;
83
84
build_ged_aml(dsdt, "\\_SB."GED_DEVICE,
85
HOTPLUG_HANDLER(lvms->acpi_ged),
86
@@ -XXX,XX +XXX,XX @@ build_la_ged_aml(Aml *dsdt, MachineState *machine)
87
AML_SYSTEM_MEMORY,
88
VIRT_GED_MEM_ADDR);
89
}
90
+
91
+ if (event & ACPI_GED_CPU_HOTPLUG_EVT) {
92
+ opts.acpi_1_compatible = false;
93
+ opts.has_legacy_cphp = false;
94
+ opts.fw_unplugs_cpu = false;
95
+ opts.smi_path = NULL;
96
+
97
+ build_cpus_aml(dsdt, machine, opts, virt_madt_cpu_entry,
98
+ VIRT_GED_CPUHP_ADDR, "\\_SB",
99
+ AML_GED_EVT_CPU_SCAN_METHOD, AML_SYSTEM_MEMORY);
100
+ }
101
+
102
acpi_dsdt_add_power_button(dsdt);
103
}
104
105
diff --git a/hw/loongarch/virt.c b/hw/loongarch/virt.c
106
index XXXXXXX..XXXXXXX 100644
107
--- a/hw/loongarch/virt.c
108
+++ b/hw/loongarch/virt.c
109
@@ -XXX,XX +XXX,XX @@ static DeviceState *create_acpi_ged(DeviceState *pch_pic,
110
{
111
DeviceState *dev;
112
MachineState *ms = MACHINE(lvms);
113
+ MachineClass *mc = MACHINE_GET_CLASS(lvms);
114
uint32_t event = ACPI_GED_PWR_DOWN_EVT;
115
116
if (ms->ram_slots) {
117
event |= ACPI_GED_MEM_HOTPLUG_EVT;
118
}
119
+
120
+ if (mc->has_hotpluggable_cpus) {
121
+ event |= ACPI_GED_CPU_HOTPLUG_EVT;
122
+ }
123
+
124
dev = qdev_new(TYPE_ACPI_GED);
125
qdev_prop_set_uint32(dev, "ged-event", event);
126
sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
127
@@ -XXX,XX +XXX,XX @@ static DeviceState *create_acpi_ged(DeviceState *pch_pic,
128
/* ged regs used for reset and power down */
129
sysbus_mmio_map(SYS_BUS_DEVICE(dev), 2, VIRT_GED_REG_ADDR);
130
131
+ if (mc->has_hotpluggable_cpus) {
132
+ sysbus_mmio_map(SYS_BUS_DEVICE(dev), 3, VIRT_GED_CPUHP_ADDR);
133
+ }
134
+
135
sysbus_connect_irq(SYS_BUS_DEVICE(dev), 0,
136
qdev_get_gpio_in(pch_pic, VIRT_SCI_IRQ - VIRT_GSI_BASE));
137
return dev;
138
diff --git a/include/hw/loongarch/virt.h b/include/hw/loongarch/virt.h
139
index XXXXXXX..XXXXXXX 100644
140
--- a/include/hw/loongarch/virt.h
141
+++ b/include/hw/loongarch/virt.h
142
@@ -XXX,XX +XXX,XX @@
143
#define VIRT_GED_EVT_ADDR 0x100e0000
144
#define VIRT_GED_MEM_ADDR (VIRT_GED_EVT_ADDR + ACPI_GED_EVT_SEL_LEN)
145
#define VIRT_GED_REG_ADDR (VIRT_GED_MEM_ADDR + MEMORY_HOTPLUG_IO_LEN)
146
+#define VIRT_GED_CPUHP_ADDR (VIRT_GED_REG_ADDR + ACPI_GED_REG_COUNT)
147
148
#define COMMAND_LINE_SIZE 512
149
36
--
150
--
37
2.34.1
151
2.43.5
diff view generated by jsdifflib