[S390] cio: fix ccwgroup online vs. ungroup race condition
Ensure atomicity of ungroup operation to prevent concurrent ungroup and online processing which may lead to use-after-release situations. Signed-off-by: Peter Oberparleiter <peter.oberparleiter@de.ibm.com> Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
This commit is contained in:
committed by
Martin Schwidefsky
parent
111e95a4ca
commit
c619d4223e
@@ -91,15 +91,23 @@ ccwgroup_ungroup_store(struct device *dev, struct device_attribute *attr, const
|
|||||||
|
|
||||||
gdev = to_ccwgroupdev(dev);
|
gdev = to_ccwgroupdev(dev);
|
||||||
|
|
||||||
if (gdev->state != CCWGROUP_OFFLINE)
|
/* Prevent concurrent online/offline processing and ungrouping. */
|
||||||
return -EINVAL;
|
if (atomic_cmpxchg(&gdev->onoff, 0, 1) != 0)
|
||||||
|
return -EAGAIN;
|
||||||
|
if (gdev->state != CCWGROUP_OFFLINE) {
|
||||||
|
rc = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
/* Note that we cannot unregister the device from one of its
|
/* Note that we cannot unregister the device from one of its
|
||||||
* attribute methods, so we have to use this roundabout approach.
|
* attribute methods, so we have to use this roundabout approach.
|
||||||
*/
|
*/
|
||||||
rc = device_schedule_callback(dev, ccwgroup_ungroup_callback);
|
rc = device_schedule_callback(dev, ccwgroup_ungroup_callback);
|
||||||
if (rc)
|
out:
|
||||||
count = rc;
|
if (rc) {
|
||||||
|
/* Release onoff "lock" when ungrouping failed. */
|
||||||
|
atomic_set(&gdev->onoff, 0);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user