Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/jdolecek-ncq]: src/sys/dev/ic convert over to new error handling world o...
details: https://anonhg.NetBSD.org/src/rev/a2c9f1eb7c96
branches: jdolecek-ncq
changeset: 352710:a2c9f1eb7c96
user: jdolecek <jdolecek%NetBSD.org@localhost>
date: Wed Jul 19 20:02:40 2017 +0000
description:
convert over to new error handling world order:
- switch to ata_timeout()
- stop using ch_status/ch_error for passing state/error, stop setting
ATACH_IRQ_WAIT in ch_flags; pass the state via the last parameter
to c_intr() routine
- add NCQ recovery and KILL_REQUEUE
- only call atastart() in c_intr() if there was no error
several siisata specific tweaks:
- improve reset to better handle PM - need to reset couple more registers
- add timeouts for unbusy wait - ATA_DELAY (10s)
- siisata_bio_complete() do not use PRSO_RTC for byte count for NCQ transfers,
they never return partial reads; and that register might contain some random
junk, at least that's the case with ahcisata cmdh_prdbc
diffstat:
sys/dev/ic/siisata.c | 341 +++++++++++++++++++++++++++++++-------------------
1 files changed, 212 insertions(+), 129 deletions(-)
diffs (truncated from 667 to 300 lines):
diff -r 5ee9ed54a902 -r a2c9f1eb7c96 sys/dev/ic/siisata.c
--- a/sys/dev/ic/siisata.c Wed Jul 19 19:46:52 2017 +0000
+++ b/sys/dev/ic/siisata.c Wed Jul 19 20:02:40 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: siisata.c,v 1.30.4.25 2017/06/27 20:13:56 jdolecek Exp $ */
+/* $NetBSD: siisata.c,v 1.30.4.26 2017/07/19 20:02:40 jdolecek Exp $ */
/* from ahcisata_core.c */
@@ -79,7 +79,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: siisata.c,v 1.30.4.25 2017/06/27 20:13:56 jdolecek Exp $");
+__KERNEL_RCSID(0, "$NetBSD: siisata.c,v 1.30.4.26 2017/07/19 20:02:40 jdolecek Exp $");
#include <sys/types.h>
#include <sys/param.h>
@@ -148,7 +148,7 @@
void siisata_cmd_start(struct ata_channel *, struct ata_xfer *);
int siisata_cmd_complete(struct ata_channel *, struct ata_xfer *, int);
-void siisata_cmd_done(struct ata_channel *, struct ata_xfer *);
+void siisata_cmd_done(struct ata_channel *, struct ata_xfer *, int);
static void siisata_cmd_done_end(struct ata_channel *, struct ata_xfer *);
void siisata_cmd_kill_xfer(struct ata_channel *, struct ata_xfer *, int);
@@ -157,14 +157,12 @@
void siisata_bio_kill_xfer(struct ata_channel *, struct ata_xfer *, int);
int siisata_exec_command(struct ata_drive_datas *, struct ata_xfer *);
-void siisata_timeout(void *);
-
static void siisata_reinit_port(struct ata_channel *);
static void siisata_device_reset(struct ata_channel *);
static void siisata_activate_prb(struct siisata_channel *, int);
static void siisata_deactivate_prb(struct siisata_channel *, int);
-static int siisata_dma_setup(struct ata_channel *chp, int, void *,
- size_t, int);
+static int siisata_dma_setup(struct ata_channel *, int, void *, size_t, int);
+static void siisata_channel_recover(struct ata_channel *, int, int);
#if NATAPIBUS > 0
void siisata_atapibus_attach(struct atabus_softc *);
@@ -478,7 +476,6 @@
struct ata_xfer *xfer;
u_int slot;
uint32_t pss, pis;
- uint32_t prbfis;
sc = (struct siisata_softc *)schp->ata_channel.ch_atac;
chp = &schp->ata_channel;
@@ -489,12 +486,15 @@
SIISATANAME(sc), __func__, chp->ch_channel, pss), DEBUG_INTR);
for (slot = 0; slot < SIISATA_MAX_SLOTS; slot++) {
- if (((schp->sch_active_slots >> slot) & 1) == 0)
+ if (((schp->sch_active_slots >> slot) & 1) == 0) {
/* there's nothing executing here, skip */
continue;
- if (((pss >> slot) & 1) != 0)
- /* execution is incomplete or unsuccessful, skip for now */
+ }
+ if (((pss >> slot) & 1) != 0) {
+ /* execution is incomplete or unsuccessful, skip
+ * for now */
continue;
+ }
xfer = ata_queue_hwslot_to_xfer(chp, slot);
if (xfer->c_intr == NULL) {
wakeup(schp);
@@ -520,54 +520,145 @@
SIISATA_DEBUG_PRINT(("%s: %s port %d, pis 0x%x ", SIISATANAME(sc),
__func__, chp->ch_channel, pis), DEBUG_INTR);
+ /* clear */
+ PRWRITE(sc, PRX(chp->ch_channel, PRO_PIS), pis);
+
if (pis & PR_PIS_CMDERRR) {
uint32_t ec;
- uint32_t ps;
- ps = PRREAD(sc, PRX(chp->ch_channel, PRO_PS));
ec = PRREAD(sc, PRX(chp->ch_channel, PRO_PCE));
SIISATA_DEBUG_PRINT(("ec %d\n", ec), DEBUG_INTR);
- slot = PR_PS_ACTIVE_SLOT(ps); /* XXX invalid for NCQ? */
-
/* emulate a CRC error by default */
- chp->ch_status = WDCS_ERR;
- chp->ch_error = WDCE_CRC;
+ int tfd = ATACH_ERR_ST(WDCE_CRC, WDCS_ERR);
- if (ec <= PR_PCE_DATAFISERROR) {
- if (ec == PR_PCE_DEVICEERROR) {
- /* read in specific information about error */
- prbfis = bus_space_read_stream_4(
- sc->sc_prt, sc->sc_prh,
- PRSX(chp->ch_channel, slot,
- PRSO_FIS));
- /* set ch_status and ch_error */
- satafis_rdh_parse(chp, (uint8_t *)&prbfis);
- }
- siisata_reinit_port(chp);
+ if (ec <= PR_PCE_DATAFISERROR && !schp->sch_recovering) {
+ siisata_channel_recover(chp, ec, tfd);
} else {
aprint_error_dev(sc->sc_atac.atac_dev, "fatal error %d"
" on channel %d (ctx 0x%x), resetting\n",
ec, chp->ch_channel,
PRREAD(sc, PRX(chp->ch_channel, PRO_PCR)));
+
/* okay, we have a "Fatal Error" */
siisata_device_reset(chp);
- }
- for (slot = 0; slot < SIISATA_MAX_SLOTS; slot++) {
- /* there's nothing executing here, skip */
- if (((schp->sch_active_slots >> slot) & 1) == 0)
- continue;
- xfer = ata_queue_hwslot_to_xfer(chp, slot);
- if (xfer == NULL)
- continue;
- xfer->c_intr(chp, xfer, 0);
+
+ /* Complete any non-finished commands with the error */
+ for (slot = 0; slot < SIISATA_MAX_SLOTS; slot++) {
+ if (((schp->sch_active_slots >> slot) & 1) == 0) {
+ /* nothing executing here, skip */
+ continue;
+ }
+ xfer = ata_queue_hwslot_to_xfer(chp, slot);
+ xfer->c_intr(chp, xfer, tfd);
+ }
}
}
- /* clear */
- PRWRITE(sc, PRX(chp->ch_channel, PRO_PIS), pis);
+ return;
+}
+
+static void
+siisata_hold(struct siisata_channel *schp)
+{
+ schp->sch_hold_slots |= schp->sch_active_slots;
+ schp->sch_active_slots = 0;
+}
+
+static void
+siisata_unhold(struct siisata_channel *schp)
+{
+ schp->sch_active_slots = schp->sch_hold_slots;
+ schp->sch_hold_slots = 0;
+}
+
+/* Recover channel after transfer aborted */
+static void
+siisata_channel_recover(struct ata_channel *chp, int ec, int tfd)
+{
+ struct siisata_channel *schp = (struct siisata_channel *)chp;
+ struct siisata_softc *sc =
+ (struct siisata_softc *)schp->ata_channel.ch_atac;
+ struct ata_drive_datas *drvp;
+ int ps, drive, error;
+ uint8_t slot, st, err;
+ struct ata_xfer *xfer;
+ uint32_t prbfis;
+
+ KASSERT(!schp->sch_recovering);
+
+ schp->sch_recovering = true;
+
+ ps = PRREAD(sc, PRX(chp->ch_channel, PRO_PS));
+
+ if (chp->ch_ndrives > PMP_PORT_CTL) {
+ /* Get PM port number for the device in error */
+ int pcr = PRREAD(sc, PRX(chp->ch_channel, PRO_PCR));
+ drive = PRO_PCR_PMP(pcr);
+ } else
+ drive = 0;
+
+ drvp = &chp->ch_drive[drive];
+
+ siisata_hold(schp);
+
+ siisata_reinit_port(chp);
- return;
+ /*
+ * When running NCQ commands, READ LOG EXT is necessary to clear the
+ * error condition and unblock the device.
+ */
+ error = ata_read_log_ext_ncq(drvp, AT_POLL, &slot, &st, &err);
+
+ siisata_unhold(schp);
+
+ if (error == EOPNOTSUPP) {
+ /*
+ * Not NCQ command or not NCQ device, there is only
+ * one outstanding tranfer, so find the active one,
+ * and send the error.
+ */
+ xfer = ata_queue_drive_active_xfer(chp, drive);
+ if (ec == PR_PCE_DEVICEERROR) {
+ slot = PR_PS_ACTIVE_SLOT(ps);
+
+ /* read in specific information about error */
+ prbfis = bus_space_read_stream_4(
+ sc->sc_prt, sc->sc_prh,
+ PRSX(chp->ch_channel, slot,
+ PRSO_FIS));
+
+ /* get status and error */
+ int ntfd = satafis_rdh_parse(chp, (uint8_t *)&prbfis);
+
+ if (ATACH_ST(ntfd) & WDCS_ERR)
+ tfd = ntfd;
+ }
+
+ xfer->c_intr(chp, xfer, tfd);
+ } else if (error == 0) {
+ xfer = ata_queue_hwslot_to_xfer(chp, slot);
+ xfer->c_intr(chp, xfer, ATACH_ERR_ST(err, st));
+ } else {
+ /* Other error, do full device reset */
+ drive = -1;
+ siisata_device_reset(chp);
+ }
+
+ /* Requeue the non-errorred commands */
+ for (slot = 0; slot < SIISATA_MAX_SLOTS; slot++) {
+ if (((schp->sch_active_slots >> slot) & 1) == 0)
+ continue;
+
+ xfer = ata_queue_hwslot_to_xfer(chp, slot);
+ if (drive >= 0 && xfer->c_drive != drive)
+ continue;
+
+ xfer->c_kill_xfer(chp, xfer,
+ drive >= 0 ? KILL_REQUEUE : KILL_RESET);
+ }
+
+ schp->sch_recovering = false;
}
void
@@ -629,10 +720,12 @@
pis = PRREAD(sc, PRX(chp->ch_channel, PRO_PIS));
const uint32_t ps = PRREAD(sc, PRX(chp->ch_channel, PRO_PS));
const u_int slot = PR_PS_ACTIVE_SLOT(ps);
+
if (slot != xfer->c_slot)
device_printf(sc->sc_atac.atac_dev, "%s port %d "
"drive %d slot %d c_slot %d", __func__,
chp->ch_channel, drvp->drive, slot, xfer->c_slot);
+
PRWRITE(sc, PRX(chp->ch_channel, PRO_PIS), pis &
PR_PIS_CMDERRR);
}
@@ -657,29 +750,6 @@
ata_free_xfer(chp, xfer);
-#if 1
- /* attempt to downgrade signaling in event of CRC error */
- /* XXX should be part of the MI (S)ATA subsystem */
- if (chp->ch_status == 0x51 && chp->ch_error == 0x84) {
- bus_space_write_4(sc->sc_prt, schp->sch_scontrol, 0,
- SControl_IPM_NONE | SControl_SPD_G1 | SControl_DET_INIT);
- DELAY(10);
- bus_space_write_4(sc->sc_prt, schp->sch_scontrol, 0,
- SControl_IPM_NONE | SControl_SPD_G1);
- DELAY(10);
- for (;;) {
- if ((bus_space_read_4(sc->sc_prt, schp->sch_sstatus, 0)
- & SStatus_DET_mask) == SStatus_DET_DEV)
- break;
- DELAY(10);
- }
- }
-#endif
-
-#if 1
- chp->ch_status = 0;
- chp->ch_error = 0;
-#endif
return;
}
@@ -902,9 +972,6 @@
chp->ch_channel, xfer->c_drive, ata_c->r_command, xfer->c_slot),
DEBUG_FUNCS|DEBUG_XFERS);
- chp->ch_status = 0;
- chp->ch_error = 0;
-
prb = schp->sch_prb[xfer->c_slot];
memset(prb, 0, SIISATA_CMD_SIZE);
@@ -936,9 +1003,8 @@
Home |
Main Index |
Thread Index |
Old Index