Module Name: src Committed By: jdolecek Date: Mon Nov 19 21:52:24 UTC 2018
Modified Files: src/sys/dev/ic: ahcisata_core.c Log Message: 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 To generate a diff of this commit: cvs rdiff -u -r1.67 -r1.68 src/sys/dev/ic/ahcisata_core.c Please note that diffs are not public domain; they are subject to the copyright notices on the relevant files.
Modified files: Index: src/sys/dev/ic/ahcisata_core.c diff -u src/sys/dev/ic/ahcisata_core.c:1.67 src/sys/dev/ic/ahcisata_core.c:1.68 --- src/sys/dev/ic/ahcisata_core.c:1.67 Mon Nov 19 19:52:08 2018 +++ src/sys/dev/ic/ahcisata_core.c Mon Nov 19 21:52:24 2018 @@ -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 @@ ahci_do_reset_drive(struct ata_channel * 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 @@ again: */ 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 @@ again: 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 @@ again: 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 @@ skip_reset: 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 @@ end: 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 @@ ahci_probe_drive(struct ata_channel *chp 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 @@ ahci_probe_drive(struct ata_channel *chp 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); }