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 |