[S390] zcrypt: Fix possible dead lock in AP bus module.
If a AP device is unconfigured __ap_poll_all() will call device_unregister() in software interrupt context which can cause dead locks. To fix this the device will be only marked as unconfigured and the device_unregister() call will be done later by either ap_scan_bus() or ap_queue_message() in process context. Cc: Martin Schwidefsky <schwidefsky@de.ibm.com> Signed-off-by: Ralph Wuerthner <rwuerthn@de.ibm.com> Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
This commit is contained in:
parent
25c61a1fe8
commit
c6a4826473
|
@ -757,10 +757,16 @@ static void ap_scan_bus(struct work_struct *unused)
|
||||||
(void *)(unsigned long)qid,
|
(void *)(unsigned long)qid,
|
||||||
__ap_scan_bus);
|
__ap_scan_bus);
|
||||||
rc = ap_query_queue(qid, &queue_depth, &device_type);
|
rc = ap_query_queue(qid, &queue_depth, &device_type);
|
||||||
if (dev && rc) {
|
if (dev) {
|
||||||
|
ap_dev = to_ap_dev(dev);
|
||||||
|
spin_lock_bh(&ap_dev->lock);
|
||||||
|
if (rc || ap_dev->unregistered) {
|
||||||
|
spin_unlock_bh(&ap_dev->lock);
|
||||||
put_device(dev);
|
put_device(dev);
|
||||||
device_unregister(dev);
|
device_unregister(dev);
|
||||||
continue;
|
continue;
|
||||||
|
} else
|
||||||
|
spin_unlock_bh(&ap_dev->lock);
|
||||||
}
|
}
|
||||||
if (dev) {
|
if (dev) {
|
||||||
put_device(dev);
|
put_device(dev);
|
||||||
|
@ -994,7 +1000,7 @@ void ap_queue_message(struct ap_device *ap_dev, struct ap_message *ap_msg)
|
||||||
ap_dev->unregistered = 1;
|
ap_dev->unregistered = 1;
|
||||||
} else {
|
} else {
|
||||||
ap_dev->drv->receive(ap_dev, ap_msg, ERR_PTR(-ENODEV));
|
ap_dev->drv->receive(ap_dev, ap_msg, ERR_PTR(-ENODEV));
|
||||||
rc = 0;
|
rc = -ENODEV;
|
||||||
}
|
}
|
||||||
spin_unlock_bh(&ap_dev->lock);
|
spin_unlock_bh(&ap_dev->lock);
|
||||||
if (rc == -ENODEV)
|
if (rc == -ENODEV)
|
||||||
|
@ -1044,18 +1050,12 @@ static void ap_poll_timeout(unsigned long unused)
|
||||||
*/
|
*/
|
||||||
static int __ap_poll_all(struct ap_device *ap_dev, unsigned long *flags)
|
static int __ap_poll_all(struct ap_device *ap_dev, unsigned long *flags)
|
||||||
{
|
{
|
||||||
int rc;
|
|
||||||
|
|
||||||
spin_lock(&ap_dev->lock);
|
spin_lock(&ap_dev->lock);
|
||||||
if (!ap_dev->unregistered) {
|
if (!ap_dev->unregistered) {
|
||||||
rc = ap_poll_queue(ap_dev, flags);
|
if (ap_poll_queue(ap_dev, flags))
|
||||||
if (rc)
|
|
||||||
ap_dev->unregistered = 1;
|
ap_dev->unregistered = 1;
|
||||||
} else
|
}
|
||||||
rc = 0;
|
|
||||||
spin_unlock(&ap_dev->lock);
|
spin_unlock(&ap_dev->lock);
|
||||||
if (rc)
|
|
||||||
device_unregister(&ap_dev->device);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue