summaryrefslogtreecommitdiff
path: root/sys/dev/ic/mfi.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic/mfi.c')
-rw-r--r--sys/dev/ic/mfi.c177
1 files changed, 94 insertions, 83 deletions
diff --git a/sys/dev/ic/mfi.c b/sys/dev/ic/mfi.c
index 61fe8b35534..0730b058bf2 100644
--- a/sys/dev/ic/mfi.c
+++ b/sys/dev/ic/mfi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mfi.c,v 1.35 2006/05/16 23:11:48 marco Exp $ */
+/* $OpenBSD: mfi.c,v 1.36 2006/05/17 16:00:52 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <marco@peereboom.us>
*
@@ -73,21 +73,21 @@ struct mfi_ccb *mfi_get_ccb(struct mfi_softc *);
void mfi_put_ccb(struct mfi_ccb *);
int mfi_init_ccb(struct mfi_softc *);
-u_int32_t mfi_read(struct mfi_softc *, bus_size_t);
-void mfi_write(struct mfi_softc *, bus_size_t, u_int32_t);
struct mfi_mem *mfi_allocmem(struct mfi_softc *, size_t);
void mfi_freemem(struct mfi_softc *, struct mfi_mem *);
+
int mfi_transition_firmware(struct mfi_softc *);
int mfi_initialize_firmware(struct mfi_softc *);
+u_int32_t mfi_read(struct mfi_softc *, bus_size_t);
+void mfi_write(struct mfi_softc *, bus_size_t, u_int32_t);
+int mfi_poll(struct mfi_ccb *);
+int mfi_despatch_cmd(struct mfi_ccb *);
-int mfi_despatch_cmd(struct mfi_softc *, struct mfi_ccb *);
-int mfi_poll(struct mfi_softc *, struct mfi_ccb *);
-int mfi_start_xs(struct mfi_softc *, struct mfi_ccb *,
- struct scsi_xfer *);
+int mfi_create_sgl(struct mfi_ccb *, int);
/* LD commands */
-int mfi_generic_scsi(struct scsi_xfer *);
-void mfi_done_generic_scsi(struct mfi_softc *, struct mfi_ccb *);
+int mfi_scsi_ld(struct mfi_ccb *, struct scsi_xfer *);
+void mfi_scsi_ld_done(struct mfi_ccb *);
#if NBIO > 0
int mfi_ioctl(struct device *, u_long, caddr_t);
@@ -134,6 +134,8 @@ mfi_put_ccb(struct mfi_ccb *ccb)
ccb->ccb_frame_size = 0;
ccb->ccb_extra_frames = 0;
ccb->ccb_sgl = NULL;
+ ccb->ccb_data = NULL;
+ ccb->ccb_len = 0;
TAILQ_INSERT_TAIL(&sc->sc_ccb_freeq, ccb, ccb_link);
splx(s);
}
@@ -381,7 +383,7 @@ mfi_initialize_firmware(struct mfi_softc *sc)
qinfo->miq_rq_entries, qinfo->miq_rq_addr_lo,
qinfo->miq_pi_addr_lo, qinfo->miq_ci_addr_lo);
- if (mfi_poll(sc, ccb)) {
+ if (mfi_poll(ccb)) {
printf("%s: mfi_initialize_firmware failed\n", DEVNAME(sc));
return (1);
}
@@ -514,29 +516,30 @@ nopcq:
}
int
-mfi_despatch_cmd(struct mfi_softc *sc, struct mfi_ccb *ccb)
+mfi_despatch_cmd(struct mfi_ccb *ccb)
{
- DNPRINTF(MFI_D_CMD, "%s: mfi_despatch_cmd\n", DEVNAME(sc));
+ DNPRINTF(MFI_D_CMD, "%s: mfi_despatch_cmd\n",
+ DEVNAME(ccb->ccb_sc));
- mfi_write(sc, MFI_IQP, htole32((ccb->ccb_pframe >> 3) |
+ mfi_write(ccb->ccb_sc, MFI_IQP, htole32((ccb->ccb_pframe >> 3) |
ccb->ccb_extra_frames));
return(0);
}
int
-mfi_poll(struct mfi_softc *sc, struct mfi_ccb *ccb)
+mfi_poll(struct mfi_ccb *ccb)
{
struct mfi_frame_header *hdr;
int to = 0;
- DNPRINTF(MFI_D_CMD, "%s: mfi_poll\n", DEVNAME(sc));
+ DNPRINTF(MFI_D_CMD, "%s: mfi_poll\n", DEVNAME(ccb->ccb_sc));
hdr = &ccb->ccb_frame->mfr_header;
hdr->mfh_cmd_status = 0xff;
hdr->mfh_flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
- mfi_despatch_cmd(sc, ccb);
+ mfi_despatch_cmd(ccb);
while (hdr->mfh_cmd_status == 0xff) {
delay(1000);
@@ -544,7 +547,7 @@ mfi_poll(struct mfi_softc *sc, struct mfi_ccb *ccb)
break;
}
if (hdr->mfh_cmd_status == 0xff) {
- printf("%s: timeout on ccb %d\n", DEVNAME(sc),
+ printf("%s: timeout on ccb %d\n", DEVNAME(ccb->ccb_sc),
hdr->mfh_context);
ccb->ccb_flags |= MFI_CCB_F_ERR;
return (1);
@@ -596,30 +599,25 @@ mfi_intr(void *arg)
}
void
-mfi_done_generic_scsi(struct mfi_softc *sc, struct mfi_ccb *ccb)
+mfi_scsi_ld_done(struct mfi_ccb *ccb)
{
#ifdef MFI_DEBUG
struct scsi_xfer *xs = ccb->ccb_xs;
struct scsi_link *link = xs->sc_link;
DNPRINTF(MFI_D_CMD, "%s: mfi_ld_done_inquiry: %.0x\n",
- DEVNAME(sc), link->target);
+ DEVNAME(ccb->ccb_sc), link->target);
#endif /* MFI_DEBUG */
}
int
-mfi_generic_scsi(struct scsi_xfer *xs)
+mfi_scsi_ld(struct mfi_ccb *ccb, struct scsi_xfer *xs)
{
struct scsi_link *link = xs->sc_link;
- struct mfi_softc *sc = link->adapter_softc;
- struct mfi_ccb *ccb;
struct mfi_pass_frame *pf;
- DNPRINTF(MFI_D_CMD, "%s: mfi_generic_scsi: %d\n",
- DEVNAME(sc), link->target);
-
- if ((ccb = mfi_get_ccb(sc)) == NULL)
- return (TRY_AGAIN_LATER);
+ DNPRINTF(MFI_D_CMD, "%s: mfi_scsi_ld: %d\n",
+ DEVNAME((struct mfi_softc *)link->adapter_softc), link->target);
pf = &ccb->ccb_frame->mfr_pass;
pf->mpf_header.mfh_cmd = MFI_CMD_LD_SCSI_IO;
@@ -636,17 +634,27 @@ mfi_generic_scsi(struct scsi_xfer *xs)
memset(pf->mpf_cdb, 0, 16);
memcpy(pf->mpf_cdb, &xs->cmdstore, xs->cmdlen);
- ccb->ccb_done = mfi_done_generic_scsi;
- ccb->ccb_xs = xs; /* XXX here or in mfi_start_xs? */
+ ccb->ccb_done = mfi_scsi_ld_done;
+ ccb->ccb_xs = xs;
ccb->ccb_frame_size = MFI_PASS_FRAME_SIZE;
ccb->ccb_sgl = &pf->mpf_sgl;
+
if (xs->flags & (SCSI_DATA_IN | SCSI_DATA_OUT))
ccb->ccb_direction = xs->flags & SCSI_DATA_IN ?
MFI_DATA_IN : MFI_DATA_OUT;
else
ccb->ccb_direction = 0;
- return (mfi_start_xs(sc, ccb, xs));
+ if (xs->data) {
+ ccb->ccb_data = xs->data;
+ ccb->ccb_len = xs->datalen;
+
+ if (mfi_create_sgl(ccb, xs->flags & SCSI_NOSLEEP) ?
+ BUS_DMA_NOWAIT : BUS_DMA_WAITOK)
+ return (1);
+ }
+
+ return (0);
}
int
@@ -666,22 +674,19 @@ mfi_scsi_cmd(struct scsi_xfer *xs)
if (!cold) {
DNPRINTF(MFI_D_CMD, "%s: no interrupt io yet %02x\n",
DEVNAME(sc), xs->cmd->opcode);
- xs->error = XS_DRIVER_STUFFUP;
- xs->flags |= ITSDONE;
- scsi_done(xs);
- return (COMPLETE);
+ goto stuffup;
}
if (target >= MFI_MAX_LD || !sc->sc_ld[target].ld_present ||
link->lun != 0) {
DNPRINTF(MFI_D_CMD, "%s: invalid target %d\n",
DEVNAME(sc), target);
- xs->error = XS_DRIVER_STUFFUP;
- xs->flags |= ITSDONE;
- scsi_done(xs);
- return (COMPLETE);
+ goto stuffup;
}
+ if ((ccb = mfi_get_ccb(sc)) == NULL)
+ return (TRY_AGAIN_LATER);
+
xs->error = XS_NOERROR;
switch (xs->cmd->opcode) {
@@ -694,43 +699,74 @@ mfi_scsi_cmd(struct scsi_xfer *xs)
/* hand it of to the firmware and let it deal with it */
default:
- return (mfi_generic_scsi(xs));
+ if (mfi_scsi_ld(ccb, xs)) {
+ mfi_put_ccb(ccb);
+ goto stuffup;
+ }
+ break;
}
DNPRINTF(MFI_D_CMD, "%s: start io %d\n", DEVNAME(sc), target);
- return (mfi_start_xs(sc, ccb, xs));
+ if (xs->flags & SCSI_POLL) {
+ if (mfi_poll(ccb)) {
+ /* XXX check for sense in ccb->ccb_sense? */
+ printf("%s: mfi_scsi_cmd poll failed\n",
+ DEVNAME(sc));
+ mfi_put_ccb(ccb);
+ bzero(&xs->sense, sizeof(xs->sense));
+ xs->sense.error_code = SSD_ERRCODE_VALID | 0x70;
+ xs->sense.flags = SKEY_ILLEGAL_REQUEST;
+ xs->sense.add_sense_code = 0x20; /* invalid opcode */
+ xs->error = XS_SENSE;
+ xs->flags |= ITSDONE;
+ scsi_done(xs);
+ return (COMPLETE);
+ }
+ DNPRINTF(MFI_D_DMA, "%s: mfi_scsi_cmd poll complete %d\n",
+ DEVNAME(sc), ccb->ccb_dmamap->dm_nsegs);
+
+ mfi_put_ccb(ccb);
+ return (COMPLETE);
+ }
+
+ mfi_despatch_cmd(ccb);
+
+ DNPRINTF(MFI_D_DMA, "%s: mfi_scsi_cmd queued %d\n", DEVNAME(sc),
+ ccb->ccb_dmamap->dm_nsegs);
+
+ return (SUCCESSFULLY_QUEUED);
+
+stuffup:
+ xs->error = XS_DRIVER_STUFFUP;
+ xs->flags |= ITSDONE;
+ scsi_done(xs);
+ return (COMPLETE);
}
int
-mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
- struct scsi_xfer *xs)
+mfi_create_sgl(struct mfi_ccb *ccb, int flags)
{
+ struct mfi_softc *sc = ccb->ccb_sc;
struct mfi_frame_header *hdr;
bus_dma_segment_t *sgd;
union mfi_sgl *sgl;
int error, i;
- DNPRINTF(MFI_D_DMA, "%s: mfi_start_xs: %p %p %d\n", DEVNAME(sc), xs,
- xs->data, xs->datalen);
+ DNPRINTF(MFI_D_DMA, "%s: mfi_create_sgl\n", DEVNAME(sc));
- if (!xs->data)
- goto skipsgl;
+ if (!ccb->ccb_data)
+ return (1);
error = bus_dmamap_load(sc->sc_dmat, ccb->ccb_dmamap,
- xs->data, xs->datalen, NULL,
- (xs->flags & SCSI_NOSLEEP) ? BUS_DMA_NOWAIT : BUS_DMA_WAITOK);
+ ccb->ccb_data, ccb->ccb_len, NULL, flags);
if (error) {
if (error == EFBIG)
- printf("more than %d dma segs\n", sc->sc_max_sgl);
+ printf("more than %d dma segs\n",
+ sc->sc_max_sgl);
else
printf("error %d loading dma map\n", error);
-
- mfi_put_ccb(ccb);
- xs->error = XS_DRIVER_STUFFUP;
- xs->flags |= ITSDONE;
- scsi_done(xs);
- return (COMPLETE);
+ return (1);
}
hdr = &ccb->ccb_frame->mfr_header;
@@ -739,8 +775,8 @@ mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
for (i = 0; i < ccb->ccb_dmamap->dm_nsegs; i++) {
sgl->sg32[i].addr = htole32(sgd[i].ds_addr);
sgl->sg32[i].len = htole32(sgd[i].ds_len);
- DNPRINTF(MFI_D_DMA, "%s: addr: %x len: %x\n", DEVNAME(sc),
- sgl->sg32[i].addr, sgl->sg32[i].len);
+ DNPRINTF(MFI_D_DMA, "%s: addr: %x len: %x\n",
+ DEVNAME(sc), sgl->sg32[i].addr, sgl->sg32[i].len);
}
if (ccb->ccb_direction == MFI_DATA_IN) {
@@ -768,32 +804,7 @@ mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
ccb->ccb_dmamap->dm_nsegs,
ccb->ccb_extra_frames);
-skipsgl:
- if (xs->flags & SCSI_POLL) {
- if (mfi_poll(sc, ccb)) {
- printf("%s: mfi_poll failed\n", DEVNAME(sc));
- bzero(&xs->sense, sizeof(xs->sense));
- xs->sense.error_code = SSD_ERRCODE_VALID | 0x70;
- xs->sense.flags = SKEY_ILLEGAL_REQUEST;
- xs->sense.add_sense_code = 0x20; /* invalid opcode */
- xs->error = XS_SENSE;
- xs->flags |= ITSDONE;
- scsi_done(xs);
- return (COMPLETE);
- }
- DNPRINTF(MFI_D_DMA, "%s: mfi_start_xs complete %d\n",
- DEVNAME(sc), ccb->ccb_dmamap->dm_nsegs);
-
- mfi_put_ccb(ccb);
- return (COMPLETE);
- }
-
- mfi_despatch_cmd(sc, ccb);
-
- DNPRINTF(MFI_D_DMA, "%s: mfi_start_xs: queued %d\n", DEVNAME(sc),
- ccb->ccb_dmamap->dm_nsegs);
-
- return (SUCCESSFULLY_QUEUED);
+ return (0);
}
int