Commit ecf5d9ef authored by Peter Oberparleiter's avatar Peter Oberparleiter Committed by Martin Schwidefsky

[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: default avatarPeter Oberparleiter <peter.oberparleiter@de.ibm.com>
Signed-off-by: default avatarMartin Schwidefsky <schwidefsky@de.ibm.com>
parent 46fbe4e4
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
#include "cio.h" #include "cio.h"
#include "cio_debug.h" #include "cio_debug.h"
#include "css.h" #include "css.h"
#include "device.h"
/* /*
* "Blacklisting" of certain devices: * "Blacklisting" of certain devices:
...@@ -191,9 +192,9 @@ static int blacklist_parse_parameters(char *str, range_action action, ...@@ -191,9 +192,9 @@ static int blacklist_parse_parameters(char *str, range_action action,
rc = blacklist_range(ra, from_ssid, to_ssid, from, to, rc = blacklist_range(ra, from_ssid, to_ssid, from, to,
msgtrigger); msgtrigger);
if (rc) if (rc)
totalrc = 1; totalrc = -EINVAL;
} else } else
totalrc = 1; totalrc = -EINVAL;
} }
return totalrc; return totalrc;
...@@ -240,8 +241,10 @@ static int blacklist_parse_proc_parameters(char *buf) ...@@ -240,8 +241,10 @@ static int blacklist_parse_proc_parameters(char *buf)
rc = blacklist_parse_parameters(buf, free, 0); rc = blacklist_parse_parameters(buf, free, 0);
else if (strcmp("add", parm) == 0) else if (strcmp("add", parm) == 0)
rc = blacklist_parse_parameters(buf, add, 0); rc = blacklist_parse_parameters(buf, add, 0);
else if (strcmp("purge", parm) == 0)
return ccw_purge_blacklisted();
else else
return 1; return -EINVAL;
css_schedule_reprobe(); css_schedule_reprobe();
...@@ -353,7 +356,7 @@ cio_ignore_write(struct file *file, const char __user *user_buf, ...@@ -353,7 +356,7 @@ cio_ignore_write(struct file *file, const char __user *user_buf,
} }
ret = blacklist_parse_proc_parameters(buf); ret = blacklist_parse_proc_parameters(buf);
if (ret) if (ret)
rc = -EINVAL; rc = ret;
else else
rc = user_len; rc = user_len;
......
...@@ -31,6 +31,7 @@ ...@@ -31,6 +31,7 @@
#include "device.h" #include "device.h"
#include "ioasm.h" #include "ioasm.h"
#include "io_sch.h" #include "io_sch.h"
#include "blacklist.h"
static struct timer_list recovery_timer; static struct timer_list recovery_timer;
static DEFINE_SPINLOCK(recovery_lock); static DEFINE_SPINLOCK(recovery_lock);
...@@ -1470,6 +1471,45 @@ static void ccw_device_schedule_recovery(void) ...@@ -1470,6 +1471,45 @@ static void ccw_device_schedule_recovery(void)
spin_unlock_irqrestore(&recovery_lock, flags); spin_unlock_irqrestore(&recovery_lock, flags);
} }
static int purge_fn(struct device *dev, void *data)
{
struct ccw_device *cdev = to_ccwdev(dev);
struct ccw_device_private *priv = cdev->private;
int unreg;
spin_lock_irq(cdev->ccwlock);
unreg = is_blacklisted(priv->dev_id.ssid, priv->dev_id.devno) &&
(priv->state == DEV_STATE_OFFLINE);
spin_unlock_irq(cdev->ccwlock);
if (!unreg)
goto out;
if (!get_device(&cdev->dev))
goto out;
CIO_MSG_EVENT(3, "ccw: purging 0.%x.%04x\n", priv->dev_id.ssid,
priv->dev_id.devno);
PREPARE_WORK(&cdev->private->kick_work, ccw_device_call_sch_unregister);
queue_work(slow_path_wq, &cdev->private->kick_work);
out:
/* Abort loop in case of pending signal. */
if (signal_pending(current))
return -EINTR;
return 0;
}
/**
* ccw_purge_blacklisted - purge unused, blacklisted devices
*
* Unregister all ccw devices that are offline and on the blacklist.
*/
int ccw_purge_blacklisted(void)
{
CIO_MSG_EVENT(2, "ccw: purging blacklisted devices\n");
bus_for_each_dev(&ccw_bus_type, NULL, NULL, purge_fn);
return 0;
}
static void device_set_disconnected(struct ccw_device *cdev) static void device_set_disconnected(struct ccw_device *cdev)
{ {
if (!cdev) if (!cdev)
......
...@@ -86,6 +86,7 @@ int ccw_device_is_orphan(struct ccw_device *); ...@@ -86,6 +86,7 @@ int ccw_device_is_orphan(struct ccw_device *);
int ccw_device_recognition(struct ccw_device *); int ccw_device_recognition(struct ccw_device *);
int ccw_device_online(struct ccw_device *); int ccw_device_online(struct ccw_device *);
int ccw_device_offline(struct ccw_device *); int ccw_device_offline(struct ccw_device *);
int ccw_purge_blacklisted(void);
/* Function prototypes for device status and basic sense stuff. */ /* Function prototypes for device status and basic sense stuff. */
void ccw_device_accumulate_irb(struct ccw_device *, struct irb *); void ccw_device_accumulate_irb(struct ccw_device *, struct irb *);
......
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