* [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu
@ 2014-03-20 6:33 Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 1/3] cpu: introduce CpuTopoInfo structure for argument simplification Chen Fan
` (3 more replies)
0 siblings, 4 replies; 5+ messages in thread
From: Chen Fan @ 2014-03-20 6:33 UTC (permalink / raw)
To: Eduardo Habkost, qemu-devel; +Cc: Igor Mammedov, Andreas Färber
at present, after hotplug a discontinuous cpu id on source, then done migration,
on target, it will fail to add the unoccupied cpu id which was skipped at source,
this cause is on target Qemu prebuild CPU with continuous cpu_index. so after
migration, the cpu infrastructure bewteen source and target are different.
I suppose we could use apic_id as instance_id which was used at registering vmstate
when create cpu. on target, we prebuild the specified cpu topology using comand line:
-device /machine/node[]/socket[]/core[]/cpu[], then migration, we could keep the same
cpu infrastructure on both side.
v1-v2: squash patch 2/4 and 3/4 together.
RFC:
V4: rename CpuTopoInfo to X86CPUTopoInfo. and move cpu_exsit() to pc_new_cpu().
V3: get rid of thread object and tie link<cpu> to <core> directly. and prebuild full
core[] and thread[] as init socket[] according to smp_cores and smp_threads.
TODO:
1. add cpu "path" property which used for specifying the QOM path.
2. add -device cpu-foo.path supported.
3. then we could introduce hot-remove cpu probably.
I don't know wether this way is right or not. pls tell me. :)
Chen Fan (3):
cpu: introduce CpuTopoInfo structure for argument simplification
i386: use CpuTopoInfo instead apic_id as argument for pc_new_cpu()
i386: introduce cpu QOM hierarchy tree
hw/i386/pc.c | 25 +++---
target-i386/Makefile.objs | 2 +-
target-i386/cpu-topology.c | 199 +++++++++++++++++++++++++++++++++++++++++++++
target-i386/cpu-topology.h | 71 ++++++++++++++++
target-i386/cpu.c | 55 +++++++++----
target-i386/cpu.h | 8 ++
target-i386/topology.h | 53 +++++++-----
tests/test-x86-cpuid.c | 165 +++++++++++++++++++++++++++++--------
8 files changed, 494 insertions(+), 84 deletions(-)
create mode 100644 target-i386/cpu-topology.c
create mode 100644 target-i386/cpu-topology.h
--
1.8.1.4
^ permalink raw reply [flat|nested] 5+ messages in thread
* [Qemu-devel] [PATCH v2 1/3] cpu: introduce CpuTopoInfo structure for argument simplification
2014-03-20 6:33 [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu Chen Fan
@ 2014-03-20 6:33 ` Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 2/3] i386: use CpuTopoInfo instead apic_id as argument for pc_new_cpu() Chen Fan
` (2 subsequent siblings)
3 siblings, 0 replies; 5+ messages in thread
From: Chen Fan @ 2014-03-20 6:33 UTC (permalink / raw)
To: Eduardo Habkost, qemu-devel; +Cc: Igor Mammedov, Andreas Färber
Signed-off-by: Chen Fan <chen.fan.fnst@cn.fujitsu.com>
Reviewed-by: Eduardo Habkost <ehabkost@redhat.com>
---
target-i386/topology.h | 33 +++++++++++++++++----------------
1 file changed, 17 insertions(+), 16 deletions(-)
diff --git a/target-i386/topology.h b/target-i386/topology.h
index 07a6c5f..e9ff89c 100644
--- a/target-i386/topology.h
+++ b/target-i386/topology.h
@@ -47,6 +47,12 @@
*/
typedef uint32_t apic_id_t;
+typedef struct X86CPUTopoInfo {
+ unsigned pkg_id;
+ unsigned core_id;
+ unsigned smt_id;
+} X86CPUTopoInfo;
+
/* Return the bit width needed for 'count' IDs
*/
static unsigned apicid_bitwidth_for_count(unsigned count)
@@ -92,13 +98,11 @@ static inline unsigned apicid_pkg_offset(unsigned nr_cores, unsigned nr_threads)
*/
static inline apic_id_t apicid_from_topo_ids(unsigned nr_cores,
unsigned nr_threads,
- unsigned pkg_id,
- unsigned core_id,
- unsigned smt_id)
+ const X86CPUTopoInfo *topo)
{
- return (pkg_id << apicid_pkg_offset(nr_cores, nr_threads)) |
- (core_id << apicid_core_offset(nr_cores, nr_threads)) |
- smt_id;
+ return (topo->pkg_id << apicid_pkg_offset(nr_cores, nr_threads)) |
+ (topo->core_id << apicid_core_offset(nr_cores, nr_threads)) |
+ topo->smt_id;
}
/* Calculate thread/core/package IDs for a specific topology,
@@ -107,14 +111,12 @@ static inline apic_id_t apicid_from_topo_ids(unsigned nr_cores,
static inline void x86_topo_ids_from_idx(unsigned nr_cores,
unsigned nr_threads,
unsigned cpu_index,
- unsigned *pkg_id,
- unsigned *core_id,
- unsigned *smt_id)
+ X86CPUTopoInfo *topo)
{
unsigned core_index = cpu_index / nr_threads;
- *smt_id = cpu_index % nr_threads;
- *core_id = core_index % nr_cores;
- *pkg_id = core_index / nr_cores;
+ topo->smt_id = cpu_index % nr_threads;
+ topo->core_id = core_index % nr_cores;
+ topo->pkg_id = core_index / nr_cores;
}
/* Make APIC ID for the CPU 'cpu_index'
@@ -125,10 +127,9 @@ static inline apic_id_t x86_apicid_from_cpu_idx(unsigned nr_cores,
unsigned nr_threads,
unsigned cpu_index)
{
- unsigned pkg_id, core_id, smt_id;
- x86_topo_ids_from_idx(nr_cores, nr_threads, cpu_index,
- &pkg_id, &core_id, &smt_id);
- return apicid_from_topo_ids(nr_cores, nr_threads, pkg_id, core_id, smt_id);
+ X86CPUTopoInfo topo;
+ x86_topo_ids_from_idx(nr_cores, nr_threads, cpu_index, &topo);
+ return apicid_from_topo_ids(nr_cores, nr_threads, &topo);
}
#endif /* TARGET_I386_TOPOLOGY_H */
--
1.8.1.4
^ permalink raw reply related [flat|nested] 5+ messages in thread
* [Qemu-devel] [PATCH v2 2/3] i386: use CpuTopoInfo instead apic_id as argument for pc_new_cpu()
2014-03-20 6:33 [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 1/3] cpu: introduce CpuTopoInfo structure for argument simplification Chen Fan
@ 2014-03-20 6:33 ` Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 3/3] i386: introduce cpu QOM hierarchy tree Chen Fan
2014-04-01 8:00 ` [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu chen.fan.fnst
3 siblings, 0 replies; 5+ messages in thread
From: Chen Fan @ 2014-03-20 6:33 UTC (permalink / raw)
To: Eduardo Habkost, qemu-devel; +Cc: Igor Mammedov, Andreas Färber
introduce x86_cpu_topo_ids_from_index() to calculate the cpu topology
information, and the compat old mode mechanism moved into there.
remove unused funciton x86_apicid_from_cpu_idx().
Unit tests included.
Signed-off-by: Chen Fan <chen.fan.fnst@cn.fujitsu.com>
---
hw/i386/pc.c | 22 ++++---
target-i386/cpu.c | 33 +++++-----
target-i386/cpu.h | 5 ++
target-i386/topology.h | 26 +++++---
tests/test-x86-cpuid.c | 165 ++++++++++++++++++++++++++++++++++++++-----------
5 files changed, 181 insertions(+), 70 deletions(-)
diff --git a/hw/i386/pc.c b/hw/i386/pc.c
index e715a33..84a017e 100644
--- a/hw/i386/pc.c
+++ b/hw/i386/pc.c
@@ -927,7 +927,7 @@ void pc_acpi_smi_interrupt(void *opaque, int irq, int level)
}
}
-static X86CPU *pc_new_cpu(const char *cpu_model, int64_t apic_id,
+static X86CPU *pc_new_cpu(const char *cpu_model, X86CPUTopoInfo *topo,
DeviceState *icc_bridge, Error **errp)
{
X86CPU *cpu;
@@ -939,7 +939,7 @@ static X86CPU *pc_new_cpu(const char *cpu_model, int64_t apic_id,
return NULL;
}
- object_property_set_int(OBJECT(cpu), apic_id, "apic-id", &local_err);
+ object_property_set_int(OBJECT(cpu), topo->apic_id, "apic-id", &local_err);
object_property_set_bool(OBJECT(cpu), true, "realized", &local_err);
if (local_err) {
@@ -955,28 +955,29 @@ static const char *current_cpu_model;
void pc_hot_add_cpu(const int64_t id, Error **errp)
{
DeviceState *icc_bridge;
- int64_t apic_id = x86_cpu_apic_id_from_index(id);
+ X86CPUTopoInfo topo;
if (id < 0) {
error_setg(errp, "Invalid CPU id: %" PRIi64, id);
return;
}
- if (cpu_exists(apic_id)) {
+ if (id >= max_cpus) {
error_setg(errp, "Unable to add CPU: %" PRIi64
- ", it already exists", id);
+ ", max allowed: %d", id, max_cpus - 1);
return;
}
- if (id >= max_cpus) {
+ x86_cpu_topo_ids_from_index(id, &topo);
+ if (cpu_exists(topo.apic_id)) {
error_setg(errp, "Unable to add CPU: %" PRIi64
- ", max allowed: %d", id, max_cpus - 1);
+ ", it already exists", id);
return;
}
icc_bridge = DEVICE(object_resolve_path_type("icc-bridge",
TYPE_ICC_BRIDGE, NULL));
- pc_new_cpu(current_cpu_model, apic_id, icc_bridge, errp);
+ pc_new_cpu(current_cpu_model, &topo, icc_bridge, errp);
}
void pc_cpus_init(const char *cpu_model, DeviceState *icc_bridge)
@@ -996,8 +997,9 @@ void pc_cpus_init(const char *cpu_model, DeviceState *icc_bridge)
current_cpu_model = cpu_model;
for (i = 0; i < smp_cpus; i++) {
- cpu = pc_new_cpu(cpu_model, x86_cpu_apic_id_from_index(i),
- icc_bridge, &error);
+ X86CPUTopoInfo topo;
+ x86_cpu_topo_ids_from_index(i, &topo);
+ cpu = pc_new_cpu(cpu_model, &topo, icc_bridge, &error);
if (error) {
error_report("%s", error_get_pretty(error));
error_free(error);
diff --git a/target-i386/cpu.c b/target-i386/cpu.c
index 0e8812a..6f2ba1c 100644
--- a/target-i386/cpu.c
+++ b/target-i386/cpu.c
@@ -23,8 +23,6 @@
#include "cpu.h"
#include "sysemu/kvm.h"
-#include "sysemu/cpus.h"
-#include "topology.h"
#include "qemu/option.h"
#include "qemu/config-file.h"
@@ -2584,6 +2582,21 @@ void enable_compat_apic_id_mode(void)
compat_apic_id_mode = true;
}
+void x86_cpu_topo_ids_from_index(unsigned int cpu_index, X86CPUTopoInfo *topo)
+{
+ static bool warned;
+
+ x86_topo_ids_from_idx(smp_cores, smp_threads, cpu_index, topo);
+ if (compat_apic_id_mode) {
+ if (cpu_index != topo->apic_id && !warned) {
+ error_report("APIC IDs set in compatibility mode, "
+ "CPU topology won't match the configuration");
+ warned = true;
+ }
+ x86_topo_ids_from_apic_id(smp_cores, smp_threads, cpu_index, topo);
+ }
+}
+
/* Calculates initial APIC ID for a specific CPU index
*
* Currently we need to be able to calculate the APIC ID from the CPU index
@@ -2593,20 +2606,10 @@ void enable_compat_apic_id_mode(void)
*/
uint32_t x86_cpu_apic_id_from_index(unsigned int cpu_index)
{
- uint32_t correct_id;
- static bool warned;
+ X86CPUTopoInfo topo;
- correct_id = x86_apicid_from_cpu_idx(smp_cores, smp_threads, cpu_index);
- if (compat_apic_id_mode) {
- if (cpu_index != correct_id && !warned) {
- error_report("APIC IDs set in compatibility mode, "
- "CPU topology won't match the configuration");
- warned = true;
- }
- return cpu_index;
- } else {
- return correct_id;
- }
+ x86_cpu_topo_ids_from_index(cpu_index, &topo);
+ return topo.apic_id;
}
static void x86_cpu_initfn(Object *obj)
diff --git a/target-i386/cpu.h b/target-i386/cpu.h
index 0014acc..a410b16 100644
--- a/target-i386/cpu.h
+++ b/target-i386/cpu.h
@@ -22,6 +22,9 @@
#include "config.h"
#include "qemu-common.h"
+#include "sysemu/cpus.h"
+#include "topology.h"
+
#ifdef TARGET_X86_64
#define TARGET_LONG_BITS 64
#else
@@ -1286,6 +1289,8 @@ void x86_cpu_compat_set_features(const char *cpu_model, FeatureWord w,
const char *get_register_name_32(unsigned int reg);
uint32_t x86_cpu_apic_id_from_index(unsigned int cpu_index);
+void x86_cpu_topo_ids_from_index(unsigned int cpu_index,
+ X86CPUTopoInfo *topo);
void enable_compat_apic_id_mode(void);
#define APIC_DEFAULT_ADDRESS 0xfee00000
diff --git a/target-i386/topology.h b/target-i386/topology.h
index e9ff89c..e89e0cc 100644
--- a/target-i386/topology.h
+++ b/target-i386/topology.h
@@ -51,6 +51,7 @@ typedef struct X86CPUTopoInfo {
unsigned pkg_id;
unsigned core_id;
unsigned smt_id;
+ apic_id_t apic_id;
} X86CPUTopoInfo;
/* Return the bit width needed for 'count' IDs
@@ -117,19 +118,26 @@ static inline void x86_topo_ids_from_idx(unsigned nr_cores,
topo->smt_id = cpu_index % nr_threads;
topo->core_id = core_index % nr_cores;
topo->pkg_id = core_index / nr_cores;
+ topo->apic_id = apicid_from_topo_ids(nr_cores, nr_threads, topo);
}
-/* Make APIC ID for the CPU 'cpu_index'
- *
- * 'cpu_index' is a sequential, contiguous ID for the CPU.
+/* Calculate CPU topology based on CPU APIC ID.
*/
-static inline apic_id_t x86_apicid_from_cpu_idx(unsigned nr_cores,
- unsigned nr_threads,
- unsigned cpu_index)
+static inline void x86_topo_ids_from_apic_id(unsigned nr_cores,
+ unsigned nr_threads,
+ apic_id_t apic_id,
+ X86CPUTopoInfo *topo)
{
- X86CPUTopoInfo topo;
- x86_topo_ids_from_idx(nr_cores, nr_threads, cpu_index, &topo);
- return apicid_from_topo_ids(nr_cores, nr_threads, &topo);
+ unsigned offset_mask;
+ topo->pkg_id = apic_id >> apicid_pkg_offset(nr_cores, nr_threads);
+
+ offset_mask = (1L << apicid_pkg_offset(nr_cores, nr_threads)) - 1;
+ topo->core_id = (apic_id & offset_mask)
+ >> apicid_core_offset(nr_cores, nr_threads);
+
+ offset_mask = (1L << apicid_core_offset(nr_cores, nr_threads)) - 1;
+ topo->smt_id = apic_id & offset_mask;
+ topo->apic_id = apic_id;
}
#endif /* TARGET_I386_TOPOLOGY_H */
diff --git a/tests/test-x86-cpuid.c b/tests/test-x86-cpuid.c
index 8d9f96a..ea2cab8 100644
--- a/tests/test-x86-cpuid.c
+++ b/tests/test-x86-cpuid.c
@@ -32,11 +32,36 @@ static void test_topo_bits(void)
g_assert_cmpuint(apicid_smt_width(1, 1), ==, 0);
g_assert_cmpuint(apicid_core_width(1, 1), ==, 0);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(1, 1, 0), ==, 0);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(1, 1, 1), ==, 1);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(1, 1, 2), ==, 2);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(1, 1, 3), ==, 3);
-
+ X86CPUTopoInfo topo = {0};
+
+ x86_topo_ids_from_idx(1, 1, 0, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 0);
+ x86_topo_ids_from_idx(1, 1, 1, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 1);
+ x86_topo_ids_from_idx(1, 1, 2, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 2);
+ x86_topo_ids_from_idx(1, 1, 3, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 3);
+
+ x86_topo_ids_from_apic_id(1, 1, 0, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+
+ x86_topo_ids_from_apic_id(1, 1, 1, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 1);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+
+ x86_topo_ids_from_apic_id(1, 1, 2, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 2);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+
+ x86_topo_ids_from_apic_id(1, 1, 3, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 3);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
/* Test field width calculation for multiple values
*/
@@ -65,37 +90,105 @@ static void test_topo_bits(void)
g_assert_cmpuint(apicid_core_width(6, 3), ==, 3);
g_assert_cmpuint(apicid_pkg_offset(6, 3), ==, 5);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 0), ==, 0);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 1), ==, 1);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 2), ==, 2);
-
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 1 * 3 + 0), ==,
- (1 << 2) | 0);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 1 * 3 + 1), ==,
- (1 << 2) | 1);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 1 * 3 + 2), ==,
- (1 << 2) | 2);
-
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 2 * 3 + 0), ==,
- (2 << 2) | 0);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 2 * 3 + 1), ==,
- (2 << 2) | 1);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 2 * 3 + 2), ==,
- (2 << 2) | 2);
-
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 5 * 3 + 0), ==,
- (5 << 2) | 0);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 5 * 3 + 1), ==,
- (5 << 2) | 1);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 5 * 3 + 2), ==,
- (5 << 2) | 2);
-
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 1 * 6 * 3 + 0 * 3 + 0), ==,
- (1 << 5));
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 1 * 6 * 3 + 1 * 3 + 1), ==,
- (1 << 5) | (1 << 2) | 1);
- g_assert_cmpuint(x86_apicid_from_cpu_idx(6, 3, 3 * 6 * 3 + 5 * 3 + 2), ==,
- (3 << 5) | (5 << 2) | 2);
+ x86_topo_ids_from_idx(6, 3, 0, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 0);
+ x86_topo_ids_from_idx(6, 3, 1, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 1);
+ x86_topo_ids_from_idx(6, 3, 2, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, 2);
+
+ x86_topo_ids_from_idx(6, 3, 1 * 3 + 0, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (1 << 2) | 0);
+ x86_topo_ids_from_idx(6, 3, 1 * 3 + 1, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (1 << 2) | 1);
+ x86_topo_ids_from_idx(6, 3, 1 * 3 + 2, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (1 << 2) | 2);
+
+ x86_topo_ids_from_idx(6, 3, 2 * 3 + 0, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (2 << 2) | 0);
+ x86_topo_ids_from_idx(6, 3, 2 * 3 + 1, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (2 << 2) | 1);
+ x86_topo_ids_from_idx(6, 3, 2 * 3 + 2, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (2 << 2) | 2);
+
+ x86_topo_ids_from_idx(6, 3, 5 * 3 + 0, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (5 << 2) | 0);
+ x86_topo_ids_from_idx(6, 3, 5 * 3 + 1, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (5 << 2) | 1);
+ x86_topo_ids_from_idx(6, 3, 5 * 3 + 2, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (5 << 2) | 2);
+
+ x86_topo_ids_from_idx(6, 3, 1 * 6 * 3 + 0 * 3 + 0, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (1 << 5) | 0);
+ x86_topo_ids_from_idx(6, 3, 1 * 6 * 3 + 1 * 3 + 1, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (1 << 5) | (1 << 2) | 1);
+ x86_topo_ids_from_idx(6, 3, 3 * 6 * 3 + 5 * 3 + 2, &topo);
+ g_assert_cmpuint(topo.apic_id, ==, (3 << 5) | (5 << 2) | 2);
+
+ x86_topo_ids_from_apic_id(6, 3, 0, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+ x86_topo_ids_from_apic_id(6, 3, 1, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 1);
+ x86_topo_ids_from_apic_id(6, 3, 2, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 2);
+
+ x86_topo_ids_from_apic_id(6, 3, (1 << 2) | 0, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 1);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+ x86_topo_ids_from_apic_id(6, 3, (1 << 2) | 1, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 1);
+ g_assert_cmpuint(topo.smt_id, ==, 1);
+ x86_topo_ids_from_apic_id(6, 3, (1 << 2) | 2, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 1);
+ g_assert_cmpuint(topo.smt_id, ==, 2);
+
+ x86_topo_ids_from_apic_id(6, 3, (2 << 2) | 0, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 2);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+ x86_topo_ids_from_apic_id(6, 3, (2 << 2) | 1, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 2);
+ g_assert_cmpuint(topo.smt_id, ==, 1);
+ x86_topo_ids_from_apic_id(6, 3, (2 << 2) | 2, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 2);
+ g_assert_cmpuint(topo.smt_id, ==, 2);
+
+ x86_topo_ids_from_apic_id(6, 3, (5 << 2) | 0, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 5);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+ x86_topo_ids_from_apic_id(6, 3, (5 << 2) | 1, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 5);
+ g_assert_cmpuint(topo.smt_id, ==, 1);
+ x86_topo_ids_from_apic_id(6, 3, (5 << 2) | 2, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 0);
+ g_assert_cmpuint(topo.core_id, ==, 5);
+ g_assert_cmpuint(topo.smt_id, ==, 2);
+
+ x86_topo_ids_from_apic_id(6, 3, (1 << 5) | 0, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 1);
+ g_assert_cmpuint(topo.core_id, ==, 0);
+ g_assert_cmpuint(topo.smt_id, ==, 0);
+ x86_topo_ids_from_apic_id(6, 3, (1 << 5) | (1 << 2) | 1, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 1);
+ g_assert_cmpuint(topo.core_id, ==, 1);
+ g_assert_cmpuint(topo.smt_id, ==, 1);
+ x86_topo_ids_from_apic_id(6, 3, (1 << 5) | (1 << 2) | 2, &topo);
+ g_assert_cmpuint(topo.pkg_id, ==, 1);
+ g_assert_cmpuint(topo.core_id, ==, 1);
+ g_assert_cmpuint(topo.smt_id, ==, 2);
}
int main(int argc, char **argv)
--
1.8.1.4
^ permalink raw reply related [flat|nested] 5+ messages in thread
* [Qemu-devel] [PATCH v2 3/3] i386: introduce cpu QOM hierarchy tree
2014-03-20 6:33 [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 1/3] cpu: introduce CpuTopoInfo structure for argument simplification Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 2/3] i386: use CpuTopoInfo instead apic_id as argument for pc_new_cpu() Chen Fan
@ 2014-03-20 6:33 ` Chen Fan
2014-04-01 8:00 ` [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu chen.fan.fnst
3 siblings, 0 replies; 5+ messages in thread
From: Chen Fan @ 2014-03-20 6:33 UTC (permalink / raw)
To: Eduardo Habkost, qemu-devel; +Cc: Igor Mammedov, Andreas Färber
add cpu-topology.h cpu-topology.c files for prebuilding cpu qom tree
"/machine/node[X]/socket[Y]/core[Z]->link cpu"
Signed-off-by: Chen Fan <chen.fan.fnst@cn.fujitsu.com>
---
hw/i386/pc.c | 3 +
target-i386/Makefile.objs | 2 +-
target-i386/cpu-topology.c | 199 +++++++++++++++++++++++++++++++++++++++++++++
target-i386/cpu-topology.h | 71 ++++++++++++++++
target-i386/cpu.c | 22 +++++
target-i386/cpu.h | 3 +
6 files changed, 299 insertions(+), 1 deletion(-)
create mode 100644 target-i386/cpu-topology.c
create mode 100644 target-i386/cpu-topology.h
diff --git a/hw/i386/pc.c b/hw/i386/pc.c
index 84a017e..8a489f5 100644
--- a/hw/i386/pc.c
+++ b/hw/i386/pc.c
@@ -941,6 +941,7 @@ static X86CPU *pc_new_cpu(const char *cpu_model, X86CPUTopoInfo *topo,
object_property_set_int(OBJECT(cpu), topo->apic_id, "apic-id", &local_err);
object_property_set_bool(OBJECT(cpu), true, "realized", &local_err);
+ x86_topo_cpu_set_link(OBJECT(cpu), topo, &local_err);
if (local_err) {
error_propagate(errp, local_err);
@@ -996,6 +997,8 @@ void pc_cpus_init(const char *cpu_model, DeviceState *icc_bridge)
}
current_cpu_model = cpu_model;
+ cpu_topo_init();
+
for (i = 0; i < smp_cpus; i++) {
X86CPUTopoInfo topo;
x86_cpu_topo_ids_from_index(i, &topo);
diff --git a/target-i386/Makefile.objs b/target-i386/Makefile.objs
index 027b94e..239474d 100644
--- a/target-i386/Makefile.objs
+++ b/target-i386/Makefile.objs
@@ -1,4 +1,4 @@
-obj-y += translate.o helper.o cpu.o
+obj-y += translate.o helper.o cpu.o cpu-topology.o
obj-y += excp_helper.o fpu_helper.o cc_helper.o int_helper.o svm_helper.o
obj-y += smm_helper.o misc_helper.o mem_helper.o seg_helper.o
obj-y += gdbstub.o
diff --git a/target-i386/cpu-topology.c b/target-i386/cpu-topology.c
new file mode 100644
index 0000000..e611bae
--- /dev/null
+++ b/target-i386/cpu-topology.c
@@ -0,0 +1,199 @@
+/*
+ * QEMU System Emulator
+ *
+ * Copyright (c) 2014 Fujitsu Ltd.
+ * Author: Chen Fan <chen.fan.fnst@cn.fujitsu.com>
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
+ * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+
+#include "qom/object.h"
+#include "qemu/module.h"
+#include "hw/hw.h"
+#include "sysemu/cpus.h"
+#include "sysemu/sysemu.h"
+#include "cpu-topology.h"
+
+static QTAILQ_HEAD(, SocketState) sockets = QTAILQ_HEAD_INITIALIZER(sockets);
+static QTAILQ_HEAD(, NodeState) nodes = QTAILQ_HEAD_INITIALIZER(nodes);
+
+static NodeState *node_get(int node_id)
+{
+ NodeState *node;
+
+ QTAILQ_FOREACH(node, &nodes, next) {
+ if (node->node_id == node_id) {
+ return node;
+ }
+ }
+ return NULL;
+}
+
+static SocketState *cpu_socket_find(int pkg_id)
+{
+ SocketState *ss;
+
+ QTAILQ_FOREACH(ss, &sockets, next) {
+ if (ss->socket_id == pkg_id) {
+ return ss;
+ }
+ }
+ return NULL;
+}
+
+void cpu_topo_init(void)
+{
+ unsigned long *node_mask;
+ int i;
+
+ node_mask = bitmap_new(MAX_NODES);
+
+ for (i = 0; i < max_cpus; i++) {
+ NodeState *node;
+ SocketState *ss;
+ gchar *name;
+ int node_id = 0, socket_id;
+ int j;
+
+ for (j = 0; j < nb_numa_nodes; j++) {
+ if (test_bit(i, node_cpumask[j])) {
+ node_id = j;
+ break;
+ }
+ }
+
+ if (test_bit(node_id, node_mask)) {
+ node = node_get(node_id);
+ } else {
+ node = NODE(object_new(TYPE_NODE));
+ node->node_id = node_id;
+ name = g_strdup_printf("node[%" PRIu32 "]", node_id);
+ object_property_add_child(qdev_get_machine(), name,
+ OBJECT(node), NULL);
+ set_bit(node_id, node_mask);
+ g_free(name);
+ QTAILQ_INSERT_TAIL(&nodes, node, next);
+ }
+
+ socket_id = i / (smp_cores * smp_threads);
+ ss = cpu_socket_find(socket_id);
+ if (!ss) {
+ ss = SOCKET(object_new(TYPE_SOCKET));
+ ss->socket_id = socket_id;
+ name = g_strdup_printf("socket[%" PRIu32 "]", socket_id);
+ object_property_add_child(OBJECT(node), name,
+ OBJECT(ss), NULL);
+ g_free(name);
+ QTAILQ_INSERT_TAIL(&sockets, ss, next);
+ }
+ }
+
+ g_free(node_mask);
+}
+
+CoreState *cpu_topo_object_core_find(X86CPUTopoInfo *topo)
+{
+ SocketState *ss;
+ CoreState *core;
+
+ ss = cpu_socket_find(topo->pkg_id);
+ if (!ss) {
+ return NULL;
+ }
+
+ QTAILQ_FOREACH(core, &ss->children, sibling) {
+ if (core->core_id == topo->core_id) {
+ return core;
+ }
+ }
+ return NULL;
+}
+
+static void socket_initfn(Object *obj)
+{
+ SocketState *ss = SOCKET(obj);
+ int i;
+
+ QTAILQ_INIT(&ss->children);
+
+ for (i = 0; i < smp_cores; i++) {
+ gchar *name;
+ CoreState *core;
+
+ core = CORE(object_new(TYPE_CORE));
+ core->core_id = i;
+ QTAILQ_INSERT_TAIL(&ss->children, core, sibling);
+
+ name = g_strdup_printf("core[%" PRIu32 "]", i);
+ object_property_add_child(obj, name, OBJECT(core), NULL);
+ g_free(name);
+ }
+}
+
+static void core_initfn(Object *obj)
+{
+ CoreState *cs = CORE(obj);
+ gchar *name;
+ int i;
+
+ cs->link_cpu = g_malloc0(sizeof(CPUState *) * smp_threads);
+ for (i = 0; i < smp_threads; i++) {
+ name = g_strdup_printf("cpu[%" PRIu32 "]", i);
+ object_property_add_link(obj, name, TYPE_CPU,
+ (Object **)&cs->link_cpu[i], NULL);
+ g_free(name);
+ }
+}
+
+static void core_fini(Object *obj)
+{
+ CoreState *cs = CORE(obj);
+
+ g_free(cs->link_cpu);
+}
+
+static const TypeInfo core_type_info = {
+ .name = TYPE_CORE,
+ .parent = TYPE_OBJECT,
+ .instance_size = sizeof(CoreState),
+ .instance_init = core_initfn,
+ .instance_finalize = core_fini,
+};
+
+static const TypeInfo socket_type_info = {
+ .name = TYPE_SOCKET,
+ .parent = TYPE_OBJECT,
+ .instance_size = sizeof(SocketState),
+ .instance_init = socket_initfn,
+};
+
+static const TypeInfo node_type_info = {
+ .name = TYPE_NODE,
+ .parent = TYPE_OBJECT,
+ .instance_size = sizeof(NodeState),
+};
+
+static void node_register_types(void)
+{
+ type_register_static(&node_type_info);
+ type_register_static(&socket_type_info);
+ type_register_static(&core_type_info);
+}
+
+type_init(node_register_types)
diff --git a/target-i386/cpu-topology.h b/target-i386/cpu-topology.h
new file mode 100644
index 0000000..6465d94
--- /dev/null
+++ b/target-i386/cpu-topology.h
@@ -0,0 +1,71 @@
+/*
+ * QEMU System Emulator
+ *
+ * Copyright (c) 2014 Fujitsu Ltd.
+ * Author: Chen Fan <chen.fan.fnst@cn.fujitsu.com>
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
+ * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+#ifndef TARGET_I386_CPU_TOPOLOGY_H
+#define TARGET_I386_CPU_TOPOLOGY_H
+
+#include "hw/qdev-core.h"
+#include "cpu.h"
+#include "qom/cpu.h"
+
+#define TYPE_NODE "node"
+#define TYPE_SOCKET "socket"
+#define TYPE_CORE "core"
+
+#define NODE(obj) OBJECT_CHECK(NodeState, (obj), TYPE_NODE)
+
+#define SOCKET(obj) OBJECT_CHECK(SocketState, (obj), TYPE_SOCKET)
+
+#define CORE(obj) OBJECT_CHECK(CoreState, (obj), TYPE_CORE)
+
+typedef struct CoreState {
+ /*< private >*/
+ Object parent_obj;
+ /*< public >*/
+ int core_id;
+ QTAILQ_ENTRY(CoreState) sibling;
+ CPUState **link_cpu;
+} CoreState;
+
+typedef struct SocketState {
+ /*< private >*/
+ Object parent_obj;
+ /*< public >*/
+ unsigned socket_id;
+ QTAILQ_HEAD(ChildHread, CoreState) children;
+ QTAILQ_ENTRY(SocketState) next;
+} SocketState;
+
+typedef struct NodeState {
+ /*< private >*/
+ Object parent_obj;
+ /*< public >*/
+ int node_id;
+ QTAILQ_ENTRY(NodeState) next;
+} NodeState;
+
+CoreState *cpu_topo_object_core_find(X86CPUTopoInfo *topo);
+void cpu_topo_init(void);
+
+#endif /* TARGET_I386_CPU_TOPOLOGY_H */
diff --git a/target-i386/cpu.c b/target-i386/cpu.c
index 6f2ba1c..9754ac2 100644
--- a/target-i386/cpu.c
+++ b/target-i386/cpu.c
@@ -2698,6 +2698,28 @@ static void x86_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock *tb)
cpu->env.eip = tb->pc - tb->cs_base;
}
+void x86_topo_cpu_set_link(Object *obj, X86CPUTopoInfo *topo, Error **errp)
+{
+ gchar *name;
+ CoreState *core;
+ Error *local_err = NULL;
+
+ core = cpu_topo_object_core_find(topo);
+ if (!core) {
+ error_set(&local_err, QERR_DEVICE_NOT_FOUND, TYPE_CORE);
+ goto out;
+ }
+
+ name = g_strdup_printf("cpu[%" PRIu32 "]", topo->smt_id);
+ object_property_set_link(OBJECT(core), obj, name, &local_err);
+ g_free(name);
+
+out:
+ if (local_err != NULL) {
+ error_propagate(errp, local_err);
+ }
+}
+
static Property x86_cpu_properties[] = {
DEFINE_PROP_BOOL("pmu", X86CPU, enable_pmu, false),
{ .name = "hv-spinlocks", .info = &qdev_prop_spinlocks },
diff --git a/target-i386/cpu.h b/target-i386/cpu.h
index a410b16..42db0fd 100644
--- a/target-i386/cpu.h
+++ b/target-i386/cpu.h
@@ -24,6 +24,7 @@
#include "sysemu/cpus.h"
#include "topology.h"
+#include "cpu-topology.h"
#ifdef TARGET_X86_64
#define TARGET_LONG_BITS 64
@@ -1293,6 +1294,8 @@ void x86_cpu_topo_ids_from_index(unsigned int cpu_index,
X86CPUTopoInfo *topo);
void enable_compat_apic_id_mode(void);
+void x86_topo_cpu_set_link(Object *obj, X86CPUTopoInfo *topo, Error **errp);
+
#define APIC_DEFAULT_ADDRESS 0xfee00000
#define APIC_SPACE_SIZE 0x100000
--
1.8.1.4
^ permalink raw reply related [flat|nested] 5+ messages in thread
* Re: [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu
2014-03-20 6:33 [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu Chen Fan
` (2 preceding siblings ...)
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 3/3] i386: introduce cpu QOM hierarchy tree Chen Fan
@ 2014-04-01 8:00 ` chen.fan.fnst
3 siblings, 0 replies; 5+ messages in thread
From: chen.fan.fnst @ 2014-04-01 8:00 UTC (permalink / raw)
To: Eduardo Habkost; +Cc: Igor Mammedov, qemu-devel@nongnu.org, Andreas Färber
Ping...
On Thu, 2014-03-20 at 14:33 +0800, Chen Fan wrote:
> at present, after hotplug a discontinuous cpu id on source, then done migration,
> on target, it will fail to add the unoccupied cpu id which was skipped at source,
> this cause is on target Qemu prebuild CPU with continuous cpu_index. so after
> migration, the cpu infrastructure bewteen source and target are different.
>
> I suppose we could use apic_id as instance_id which was used at registering vmstate
> when create cpu. on target, we prebuild the specified cpu topology using comand line:
> -device /machine/node[]/socket[]/core[]/cpu[], then migration, we could keep the same
> cpu infrastructure on both side.
>
> v1-v2: squash patch 2/4 and 3/4 together.
>
> RFC:
> V4: rename CpuTopoInfo to X86CPUTopoInfo. and move cpu_exsit() to pc_new_cpu().
>
> V3: get rid of thread object and tie link<cpu> to <core> directly. and prebuild full
> core[] and thread[] as init socket[] according to smp_cores and smp_threads.
>
> TODO:
> 1. add cpu "path" property which used for specifying the QOM path.
> 2. add -device cpu-foo.path supported.
> 3. then we could introduce hot-remove cpu probably.
>
> I don't know wether this way is right or not. pls tell me. :)
>
> Chen Fan (3):
> cpu: introduce CpuTopoInfo structure for argument simplification
> i386: use CpuTopoInfo instead apic_id as argument for pc_new_cpu()
> i386: introduce cpu QOM hierarchy tree
>
> hw/i386/pc.c | 25 +++---
> target-i386/Makefile.objs | 2 +-
> target-i386/cpu-topology.c | 199 +++++++++++++++++++++++++++++++++++++++++++++
> target-i386/cpu-topology.h | 71 ++++++++++++++++
> target-i386/cpu.c | 55 +++++++++----
> target-i386/cpu.h | 8 ++
> target-i386/topology.h | 53 +++++++-----
> tests/test-x86-cpuid.c | 165 +++++++++++++++++++++++++++++--------
> 8 files changed, 494 insertions(+), 84 deletions(-)
> create mode 100644 target-i386/cpu-topology.c
> create mode 100644 target-i386/cpu-topology.h
>
^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2014-04-01 8:34 UTC | newest]
Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-03-20 6:33 [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 1/3] cpu: introduce CpuTopoInfo structure for argument simplification Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 2/3] i386: use CpuTopoInfo instead apic_id as argument for pc_new_cpu() Chen Fan
2014-03-20 6:33 ` [Qemu-devel] [PATCH v2 3/3] i386: introduce cpu QOM hierarchy tree Chen Fan
2014-04-01 8:00 ` [Qemu-devel] [PATCH v2 0/3] prebuild cpu QOM tree /machine/node/socket/core ->link-cpu chen.fan.fnst
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).