diff options
author | Sebastian Ott <sebott@linux.vnet.ibm.com> | 2010-02-26 16:37:25 -0500 |
---|---|---|
committer | Martin Schwidefsky <sky@mschwide.boeblingen.de.ibm.com> | 2010-02-26 16:37:29 -0500 |
commit | 879acca58a904c25487c89ab11e23eb556fb13d3 (patch) | |
tree | 6b645121090442eb74807ad8388f9986b551def3 /drivers | |
parent | be5d3823f29c09676abd2eeea4f9767bc4a1a531 (diff) |
[S390] cio: introduce cio_settle
This patch introduces a proc file cio_settle. A write request to
this file is blocked until all queued cio actions are handled.
This will allow userspace to wait for pending work affecting
device availability after changing cio_ignore or the hardware
configuration.
Signed-off-by: Sebastian Ott <sebott@linux.vnet.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/s390/cio/css.c | 47 |
1 files changed, 42 insertions, 5 deletions
diff --git a/drivers/s390/cio/css.c b/drivers/s390/cio/css.c index 99fcf9d0ea1..366affcd9bb 100644 --- a/drivers/s390/cio/css.c +++ b/drivers/s390/cio/css.c | |||
@@ -18,6 +18,7 @@ | |||
18 | #include <linux/list.h> | 18 | #include <linux/list.h> |
19 | #include <linux/reboot.h> | 19 | #include <linux/reboot.h> |
20 | #include <linux/suspend.h> | 20 | #include <linux/suspend.h> |
21 | #include <linux/proc_fs.h> | ||
21 | #include <asm/isc.h> | 22 | #include <asm/isc.h> |
22 | #include <asm/crw.h> | 23 | #include <asm/crw.h> |
23 | 24 | ||
@@ -1019,6 +1020,18 @@ static int css_settle(struct device_driver *drv, void *unused) | |||
1019 | return 0; | 1020 | return 0; |
1020 | } | 1021 | } |
1021 | 1022 | ||
1023 | static inline void css_complete_work(void) | ||
1024 | { | ||
1025 | int ret; | ||
1026 | |||
1027 | /* Wait for the evaluation of subchannels to finish. */ | ||
1028 | wait_event(css_eval_wq, atomic_read(&css_eval_scheduled) == 0); | ||
1029 | flush_workqueue(cio_work_q); | ||
1030 | /* Wait for the subchannel type specific initialization to finish */ | ||
1031 | ret = bus_for_each_drv(&css_bus_type, NULL, NULL, css_settle); | ||
1032 | } | ||
1033 | |||
1034 | |||
1022 | /* | 1035 | /* |
1023 | * Wait for the initialization of devices to finish, to make sure we are | 1036 | * Wait for the initialization of devices to finish, to make sure we are |
1024 | * done with our setup if the search for the root device starts. | 1037 | * done with our setup if the search for the root device starts. |
@@ -1027,14 +1040,38 @@ static int __init channel_subsystem_init_sync(void) | |||
1027 | { | 1040 | { |
1028 | /* Start initial subchannel evaluation. */ | 1041 | /* Start initial subchannel evaluation. */ |
1029 | css_schedule_eval_all(); | 1042 | css_schedule_eval_all(); |
1030 | /* Wait for the evaluation of subchannels to finish. */ | 1043 | css_complete_work(); |
1031 | wait_event(css_eval_wq, atomic_read(&css_eval_scheduled) == 0); | 1044 | return 0; |
1032 | flush_workqueue(cio_work_q); | ||
1033 | /* Wait for the subchannel type specific initialization to finish */ | ||
1034 | return bus_for_each_drv(&css_bus_type, NULL, NULL, css_settle); | ||
1035 | } | 1045 | } |
1036 | subsys_initcall_sync(channel_subsystem_init_sync); | 1046 | subsys_initcall_sync(channel_subsystem_init_sync); |
1037 | 1047 | ||
1048 | #ifdef CONFIG_PROC_FS | ||
1049 | static ssize_t cio_settle_write(struct file *file, const char __user *buf, | ||
1050 | size_t count, loff_t *ppos) | ||
1051 | { | ||
1052 | /* Handle pending CRW's. */ | ||
1053 | crw_wait_for_channel_report(); | ||
1054 | css_complete_work(); | ||
1055 | return count; | ||
1056 | } | ||
1057 | |||
1058 | static const struct file_operations cio_settle_proc_fops = { | ||
1059 | .write = cio_settle_write, | ||
1060 | }; | ||
1061 | |||
1062 | static int __init cio_settle_init(void) | ||
1063 | { | ||
1064 | struct proc_dir_entry *entry; | ||
1065 | |||
1066 | entry = proc_create("cio_settle", S_IWUSR, NULL, | ||
1067 | &cio_settle_proc_fops); | ||
1068 | if (!entry) | ||
1069 | return -ENOMEM; | ||
1070 | return 0; | ||
1071 | } | ||
1072 | device_initcall(cio_settle_init); | ||
1073 | #endif /*CONFIG_PROC_FS*/ | ||
1074 | |||
1038 | int sch_is_pseudo_sch(struct subchannel *sch) | 1075 | int sch_is_pseudo_sch(struct subchannel *sch) |
1039 | { | 1076 | { |
1040 | return sch == to_css(sch->dev.parent)->pseudo_subchannel; | 1077 | return sch == to_css(sch->dev.parent)->pseudo_subchannel; |