Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/dev/ata Add WDF_OPEN flag to really disallow opening of ...



details:   https://anonhg.NetBSD.org/src/rev/7e1e28b6ddbb
branches:  trunk
changeset: 357257:7e1e28b6ddbb
user:      mlelstv <mlelstv%NetBSD.org@localhost>
date:      Fri Nov 03 13:01:26 2017 +0000

description:
Add WDF_OPEN flag to really disallow opening of a disk that has been invalidated.

Restore wdbiorestart function to actually retry the failed I/O request instead
of just restarting the queue.

Fix compilation without ATADEBUG.

diffstat:

 sys/dev/ata/wd.c    |  63 ++++++++++++++++++++++++++++++++++------------------
 sys/dev/ata/wdvar.h |   3 +-
 2 files changed, 43 insertions(+), 23 deletions(-)

diffs (156 lines):

diff -r 37a13bdcdc49 -r 7e1e28b6ddbb sys/dev/ata/wd.c
--- a/sys/dev/ata/wd.c  Fri Nov 03 09:59:07 2017 +0000
+++ b/sys/dev/ata/wd.c  Fri Nov 03 13:01:26 2017 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: wd.c,v 1.434 2017/11/01 19:34:46 mlelstv Exp $ */
+/*     $NetBSD: wd.c,v 1.435 2017/11/03 13:01:26 mlelstv Exp $ */
 
 /*
  * Copyright (c) 1998, 2001 Manuel Bouyer.  All rights reserved.
@@ -54,7 +54,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.434 2017/11/01 19:34:46 mlelstv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.435 2017/11/03 13:01:26 mlelstv Exp $");
 
 #include "opt_ata.h"
 #include "opt_wd.h"
@@ -194,6 +194,7 @@
 static void    wdioctlstrategy(struct buf *);
 
 static void    wdstart(device_t);
+static void    wdstart1(struct wd_softc *, struct buf *, struct ata_xfer *);
 static int     wd_diskstart(device_t, struct buf *);
 static int     wd_dumpblocks(device_t, void *, daddr_t, int);
 static void    wd_iosize(device_t, int *);
@@ -607,23 +608,10 @@
        biodone(bp);
 }
 
-static int
-wd_diskstart(device_t dev, struct buf *bp)
+static void
+wdstart1(struct wd_softc *wd, struct buf *bp, struct ata_xfer *xfer)
 {
-       struct wd_softc *wd = device_private(dev);
        struct dk_softc *dksc = &wd->sc_dksc;
-       struct ata_xfer *xfer;
-
-       mutex_enter(&wd->sc_lock);
-
-       xfer = ata_get_xfer_ext(wd->drvp->chnl_softc, 0,
-           WD_USE_NCQ(wd) ? WD_MAX_OPENINGS(wd) : 0);
-       if (xfer == NULL) {
-               ATADEBUG_PRINT(("wdstart %s no xfer\n",
-                   dksc->sc_xname), DEBUG_XFERS);
-               mutex_exit(&wd->sc_lock);
-               return EAGAIN;
-       }
 
        KASSERT(bp == xfer->c_bio.bp || xfer->c_bio.bp == NULL);
        KASSERT((xfer->c_flags & (C_WAITACT|C_FREE)) == 0);
@@ -722,6 +710,29 @@
        default:
                panic("wdstart1: bad return code from ata_bio()");
        }
+}
+
+static int
+wd_diskstart(device_t dev, struct buf *bp)
+{
+       struct wd_softc *wd = device_private(dev);
+#ifdef ATADEBUG
+       struct dk_softc *dksc = &wd->sc_dksc;
+#endif
+       struct ata_xfer *xfer;
+
+       mutex_enter(&wd->sc_lock);
+
+       xfer = ata_get_xfer_ext(wd->drvp->chnl_softc, 0,
+           WD_USE_NCQ(wd) ? WD_MAX_OPENINGS(wd) : 0);
+       if (xfer == NULL) {
+               ATADEBUG_PRINT(("wd_diskstart %s no xfer\n",
+                   dksc->sc_xname), DEBUG_XFERS);
+               mutex_exit(&wd->sc_lock);
+               return EAGAIN;
+       }
+
+       wdstart1(wd, bp, xfer);
 
        mutex_exit(&wd->sc_lock);
 
@@ -825,7 +836,7 @@
                        /* Rerun ASAP if just requeued */
                        callout_reset(&xfer->c_retry_callout,
                            (xfer->c_bio.error == REQUEUE) ? 1 : RECOVERYTIME,
-                           wdbiorestart, wd);
+                           wdbiorestart, xfer);
 
                        mutex_exit(&wd->sc_lock);
                        return;
@@ -896,13 +907,19 @@
 static void
 wdbiorestart(void *v)
 {
-       struct wd_softc *wd = v;
+       struct ata_xfer *xfer = v;
+       struct buf *bp = xfer->c_bio.bp;
+       struct wd_softc *wd = device_lookup_private(&wd_cd, WDUNIT(bp->b_dev));
+#ifdef ATADEBUG
        struct dk_softc *dksc = &wd->sc_dksc;
+#endif
 
-       ATADEBUG_PRINT(("wdrestart %s\n", dksc->sc_xname),
+       ATADEBUG_PRINT(("wdbiorestart %s\n", dksc->sc_xname),
            DEBUG_XFERS);
 
-       dk_start(dksc, NULL);
+       mutex_enter(&wd->sc_lock);
+       wdstart1(wd, bp, xfer);
+       mutex_exit(&wd->sc_lock);
 }
 
 static void
@@ -985,7 +1002,7 @@
         * If any partition is open, but the disk has been invalidated,
         * disallow further opens.
         */
-       if ((wd->sc_flags & WDF_LOADED) == 0) {
+       if ((wd->sc_flags & (WDF_OPEN | WDF_LOADED)) == WDF_OPEN) {
                if (part != RAW_PART || fmt != S_IFCHR)
                        return EIO;
        }
@@ -1023,6 +1040,7 @@
                wd->sc_flags |= WDF_LOADED;
        }
 
+       wd->sc_flags |= WDF_OPEN;
        return 0;
 
 bad:
@@ -1041,6 +1059,7 @@
        wd_flushcache(wd, AT_WAIT, false);
 
        wd->atabus->ata_delref(wd->drvp);
+       wd->sc_flags &= ~WDF_OPEN;
 
        return 0;
 }
diff -r 37a13bdcdc49 -r 7e1e28b6ddbb sys/dev/ata/wdvar.h
--- a/sys/dev/ata/wdvar.h       Fri Nov 03 09:59:07 2017 +0000
+++ b/sys/dev/ata/wdvar.h       Fri Nov 03 13:01:26 2017 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: wdvar.h,v 1.45 2017/11/01 19:34:46 mlelstv Exp $       */
+/*     $NetBSD: wdvar.h,v 1.46 2017/11/03 13:01:26 mlelstv Exp $       */
 
 /*
  * Copyright (c) 1998, 2001 Manuel Bouyer.
@@ -54,6 +54,7 @@
 #define WDF_LBA                0x040 /* using LBA mode */
 #define WDF_LBA48      0x100 /* using 48-bit LBA mode */
 #define WDF_FLUSH_PEND 0x200 /* cache flush waits for free xfer */
+#define WDF_OPEN       0x400 /* device is open */
        uint64_t sc_capacity; /* full capacity of the device */
        uint64_t sc_capacity512; /* ... in DEV_BSIZE blocks */
        uint32_t sc_capacity28; /* capacity accessible with LBA28 commands */



Home | Main Index | Thread Index | Old Index