2 * css bridge implementation
4 * Copyright 2012,2016 IBM Corp.
5 * Author(s): Cornelia Huck <cornelia.huck@de.ibm.com>
6 * Pierre Morel <pmorel@linux.vnet.ibm.com>
8 * This work is licensed under the terms of the GNU GPL, version 2 or (at
9 * your option) any later version. See the COPYING file in the top-level
13 #include "qemu/osdep.h"
14 #include "qapi/error.h"
15 #include "hw/hotplug.h"
16 #include "hw/qdev-properties.h"
17 #include "hw/sysbus.h"
18 #include "qemu/bitops.h"
19 #include "qemu/module.h"
20 #include "hw/s390x/css.h"
21 #include "ccw-device.h"
22 #include "hw/s390x/css-bridge.h"
26 * Invoke device-specific unplug handler, disable the subchannel
27 * (including sending a channel report to the guest) and remove the
28 * device from the virtual css bus.
30 static void ccw_device_unplug(HotplugHandler
*hotplug_dev
,
31 DeviceState
*dev
, Error
**errp
)
33 CcwDevice
*ccw_dev
= CCW_DEVICE(dev
);
34 CCWDeviceClass
*k
= CCW_DEVICE_GET_CLASS(ccw_dev
);
35 SubchDev
*sch
= ccw_dev
->sch
;
39 k
->unplug(hotplug_dev
, dev
, &err
);
41 error_propagate(errp
, err
);
47 * We should arrive here only for device_del, since we don't support
48 * direct hot(un)plug of channels.
51 /* Subchannel is now disabled and no longer valid. */
52 sch
->curr_status
.pmcw
.flags
&= ~(PMCW_FLAGS_MASK_ENA
|
55 css_generate_sch_crws(sch
->cssid
, sch
->ssid
, sch
->schid
, 1, 0);
60 static void virtual_css_bus_reset(BusState
*qbus
)
62 /* This should actually be modelled via the generic css */
66 static char *virtual_css_bus_get_dev_path(DeviceState
*dev
)
68 CcwDevice
*ccw_dev
= CCW_DEVICE(dev
);
69 SubchDev
*sch
= ccw_dev
->sch
;
70 VirtualCssBridge
*bridge
=
71 VIRTUAL_CSS_BRIDGE(qdev_get_parent_bus(dev
)->parent
);
74 * We can't provide a dev path for backward compatibility on
75 * older machines, as it is visible in the migration stream.
77 return bridge
->css_dev_path
?
78 g_strdup_printf("/%02x.%1x.%04x", sch
->cssid
, sch
->ssid
, sch
->devno
) :
82 static void virtual_css_bus_class_init(ObjectClass
*klass
, void *data
)
84 BusClass
*k
= BUS_CLASS(klass
);
86 k
->reset
= virtual_css_bus_reset
;
87 k
->get_dev_path
= virtual_css_bus_get_dev_path
;
90 static const TypeInfo virtual_css_bus_info
= {
91 .name
= TYPE_VIRTUAL_CSS_BUS
,
93 .instance_size
= sizeof(VirtualCssBus
),
94 .class_init
= virtual_css_bus_class_init
,
97 VirtualCssBus
*virtual_css_bus_init(void)
103 /* Create bridge device */
104 dev
= qdev_new(TYPE_VIRTUAL_CSS_BRIDGE
);
105 object_property_add_child(qdev_get_machine(), TYPE_VIRTUAL_CSS_BRIDGE
,
107 sysbus_realize_and_unref(SYS_BUS_DEVICE(dev
), &error_fatal
);
109 /* Create bus on bridge device */
110 bus
= qbus_create(TYPE_VIRTUAL_CSS_BUS
, dev
, "virtual-css");
111 cbus
= VIRTUAL_CSS_BUS(bus
);
113 /* Enable hotplugging */
114 qbus_set_hotplug_handler(bus
, OBJECT(dev
));
116 css_register_io_adapters(CSS_IO_ADAPTER_VIRTIO
, true, false,
122 /***************** Virtual-css Bus Bridge Device ********************/
124 static Property virtual_css_bridge_properties
[] = {
125 DEFINE_PROP_BOOL("css_dev_path", VirtualCssBridge
, css_dev_path
,
127 DEFINE_PROP_END_OF_LIST(),
130 static bool prop_get_true(Object
*obj
, Error
**errp
)
135 static void virtual_css_bridge_class_init(ObjectClass
*klass
, void *data
)
137 HotplugHandlerClass
*hc
= HOTPLUG_HANDLER_CLASS(klass
);
138 DeviceClass
*dc
= DEVICE_CLASS(klass
);
140 hc
->unplug
= ccw_device_unplug
;
141 set_bit(DEVICE_CATEGORY_BRIDGE
, dc
->categories
);
142 device_class_set_props(dc
, virtual_css_bridge_properties
);
143 object_class_property_add_bool(klass
, "cssid-unrestricted",
144 prop_get_true
, NULL
);
145 object_class_property_set_description(klass
, "cssid-unrestricted",
146 "A css device can use any cssid, regardless whether virtual"
147 " or not (read only, always true)");
150 static const TypeInfo virtual_css_bridge_info
= {
151 .name
= TYPE_VIRTUAL_CSS_BRIDGE
,
152 .parent
= TYPE_SYS_BUS_DEVICE
,
153 .instance_size
= sizeof(VirtualCssBridge
),
154 .class_init
= virtual_css_bridge_class_init
,
155 .interfaces
= (InterfaceInfo
[]) {
156 { TYPE_HOTPLUG_HANDLER
},
161 static void virtual_css_register(void)
163 type_register_static(&virtual_css_bridge_info
);
164 type_register_static(&virtual_css_bus_info
);
167 type_init(virtual_css_register
)