@@ -814,6 +814,8 @@ void ppce500_init(MachineState *machine, PPCE500Params *params)
machine->cpu_model = "e500v2_v30";
}
+ ppc_cpu_parse_features(machine->cpu_model);
+
irqs = g_malloc0(smp_cpus * sizeof(qemu_irq *));
irqs[0] = g_malloc0(smp_cpus * sizeof(qemu_irq) * OPENPIC_OUTPUT_NB);
for (i = 0; i < smp_cpus; i++) {
@@ -192,6 +192,7 @@ static void ppc_core99_init(MachineState *machine)
machine->cpu_model = "G4";
#endif
}
+ ppc_cpu_parse_features(machine->cpu_model);
for (i = 0; i < smp_cpus; i++) {
cpu = ppc_cpu_init(machine->cpu_model);
if (cpu == NULL) {
@@ -112,6 +112,7 @@ static void ppc_heathrow_init(MachineState *machine)
/* init CPUs */
if (machine->cpu_model == NULL)
machine->cpu_model = "G3";
+ ppc_cpu_parse_features(machine->cpu_model);
for (i = 0; i < smp_cpus; i++) {
cpu = ppc_cpu_init(machine->cpu_model);
if (cpu == NULL) {
@@ -33,6 +33,7 @@
#include "hw/timer/m48t59.h"
#include "qemu/log.h"
#include "qemu/error-report.h"
+#include "qapi/error.h"
#include "hw/loader.h"
#include "sysemu/kvm.h"
#include "kvm_ppc.h"
@@ -1355,3 +1356,28 @@ PowerPCCPU *ppc_cpu_init(const char *cpu_model)
{
return POWERPC_CPU(cpu_generic_init(TYPE_POWERPC_CPU, cpu_model));
}
+
+void ppc_cpu_parse_features(const char *cpu_model)
+{
+ CPUClass *cc;
+ ObjectClass *oc;
+ const char *typename;
+ gchar **model_pieces;
+
+ model_pieces = g_strsplit(cpu_model, ",", 2);
+ if (!model_pieces[0]) {
+ error_report("Invalid/empty CPU model name");
+ exit(1);
+ }
+
+ oc = cpu_class_by_name(TYPE_POWERPC_CPU, model_pieces[0]);
+ if (oc == NULL) {
+ error_report("Unable to find CPU definition: %s", model_pieces[0]);
+ exit(1);
+ }
+
+ typename = object_class_get_name(oc);
+ cc = CPU_CLASS(oc);
+ cc->parse_features(typename, model_pieces[1], &error_fatal);
+ g_strfreev(model_pieces);
+}
@@ -186,6 +186,7 @@ static void bamboo_init(MachineState *machine)
if (machine->cpu_model == NULL) {
machine->cpu_model = "440EP";
}
+ ppc_cpu_parse_features(machine->cpu_model);
cpu = ppc_cpu_init(machine->cpu_model);
if (cpu == NULL) {
fprintf(stderr, "Unable to initialize CPU!\n");
@@ -56,6 +56,7 @@ PowerPCCPU *ppc4xx_init(const char *cpu_model,
CPUPPCState *env;
/* init CPUs */
+ ppc_cpu_parse_features(cpu_model);
cpu = ppc_cpu_init(cpu_model);
if (cpu == NULL) {
fprintf(stderr, "Unable to find PowerPC %s CPU definition\n",
@@ -508,6 +508,7 @@ static void ppc_prep_init(MachineState *machine)
/* init CPUs */
if (machine->cpu_model == NULL)
machine->cpu_model = "602";
+ ppc_cpu_parse_features(machine->cpu_model);
for (i = 0; i < smp_cpus; i++) {
cpu = ppc_cpu_init(machine->cpu_model);
if (cpu == NULL) {
@@ -1803,6 +1803,8 @@ static void ppc_spapr_init(MachineState *machine)
machine->cpu_model = kvm_enabled() ? "host" : "POWER7";
}
+ ppc_cpu_parse_features(machine->cpu_model);
+
if (smc->dr_cpu_enabled) {
char *type = spapr_get_cpu_core_type(machine->cpu_model);
@@ -96,6 +96,7 @@ static PowerPCCPU *ppc440_init_xilinx(ram_addr_t *ram_size,
CPUPPCState *env;
qemu_irq *irqs;
+ ppc_cpu_parse_features(cpu_model);
cpu = ppc_cpu_init(cpu_model);
if (cpu == NULL) {
fprintf(stderr, "Unable to initialize CPU!\n");
@@ -107,4 +107,5 @@ enum {
void ppc_booke_timers_init(PowerPCCPU *cpu, uint32_t freq, uint32_t flags);
PowerPCCPU *ppc_cpu_init(const char *cpu_model);
+void ppc_cpu_parse_features(const char *cpu_model);
#endif
Considering that features are converted to global properties and global properties are automatically applied to every new instance of created CPU (at object_new() time), there is no point in parsing cpu_model string every time a CPU created. So move parsing outside CPU creation loop and do it only once. Parsing also should be done before any CPU is created so that features would affect the first CPU a well. This patch does that for all PowerPC machine types. It is based on previous work from Bharata: https://lists.nongnu.org/archive/html/qemu-devel/2016-06/msg07564.html Signed-off-by: Greg Kurz <groug@kaod.org> --- hw/ppc/e500.c | 2 ++ hw/ppc/mac_newworld.c | 1 + hw/ppc/mac_oldworld.c | 1 + hw/ppc/ppc.c | 26 ++++++++++++++++++++++++++ hw/ppc/ppc440_bamboo.c | 1 + hw/ppc/ppc4xx_devs.c | 1 + hw/ppc/prep.c | 1 + hw/ppc/spapr.c | 2 ++ hw/ppc/virtex_ml507.c | 1 + include/hw/ppc/ppc.h | 1 + 10 files changed, 37 insertions(+)