s390/cio: skip broken paths
authorSebastian Ott <sebott@linux.vnet.ibm.com>
Mon, 28 Jan 2013 18:31:50 +0000 (19:31 +0100)
committerMartin Schwidefsky <schwidefsky@de.ibm.com>
Thu, 14 Feb 2013 14:55:07 +0000 (15:55 +0100)
Omit known to be broken paths (those set in path_noirq_mask) for the
sense/set PGID and nop IO commands.

Note: path_noirq_mask will be reset in ccw_device_verify_start (the
paths could be healthy again). However if we restart a path
verification via verify_start this mask will not be reset (there is
no need to let the wait for an interrupt time out again - plus we
do not want to loop once we deal with the paths in unknown path group
state).

Reviewed-by: Peter Oberparleiter <peter.oberparleiter@de.ibm.com>
Signed-off-by: Sebastian Ott <sebott@linux.vnet.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
drivers/s390/cio/device_pgid.c

index 6f2987d8da998358d62867dbc3f2eea7deeb397e..f12beb72f263a6f50e403d2cea294adba66f9291 100644 (file)
@@ -70,8 +70,8 @@ static void nop_do(struct ccw_device *cdev)
        struct subchannel *sch = to_subchannel(cdev->dev.parent);
        struct ccw_request *req = &cdev->private->req;
 
-       /* Adjust lpm. */
-       req->lpm = lpm_adjust(req->lpm, sch->schib.pmcw.pam & sch->opm);
+       req->lpm = lpm_adjust(req->lpm, sch->schib.pmcw.pam & sch->opm &
+                             ~cdev->private->path_noirq_mask);
        if (!req->lpm)
                goto out_nopath;
        nop_build_cp(cdev);
@@ -345,8 +345,9 @@ static void snid_done(struct ccw_device *cdev, int rc)
        else {
                donepm = pgid_to_donepm(cdev);
                sch->vpm = donepm & sch->opm;
-               cdev->private->pgid_todo_mask &= ~donepm;
                cdev->private->pgid_reset_mask |= reset;
+               cdev->private->pgid_todo_mask &=
+                       ~(donepm | cdev->private->path_noirq_mask);
                pgid_fill(cdev, pgid);
        }
 out:
@@ -400,8 +401,8 @@ static void snid_do(struct ccw_device *cdev)
        struct subchannel *sch = to_subchannel(cdev->dev.parent);
        struct ccw_request *req = &cdev->private->req;
 
-       /* Adjust lpm if paths are not set in pam. */
-       req->lpm = lpm_adjust(req->lpm, sch->schib.pmcw.pam);
+       req->lpm = lpm_adjust(req->lpm, sch->schib.pmcw.pam &
+                             ~cdev->private->path_noirq_mask);
        if (!req->lpm)
                goto out_nopath;
        snid_build_cp(cdev);