Source-Changes-HG archive

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

[src/trunk]: src/sys/dev/ic if softreset during PMP detection fails, disable ...



details:   https://anonhg.NetBSD.org/src/rev/6849c5e3eb96
branches:  trunk
changeset: 446001:6849c5e3eb96
user:      jdolecek <jdolecek%NetBSD.org@localhost>
date:      Mon Nov 19 21:52:24 2018 +0000

description:
if softreset during PMP detection fails, disable PMP, reset port and continue
with sig detected after the initial COMRESET

this does not yet fix the infamous 'clearing WDCTL_RST failed', but at least
now the disk is detected and usable when it happens

tested on AMD system with ASUS Prime A320M-K, similar to one from PR kern/53524

diffstat:

 sys/dev/ic/ahcisata_core.c |  53 ++++++++++++++++++++++++---------------------
 1 files changed, 28 insertions(+), 25 deletions(-)

diffs (134 lines):

diff -r aad0f6e77cd2 -r 6849c5e3eb96 sys/dev/ic/ahcisata_core.c
--- a/sys/dev/ic/ahcisata_core.c        Mon Nov 19 21:45:37 2018 +0000
+++ b/sys/dev/ic/ahcisata_core.c        Mon Nov 19 21:52:24 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: ahcisata_core.c,v 1.67 2018/11/19 19:52:08 jdolecek Exp $      */
+/*     $NetBSD: ahcisata_core.c,v 1.68 2018/11/19 21:52:24 jdolecek Exp $      */
 
 /*
  * Copyright (c) 2006 Manuel Bouyer.
@@ -26,7 +26,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: ahcisata_core.c,v 1.67 2018/11/19 19:52:08 jdolecek Exp $");
+__KERNEL_RCSID(0, "$NetBSD: ahcisata_core.c,v 1.68 2018/11/19 21:52:24 jdolecek Exp $");
 
 #include <sys/types.h>
 #include <sys/malloc.h>
@@ -791,13 +791,11 @@
        struct ahci_softc *sc = (struct ahci_softc *)chp->ch_atac;
        struct ahci_cmd_tbl *cmd_tbl;
        struct ahci_cmd_header *cmd_h;
-       int i;
+       int i, error = 0;
        uint32_t sig;
 
-       KASSERT((AHCI_READ(sc, AHCI_P_CMD(chp->ch_channel)) & AHCI_P_CMD_CR) == 0);
        ata_channel_lock_owned(chp);
 
-again:
        /* clear port interrupt register */
        AHCI_WRITE(sc, AHCI_P_IS(chp->ch_channel), 0xffffffff);
        /* clear SErrors and start operations */
@@ -809,6 +807,9 @@
                 */
                ahci_channel_start(sc, chp, flags, 1);
        } else {
+               /* Can't handle command still running without CLO */
+               KASSERT((AHCI_READ(sc, AHCI_P_CMD(chp->ch_channel)) & AHCI_P_CMD_CR) == 0);
+
                ahci_channel_start(sc, chp, flags, 0);
        }
        if (drive > 0) {
@@ -834,12 +835,12 @@
        case TIMEOUT:
                aprint_error("%s port %d: setting WDCTL_RST failed "
                    "for drive %d\n", AHCINAME(sc), chp->ch_channel, drive);
-               if (sigp)
-                       *sigp = 0xffffffff;
+               error = EBUSY;
                goto end;
        default:
                break;
        }
+
        cmd_h->cmdh_flags = htole16(RHD_FISLEN / 4 |
            (drive << AHCI_CMDH_F_PMP_SHIFT));
        cmd_h->cmdh_prdbc = 0;
@@ -850,21 +851,9 @@
        switch (ahci_exec_fis(chp, 310, flags, c_slot)) {
        case ERR_DF:
        case TIMEOUT:
-               if ((sc->sc_ahci_quirks & AHCI_QUIRK_BADPMPRESET) != 0 &&
-                   drive == PMP_PORT_CTL) {
-                       /*
-                        * some controllers fails to reset when
-                        * targeting a PMP but a single drive is attached.
-                        * try again with port 0
-                        */
-                       drive = 0;
-                       ahci_channel_stop(sc, chp, flags);
-                       goto again;
-               }
                aprint_error("%s port %d: clearing WDCTL_RST failed "
                    "for drive %d\n", AHCINAME(sc), chp->ch_channel, drive);
-               if (sigp)
-                       *sigp = 0xffffffff;
+               error = EBUSY;
                goto end;
        default:
                break;
@@ -885,8 +874,6 @@
        if (i == AHCI_RST_WAIT) {
                aprint_error("%s: BSY never cleared, TD 0x%x\n",
                    AHCINAME(sc), sig);
-               if (sigp)
-                       *sigp = 0xffffffff;
                goto end;
        }
        AHCIDEBUG_PRINT(("%s: BSY took %d ms\n", AHCINAME(sc), i * 10),
@@ -904,7 +891,7 @@
        AHCI_WRITE(sc, AHCI_P_IS(chp->ch_channel), 0xffffffff);
        ahci_channel_start(sc, chp, flags,
            (sc->sc_ahci_cap & AHCI_CAP_CLO) ? 1 : 0);
-       return 0;
+       return error;
 }
 
 static void
@@ -972,6 +959,7 @@
        struct ahci_channel *achp = (struct ahci_channel *)chp;
        uint32_t sig;
        uint8_t c_slot;
+       int error;
 
        ata_channel_lock(chp);
 
@@ -993,9 +981,24 @@
            achp->ahcic_sstatus, AT_WAIT)) {
        case SStatus_DET_DEV:
                ata_delay(chp, 500, "ahcidv", AT_WAIT);
+
+               /* Initial value, used in case the soft reset fails */
+               sig = AHCI_READ(sc, AHCI_P_SIG(chp->ch_channel));
+
                if (sc->sc_ahci_cap & AHCI_CAP_SPM) {
-                       ahci_do_reset_drive(chp, PMP_PORT_CTL, AT_WAIT, &sig,
-                           c_slot);
+                       error = ahci_do_reset_drive(chp, PMP_PORT_CTL, AT_WAIT,
+                           &sig, c_slot);
+
+                       /* If probe for PMP failed, just fallback to drive 0 */
+                       if (error) {
+                               aprint_error("%s port %d: drive %d reset "
+                                   "failing, disabling PMP\n",
+                                   AHCINAME(sc), chp->ch_channel,
+                               PMP_PORT_CTL);
+
+                               sc->sc_ahci_cap &= ~AHCI_CAP_SPM;
+                               ahci_reset_channel(chp, AT_WAIT);
+                       }
                } else {
                        ahci_do_reset_drive(chp, 0, AT_WAIT, &sig, c_slot);
                }



Home | Main Index | Thread Index | Old Index