diff options
author | Cornelia Huck <cornelia.huck@de.ibm.com> | 2006-12-04 09:41:01 -0500 |
---|---|---|
committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2006-12-04 09:41:01 -0500 |
commit | 24cb5b4846ebae5543869b5c596c2650f380df53 (patch) | |
tree | fd45cf7fab0a065c7529c5811e3258a66e877095 | |
parent | 9163bb2e556f6c7879961df94540f0879db4717b (diff) |
[S390] cio: Use path verification for last path gone after vary off.
If the last path to a device is gone after a chpid has been varied
off, putting it on the slow queue doesn't prevent a device driver
from still attempting to use it (it may stay on the slow queue for a
long time). Instead, trigger a verify event which will prevent I/O
attempts from the device driver immediately.
Signed-off-by: Cornelia Huck <cornelia.huck@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
-rw-r--r-- | drivers/s390/cio/chsc.c | 22 | ||||
-rw-r--r-- | drivers/s390/cio/css.h | 1 | ||||
-rw-r--r-- | drivers/s390/cio/device_fsm.c | 11 |
3 files changed, 24 insertions, 10 deletions
diff --git a/drivers/s390/cio/chsc.c b/drivers/s390/cio/chsc.c index 9d92540bd99e..11900de94cb3 100644 --- a/drivers/s390/cio/chsc.c +++ b/drivers/s390/cio/chsc.c | |||
@@ -744,20 +744,22 @@ __s390_subchannel_vary_chpid(struct subchannel *sch, __u8 chpid, int on) | |||
744 | device_trigger_reprobe(sch); | 744 | device_trigger_reprobe(sch); |
745 | else if (sch->driver && sch->driver->verify) | 745 | else if (sch->driver && sch->driver->verify) |
746 | sch->driver->verify(&sch->dev); | 746 | sch->driver->verify(&sch->dev); |
747 | } else { | 747 | break; |
748 | sch->opm &= ~(0x80 >> chp); | 748 | } |
749 | sch->lpm &= ~(0x80 >> chp); | 749 | sch->opm &= ~(0x80 >> chp); |
750 | if (check_for_io_on_path(sch, chp)) | 750 | sch->lpm &= ~(0x80 >> chp); |
751 | /* Path verification is done after killing. */ | 751 | if (check_for_io_on_path(sch, chp)) |
752 | device_kill_io(sch); | 752 | /* Path verification is done after killing. */ |
753 | else if (!sch->lpm) { | 753 | device_kill_io(sch); |
754 | else if (!sch->lpm) { | ||
755 | if (device_trigger_verify(sch) != 0) { | ||
754 | if (css_enqueue_subchannel_slow(sch->schid)) { | 756 | if (css_enqueue_subchannel_slow(sch->schid)) { |
755 | css_clear_subchannel_slow_list(); | 757 | css_clear_subchannel_slow_list(); |
756 | need_rescan = 1; | 758 | need_rescan = 1; |
757 | } | 759 | } |
758 | } else if (sch->driver && sch->driver->verify) | 760 | } |
759 | sch->driver->verify(&sch->dev); | 761 | } else if (sch->driver && sch->driver->verify) |
760 | } | 762 | sch->driver->verify(&sch->dev); |
761 | break; | 763 | break; |
762 | } | 764 | } |
763 | spin_unlock_irqrestore(&sch->lock, flags); | 765 | spin_unlock_irqrestore(&sch->lock, flags); |
diff --git a/drivers/s390/cio/css.h b/drivers/s390/cio/css.h index 4c2ff8336288..da73453fa54b 100644 --- a/drivers/s390/cio/css.h +++ b/drivers/s390/cio/css.h | |||
@@ -171,6 +171,7 @@ void device_trigger_reprobe(struct subchannel *); | |||
171 | /* Helper functions for vary on/off. */ | 171 | /* Helper functions for vary on/off. */ |
172 | int device_is_online(struct subchannel *); | 172 | int device_is_online(struct subchannel *); |
173 | void device_kill_io(struct subchannel *); | 173 | void device_kill_io(struct subchannel *); |
174 | int device_trigger_verify(struct subchannel *sch); | ||
174 | 175 | ||
175 | /* Machine check helper function. */ | 176 | /* Machine check helper function. */ |
176 | void device_kill_pending_timer(struct subchannel *); | 177 | void device_kill_pending_timer(struct subchannel *); |
diff --git a/drivers/s390/cio/device_fsm.c b/drivers/s390/cio/device_fsm.c index de3d0857db9f..7665000e8dfe 100644 --- a/drivers/s390/cio/device_fsm.c +++ b/drivers/s390/cio/device_fsm.c | |||
@@ -59,6 +59,17 @@ device_set_disconnected(struct subchannel *sch) | |||
59 | cdev->private->state = DEV_STATE_DISCONNECTED; | 59 | cdev->private->state = DEV_STATE_DISCONNECTED; |
60 | } | 60 | } |
61 | 61 | ||
62 | int device_trigger_verify(struct subchannel *sch) | ||
63 | { | ||
64 | struct ccw_device *cdev; | ||
65 | |||
66 | cdev = sch->dev.driver_data; | ||
67 | if (!cdev || !cdev->online) | ||
68 | return -EINVAL; | ||
69 | dev_fsm_event(cdev, DEV_EVENT_VERIFY); | ||
70 | return 0; | ||
71 | } | ||
72 | |||
62 | /* | 73 | /* |
63 | * Timeout function. It just triggers a DEV_EVENT_TIMEOUT. | 74 | * Timeout function. It just triggers a DEV_EVENT_TIMEOUT. |
64 | */ | 75 | */ |