diff options
author | Peter Oberparleiter <peter.oberparleiter@de.ibm.com> | 2008-10-10 15:33:06 -0400 |
---|---|---|
committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2008-10-10 15:33:47 -0400 |
commit | ecf5d9ef68b868b293b40b89615a92de7310d065 (patch) | |
tree | ec92f05f79e222075e2425129e15b0ee9131364e /drivers/s390/cio/device.c | |
parent | 46fbe4e46ddb88805245a24f684400b50ead68a7 (diff) |
[S390] cio: introduce purge function for /proc/cio_ignore
Allow users to remove blacklisted ccw devices by using the
/proc/cio_ignore interface:
echo purge > /proc/cio_ignore
will remove all devices which are offline and blacklisted.
Signed-off-by: Peter Oberparleiter <peter.oberparleiter@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Diffstat (limited to 'drivers/s390/cio/device.c')
-rw-r--r-- | drivers/s390/cio/device.c | 40 |
1 files changed, 40 insertions, 0 deletions
diff --git a/drivers/s390/cio/device.c b/drivers/s390/cio/device.c index 38a79ecfc743..8575d263b13e 100644 --- a/drivers/s390/cio/device.c +++ b/drivers/s390/cio/device.c | |||
@@ -31,6 +31,7 @@ | |||
31 | #include "device.h" | 31 | #include "device.h" |
32 | #include "ioasm.h" | 32 | #include "ioasm.h" |
33 | #include "io_sch.h" | 33 | #include "io_sch.h" |
34 | #include "blacklist.h" | ||
34 | 35 | ||
35 | static struct timer_list recovery_timer; | 36 | static struct timer_list recovery_timer; |
36 | static DEFINE_SPINLOCK(recovery_lock); | 37 | static DEFINE_SPINLOCK(recovery_lock); |
@@ -1470,6 +1471,45 @@ static void ccw_device_schedule_recovery(void) | |||
1470 | spin_unlock_irqrestore(&recovery_lock, flags); | 1471 | spin_unlock_irqrestore(&recovery_lock, flags); |
1471 | } | 1472 | } |
1472 | 1473 | ||
1474 | static int purge_fn(struct device *dev, void *data) | ||
1475 | { | ||
1476 | struct ccw_device *cdev = to_ccwdev(dev); | ||
1477 | struct ccw_device_private *priv = cdev->private; | ||
1478 | int unreg; | ||
1479 | |||
1480 | spin_lock_irq(cdev->ccwlock); | ||
1481 | unreg = is_blacklisted(priv->dev_id.ssid, priv->dev_id.devno) && | ||
1482 | (priv->state == DEV_STATE_OFFLINE); | ||
1483 | spin_unlock_irq(cdev->ccwlock); | ||
1484 | if (!unreg) | ||
1485 | goto out; | ||
1486 | if (!get_device(&cdev->dev)) | ||
1487 | goto out; | ||
1488 | CIO_MSG_EVENT(3, "ccw: purging 0.%x.%04x\n", priv->dev_id.ssid, | ||
1489 | priv->dev_id.devno); | ||
1490 | PREPARE_WORK(&cdev->private->kick_work, ccw_device_call_sch_unregister); | ||
1491 | queue_work(slow_path_wq, &cdev->private->kick_work); | ||
1492 | |||
1493 | out: | ||
1494 | /* Abort loop in case of pending signal. */ | ||
1495 | if (signal_pending(current)) | ||
1496 | return -EINTR; | ||
1497 | |||
1498 | return 0; | ||
1499 | } | ||
1500 | |||
1501 | /** | ||
1502 | * ccw_purge_blacklisted - purge unused, blacklisted devices | ||
1503 | * | ||
1504 | * Unregister all ccw devices that are offline and on the blacklist. | ||
1505 | */ | ||
1506 | int ccw_purge_blacklisted(void) | ||
1507 | { | ||
1508 | CIO_MSG_EVENT(2, "ccw: purging blacklisted devices\n"); | ||
1509 | bus_for_each_dev(&ccw_bus_type, NULL, NULL, purge_fn); | ||
1510 | return 0; | ||
1511 | } | ||
1512 | |||
1473 | static void device_set_disconnected(struct ccw_device *cdev) | 1513 | static void device_set_disconnected(struct ccw_device *cdev) |
1474 | { | 1514 | { |
1475 | if (!cdev) | 1515 | if (!cdev) |