s390/cio: fix unbind of io_subchannel_driver
authorSebastian Ott <sebott@linux.vnet.ibm.com>
Thu, 15 Mar 2018 14:03:43 +0000 (15:03 +0100)
committerMartin Schwidefsky <schwidefsky@de.ibm.com>
Mon, 26 Mar 2018 14:13:06 +0000 (16:13 +0200)
If the io_subchannel_driver is unbound from a subchannel it bluntly kills
all I/O on the subchannel and sets the ccw_device state to not operable
before deregistering the ccw_device. However, for online devices we should
set the device offline (disband path groups etc.) which does not happen if
the device is in not oper state.

Simply deregister the ccw device - ccw_device_remove is smart enough to set
the device offline properly. If everything fails call io_subchannel_quiesce
afterwards as a safeguard.

Reported-by: Shalini Chellathurai Saroja <shalini@de.ibm.com>
Signed-off-by: Sebastian Ott <sebott@linux.vnet.ibm.com>
Acked-by: Peter Oberparleiter <oberpar@linux.vnet.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
drivers/s390/cio/device.c

index f50ea03..1540229 100644 (file)
@@ -1073,8 +1073,7 @@ out_schedule:
        return 0;
 }
 
-static int
-io_subchannel_remove (struct subchannel *sch)
+static int io_subchannel_remove(struct subchannel *sch)
 {
        struct io_subchannel_private *io_priv = to_io_private(sch);
        struct ccw_device *cdev;
@@ -1082,14 +1081,12 @@ io_subchannel_remove (struct subchannel *sch)
        cdev = sch_get_cdev(sch);
        if (!cdev)
                goto out_free;
-       io_subchannel_quiesce(sch);
-       /* Set ccw device to not operational and drop reference. */
-       spin_lock_irq(cdev->ccwlock);
+
+       ccw_device_unregister(cdev);
+       spin_lock_irq(sch->lock);
        sch_set_cdev(sch, NULL);
        set_io_private(sch, NULL);
-       cdev->private->state = DEV_STATE_NOT_OPER;
-       spin_unlock_irq(cdev->ccwlock);
-       ccw_device_unregister(cdev);
+       spin_unlock_irq(sch->lock);
 out_free:
        kfree(io_priv);
        sysfs_remove_group(&sch->dev.kobj, &io_subchannel_attr_group);
@@ -1721,6 +1718,7 @@ static int ccw_device_remove(struct device *dev)
 {
        struct ccw_device *cdev = to_ccwdev(dev);
        struct ccw_driver *cdrv = cdev->drv;
+       struct subchannel *sch;
        int ret;
 
        if (cdrv->remove)
@@ -1746,7 +1744,9 @@ static int ccw_device_remove(struct device *dev)
        ccw_device_set_timeout(cdev, 0);
        cdev->drv = NULL;
        cdev->private->int_class = IRQIO_CIO;
+       sch = to_subchannel(cdev->dev.parent);
        spin_unlock_irq(cdev->ccwlock);
+       io_subchannel_quiesce(sch);
        __disable_cmf(cdev);
 
        return 0;