Commit a6ef1565 authored by Sebastian Ott's avatar Sebastian Ott Committed by Martin Schwidefsky

s390/cio: fix use after free in cmb processing

Devices with active channel measurement are included in a list. When a
device is removed without deactivating channel measurement first the
list_head is freed but still used. Fix this by making sure that
channel measurement is deactivated during device deregistration.

For devices that we deregister because they are no longer accessible
deactivating channel measurement will fail. In this case we can report
success because the FW will no longer access the measurement block.

In addition to these steps keep an extra device reference while
channel measurement is active.
Signed-off-by: default avatarSebastian Ott <sebott@linux.vnet.ibm.com>
Reviewed-by: default avatarCornelia Huck <cornelia.huck@de.ibm.com>
Signed-off-by: default avatarMartin Schwidefsky <schwidefsky@de.ibm.com>
parent 1bc6664b
...@@ -186,9 +186,8 @@ static inline void cmf_activate(void *area, unsigned int onoff) ...@@ -186,9 +186,8 @@ static inline void cmf_activate(void *area, unsigned int onoff)
static int set_schib(struct ccw_device *cdev, u32 mme, int mbfc, static int set_schib(struct ccw_device *cdev, u32 mme, int mbfc,
unsigned long address) unsigned long address)
{ {
struct subchannel *sch; struct subchannel *sch = to_subchannel(cdev->dev.parent);
int ret;
sch = to_subchannel(cdev->dev.parent);
sch->config.mme = mme; sch->config.mme = mme;
sch->config.mbfc = mbfc; sch->config.mbfc = mbfc;
...@@ -198,7 +197,15 @@ static int set_schib(struct ccw_device *cdev, u32 mme, int mbfc, ...@@ -198,7 +197,15 @@ static int set_schib(struct ccw_device *cdev, u32 mme, int mbfc,
else else
sch->config.mbi = address; sch->config.mbi = address;
return cio_commit_config(sch); ret = cio_commit_config(sch);
if (!mme && ret == -ENODEV) {
/*
* The task was to disable measurement block updates but
* the subchannel is already gone. Report success.
*/
ret = 0;
}
return ret;
} }
struct set_schib_struct { struct set_schib_struct {
...@@ -606,12 +613,6 @@ static void free_cmb(struct ccw_device *cdev) ...@@ -606,12 +613,6 @@ static void free_cmb(struct ccw_device *cdev)
spin_lock_irq(cdev->ccwlock); spin_lock_irq(cdev->ccwlock);
priv = cdev->private; priv = cdev->private;
if (list_empty(&priv->cmb_list)) {
/* already freed */
goto out;
}
cmb_data = priv->cmb; cmb_data = priv->cmb;
priv->cmb = NULL; priv->cmb = NULL;
if (cmb_data) if (cmb_data)
...@@ -626,7 +627,6 @@ static void free_cmb(struct ccw_device *cdev) ...@@ -626,7 +627,6 @@ static void free_cmb(struct ccw_device *cdev)
free_pages((unsigned long)cmb_area.mem, get_order(size)); free_pages((unsigned long)cmb_area.mem, get_order(size));
cmb_area.mem = NULL; cmb_area.mem = NULL;
} }
out:
spin_unlock_irq(cdev->ccwlock); spin_unlock_irq(cdev->ccwlock);
spin_unlock(&cmb_area.lock); spin_unlock(&cmb_area.lock);
} }
...@@ -1227,6 +1227,7 @@ int enable_cmf(struct ccw_device *cdev) ...@@ -1227,6 +1227,7 @@ int enable_cmf(struct ccw_device *cdev)
int ret; int ret;
device_lock(&cdev->dev); device_lock(&cdev->dev);
get_device(&cdev->dev);
ret = cmbops->alloc(cdev); ret = cmbops->alloc(cdev);
if (ret) if (ret)
goto out; goto out;
...@@ -1242,6 +1243,9 @@ int enable_cmf(struct ccw_device *cdev) ...@@ -1242,6 +1243,9 @@ int enable_cmf(struct ccw_device *cdev)
cmbops->free(cdev); cmbops->free(cdev);
} }
out: out:
if (ret)
put_device(&cdev->dev);
device_unlock(&cdev->dev); device_unlock(&cdev->dev);
return ret; return ret;
} }
...@@ -1265,6 +1269,7 @@ int __disable_cmf(struct ccw_device *cdev) ...@@ -1265,6 +1269,7 @@ int __disable_cmf(struct ccw_device *cdev)
sysfs_remove_group(&cdev->dev.kobj, cmbops->attr_group); sysfs_remove_group(&cdev->dev.kobj, cmbops->attr_group);
cmbops->free(cdev); cmbops->free(cdev);
put_device(&cdev->dev);
return ret; return ret;
} }
......
...@@ -1787,6 +1787,8 @@ static int ccw_device_remove(struct device *dev) ...@@ -1787,6 +1787,8 @@ static int ccw_device_remove(struct device *dev)
cdev->drv = NULL; cdev->drv = NULL;
cdev->private->int_class = IRQIO_CIO; cdev->private->int_class = IRQIO_CIO;
spin_unlock_irq(cdev->ccwlock); spin_unlock_irq(cdev->ccwlock);
__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