summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/dev/ic/mfi.c199
-rw-r--r--sys/dev/ic/mfireg.h3
-rw-r--r--sys/dev/ic/mfivar.h11
3 files changed, 197 insertions, 16 deletions
diff --git a/sys/dev/ic/mfi.c b/sys/dev/ic/mfi.c
index bf2bb156d2c..fefd68bd9ab 100644
--- a/sys/dev/ic/mfi.c
+++ b/sys/dev/ic/mfi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mfi.c,v 1.25 2006/05/10 21:48:50 marco Exp $ */
+/* $OpenBSD: mfi.c,v 1.26 2006/05/15 23:20:56 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <marco@peereboom.us>
*
@@ -85,6 +85,10 @@ int mfi_poll(struct mfi_softc *, struct mfi_ccb *);
int mfi_start_xs(struct mfi_softc *, struct mfi_ccb *,
struct scsi_xfer *);
+/* LD commands */
+int mfi_ld_inquiry(struct scsi_xfer *);
+void mfi_done_ld_inquiry(struct mfi_softc *, struct mfi_ccb *);
+
#if NBIO > 0
int mfi_ioctl(struct device *, u_long, caddr_t);
int mfi_ioctl_inq(struct mfi_softc *, struct bioc_inq *);
@@ -443,9 +447,6 @@ mfi_attach(struct mfi_softc *sc)
goto noinit;
}
- /* enable interrupts */
- mfi_write(sc, MFI_OMSK, MFI_ENABLE_INTR);
-
#if NBIO > 0
if (bio_register(&sc->sc_dev, mfi_ioctl) != 0)
panic("%s: controller registration failed", DEVNAME(sc));
@@ -453,8 +454,36 @@ mfi_attach(struct mfi_softc *sc)
sc->sc_ioctl = mfi_ioctl;
#endif /* NBIO > 0 */
+ sc->sc_max_ld = MFI_MAX_LD;
+
+ /* XXX fake one ld for now */
+ sc->sc_ld_cnt = 2;
+ sc->sc_ld[0].ld_present = 1;
+ sc->sc_ld[1].ld_present = 1;
+ sc->sc_max_ld = 2;
+
+ if (sc->sc_ld_cnt)
+ sc->sc_link.openings = sc->sc_max_cmds / sc->sc_ld_cnt;
+ else
+ sc->sc_link.openings = sc->sc_max_cmds;
+
+ sc->sc_link.device = &mfi_dev;
+ sc->sc_link.adapter_softc = sc;
+ sc->sc_link.adapter = &mfi_switch;
+ sc->sc_link.adapter_target = sc->sc_max_ld;
+ sc->sc_link.adapter_buswidth = sc->sc_max_ld;
+#if 0
+ printf(", FW %s, BIOS v%s, %dMB RAM\n"
+ "%s: %d channels, %d %ss, %d logical drives\n",
+ sc->sc_fwver, sc->sc_biosver, sc->sc_memory, DEVNAME(sc),
+ sc->sc_channels, sc->sc_targets, p, sc->sc_ld_cnt);
+#endif
+
config_found(&sc->sc_dev, &sc->sc_link, scsiprint);
+ /* enable interrupts */
+ mfi_write(sc, MFI_OMSK, MFI_ENABLE_INTR);
+
return (0);
noinit:
mfi_freemem(sc, sc->sc_sense);
@@ -547,6 +576,55 @@ mfi_intr(void *arg)
return (claimed);
}
+void
+mfi_done_ld_inquiry(struct mfi_softc *sc, struct mfi_ccb *ccb)
+{
+ struct scsi_xfer *xs = ccb->ccb_xs;
+ struct scsi_link *link = xs->sc_link;
+
+ DNPRINTF(MFI_D_CMD, "%s: mfi_ld_inquiry_done: %.0x\n",
+ DEVNAME(sc), link->target);
+}
+
+int
+mfi_ld_inquiry(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_ld_inquiry: %.0x\n",
+ DEVNAME(sc), link->target);
+
+ if ((ccb = mfi_get_ccb(sc)) == NULL)
+ return (TRY_AGAIN_LATER);
+
+ pf = &ccb->ccb_frame->mfr_pass;
+ pf->mpf_header.mfh_cmd = MFI_CMD_LD_SCSI_IO;
+ pf->mpf_header.mfh_target_id = link->target;
+ pf->mpf_header.mfh_lun_id = 0;
+ pf->mpf_header.mfh_cdb_len = 6;
+ pf->mpf_header.mfh_timeout = 0;
+ pf->mpf_header.mfh_data_len= sizeof(struct scsi_inquiry_data); /* XXX */
+ pf->mpf_header.mfh_sense_len = MFI_SENSE_SIZE;
+
+ pf->mpf_sense_addr_hi = 0;
+ pf->mpf_sense_addr_lo = htole32(ccb->ccb_psense);
+
+ memset(pf->mpf_cdb, 0, 16);
+ pf->mpf_cdb[0] = INQUIRY;
+ pf->mpf_cdb[4] = sizeof(struct scsi_inquiry_data);
+
+ ccb->ccb_done = mfi_done_ld_inquiry;
+ ccb->ccb_xs = xs; /* XXX here or in mfi_start_xs? */
+ ccb->ccb_sgl = &pf->mpf_sgl;
+ ccb->ccb_direction = MFI_DATA_IN;
+ ccb->ccb_frame_size = MFI_PASS_FRAME_SIZE;
+
+ return (mfi_start_xs(sc, ccb, xs));
+}
+
int
mfi_scsi_cmd(struct scsi_xfer *xs)
{
@@ -556,13 +634,14 @@ mfi_scsi_cmd(struct scsi_xfer *xs)
struct mfi_ccb *ccb;
u_int8_t target = link->target;
- DNPRINTF(MFI_D_CMD, "mfi_scsi_cmd\n");
+ DNPRINTF(MFI_D_CMD, "%s: mfi_scsi_cmd opcode: %.0x\n",
+ DEVNAME(sc), xs->cmd->opcode);
/* only issue IO through this path, create seperate path for mgmt */
if (target >= MFI_MAX_LD || !sc->sc_ld[target].ld_present ||
link->lun != 0) {
- DNPRINTF(MFI_D_CMD, "%s: invalid target %d ",
+ DNPRINTF(MFI_D_CMD, "%s: invalid target %d\n",
DEVNAME(sc), target);
xs->error = XS_DRIVER_STUFFUP;
xs->flags |= ITSDONE;
@@ -572,32 +651,124 @@ mfi_scsi_cmd(struct scsi_xfer *xs)
xs->error = XS_NOERROR;
-
switch (xs->cmd->opcode) {
- case READ_COMMAND: /* IO path */
+ /* IO path */
+ case READ_COMMAND:
case READ_BIG:
case WRITE_COMMAND:
case WRITE_BIG:
break;
- default: /* the rest is a DCDB */
- break;
+ /* DCDB */
+ case INQUIRY:
+ return (mfi_ld_inquiry(xs));
+ /* NOTREACHED */
+
+ case SYNCHRONIZE_CACHE:
+ case TEST_UNIT_READY:
+ case START_STOP:
+#if 0
+ case VERIFY:
+#endif
+ case PREVENT_ALLOW:
+ case REQUEST_SENSE:
+ case READ_CAPACITY:
+ DNPRINTF(MFI_D_CMD, "%s: not implemented yet %.0x\n",
+ DEVNAME(sc), xs->cmd->opcode);
+ xs->error = XS_DRIVER_STUFFUP;
+ xs->flags |= ITSDONE;
+ scsi_done(xs);
+ return (COMPLETE);
+ /* NOTREACHED */
+
+ /* illegal opcode */
+ default:
+ 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);
+ /* NOTREACHED */
}
return (mfi_start_xs(sc, ccb, xs));
}
-int mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
+int
+mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
struct scsi_xfer *xs)
{
- return (1); /* XXX not yet */
+ 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:\n", DEVNAME(sc));
+
+ 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);
+ if (error) {
+ if (error == EFBIG)
+ 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;
+ scsi_done(xs);
+ return (COMPLETE);
+ }
+
+ hdr = &ccb->ccb_frame->mfr_header;
+ sgl = ccb->ccb_sgl;
+ sgd = ccb->ccb_dmamap->dm_segs;
+ 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);
+ }
+
+ if (ccb->ccb_direction == MFI_DATA_IN) {
+ hdr->mfh_flags |= MFI_FRAME_DIR_READ;
+ bus_dmamap_sync(sc->sc_dmat, ccb->ccb_dmamap, 0,
+ ccb->ccb_dmamap->dm_mapsize, BUS_DMASYNC_PREWRITE);
+ } else {
+ hdr->mfh_flags |= MFI_FRAME_DIR_WRITE;
+ bus_dmamap_sync(sc->sc_dmat, ccb->ccb_dmamap, 0,
+ ccb->ccb_dmamap->dm_mapsize, BUS_DMASYNC_PREWRITE);
+ }
+
+ hdr->mfh_sg_count = ccb->ccb_dmamap->dm_nsegs;
+ ccb->ccb_frame_size += sc->sc_frames_size * ccb->ccb_dmamap->dm_nsegs;
+ ccb->ccb_extra_frames = (ccb->ccb_frame_size - 1) / MFI_FRAME_SIZE;
+
+ mfi_despatch_cmd(sc, ccb);
+
+ if (ccb->ccb_xs->flags & SCSI_POLL) {
+ if (mfi_poll(sc, ccb)) {
+ printf("%s: mfi_poll failed\n", DEVNAME(sc));
+ xs->error = XS_DRIVER_STUFFUP;
+ }
+ DNPRINTF(MFI_D_DMA, "%s: mfi_start_xs complete %d\n",
+ DEVNAME(sc), ccb->ccb_dmamap->dm_nsegs);
+ mfi_put_ccb(ccb);
+ return (COMPLETE);
+ }
+
+ DNPRINTF(MFI_D_DMA, "%s: mfi_start_xs: queued %d\n", DEVNAME(sc),
+ ccb->ccb_dmamap->dm_nsegs);
+
+ return (SUCCESSFULLY_QUEUED);
}
int
mfi_scsi_ioctl(struct scsi_link *link, u_long cmd, caddr_t addr, int flag,
struct proc *p)
{
- struct mfi_softc *sc = (struct mfi_softc *)link->adapter_softc;
+ struct mfi_softc *sc = (struct mfi_softc *)link->adapter_softc;
DNPRINTF(MFI_D_IOCTL, "mfi_scsi_ioctl\n");
@@ -611,7 +782,7 @@ mfi_scsi_ioctl(struct scsi_link *link, u_long cmd, caddr_t addr, int flag,
int
mfi_ioctl(struct device *dev, u_long cmd, caddr_t addr)
{
- struct mfi_softc *sc = (struct mfi_softc *)dev;
+ struct mfi_softc *sc = (struct mfi_softc *)dev;
int error = 0;
DNPRINTF(MFI_D_IOCTL, "%s: ioctl ", DEVNAME(sc));
diff --git a/sys/dev/ic/mfireg.h b/sys/dev/ic/mfireg.h
index d917e22ad5b..1d4dfb3bb16 100644
--- a/sys/dev/ic/mfireg.h
+++ b/sys/dev/ic/mfireg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: mfireg.h,v 1.10 2006/05/10 21:48:50 marco Exp $ */
+/* $OpenBSD: mfireg.h,v 1.11 2006/05/15 23:20:57 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <marco@peereboom.us>
*
@@ -291,6 +291,7 @@ struct mfi_io_frame {
union mfi_sgl mif_sgl;
} __packed;
+#define MFI_PASS_FRAME_SIZE 48
struct mfi_pass_frame {
struct mfi_frame_header mpf_header;
uint32_t mpf_sense_addr_lo;
diff --git a/sys/dev/ic/mfivar.h b/sys/dev/ic/mfivar.h
index 8c0a244ee27..80c1a111202 100644
--- a/sys/dev/ic/mfivar.h
+++ b/sys/dev/ic/mfivar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: mfivar.h,v 1.14 2006/05/10 21:48:50 marco Exp $ */
+/* $OpenBSD: mfivar.h,v 1.15 2006/05/15 23:20:57 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <marco@peereboom.us>
*
@@ -57,6 +57,7 @@ struct mfi_ccb {
union mfi_frame *ccb_frame;
paddr_t ccb_pframe;
+ uint32_t ccb_frame_size;
uint32_t ccb_extra_frames;
struct mfi_sense *ccb_sense;
@@ -64,6 +65,12 @@ struct mfi_ccb {
bus_dmamap_t ccb_dmamap;
+ union mfi_sgl *ccb_sgl;
+
+ uint32_t ccb_direction;
+#define MFI_DATA_IN 0
+#define MFI_DATA_OUT 1
+
struct scsi_xfer *ccb_xs;
void (*ccb_done)(struct mfi_softc *sc,
@@ -106,6 +113,8 @@ struct mfi_softc {
/* firmware determined max and totals */
uint32_t sc_max_cmds;
uint32_t sc_max_sgl;
+ uint32_t sc_max_ld;
+ uint32_t sc_ld_cnt;
/* all commands */
struct mfi_ccb *sc_ccb;