cpu: Use CPUClass->parse_features() as convertor to global properties
Currently CPUClass->parse_features() is used to parse -cpu features string and set properties on created CPU instances. But considering that features specified by -cpu apply to every created CPU instance, it doesn't make sense to parse the same features string for every CPU created. It also makes every target that cares about parsing features string explicitly call CPUClass->parse_features() parser, which gets in a way if we consider using generic device_add for CPU hotplug as device_add has not a clue about CPU specific hooks. Turns out we can use global properties mechanism to set properties on every created CPU instance for a given type. That way it's possible to convert CPU features into a set of global properties for CPU type specified by -cpu cpu_model and common Device.device_post_init() will apply them to CPU of given type automatically regardless whether it's manually created CPU or CPU created with help of device_add. Signed-off-by: Igor Mammedov <imammedo@redhat.com> Reviewed-by: Eduardo Habkost <ehabkost@redhat.com> Signed-off-by: Eduardo Habkost <ehabkost@redhat.com>
This commit is contained in:
parent
cf2887c973
commit
62a48a2a57
@ -1261,6 +1261,7 @@ static void machvirt_init(MachineState *machine)
|
|||||||
|
|
||||||
for (n = 0; n < smp_cpus; n++) {
|
for (n = 0; n < smp_cpus; n++) {
|
||||||
ObjectClass *oc = cpu_class_by_name(TYPE_ARM_CPU, cpustr[0]);
|
ObjectClass *oc = cpu_class_by_name(TYPE_ARM_CPU, cpustr[0]);
|
||||||
|
const char *typename = object_class_get_name(oc);
|
||||||
CPUClass *cc = CPU_CLASS(oc);
|
CPUClass *cc = CPU_CLASS(oc);
|
||||||
Object *cpuobj;
|
Object *cpuobj;
|
||||||
Error *err = NULL;
|
Error *err = NULL;
|
||||||
@ -1270,10 +1271,10 @@ static void machvirt_init(MachineState *machine)
|
|||||||
error_report("Unable to find CPU definition");
|
error_report("Unable to find CPU definition");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
cpuobj = object_new(object_class_get_name(oc));
|
/* convert -smp CPU options specified by the user into global props */
|
||||||
|
cc->parse_features(typename, cpuopts, &err);
|
||||||
|
cpuobj = object_new(typename);
|
||||||
|
|
||||||
/* Handle any CPU options specified by the user */
|
|
||||||
cc->parse_features(CPU(cpuobj), cpuopts, &err);
|
|
||||||
g_free(cpuopts);
|
g_free(cpuopts);
|
||||||
if (err) {
|
if (err) {
|
||||||
error_report_err(err);
|
error_report_err(err);
|
||||||
|
@ -134,7 +134,7 @@ typedef struct CPUClass {
|
|||||||
/*< public >*/
|
/*< public >*/
|
||||||
|
|
||||||
ObjectClass *(*class_by_name)(const char *cpu_model);
|
ObjectClass *(*class_by_name)(const char *cpu_model);
|
||||||
void (*parse_features)(CPUState *cpu, char *str, Error **errp);
|
void (*parse_features)(const char *typename, char *str, Error **errp);
|
||||||
|
|
||||||
void (*reset)(CPUState *cpu);
|
void (*reset)(CPUState *cpu);
|
||||||
int reset_dump_flags;
|
int reset_dump_flags;
|
||||||
|
41
qom/cpu.c
41
qom/cpu.c
@ -28,6 +28,7 @@
|
|||||||
#include "exec/log.h"
|
#include "exec/log.h"
|
||||||
#include "qemu/error-report.h"
|
#include "qemu/error-report.h"
|
||||||
#include "sysemu/sysemu.h"
|
#include "sysemu/sysemu.h"
|
||||||
|
#include "hw/qdev-properties.h"
|
||||||
|
|
||||||
bool cpu_exists(int64_t id)
|
bool cpu_exists(int64_t id)
|
||||||
{
|
{
|
||||||
@ -46,7 +47,7 @@ bool cpu_exists(int64_t id)
|
|||||||
CPUState *cpu_generic_init(const char *typename, const char *cpu_model)
|
CPUState *cpu_generic_init(const char *typename, const char *cpu_model)
|
||||||
{
|
{
|
||||||
char *str, *name, *featurestr;
|
char *str, *name, *featurestr;
|
||||||
CPUState *cpu;
|
CPUState *cpu = NULL;
|
||||||
ObjectClass *oc;
|
ObjectClass *oc;
|
||||||
CPUClass *cc;
|
CPUClass *cc;
|
||||||
Error *err = NULL;
|
Error *err = NULL;
|
||||||
@ -60,16 +61,18 @@ CPUState *cpu_generic_init(const char *typename, const char *cpu_model)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
cpu = CPU(object_new(object_class_get_name(oc)));
|
cc = CPU_CLASS(oc);
|
||||||
cc = CPU_GET_CLASS(cpu);
|
|
||||||
|
|
||||||
featurestr = strtok(NULL, ",");
|
featurestr = strtok(NULL, ",");
|
||||||
cc->parse_features(cpu, featurestr, &err);
|
/* TODO: all callers of cpu_generic_init() need to be converted to
|
||||||
|
* call parse_features() only once, before calling cpu_generic_init().
|
||||||
|
*/
|
||||||
|
cc->parse_features(object_class_get_name(oc), featurestr, &err);
|
||||||
g_free(str);
|
g_free(str);
|
||||||
if (err != NULL) {
|
if (err != NULL) {
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cpu = CPU(object_new(object_class_get_name(oc)));
|
||||||
object_property_set_bool(OBJECT(cpu), true, "realized", &err);
|
object_property_set_bool(OBJECT(cpu), true, "realized", &err);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
@ -282,25 +285,39 @@ static ObjectClass *cpu_common_class_by_name(const char *cpu_model)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cpu_common_parse_features(CPUState *cpu, char *features,
|
static void cpu_common_parse_features(const char *typename, char *features,
|
||||||
Error **errp)
|
Error **errp)
|
||||||
{
|
{
|
||||||
char *featurestr; /* Single "key=value" string being parsed */
|
char *featurestr; /* Single "key=value" string being parsed */
|
||||||
char *val;
|
char *val;
|
||||||
Error *err = NULL;
|
static bool cpu_globals_initialized;
|
||||||
|
|
||||||
|
/* TODO: all callers of ->parse_features() need to be changed to
|
||||||
|
* call it only once, so we can remove this check (or change it
|
||||||
|
* to assert(!cpu_globals_initialized).
|
||||||
|
* Current callers of ->parse_features() are:
|
||||||
|
* - machvirt_init()
|
||||||
|
* - cpu_generic_init()
|
||||||
|
* - cpu_x86_create()
|
||||||
|
*/
|
||||||
|
if (cpu_globals_initialized) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cpu_globals_initialized = true;
|
||||||
|
|
||||||
featurestr = features ? strtok(features, ",") : NULL;
|
featurestr = features ? strtok(features, ",") : NULL;
|
||||||
|
|
||||||
while (featurestr) {
|
while (featurestr) {
|
||||||
val = strchr(featurestr, '=');
|
val = strchr(featurestr, '=');
|
||||||
if (val) {
|
if (val) {
|
||||||
|
GlobalProperty *prop = g_new0(typeof(*prop), 1);
|
||||||
*val = 0;
|
*val = 0;
|
||||||
val++;
|
val++;
|
||||||
object_property_parse(OBJECT(cpu), val, featurestr, &err);
|
prop->driver = typename;
|
||||||
if (err) {
|
prop->property = g_strdup(featurestr);
|
||||||
error_propagate(errp, err);
|
prop->value = g_strdup(val);
|
||||||
return;
|
prop->errp = &error_fatal;
|
||||||
}
|
qdev_prop_register_global(prop);
|
||||||
} else {
|
} else {
|
||||||
error_setg(errp, "Expected key=value format, found %s.",
|
error_setg(errp, "Expected key=value format, found %s.",
|
||||||
featurestr);
|
featurestr);
|
||||||
|
@ -1958,12 +1958,17 @@ static FeatureWordArray minus_features = { 0 };
|
|||||||
|
|
||||||
/* Parse "+feature,-feature,feature=foo" CPU feature string
|
/* Parse "+feature,-feature,feature=foo" CPU feature string
|
||||||
*/
|
*/
|
||||||
static void x86_cpu_parse_featurestr(CPUState *cs, char *features,
|
static void x86_cpu_parse_featurestr(const char *typename, char *features,
|
||||||
Error **errp)
|
Error **errp)
|
||||||
{
|
{
|
||||||
X86CPU *cpu = X86_CPU(cs);
|
|
||||||
char *featurestr; /* Single 'key=value" string being parsed */
|
char *featurestr; /* Single 'key=value" string being parsed */
|
||||||
Error *local_err = NULL;
|
Error *local_err = NULL;
|
||||||
|
static bool cpu_globals_initialized;
|
||||||
|
|
||||||
|
if (cpu_globals_initialized) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cpu_globals_initialized = true;
|
||||||
|
|
||||||
if (!features) {
|
if (!features) {
|
||||||
return;
|
return;
|
||||||
@ -1976,6 +1981,7 @@ static void x86_cpu_parse_featurestr(CPUState *cs, char *features,
|
|||||||
const char *val = NULL;
|
const char *val = NULL;
|
||||||
char *eq = NULL;
|
char *eq = NULL;
|
||||||
char num[32];
|
char num[32];
|
||||||
|
GlobalProperty *prop;
|
||||||
|
|
||||||
/* Compatibility syntax: */
|
/* Compatibility syntax: */
|
||||||
if (featurestr[0] == '+') {
|
if (featurestr[0] == '+') {
|
||||||
@ -2013,7 +2019,12 @@ static void x86_cpu_parse_featurestr(CPUState *cs, char *features,
|
|||||||
name = "tsc-frequency";
|
name = "tsc-frequency";
|
||||||
}
|
}
|
||||||
|
|
||||||
object_property_parse(OBJECT(cpu), val, name, &local_err);
|
prop = g_new0(typeof(*prop), 1);
|
||||||
|
prop->driver = typename;
|
||||||
|
prop->property = g_strdup(name);
|
||||||
|
prop->value = g_strdup(val);
|
||||||
|
prop->errp = &error_fatal;
|
||||||
|
qdev_prop_register_global(prop);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (local_err) {
|
if (local_err) {
|
||||||
@ -2202,9 +2213,11 @@ X86CPU *cpu_x86_create(const char *cpu_model, Error **errp)
|
|||||||
{
|
{
|
||||||
X86CPU *cpu = NULL;
|
X86CPU *cpu = NULL;
|
||||||
ObjectClass *oc;
|
ObjectClass *oc;
|
||||||
|
CPUClass *cc;
|
||||||
gchar **model_pieces;
|
gchar **model_pieces;
|
||||||
char *name, *features;
|
char *name, *features;
|
||||||
Error *error = NULL;
|
Error *error = NULL;
|
||||||
|
const char *typename;
|
||||||
|
|
||||||
model_pieces = g_strsplit(cpu_model, ",", 2);
|
model_pieces = g_strsplit(cpu_model, ",", 2);
|
||||||
if (!model_pieces[0]) {
|
if (!model_pieces[0]) {
|
||||||
@ -2219,10 +2232,11 @@ X86CPU *cpu_x86_create(const char *cpu_model, Error **errp)
|
|||||||
error_setg(&error, "Unable to find CPU definition: %s", name);
|
error_setg(&error, "Unable to find CPU definition: %s", name);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
cc = CPU_CLASS(oc);
|
||||||
|
typename = object_class_get_name(oc);
|
||||||
|
|
||||||
cpu = X86_CPU(object_new(object_class_get_name(oc)));
|
cc->parse_features(typename, features, &error);
|
||||||
|
cpu = X86_CPU(object_new(typename));
|
||||||
x86_cpu_parse_featurestr(CPU(cpu), features, &error);
|
|
||||||
if (error) {
|
if (error) {
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user