Commit 135a8b4c authored by Sebastian Ott's avatar Sebastian Ott Committed by Martin Schwidefsky

s390/cio: fix unbind of io_subchannel_driver

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: default avatarShalini Chellathurai Saroja <shalini@de.ibm.com>
Signed-off-by: default avatarSebastian Ott <sebott@linux.vnet.ibm.com>
Acked-by: default avatarPeter Oberparleiter <oberpar@linux.vnet.ibm.com>
Signed-off-by: default avatarMartin Schwidefsky <schwidefsky@de.ibm.com>
parent 88bf319f
...@@ -1073,8 +1073,7 @@ static int io_subchannel_probe(struct subchannel *sch) ...@@ -1073,8 +1073,7 @@ static int io_subchannel_probe(struct subchannel *sch)
return 0; return 0;
} }
static int static int io_subchannel_remove(struct subchannel *sch)
io_subchannel_remove (struct subchannel *sch)
{ {
struct io_subchannel_private *io_priv = to_io_private(sch); struct io_subchannel_private *io_priv = to_io_private(sch);
struct ccw_device *cdev; struct ccw_device *cdev;
...@@ -1082,14 +1081,12 @@ io_subchannel_remove (struct subchannel *sch) ...@@ -1082,14 +1081,12 @@ io_subchannel_remove (struct subchannel *sch)
cdev = sch_get_cdev(sch); cdev = sch_get_cdev(sch);
if (!cdev) if (!cdev)
goto out_free; goto out_free;
io_subchannel_quiesce(sch);
/* Set ccw device to not operational and drop reference. */ ccw_device_unregister(cdev);
spin_lock_irq(cdev->ccwlock); spin_lock_irq(sch->lock);
sch_set_cdev(sch, NULL); sch_set_cdev(sch, NULL);
set_io_private(sch, NULL); set_io_private(sch, NULL);
cdev->private->state = DEV_STATE_NOT_OPER; spin_unlock_irq(sch->lock);
spin_unlock_irq(cdev->ccwlock);
ccw_device_unregister(cdev);
out_free: out_free:
kfree(io_priv); kfree(io_priv);
sysfs_remove_group(&sch->dev.kobj, &io_subchannel_attr_group); sysfs_remove_group(&sch->dev.kobj, &io_subchannel_attr_group);
...@@ -1721,6 +1718,7 @@ static int ccw_device_remove(struct device *dev) ...@@ -1721,6 +1718,7 @@ static int ccw_device_remove(struct device *dev)
{ {
struct ccw_device *cdev = to_ccwdev(dev); struct ccw_device *cdev = to_ccwdev(dev);
struct ccw_driver *cdrv = cdev->drv; struct ccw_driver *cdrv = cdev->drv;
struct subchannel *sch;
int ret; int ret;
if (cdrv->remove) if (cdrv->remove)
...@@ -1746,7 +1744,9 @@ static int ccw_device_remove(struct device *dev) ...@@ -1746,7 +1744,9 @@ static int ccw_device_remove(struct device *dev)
ccw_device_set_timeout(cdev, 0); ccw_device_set_timeout(cdev, 0);
cdev->drv = NULL; cdev->drv = NULL;
cdev->private->int_class = IRQIO_CIO; cdev->private->int_class = IRQIO_CIO;
sch = to_subchannel(cdev->dev.parent);
spin_unlock_irq(cdev->ccwlock); spin_unlock_irq(cdev->ccwlock);
io_subchannel_quiesce(sch);
__disable_cmf(cdev); __disable_cmf(cdev);
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment