summaryrefslogtreecommitdiff
path: root/sys/dev/ic
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic')
-rw-r--r--sys/dev/ic/mfi.c67
1 files changed, 61 insertions, 6 deletions
diff --git a/sys/dev/ic/mfi.c b/sys/dev/ic/mfi.c
index 1d19de5e233..9c56f4a3551 100644
--- a/sys/dev/ic/mfi.c
+++ b/sys/dev/ic/mfi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mfi.c,v 1.30 2006/05/16 01:15:29 marco Exp $ */
+/* $OpenBSD: mfi.c,v 1.31 2006/05/16 01:58:46 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <marco@peereboom.us>
*
@@ -90,6 +90,8 @@ int mfi_ld_inquiry(struct scsi_xfer *);
void mfi_done_ld_inquiry(struct mfi_softc *, struct mfi_ccb *);
int mfi_ld_tur(struct scsi_xfer *);
void mfi_done_ld_tur(struct mfi_softc *, struct mfi_ccb *);
+int mfi_ld_readcap(struct scsi_xfer *);
+void mfi_done_ld_readcap(struct mfi_softc *, struct mfi_ccb *);
#if NBIO > 0
int mfi_ioctl(struct device *, u_long, caddr_t);
@@ -630,6 +632,16 @@ mfi_ld_inquiry(struct scsi_xfer *xs)
return (mfi_start_xs(sc, ccb, xs));
}
+void
+mfi_done_ld_tur(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_done_ld_tur: %.0x\n",
+ DEVNAME(sc), link->target);
+}
+
int
mfi_ld_tur(struct scsi_xfer *xs)
{
@@ -665,6 +677,7 @@ mfi_ld_tur(struct scsi_xfer *xs)
ccb->ccb_direction = 0;
ccb->ccb_frame_size = MFI_PASS_FRAME_SIZE;
+ /* XXX don't do this here, make something generic */
if (ccb->ccb_xs->flags & SCSI_POLL) {
if (mfi_poll(sc, ccb)) {
printf("%s: mfi_poll failed\n", DEVNAME(sc));
@@ -687,16 +700,54 @@ mfi_ld_tur(struct scsi_xfer *xs)
}
void
-mfi_done_ld_tur(struct mfi_softc *sc, struct mfi_ccb *ccb)
+mfi_done_ld_readcap(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_done_ld_tur: %.0x\n",
+ DNPRINTF(MFI_D_CMD, "%s: mfi_ld_done_readcap: %.0x\n",
DEVNAME(sc), link->target);
}
int
+mfi_ld_readcap(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_readcap: %d\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_read_capacity);
+ 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] = READ_CAPACITY; /* XXX other drivers use READCAP 16 */
+
+ ccb->ccb_done = mfi_done_ld_readcap;
+ 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)
{
struct scsi_link *link = xs->sc_link;
@@ -753,13 +804,16 @@ mfi_scsi_cmd(struct scsi_xfer *xs)
DEVNAME(sc), target);
return (COMPLETE);
- case SYNCHRONIZE_CACHE:
+ case READ_CAPACITY:
+ return (mfi_ld_readcap(xs));
+ /* NOTREACHED */
+
#if 0
case VERIFY:
#endif
+ case SYNCHRONIZE_CACHE:
case PREVENT_ALLOW:
case REQUEST_SENSE:
- case READ_CAPACITY:
DNPRINTF(MFI_D_CMD, "%s: not implemented yet %02x\n",
DEVNAME(sc), xs->cmd->opcode);
xs->error = XS_DRIVER_STUFFUP;
@@ -808,6 +862,7 @@ mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
mfi_put_ccb(ccb);
xs->error = XS_DRIVER_STUFFUP;
+ xs->flags |= ITSDONE;
scsi_done(xs);
return (COMPLETE);
}
@@ -834,7 +889,7 @@ mfi_start_xs(struct mfi_softc *sc, struct mfi_ccb *ccb,
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;
- if (ccb->ccb_xs->flags & SCSI_POLL) {
+ if (xs->flags & SCSI_POLL) {
if (mfi_poll(sc, ccb)) {
printf("%s: mfi_poll failed\n", DEVNAME(sc));
xs->error = XS_DRIVER_STUFFUP;