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
12 #include "qemu/osdep.h"
13 #include "qapi/error.h"
14 #include "hw/hotplug.h"
15 #include "hw/sysbus.h"
16 #include "qemu/bitops.h"
17 #include "hw/s390x/css.h"
18 #include "ccw-device.h"
19 #include "hw/s390x/css-bridge.h"
23 * Invoke device-specific unplug handler, disable the subchannel
24 * (including sending a channel report to the guest) and remove the
25 * device from the virtual css bus.
27 static void ccw_device_unplug(HotplugHandler
*hotplug_dev
,
28 DeviceState
*dev
, Error
**errp
)
30 CcwDevice
*ccw_dev
= CCW_DEVICE(dev
);
31 CCWDeviceClass
*k
= CCW_DEVICE_GET_CLASS(ccw_dev
);
32 SubchDev
*sch
= ccw_dev
->sch
;
36 k
->unplug(hotplug_dev
, dev
, &err
);
38 error_propagate(errp
, err
);
44 * We should arrive here only for device_del, since we don't support
45 * direct hot(un)plug of channels.
48 /* Subchannel is now disabled and no longer valid. */
49 sch
->curr_status
.pmcw
.flags
&= ~(PMCW_FLAGS_MASK_ENA
|
52 css_generate_sch_crws(sch
->cssid
, sch
->ssid
, sch
->schid
, 1, 0);
54 object_unparent(OBJECT(dev
));
57 static void virtual_css_bus_reset(BusState
*qbus
)
59 /* This should actually be modelled via the generic css */
63 static char *virtual_css_bus_get_dev_path(DeviceState
*dev
)
65 CcwDevice
*ccw_dev
= CCW_DEVICE(dev
);
66 SubchDev
*sch
= ccw_dev
->sch
;
67 VirtualCssBridge
*bridge
=
68 VIRTUAL_CSS_BRIDGE(qdev_get_parent_bus(dev
)->parent
);
71 * We can't provide a dev path for backward compatibility on
72 * older machines, as it is visible in the migration stream.
74 return bridge
->css_dev_path
?
75 g_strdup_printf("/%02x.%1x.%04x", sch
->cssid
, sch
->ssid
, sch
->devno
) :
79 static void virtual_css_bus_class_init(ObjectClass
*klass
, void *data
)
81 BusClass
*k
= BUS_CLASS(klass
);
83 k
->reset
= virtual_css_bus_reset
;
84 k
->get_dev_path
= virtual_css_bus_get_dev_path
;
87 static const TypeInfo virtual_css_bus_info
= {
88 .name
= TYPE_VIRTUAL_CSS_BUS
,
90 .instance_size
= sizeof(VirtualCssBus
),
91 .class_init
= virtual_css_bus_class_init
,
94 VirtualCssBus
*virtual_css_bus_init(void)
100 /* Create bridge device */
101 dev
= qdev_create(NULL
, TYPE_VIRTUAL_CSS_BRIDGE
);
102 object_property_add_child(qdev_get_machine(), TYPE_VIRTUAL_CSS_BRIDGE
,
104 qdev_init_nofail(dev
);
106 /* Create bus on bridge device */
107 bus
= qbus_create(TYPE_VIRTUAL_CSS_BUS
, dev
, "virtual-css");
108 cbus
= VIRTUAL_CSS_BUS(bus
);
109 cbus
->squash_mcss
= s390_get_squash_mcss();
111 /* Enable hotplugging */
112 qbus_set_hotplug_handler(bus
, dev
, &error_abort
);
114 css_register_io_adapters(CSS_IO_ADAPTER_VIRTIO
, true, false,
120 /***************** Virtual-css Bus Bridge Device ********************/
122 static Property virtual_css_bridge_properties
[] = {
123 DEFINE_PROP_BOOL("css_dev_path", VirtualCssBridge
, css_dev_path
,
125 DEFINE_PROP_END_OF_LIST(),
128 static bool prop_get_true(Object
*obj
, Error
**errp
)
133 static void virtual_css_bridge_class_init(ObjectClass
*klass
, void *data
)
135 HotplugHandlerClass
*hc
= HOTPLUG_HANDLER_CLASS(klass
);
136 DeviceClass
*dc
= DEVICE_CLASS(klass
);
138 hc
->unplug
= ccw_device_unplug
;
139 set_bit(DEVICE_CATEGORY_BRIDGE
, dc
->categories
);
140 dc
->props
= virtual_css_bridge_properties
;
141 object_class_property_add_bool(klass
, "cssid-unrestricted",
142 prop_get_true
, NULL
, NULL
);
143 object_class_property_set_description(klass
, "cssid-unrestricted",
144 "A css device can use any cssid, regardless whether virtual"
145 " or not (read only, always true)",
149 static const TypeInfo virtual_css_bridge_info
= {
150 .name
= TYPE_VIRTUAL_CSS_BRIDGE
,
151 .parent
= TYPE_SYS_BUS_DEVICE
,
152 .instance_size
= sizeof(VirtualCssBridge
),
153 .class_init
= virtual_css_bridge_class_init
,
154 .interfaces
= (InterfaceInfo
[]) {
155 { TYPE_HOTPLUG_HANDLER
},
160 static void virtual_css_register(void)
162 type_register_static(&virtual_css_bridge_info
);
163 type_register_static(&virtual_css_bus_info
);
166 type_init(virtual_css_register
)