summaryrefslogtreecommitdiff
path: root/sys/scsi/ss_scanjet.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/scsi/ss_scanjet.c')
-rw-r--r--sys/scsi/ss_scanjet.c93
1 files changed, 68 insertions, 25 deletions
diff --git a/sys/scsi/ss_scanjet.c b/sys/scsi/ss_scanjet.c
index ccf789fedf9..3777494300b 100644
--- a/sys/scsi/ss_scanjet.c
+++ b/sys/scsi/ss_scanjet.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ss_scanjet.c,v 1.33 2008/06/22 16:32:05 krw Exp $ */
+/* $OpenBSD: ss_scanjet.c,v 1.34 2010/01/02 23:28:51 dlg Exp $ */
/* $NetBSD: ss_scanjet.c,v 1.6 1996/05/18 22:58:01 christos Exp $ */
/*
@@ -54,7 +54,8 @@
int scanjet_set_params(struct ss_softc *, struct scan_io *);
int scanjet_trigger_scanner(struct ss_softc *);
-int scanjet_read(struct ss_softc *, struct buf *);
+int scanjet_read(struct ss_softc *, struct scsi_xfer *, struct buf *);
+void scanjet_read_done(struct scsi_xfer *);
/* only used internally */
int scanjet_ctl_write(struct ss_softc *, char *, u_int, int);
@@ -276,43 +277,85 @@ scanjet_trigger_scanner(ss)
}
int
-scanjet_read(ss, bp)
+scanjet_read(ss, xs, bp)
struct ss_softc *ss;
+ struct scsi_xfer *xs;
struct buf *bp;
{
- struct scsi_rw_scanner cmd;
- struct scsi_link *sc_link = ss->sc_link;
+ struct scsi_rw_scanner *cdb;
- /*
- * Fill out the scsi command
- */
- bzero(&cmd, sizeof(cmd));
- cmd.opcode = READ;
+ SC_DEBUG(sc_link, SDEV_DB1, ("scanjet_read: start\n"));
- /*
- * Handle "fixed-block-mode" tape drives by using the
- * block count instead of the length.
- */
- _lto3b(bp->b_bcount, cmd.len);
+ cdb = (struct scsi_rw_scanner *)xs->cmd;
+ xs->cmdlen = sizeof(*cdb);
- /*
- * go ask the adapter to do all this for us
- */
- if (scsi_scsi_cmd(sc_link, (struct scsi_generic *) &cmd, sizeof(cmd),
- (u_char *) bp->b_data, bp->b_bcount, SCSI_RETRIES, 100000, bp,
- SCSI_NOSLEEP | SCSI_DATA_IN) != SUCCESSFULLY_QUEUED)
- printf("%s: not queued\n", ss->sc_dev.dv_xname);
- else {
+ cdb->opcode = READ;
+ _lto3b(bp->b_bcount, cdb->len);
+
+ xs->data = bp->b_data;
+ xs->datalen = bp->b_bcount;
+ xs->flags |= SCSI_DATA_IN;
+ xs->timeout = 100000;
+ xs->done = scanjet_read_done;
+ xs->cookie = bp;
+
+ scsi_xs_exec(xs);
+
+ return (0);
+}
+
+void
+scanjet_read_done(struct scsi_xfer *xs)
+{
+ struct ss_softc *ss = xs->sc_link->device_softc;
+ struct buf *bp = xs->cookie;
+
+ splassert(IPL_BIO);
+
+ switch (xs->error) {
+ case XS_NOERROR:
if (bp->b_bcount >= ss->sio.scan_window_size)
ss->sio.scan_window_size = 0;
else
ss->sio.scan_window_size -= bp->b_bcount;
+
+ bp->b_error = 0;
+ bp->b_resid = xs->resid;
+ break;
+
+ case XS_NO_CCB:
+ /* The adapter is busy, requeue the buf and try it later. */
+ ss_buf_requeue(ss, bp);
+ scsi_xs_put(xs);
+ SET(ss->flags, SSF_WAITING); /* break out of cdstart loop */
+ timeout_add(&ss->timeout, 1);
+ return;
+
+ case XS_SENSE:
+ case XS_SHORTSENSE:
+ if (scsi_interpret_sense(xs) != ERESTART)
+ xs->retries = 0;
+
+ /* FALLTHROUGH */
+ case XS_BUSY:
+ case XS_TIMEOUT:
+ if (xs->retries--) {
+ scsi_xs_exec(xs);
+ return;
+ }
+
+ /* FALLTHROUGH */
+ default:
+ bp->b_error = EIO;
+ bp->b_flags |= B_ERROR;
+ bp->b_resid = bp->b_bcount;
+ break;
}
- return (0);
+ biodone(bp);
+ scsi_xs_put(xs);
}
-
/*
* Do a synchronous write. Used to send control messages.
*/