summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/ic/sili.c118
1 files changed, 90 insertions, 28 deletions
diff --git a/sys/dev/ic/sili.c b/sys/dev/ic/sili.c
index 6a05090ac19..dd20133fc99 100644
--- a/sys/dev/ic/sili.c
+++ b/sys/dev/ic/sili.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: sili.c,v 1.15 2007/04/07 06:10:09 dlg Exp $ */
+/* $OpenBSD: sili.c,v 1.16 2007/04/07 09:24:12 pascoe Exp $ */
/*
* Copyright (c) 2007 David Gwynne <dlg@openbsd.org>
@@ -72,6 +72,8 @@ struct sili_port {
struct sili_ccb *sp_ccbs;
struct sili_dmamem *sp_cmds;
+
+ TAILQ_HEAD(, sili_ccb) sp_free_ccbs;
};
int sili_ports_alloc(struct sili_softc *);
@@ -102,10 +104,14 @@ struct sili_ccb {
bus_dmamap_t ccb_dmamap;
struct sili_port *ccb_port;
+
+ TAILQ_ENTRY(sili_ccb) ccb_entry;
};
int sili_ccb_alloc(struct sili_port *);
void sili_ccb_free(struct sili_port *);
+struct sili_ccb *sili_get_ccb(struct sili_port *);
+void sili_put_ccb(struct sili_ccb *);
/* bus space ops */
u_int32_t sili_read(struct sili_softc *, bus_size_t);
@@ -159,7 +165,7 @@ sili_attach(struct sili_softc *sc)
aaa.aaa_methods = &sili_atascsi_methods;
aaa.aaa_minphys = minphys;
aaa.aaa_nports = sc->sc_nports;
- aaa.aaa_ncmds = 1 /* XXX SILI_MAX_CMDS */;
+ aaa.aaa_ncmds = SILI_MAX_CMDS;
sc->sc_atascsi = atascsi_attach(&sc->sc_dev, &aaa);
@@ -236,31 +242,46 @@ sili_ccb_alloc(struct sili_port *sp)
{
struct sili_softc *sc = sp->sp_sc;
struct sili_ccb *ccb;
+ struct sili_prb *prb;
+ int i;
- /* XXX this should allocate multiple ccbs */
+ TAILQ_INIT(&sp->sp_free_ccbs);
- sp->sp_ccbs = malloc(sizeof(struct sili_ccb), M_DEVBUF, M_WAITOK);
- sp->sp_cmds = sili_dmamem_alloc(sc, SILI_CMD_LEN /* * SILI_MAX_CMDS */,
+ sp->sp_ccbs = malloc(sizeof(struct sili_ccb) * SILI_MAX_CMDS,
+ M_DEVBUF, M_WAITOK);
+ sp->sp_cmds = sili_dmamem_alloc(sc, SILI_CMD_LEN * SILI_MAX_CMDS,
SILI_PRB_ALIGN);
if (sp->sp_cmds == NULL)
goto free_ccbs;
- ccb = sp->sp_ccbs;
- bzero(ccb, sizeof(struct sili_ccb));
- ccb->ccb_port = sp;
- ccb->ccb_cmd = SILI_DMA_KVA(sp->sp_cmds);
- ccb->ccb_cmd_dva = SILI_DMA_DVA(sp->sp_cmds);
- if (bus_dmamap_create(sc->sc_dmat, MAXPHYS, SILI_DMA_SEGS,
- MAXPHYS, 0, BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW,
- &sp->sp_ccbs->ccb_dmamap) != 0)
- goto free_cmds;
+ bzero(sp->sp_ccbs, sizeof(struct sili_ccb) * SILI_MAX_CMDS);
+
+ for (i = 0; i < SILI_MAX_CMDS; i++) {
+ ccb = &sp->sp_ccbs[i];
+ ccb->ccb_port = sp;
+ ccb->ccb_cmd = SILI_DMA_KVA(sp->sp_cmds) + i * SILI_CMD_LEN;
+ ccb->ccb_cmd_dva = SILI_DMA_DVA(sp->sp_cmds) + i * SILI_CMD_LEN;
+ if (bus_dmamap_create(sc->sc_dmat, MAXPHYS, SILI_DMA_SEGS,
+ MAXPHYS, 0, BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW,
+ &ccb->ccb_dmamap) != 0)
+ goto free_cmds;
+
+ prb = ccb->ccb_cmd;
+ ccb->ccb_xa.fis = (struct ata_fis_h2d *)&prb->fis;
+ ccb->ccb_xa.packetcmd = ((struct sili_prb_packet *)prb)->cdb;
+ ccb->ccb_xa.tag = i;
+ ccb->ccb_xa.ata_put_xfer = sili_ata_put_xfer;
+ ccb->ccb_xa.state = ATA_S_COMPLETE;
+
+ sili_put_ccb(ccb);
+ }
return (0);
free_cmds:
sili_dmamem_free(sc, sp->sp_cmds);
free_ccbs:
- free(sp->sp_ccbs, M_DEVBUF);
+ sili_ccb_free(sp);
return (1);
}
@@ -270,13 +291,47 @@ sili_ccb_free(struct sili_port *sp)
struct sili_softc *sc = sp->sp_sc;
struct sili_ccb *ccb;
- ccb = sp->sp_ccbs;
- bus_dmamap_destroy(sc->sc_dmat, ccb->ccb_dmamap);
+ while ((ccb = sili_get_ccb(sp)) != NULL)
+ bus_dmamap_destroy(sc->sc_dmat, ccb->ccb_dmamap);
free(sp->sp_ccbs, M_DEVBUF);
sp->sp_ccbs = NULL;
}
+struct sili_ccb *
+sili_get_ccb(struct sili_port *sp)
+{
+ struct sili_ccb *ccb;
+
+ ccb = TAILQ_FIRST(&sp->sp_free_ccbs);
+ if (ccb != NULL) {
+ KASSERT(ccb->ccb_xa.state == ATA_S_PUT);
+ TAILQ_REMOVE(&sp->sp_free_ccbs, ccb, ccb_entry);
+ ccb->ccb_xa.state = ATA_S_SETUP;
+ }
+
+ return (ccb);
+}
+
+void
+sili_put_ccb(struct sili_ccb *ccb)
+{
+ struct sili_port *sp = ccb->ccb_port;
+
+#ifdef DIAGNOSTIC
+ if (ccb->ccb_xa.state != ATA_S_COMPLETE &&
+ ccb->ccb_xa.state != ATA_S_TIMEOUT &&
+ ccb->ccb_xa.state != ATA_S_ERROR) {
+ printf("%s: invalid ata_xfer state %02x in sili_put_ccb, "
+ "slot %d\n", DEVNAME(sp->sp_sc), ccb->ccb_xa.state,
+ ccb->ccb_xa.tag);
+ }
+#endif
+
+ ccb->ccb_xa.state = ATA_S_PUT;
+ TAILQ_INSERT_TAIL(&sp->sp_free_ccbs, ccb, ccb_entry);
+}
+
struct sili_dmamem *
sili_dmamem_alloc(struct sili_softc *sc, bus_size_t size, bus_size_t align)
{
@@ -512,6 +567,7 @@ sili_ata_cmd(struct ata_xfer *xa)
sgl = atapi->sgl;
sgllen = sizeofa(atapi->sgl);
+ xa->state = ATA_S_ERROR;
return (ATA_ERROR);
} else {
ata = ccb->ccb_cmd;
@@ -522,8 +578,10 @@ sili_ata_cmd(struct ata_xfer *xa)
sgllen = sizeofa(ata->sgl);
}
- if (sili_load(ccb, sgl, sgllen) != 0)
+ if (sili_load(ccb, sgl, sgllen) != 0) {
+ xa->state = ATA_S_ERROR;
return (ATA_ERROR);
+ }
bus_dmamap_sync(sc->sc_dmat, SILI_DMA_MAP(sp->sp_cmds),
0, SILI_CMD_LEN, BUS_DMASYNC_PREWRITE);
@@ -531,8 +589,10 @@ sili_ata_cmd(struct ata_xfer *xa)
sili_post_direct(sp, 0, &ccb->ccb_prb, sizeof(ccb->ccb_prb));
#endif
sili_post_indirect(sp, ccb);
- if (!sili_pwait_eq(sp, SILI_PREG_PSS, (1 << 0), 0, 1000)) {
+ if (!sili_pwait_eq(sp, SILI_PREG_PSS, (1 << ccb->ccb_xa.tag), 0,
+ ccb->ccb_xa.timeout)) {
printf("%s: cmd failed\n", DEVNAME(sp->sp_sc));
+ xa->state = ATA_S_ERROR;
return (ATA_ERROR);
}
@@ -635,21 +695,23 @@ sili_ata_get_xfer(void *xsc, int port)
{
struct sili_softc *sc = xsc;
struct sili_port *sp = &sc->sc_ports[port];
- struct sili_ccb *ccb = sp->sp_ccbs;
- struct sili_prb *prb = ccb->ccb_cmd;
- struct ata_xfer *xa;
+ struct sili_ccb *ccb;
- bzero(ccb->ccb_cmd, SILI_CMD_LEN);
+ ccb = sili_get_ccb(sp);
+ if (ccb == NULL) {
+ printf("sili_ata_get_xfer: NULL ccb\n");
+ return (NULL);
+ }
- xa = &ccb->ccb_xa;
- xa->fis = (struct ata_fis_h2d *)&prb->fis;
- xa->ata_put_xfer = sili_ata_put_xfer;
+ bzero(ccb->ccb_cmd, SILI_CMD_LEN);
- return (xa);
+ return ((struct ata_xfer *)ccb);
}
void
sili_ata_put_xfer(struct ata_xfer *xa)
{
- /* this does nothing (yet) */
+ struct sili_ccb *ccb = (struct sili_ccb *)xa;
+
+ sili_put_ccb(ccb);
}