summaryrefslogtreecommitdiff
path: root/sys/dev/pci/ahci.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pci/ahci.c')
-rw-r--r--sys/dev/pci/ahci.c100
1 files changed, 90 insertions, 10 deletions
diff --git a/sys/dev/pci/ahci.c b/sys/dev/pci/ahci.c
index c33da5a666c..f6afe32a513 100644
--- a/sys/dev/pci/ahci.c
+++ b/sys/dev/pci/ahci.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ahci.c,v 1.84 2007/03/20 07:38:45 pascoe Exp $ */
+/* $OpenBSD: ahci.c,v 1.85 2007/03/20 08:47:46 pascoe Exp $ */
/*
* Copyright (c) 2006 David Gwynne <dlg@openbsd.org>
@@ -36,7 +36,8 @@
#define AHCI_DEBUG
#ifdef AHCI_DEBUG
-#define DPRINTF(m, f...) do { if (ahcidebug & (m)) printf(f); } while (0)
+#define DPRINTF(m, f...) do { if ((ahcidebug & (m)) == (m)) printf(f); } while (0)
+#define AHCI_D_TIMEOUT 0x00
#define AHCI_D_VERBOSE 0x01
#define AHCI_D_INTR 0x02
#define AHCI_D_XFER 0x08
@@ -417,7 +418,7 @@ int ahci_port_portreset(struct ahci_port *);
int ahci_load_prdt(struct ahci_ccb *);
void ahci_unload_prdt(struct ahci_ccb *);
-int ahci_poll(struct ahci_ccb *, int);
+int ahci_poll(struct ahci_ccb *, int, void (*)(void *));
void ahci_start(struct ahci_ccb *);
int ahci_intr(void *);
@@ -465,6 +466,7 @@ struct atascsi_methods ahci_atascsi_methods = {
/* ccb completions */
void ahci_ata_cmd_done(struct ahci_ccb *);
+void ahci_ata_cmd_timeout(void *);
void ahci_empty_done(struct ahci_ccb *);
const struct ahci_device *
@@ -1106,7 +1108,8 @@ ahci_port_softreset(struct ahci_port *ap)
cmd_slot->flags |= htole16(AHCI_CMD_LIST_FLAG_R); /* Reset */
cmd_slot->flags |= htole16(AHCI_CMD_LIST_FLAG_W); /* Write */
- if (ahci_poll(ccb, 1000) != 0)
+ ccb->ccb_xa.state = ATA_S_PENDING;
+ if (ahci_poll(ccb, 1000, NULL) != 0)
goto err;
/* Prep second D2H command to read status and complete reset sequence */
@@ -1120,7 +1123,8 @@ ahci_port_softreset(struct ahci_port *ap)
cmd_slot->flags = htole16(5); /* FIS length: 5 DWORDS */
cmd_slot->flags |= htole16(AHCI_CMD_LIST_FLAG_W);
- if (ahci_poll(ccb, 1000) != 0)
+ ccb->ccb_xa.state = ATA_S_PENDING;
+ if (ahci_poll(ccb, 1000, NULL) != 0)
goto err;
if (ahci_pwait_clr(ap, AHCI_PREG_TFD, AHCI_PREG_TFD_STS_BSY |
@@ -1275,7 +1279,7 @@ ahci_unload_prdt(struct ahci_ccb *ccb)
int
-ahci_poll(struct ahci_ccb *ccb, int timeout)
+ahci_poll(struct ahci_ccb *ccb, int timeout, void (*timeout_fn)(void *))
{
struct ahci_port *ap = ccb->ccb_port;
int s;
@@ -1290,9 +1294,12 @@ ahci_poll(struct ahci_ccb *ccb, int timeout)
delay(1000);
} while (--timeout > 0);
- splx(s);
- /* XXX cleanup! */
+ /* Run timeout while at splbio, otherwise ahci_intr could interfere. */
+ if (timeout_fn != NULL)
+ timeout_fn(ccb);
+
+ splx(s);
return (1);
}
@@ -1318,6 +1325,7 @@ ahci_start(struct ahci_ccb *ccb)
bus_dmamap_sync(sc->sc_dmat, AHCI_DMA_MAP(ap->ap_dmamem_rfis), 0,
sizeof(struct ahci_rfis), BUS_DMASYNC_PREREAD);
+ ccb->ccb_xa.state = ATA_S_ONCHIP;
ap->ap_active |= 1 << ccb->ccb_slot;
ahci_pwrite(ap, AHCI_PREG_CI, 1 << ccb->ccb_slot);
}
@@ -1672,14 +1680,18 @@ ahci_ata_cmd(struct ata_xfer *xa)
if (ahci_load_prdt(ccb) != 0)
goto failcmd;
+ timeout_set(&xa->stimeout, ahci_ata_cmd_timeout, ccb);
+
xa->state = ATA_S_PENDING;
if (xa->flags & ATA_F_POLL) {
- if (ahci_poll(ccb, 1000) != 0)
+ if (ahci_poll(ccb, 1000, ahci_ata_cmd_timeout) != 0)
return (ATA_ERROR);
return (ATA_COMPLETE);
}
+ timeout_add(&xa->stimeout, (xa->timeout * hz) / 1000);
+
s = splbio();
ahci_start(ccb);
splx(s);
@@ -1698,10 +1710,78 @@ ahci_ata_cmd_done(struct ahci_ccb *ccb)
{
struct ata_xfer *xa = &ccb->ccb_xa;
+ timeout_del(&xa->stimeout);
+
ahci_unload_prdt(ccb);
- xa->state = ATA_S_COMPLETE;
+ if (xa->state == ATA_S_ONCHIP)
+ xa->state = ATA_S_COMPLETE;
+ if (xa->state != ATA_S_TIMEOUT)
+ xa->complete(xa);
+}
+
+void
+ahci_ata_cmd_timeout(void *arg)
+{
+ struct ahci_ccb *ccb = arg;
+ struct ata_xfer *xa = &ccb->ccb_xa;
+ struct ahci_port *ap = ccb->ccb_port;
+ int s;
+
+ s = splbio();
+
+ if (ccb->ccb_xa.state == ATA_S_PENDING) {
+ DPRINTF(AHCI_D_TIMEOUT, "%s: command for slot %d timed out "
+ "before it got on chip\n", PORTNAME(ap), ccb->ccb_slot);
+ } else if (ccb->ccb_xa.state == ATA_S_ONCHIP && ahci_port_intr(ap,
+ 1 << ccb->ccb_slot)) {
+ DPRINTF(AHCI_D_TIMEOUT, "%s: final poll of port completed "
+ "command in slot %d\n", PORTNAME(ap), ccb->ccb_slot);
+ goto ret;
+ } else if (ccb->ccb_xa.state != ATA_S_ONCHIP) {
+ DPRINTF(AHCI_D_TIMEOUT, "%s: command slot %d already "
+ "handled%s\n", PORTNAME(ap), ccb->ccb_slot,
+ ISSET(ap->ap_active, 1 << ccb->ccb_slot) ?
+ " but slot is still active?" : ".");
+ goto ret;
+ } else if (!ISSET(ahci_pread(ap, AHCI_PREG_CI), 1 << ccb->ccb_slot) &&
+ ISSET(ap->ap_active, 1 << ccb->ccb_slot)) {
+ DPRINTF(AHCI_D_TIMEOUT, "%s: command slot %d completed but "
+ "IRQ handler didn't detect it. Why?\n", PORTNAME(ap),
+ ccb->ccb_slot);
+ ap->ap_active &= ~(1 << ccb->ccb_slot);
+ ccb->ccb_done(ccb);
+ goto ret;
+ }
+
+ /* Complete the slot with timeout flag set. */
+ ccb->ccb_xa.state = ATA_S_TIMEOUT;
+ ap->ap_active &= ~(1 << ccb->ccb_slot);
+ DPRINTF(AHCI_D_TIMEOUT, "%s: run completion (1)\n", PORTNAME(ap));
+ ccb->ccb_done(ccb);
+
+ /* Reset port. */
+ DPRINTF(AHCI_D_TIMEOUT, "%s: resetting port to abort command in slot %d, "
+ "active %08x\n", PORTNAME(ap), ccb->ccb_slot, ap->ap_active);
+ if (ahci_port_softreset(ap) != 0 && ahci_port_portreset(ap) != 0) {
+ DPRINTF(AHCI_D_TIMEOUT, "%s: failed to reset port\n",
+ PORTNAME(ap));
+ }
+
+ /* Restart any outstanding commands that were stopped by the reset. */
+ if (ap->ap_active) {
+ DPRINTF(AHCI_D_TIMEOUT, "%s: re-enabling slots %08x\n",
+ PORTNAME(ap), ap->ap_active);
+ ahci_pwrite(ap, AHCI_PREG_CI, ap->ap_active);
+ }
+
+ /* Complete the timed out ata_xfer I/O (may generate new I/O). */
+ DPRINTF(AHCI_D_TIMEOUT, "%s: run completion (2)\n", PORTNAME(ap));
xa->complete(xa);
+
+ DPRINTF(AHCI_D_TIMEOUT, "%s: splx\n", PORTNAME(ap));
+ret:
+ splx(s);
}
void