mirror of https://github.com/xemu-project/xemu.git
ppc/xics: remove set_nr_servers() handler from XICSStateClass
Today, the ICP (Interrupt Controller Presenter) objects are created by the 'nr_servers' property handler of the XICS object and a class handler. They are realized in the XICS object realize routine. Let's simplify the process by creating the ICP objects along with the XICS object at the machine level. Signed-off-by: Cédric Le Goater <clg@kaod.org> Reviewed-by: David Gibson <david@gibson.dropbear.id.au> Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
This commit is contained in:
parent
4e4169f7a2
commit
817bb6a446
|
@ -151,67 +151,11 @@ static void xics_common_reset(DeviceState *d)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void xics_set_nr_servers(XICSState *xics, uint32_t nr_servers,
|
|
||||||
const char *typename, Error **errp)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
xics->nr_servers = nr_servers;
|
|
||||||
|
|
||||||
xics->ss = g_malloc0(xics->nr_servers * sizeof(ICPState));
|
|
||||||
for (i = 0; i < xics->nr_servers; i++) {
|
|
||||||
char name[32];
|
|
||||||
ICPState *icp = &xics->ss[i];
|
|
||||||
|
|
||||||
object_initialize(icp, sizeof(*icp), typename);
|
|
||||||
snprintf(name, sizeof(name), "icp[%d]", i);
|
|
||||||
object_property_add_child(OBJECT(xics), name, OBJECT(icp), errp);
|
|
||||||
icp->xics = xics;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void xics_prop_get_nr_servers(Object *obj, Visitor *v,
|
|
||||||
const char *name, void *opaque,
|
|
||||||
Error **errp)
|
|
||||||
{
|
|
||||||
XICSState *xics = XICS_COMMON(obj);
|
|
||||||
int64_t value = xics->nr_servers;
|
|
||||||
|
|
||||||
visit_type_int(v, name, &value, errp);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void xics_prop_set_nr_servers(Object *obj, Visitor *v,
|
|
||||||
const char *name, void *opaque,
|
|
||||||
Error **errp)
|
|
||||||
{
|
|
||||||
XICSState *xics = XICS_COMMON(obj);
|
|
||||||
XICSStateClass *xsc = XICS_COMMON_GET_CLASS(xics);
|
|
||||||
Error *error = NULL;
|
|
||||||
int64_t value;
|
|
||||||
|
|
||||||
visit_type_int(v, name, &value, &error);
|
|
||||||
if (error) {
|
|
||||||
error_propagate(errp, error);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (xics->nr_servers) {
|
|
||||||
error_setg(errp, "Number of servers is already set to %u",
|
|
||||||
xics->nr_servers);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(xsc->set_nr_servers);
|
|
||||||
xsc->set_nr_servers(xics, value, errp);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void xics_common_initfn(Object *obj)
|
static void xics_common_initfn(Object *obj)
|
||||||
{
|
{
|
||||||
XICSState *xics = XICS_COMMON(obj);
|
XICSState *xics = XICS_COMMON(obj);
|
||||||
|
|
||||||
QLIST_INIT(&xics->ics);
|
QLIST_INIT(&xics->ics);
|
||||||
object_property_add(obj, "nr_servers", "int",
|
|
||||||
xics_prop_get_nr_servers, xics_prop_set_nr_servers,
|
|
||||||
NULL, NULL, NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void xics_common_class_init(ObjectClass *oc, void *data)
|
static void xics_common_class_init(ObjectClass *oc, void *data)
|
||||||
|
@ -450,12 +394,30 @@ static void icp_reset(DeviceState *dev)
|
||||||
qemu_set_irq(icp->output, 0);
|
qemu_set_irq(icp->output, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void icp_realize(DeviceState *dev, Error **errp)
|
||||||
|
{
|
||||||
|
ICPState *icp = ICP(dev);
|
||||||
|
Object *obj;
|
||||||
|
Error *err = NULL;
|
||||||
|
|
||||||
|
obj = object_property_get_link(OBJECT(dev), "xics", &err);
|
||||||
|
if (!obj) {
|
||||||
|
error_setg(errp, "%s: required link 'xics' not found: %s",
|
||||||
|
__func__, error_get_pretty(err));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
icp->xics = XICS_COMMON(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static void icp_class_init(ObjectClass *klass, void *data)
|
static void icp_class_init(ObjectClass *klass, void *data)
|
||||||
{
|
{
|
||||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||||
|
|
||||||
dc->reset = icp_reset;
|
dc->reset = icp_reset;
|
||||||
dc->vmsd = &vmstate_icp_server;
|
dc->vmsd = &vmstate_icp_server;
|
||||||
|
dc->realize = icp_realize;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const TypeInfo icp_info = {
|
static const TypeInfo icp_info = {
|
||||||
|
|
|
@ -358,12 +358,6 @@ static void xics_kvm_cpu_setup(XICSState *xics, PowerPCCPU *cpu)
|
||||||
ss->cap_irq_xics_enabled = true;
|
ss->cap_irq_xics_enabled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void xics_kvm_set_nr_servers(XICSState *xics, uint32_t nr_servers,
|
|
||||||
Error **errp)
|
|
||||||
{
|
|
||||||
xics_set_nr_servers(xics, nr_servers, TYPE_KVM_ICP, errp);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void rtas_dummy(PowerPCCPU *cpu, sPAPRMachineState *spapr,
|
static void rtas_dummy(PowerPCCPU *cpu, sPAPRMachineState *spapr,
|
||||||
uint32_t token,
|
uint32_t token,
|
||||||
uint32_t nargs, target_ulong args,
|
uint32_t nargs, target_ulong args,
|
||||||
|
@ -376,9 +370,7 @@ static void rtas_dummy(PowerPCCPU *cpu, sPAPRMachineState *spapr,
|
||||||
static void xics_kvm_realize(DeviceState *dev, Error **errp)
|
static void xics_kvm_realize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
KVMXICSState *xicskvm = XICS_SPAPR_KVM(dev);
|
KVMXICSState *xicskvm = XICS_SPAPR_KVM(dev);
|
||||||
XICSState *xics = XICS_COMMON(dev);
|
int rc;
|
||||||
int i, rc;
|
|
||||||
Error *error = NULL;
|
|
||||||
struct kvm_create_device xics_create_device = {
|
struct kvm_create_device xics_create_device = {
|
||||||
.type = KVM_DEV_TYPE_XICS,
|
.type = KVM_DEV_TYPE_XICS,
|
||||||
.flags = 0,
|
.flags = 0,
|
||||||
|
@ -428,16 +420,6 @@ static void xics_kvm_realize(DeviceState *dev, Error **errp)
|
||||||
|
|
||||||
xicskvm->kernel_xics_fd = xics_create_device.fd;
|
xicskvm->kernel_xics_fd = xics_create_device.fd;
|
||||||
|
|
||||||
assert(xics->nr_servers);
|
|
||||||
for (i = 0; i < xics->nr_servers; i++) {
|
|
||||||
object_property_set_bool(OBJECT(&xics->ss[i]), true, "realized",
|
|
||||||
&error);
|
|
||||||
if (error) {
|
|
||||||
error_propagate(errp, error);
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
kvm_kernel_irqchip = true;
|
kvm_kernel_irqchip = true;
|
||||||
kvm_msi_via_irqfd_allowed = true;
|
kvm_msi_via_irqfd_allowed = true;
|
||||||
kvm_gsi_direct_mapping = true;
|
kvm_gsi_direct_mapping = true;
|
||||||
|
@ -458,7 +440,6 @@ static void xics_kvm_class_init(ObjectClass *oc, void *data)
|
||||||
|
|
||||||
dc->realize = xics_kvm_realize;
|
dc->realize = xics_kvm_realize;
|
||||||
xsc->cpu_setup = xics_kvm_cpu_setup;
|
xsc->cpu_setup = xics_kvm_cpu_setup;
|
||||||
xsc->set_nr_servers = xics_kvm_set_nr_servers;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static const TypeInfo xics_spapr_kvm_info = {
|
static const TypeInfo xics_spapr_kvm_info = {
|
||||||
|
|
|
@ -239,23 +239,8 @@ static void rtas_int_on(PowerPCCPU *cpu, sPAPRMachineState *spapr,
|
||||||
rtas_st(rets, 0, RTAS_OUT_SUCCESS);
|
rtas_st(rets, 0, RTAS_OUT_SUCCESS);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void xics_spapr_set_nr_servers(XICSState *xics, uint32_t nr_servers,
|
|
||||||
Error **errp)
|
|
||||||
{
|
|
||||||
xics_set_nr_servers(xics, nr_servers, TYPE_ICP, errp);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void xics_spapr_realize(DeviceState *dev, Error **errp)
|
static void xics_spapr_realize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
XICSState *xics = XICS_SPAPR(dev);
|
|
||||||
Error *error = NULL;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
if (!xics->nr_servers) {
|
|
||||||
error_setg(errp, "Number of servers needs to be greater 0");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Registration of global state belongs into realize */
|
/* Registration of global state belongs into realize */
|
||||||
spapr_rtas_register(RTAS_IBM_SET_XIVE, "ibm,set-xive", rtas_set_xive);
|
spapr_rtas_register(RTAS_IBM_SET_XIVE, "ibm,set-xive", rtas_set_xive);
|
||||||
spapr_rtas_register(RTAS_IBM_GET_XIVE, "ibm,get-xive", rtas_get_xive);
|
spapr_rtas_register(RTAS_IBM_GET_XIVE, "ibm,get-xive", rtas_get_xive);
|
||||||
|
@ -268,24 +253,13 @@ static void xics_spapr_realize(DeviceState *dev, Error **errp)
|
||||||
spapr_register_hypercall(H_XIRR_X, h_xirr_x);
|
spapr_register_hypercall(H_XIRR_X, h_xirr_x);
|
||||||
spapr_register_hypercall(H_EOI, h_eoi);
|
spapr_register_hypercall(H_EOI, h_eoi);
|
||||||
spapr_register_hypercall(H_IPOLL, h_ipoll);
|
spapr_register_hypercall(H_IPOLL, h_ipoll);
|
||||||
|
|
||||||
for (i = 0; i < xics->nr_servers; i++) {
|
|
||||||
object_property_set_bool(OBJECT(&xics->ss[i]), true, "realized",
|
|
||||||
&error);
|
|
||||||
if (error) {
|
|
||||||
error_propagate(errp, error);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void xics_spapr_class_init(ObjectClass *oc, void *data)
|
static void xics_spapr_class_init(ObjectClass *oc, void *data)
|
||||||
{
|
{
|
||||||
DeviceClass *dc = DEVICE_CLASS(oc);
|
DeviceClass *dc = DEVICE_CLASS(oc);
|
||||||
XICSStateClass *xsc = XICS_SPAPR_CLASS(oc);
|
|
||||||
|
|
||||||
dc->realize = xics_spapr_realize;
|
dc->realize = xics_spapr_realize;
|
||||||
xsc->set_nr_servers = xics_spapr_set_nr_servers;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static const TypeInfo xics_spapr_info = {
|
static const TypeInfo xics_spapr_info = {
|
||||||
|
|
|
@ -96,17 +96,17 @@
|
||||||
#define HTAB_SIZE(spapr) (1ULL << ((spapr)->htab_shift))
|
#define HTAB_SIZE(spapr) (1ULL << ((spapr)->htab_shift))
|
||||||
|
|
||||||
static XICSState *try_create_xics(const char *type, const char *type_ics,
|
static XICSState *try_create_xics(const char *type, const char *type_ics,
|
||||||
int nr_servers, int nr_irqs, Error **errp)
|
const char *type_icp, int nr_servers,
|
||||||
|
int nr_irqs, Error **errp)
|
||||||
{
|
{
|
||||||
Error *err = NULL, *local_err = NULL;
|
Error *err = NULL, *local_err = NULL;
|
||||||
XICSState *xics;
|
XICSState *xics;
|
||||||
ICSState *ics = NULL;
|
ICSState *ics = NULL;
|
||||||
|
int i;
|
||||||
|
|
||||||
xics = XICS_COMMON(object_new(type));
|
xics = XICS_COMMON(object_new(type));
|
||||||
qdev_set_parent_bus(DEVICE(xics), sysbus_get_default());
|
qdev_set_parent_bus(DEVICE(xics), sysbus_get_default());
|
||||||
object_property_set_int(OBJECT(xics), nr_servers, "nr_servers", &err);
|
object_property_set_bool(OBJECT(xics), true, "realized", &err);
|
||||||
object_property_set_bool(OBJECT(xics), true, "realized", &local_err);
|
|
||||||
error_propagate(&err, local_err);
|
|
||||||
if (err) {
|
if (err) {
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
@ -122,6 +122,22 @@ static XICSState *try_create_xics(const char *type, const char *type_ics,
|
||||||
}
|
}
|
||||||
QLIST_INSERT_HEAD(&xics->ics, ics, list);
|
QLIST_INSERT_HEAD(&xics->ics, ics, list);
|
||||||
|
|
||||||
|
xics->ss = g_malloc0(nr_servers * sizeof(ICPState));
|
||||||
|
xics->nr_servers = nr_servers;
|
||||||
|
|
||||||
|
for (i = 0; i < nr_servers; i++) {
|
||||||
|
ICPState *icp = &xics->ss[i];
|
||||||
|
|
||||||
|
object_initialize(icp, sizeof(*icp), type_icp);
|
||||||
|
object_property_add_child(OBJECT(xics), "icp[*]", OBJECT(icp), NULL);
|
||||||
|
object_property_add_const_link(OBJECT(icp), "xics", OBJECT(xics), NULL);
|
||||||
|
object_property_set_bool(OBJECT(icp), true, "realized", &err);
|
||||||
|
if (err) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
object_unref(OBJECT(icp));
|
||||||
|
}
|
||||||
|
|
||||||
return xics;
|
return xics;
|
||||||
|
|
||||||
error:
|
error:
|
||||||
|
@ -143,7 +159,7 @@ static XICSState *xics_system_init(MachineState *machine,
|
||||||
|
|
||||||
if (machine_kernel_irqchip_allowed(machine)) {
|
if (machine_kernel_irqchip_allowed(machine)) {
|
||||||
xics = try_create_xics(TYPE_XICS_SPAPR_KVM, TYPE_ICS_KVM,
|
xics = try_create_xics(TYPE_XICS_SPAPR_KVM, TYPE_ICS_KVM,
|
||||||
nr_servers, nr_irqs, &err);
|
TYPE_KVM_ICP, nr_servers, nr_irqs, &err);
|
||||||
}
|
}
|
||||||
if (machine_kernel_irqchip_required(machine) && !xics) {
|
if (machine_kernel_irqchip_required(machine) && !xics) {
|
||||||
error_reportf_err(err,
|
error_reportf_err(err,
|
||||||
|
@ -154,8 +170,8 @@ static XICSState *xics_system_init(MachineState *machine,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!xics) {
|
if (!xics) {
|
||||||
xics = try_create_xics(TYPE_XICS_SPAPR, TYPE_ICS_SIMPLE, nr_servers,
|
xics = try_create_xics(TYPE_XICS_SPAPR, TYPE_ICS_SIMPLE, TYPE_ICP,
|
||||||
nr_irqs, errp);
|
nr_servers, nr_irqs, errp);
|
||||||
}
|
}
|
||||||
|
|
||||||
return xics;
|
return xics;
|
||||||
|
|
|
@ -74,7 +74,6 @@ struct XICSStateClass {
|
||||||
DeviceClass parent_class;
|
DeviceClass parent_class;
|
||||||
|
|
||||||
void (*cpu_setup)(XICSState *icp, PowerPCCPU *cpu);
|
void (*cpu_setup)(XICSState *icp, PowerPCCPU *cpu);
|
||||||
void (*set_nr_servers)(XICSState *icp, uint32_t nr_servers, Error **errp);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct XICSState {
|
struct XICSState {
|
||||||
|
@ -190,8 +189,6 @@ void spapr_dt_xics(XICSState *xics, void *fdt, uint32_t phandle);
|
||||||
|
|
||||||
void xics_cpu_setup(XICSState *icp, PowerPCCPU *cpu);
|
void xics_cpu_setup(XICSState *icp, PowerPCCPU *cpu);
|
||||||
void xics_cpu_destroy(XICSState *icp, PowerPCCPU *cpu);
|
void xics_cpu_destroy(XICSState *icp, PowerPCCPU *cpu);
|
||||||
void xics_set_nr_servers(XICSState *xics, uint32_t nr_servers,
|
|
||||||
const char *typename, Error **errp);
|
|
||||||
|
|
||||||
/* Internal XICS interfaces */
|
/* Internal XICS interfaces */
|
||||||
int xics_get_cpu_index_by_dt_id(int cpu_dt_id);
|
int xics_get_cpu_index_by_dt_id(int cpu_dt_id);
|
||||||
|
|
Loading…
Reference in New Issue