summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/softraid.c120
-rw-r--r--sys/dev/softraidvar.h23
2 files changed, 122 insertions, 21 deletions
diff --git a/sys/dev/softraid.c b/sys/dev/softraid.c
index 87f9e6c5bf3..fc7bbcfe25b 100644
--- a/sys/dev/softraid.c
+++ b/sys/dev/softraid.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: softraid.c,v 1.14 2007/03/31 12:57:08 marco Exp $ */
+/* $OpenBSD: softraid.c,v 1.15 2007/04/11 22:05:09 marco Exp $ */
/*
* Copyright (c) 2007 Marco Peereboom <marco@peereboom.us>
*
@@ -56,7 +56,7 @@ uint32_t sr_debug = 0
/* | SR_D_WU */
/* | SR_D_META */
/* | SR_D_DIS */
- | SR_D_STATE
+ /* | SR_D_STATE */
;
#endif
@@ -118,6 +118,7 @@ void sr_raid1_intr(struct buf *);
void sr_raid1_set_chunk_state(struct sr_discipline *,
int, int);
void sr_raid1_set_vol_state(struct sr_discipline *);
+void sr_raid1_startwu(struct sr_workunit *);
struct scsi_adapter sr_switch = {
sr_scsi_cmd, sr_minphys, NULL, NULL, sr_scsi_ioctl
@@ -293,6 +294,8 @@ sr_alloc_wu(struct sr_discipline *sd)
M_DEVBUF, M_WAITOK);
memset(sd->sd_wu, 0, sizeof(struct sr_workunit) * no_wu);
TAILQ_INIT(&sd->sd_wu_freeq);
+ TAILQ_INIT(&sd->sd_wu_pendq);
+ TAILQ_INIT(&sd->sd_wu_defq);
for (i = 0; i < no_wu; i++) {
wu = &sd->sd_wu[i];
wu->swu_dis = sd;
@@ -314,6 +317,10 @@ sr_free_wu(struct sr_discipline *sd)
while ((wu = TAILQ_FIRST(&sd->sd_wu_freeq)) != NULL)
TAILQ_REMOVE(&sd->sd_wu_freeq, wu, swu_link);
+ while ((wu = TAILQ_FIRST(&sd->sd_wu_pendq)) != NULL)
+ TAILQ_REMOVE(&sd->sd_wu_pendq, wu, swu_link);
+ while ((wu = TAILQ_FIRST(&sd->sd_wu_defq)) != NULL)
+ TAILQ_REMOVE(&sd->sd_wu_defq, wu, swu_link);
if (sd->sd_wu)
free(sd->sd_wu, M_DEVBUF);
@@ -323,6 +330,8 @@ void
sr_put_wu(struct sr_workunit *wu)
{
struct sr_discipline *sd = wu->swu_dis;
+ struct sr_ccb *ccb;
+
int s;
DNPRINTF(SR_D_WU, "%s: sr_put_wu: %p\n", DEVNAME(sd->sd_sc), wu);
@@ -333,7 +342,15 @@ sr_put_wu(struct sr_workunit *wu)
wu->swu_state = SR_WU_FREE;
wu->swu_ios_complete = 0;
wu->swu_io_count = 0;
- wu->swu_ios = NULL;
+ wu->swu_blk_start = 0;
+ wu->swu_blk_end = 0;
+ wu->swu_collider = NULL;
+
+ while ((ccb = TAILQ_FIRST(&wu->swu_ccb)) != NULL) {
+ TAILQ_REMOVE(&wu->swu_ccb, ccb, ccb_link);
+ sr_put_ccb(ccb);
+ }
+ TAILQ_INIT(&wu->swu_ccb);
TAILQ_INSERT_TAIL(&sd->sd_wu_freeq, wu, swu_link);
splx(s);
@@ -783,7 +800,7 @@ sr_ioctl_createraid(struct sr_softc *sc, struct bioc_createraid *bc)
sd->sd_alloc_resources(sd);
/* metadata SHALL be fully filled in at this point */
- sd->sd_link.openings = 1; /* sc->sc_max_cmds; */
+ sd->sd_link.openings = sd->sd_max_wu;
sd->sd_link.device = &sr_dev;
sd->sd_link.device_softc = sc;
sd->sd_link.adapter_softc = sc;
@@ -1224,9 +1241,10 @@ sr_raid1_rw(struct sr_workunit *wu)
{
struct sr_discipline *sd = wu->swu_dis;
struct scsi_xfer *xs = wu->swu_xs;
+ struct sr_workunit *wup;
struct sr_ccb *ccb;
- int rv = 1, ios, x, i;
- daddr_t blk;
+ int rv = 1, ios, x, i, s;
+ daddr64_t blk;
DNPRINTF(SR_D_DIS, "%s: sr_raid1_rw 0x%02x\n", DEVNAME(sd->sd_sc),
xs->cmd->opcode);
@@ -1253,6 +1271,9 @@ sr_raid1_rw(struct sr_workunit *wu)
else
ios = sd->sd_vol.sv_meta.svm_no_chunk;
+ wu->swu_blk_start = blk;
+ wu->swu_blk_end = blk + xs->datalen - 1;
+
for (i = 0; i < ios; i++) {
ccb = sr_get_ccb(sd);
if (!ccb) {
@@ -1299,12 +1320,15 @@ sr_raid1_rw(struct sr_workunit *wu)
ccb->ccb_wu = wu;
+ TAILQ_INSERT_TAIL(&wu->swu_ccb, ccb, ccb_link);
+
DNPRINTF(SR_D_DIS, "%s: %s: sr_raid1: b_bcount: %d "
"b_blkno: %x b_flags 0x%0x b_data %p\n",
DEVNAME(sd->sd_sc), sd->sd_vol.sv_meta.svm_devname,
ccb->ccb_buf.b_bcount, ccb->ccb_buf.b_blkno,
ccb->ccb_buf.b_flags, ccb->ccb_buf.b_data);
+#if 0
/* vprint("despatch: ", ccb->ccb_buf.b_vp); */
ccb->ccb_buf.b_vp->v_numoutput++;
VOP_STRATEGY(&ccb->ccb_buf);
@@ -1321,23 +1345,66 @@ sr_raid1_rw(struct sr_workunit *wu)
}
sr_put_ccb(ccb);
}
+#endif
+ }
+
+ /* walk queue backwards and fill in collider if we have one */
+ s = splbio();
+ TAILQ_FOREACH_REVERSE(wup, &sd->sd_wu_pendq, sr_wu_list, swu_link) {
+ if (wu->swu_blk_end < wup->swu_blk_start ||
+ wup->swu_blk_end < wu->swu_blk_start)
+ continue;
+
+ /* we have an LBA collision, defer wu */
+ wu->swu_state = SR_WU_DEFERRED;
+ if (wup->swu_collider)
+ /* wu is on deferred queue, append to last wu */
+ while (wup->swu_collider)
+ wup = wup->swu_collider;
+
+ wup->swu_collider = wu;
+ TAILQ_INSERT_TAIL(&sd->sd_wu_defq, wu, swu_link);
+ goto queued;
}
+ /* XXX deal with polling */
+
+ sr_raid1_startwu(wu);
+
+queued:
+ splx(s);
rv = 0;
bad:
-
+ /* unwind the whole wu */
return (rv);
}
void
+sr_raid1_startwu(struct sr_workunit *wu)
+{
+ struct sr_discipline *sd = wu->swu_dis;
+ struct sr_ccb *ccb;
+
+ splassert(IPL_BIO);
+
+ /* move io to pending queue */
+ TAILQ_INSERT_TAIL(&sd->sd_wu_pendq, wu, swu_link);
+ /* XXX create function */
+ TAILQ_FOREACH(ccb, &wu->swu_ccb, ccb_link) {
+ ccb->ccb_buf.b_vp->v_numoutput++;
+ VOP_STRATEGY(&ccb->ccb_buf);
+ }
+}
+
+void
sr_raid1_intr(struct buf *bp)
{
struct sr_ccb *ccb = (struct sr_ccb *)bp;
- struct sr_workunit *wu = ccb->ccb_wu;
+ struct sr_workunit *wu = ccb->ccb_wu, *wup;
struct sr_discipline *sd = wu->swu_dis;
struct scsi_xfer *xs = wu->swu_xs;
struct sr_softc *sc = sd->sd_sc;
- int s;
+ int s, pend = 0;
DNPRINTF(SR_D_INTR, "%s: sr_intr bp %x xs %x\n",
DEVNAME(sc), bp, xs);
@@ -1346,14 +1413,14 @@ sr_raid1_intr(struct buf *bp)
" b_flags: 0x%0x\n", DEVNAME(sc), ccb->ccb_buf.b_bcount,
ccb->ccb_buf.b_resid, ccb->ccb_buf.b_flags);
+ s = splbio();
+
if (ccb->ccb_buf.b_flags & B_ERROR) {
- printf("%s: i/o error on block %d\n", DEVNAME(sc),
+ printf("%s: i/o error on block %llu\n", DEVNAME(sc),
ccb->ccb_buf.b_blkno);
wu->swu_state = SR_WU_FAILED;
}
- sr_put_ccb(ccb);
-
wu->swu_ios_complete++;
DNPRINTF(SR_D_INTR, "%s: sr_intr: comp: %d count: %d\n",
DEVNAME(sc), wu->swu_ios_complete, wu->swu_io_count);
@@ -1367,11 +1434,34 @@ sr_raid1_intr(struct buf *bp)
xs->resid = 0;
xs->flags |= ITSDONE;
- s = splbio();
- scsi_done(xs);
- splx(s);
+
+ TAILQ_FOREACH(wup, &sd->sd_wu_pendq, swu_link) {
+ if (wu == wup) {
+ /* io on pendq, remove */
+ TAILQ_REMOVE(&sd->sd_wu_pendq, wu, swu_link);
+ pend = 1;
+
+ if (wu->swu_collider) {
+ /* restart deferred wu */
+ wu->swu_collider->swu_state =
+ SR_WU_INPROGRESS;
+ TAILQ_REMOVE(&sd->sd_wu_defq,
+ wu->swu_collider, swu_link);
+ sr_raid1_startwu(wu->swu_collider);
+ }
+ break;
+ }
+ }
+
+ if (!pend)
+ printf("wu: %p not on pending queue\n", wu);
+
+ /* do not change the order of these 2 functions */
sr_put_wu(wu);
+ scsi_done(xs);
}
+
+ splx(s);
}
void
diff --git a/sys/dev/softraidvar.h b/sys/dev/softraidvar.h
index 86bd3786560..4a7ab9d78c8 100644
--- a/sys/dev/softraidvar.h
+++ b/sys/dev/softraidvar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: softraidvar.h,v 1.5 2007/03/30 23:15:30 marco Exp $ */
+/* $OpenBSD: softraidvar.h,v 1.6 2007/04/11 22:05:09 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <sro@peereboom.us>
*
@@ -86,6 +86,12 @@ struct sr_workunit {
#define SR_WU_OK 2
#define SR_WU_FAILED 3
#define SR_WU_PARTIALLYFAILED 4
+#define SR_WU_DEFERRED 5
+#define SR_WU_PENDING 6
+
+ /* workunit io range */
+ daddr64_t swu_blk_start;
+ daddr64_t swu_blk_end;
/* in flight totals */
u_int32_t swu_ios_complete;
@@ -95,9 +101,11 @@ struct sr_workunit {
/* number of ios that makes up the whole work unit */
u_int32_t swu_io_count;
- /* index into all ios */
- /* XXX make this a list ? */
- struct sr_ccb **swu_ios;
+ /* colliding wu */
+ struct sr_workunit *swu_collider;
+
+ /* all ios that make up this workunit */
+ struct sr_ccb_list swu_ccb;
TAILQ_ENTRY(sr_workunit) swu_link;
};
@@ -213,10 +221,13 @@ struct sr_discipline {
struct sr_ccb_list sd_ccb_freeq;
u_int32_t sd_max_ccb_per_wu;
- struct sr_workunit *sd_wu;
- struct sr_wu_list sd_wu_freeq;
+ struct sr_workunit *sd_wu; /* all workunits */
u_int32_t sd_max_wu;
+ struct sr_wu_list sd_wu_freeq; /* free wu queue */
+ struct sr_wu_list sd_wu_pendq; /* pending wu queue */
+ struct sr_wu_list sd_wu_defq; /* deferred wu queue */
+
/* discipline functions */
int (*sd_alloc_resources)(struct sr_discipline *);
int (*sd_assemble_volume)(void *);