summaryrefslogtreecommitdiff
path: root/sys/dev/softraid.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/softraid.c')
-rw-r--r--sys/dev/softraid.c117
1 files changed, 86 insertions, 31 deletions
diff --git a/sys/dev/softraid.c b/sys/dev/softraid.c
index a24bdcd2dc8..56078b97784 100644
--- a/sys/dev/softraid.c
+++ b/sys/dev/softraid.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: softraid.c,v 1.128 2009/04/26 17:04:53 marco Exp $ */
+/* $OpenBSD: softraid.c,v 1.129 2009/04/28 02:54:53 marco Exp $ */
/*
* Copyright (c) 2007 Marco Peereboom <marco@peereboom.us>
* Copyright (c) 2008 Chris Kuethe <ckuethe@openbsd.org>
@@ -54,6 +54,7 @@
#include <dev/rndvar.h>
/* #define SR_FANCY_STATS */
+/* #define SR_UNIT_TEST */
#ifdef SR_DEBUG
#define SR_FANCY_STATS
@@ -189,7 +190,7 @@ sr_meta_attach(struct sr_discipline *sd, int force)
struct sr_softc *sc = sd->sd_sc;
struct sr_chunk_head *cl;
struct sr_chunk *ch_entry;
- int rv = 1, i;
+ int rv = 1, i = 0;
DNPRINTF(SR_D_META, "%s: sr_meta_attach(%d)\n", DEVNAME(sc));
@@ -213,15 +214,23 @@ sr_meta_attach(struct sr_discipline *sd, int force)
}
}
- if (smd[sd->sd_meta_type].smd_attach(sd, force))
- goto bad;
-
- /* fill out chunk array */
+ /* we have a valid list now create an array index */
cl = &sd->sd_vol.sv_chunk_list;
+ SLIST_FOREACH(ch_entry, cl, src_link) {
+ i++;
+ }
+ sd->sd_vol.sv_chunks = malloc(sizeof(struct sr_chunk *) * i,
+ M_DEVBUF, M_WAITOK | M_ZERO);
+
+ /* fill out chunk array */
i = 0;
SLIST_FOREACH(ch_entry, cl, src_link)
sd->sd_vol.sv_chunks[i++] = ch_entry;
+ /* attach metadata */
+ if (smd[sd->sd_meta_type].smd_attach(sd, force))
+ goto bad;
+
rv = 0;
bad:
return (rv);
@@ -633,7 +642,7 @@ sr_meta_read(struct sr_discipline *sd)
struct sr_meta_driver *s;
struct sr_meta_opt *om;
void *fm = NULL;
- int no_disk = 0;
+ int no_disk = 0, got_meta = 0;
DNPRINTF(SR_D_META, "%s: sr_meta_read\n", DEVNAME(sc));
@@ -644,9 +653,18 @@ sr_meta_read(struct sr_discipline *sd)
cp = (struct sr_meta_chunk *)(sm + 1);
SLIST_FOREACH(ch_entry, cl, src_link) {
- /* read and translate */
- if (s->smd_read(sd, ch_entry->src_dev_mm, sm, fm)) {
- /* XXX mark disk offline */
+ /* skip disks that are offline */
+ if (ch_entry->src_meta.scm_status == BIOC_SDOFFLINE) {
+ DNPRINTF(SR_D_META,
+ "%s: %s chunk marked offline, spoofing status\n",
+ DEVNAME(sc), ch_entry->src_devname);
+ cp++; /* adjust chunk pointer to match failure */
+ continue;
+ } else if (s->smd_read(sd, ch_entry->src_dev_mm, sm, fm)) {
+ /* read and translate */
+ /* XXX mark chunk offline, elsewhere!! */
+ ch_entry->src_meta.scm_status = BIOC_SDOFFLINE;
+ cp++; /* adjust chunk pointer to match failure */
DNPRINTF(SR_D_META, "%s: sr_meta_read failed\n",
DEVNAME(sc));
continue;
@@ -666,9 +684,11 @@ sr_meta_read(struct sr_discipline *sd)
goto done;
}
- /* assume chunk 0 contains metadata */
- if (no_disk == 0)
+ /* assume first chunk contains metadata */
+ if (got_meta == 0) {
bcopy(sm, sd->sd_meta, sizeof(*sd->sd_meta));
+ got_meta = 1;
+ }
bcopy(cp, &ch_entry->src_meta, sizeof(ch_entry->src_meta));
@@ -1050,9 +1070,10 @@ sr_meta_native_attach(struct sr_discipline *sd, int force)
struct sr_softc *sc = sd->sd_sc;
struct sr_chunk_head *cl = &sd->sd_vol.sv_chunk_list;
struct sr_metadata *md = NULL;
- struct sr_chunk *ch_entry;
+ struct sr_chunk *ch_entry, *ch_next;
struct sr_uuid uuid;
- int sr, not_sr, rv = 1, d, expected = -1;
+ u_int64_t version = 0;
+ int sr, not_sr, rv = 1, d, expected = -1, old_meta = 0;
DNPRINTF(SR_D_META, "%s: sr_meta_native_attach\n", DEVNAME(sc));
@@ -1078,6 +1099,8 @@ sr_meta_native_attach(struct sr_discipline *sd, int force)
if (d == 0) {
bcopy(&md->ssdi.ssd_uuid, &uuid, sizeof uuid);
expected = md->ssdi.ssd_chunk_no;
+ version = md->ssd_ondisk;
+ d++;
continue;
} else if (bcmp(&md->ssdi.ssd_uuid, &uuid,
sizeof uuid)) {
@@ -1085,10 +1108,12 @@ sr_meta_native_attach(struct sr_discipline *sd, int force)
DEVNAME(sc));
goto bad;
}
+ if (md->ssd_ondisk != version) {
+ old_meta++;
+ version = MAX(md->ssd_ondisk, version);
+ }
} else
not_sr++;
-
- d++;
}
if (sr && not_sr) {
@@ -1096,6 +1121,25 @@ sr_meta_native_attach(struct sr_discipline *sd, int force)
DEVNAME(sc));
goto bad;
}
+
+ /* mixed metadata versions; mark bad disks offline */
+ if (old_meta) {
+ d = 0;
+ for (ch_entry = SLIST_FIRST(cl); ch_entry != SLIST_END(cl);
+ ch_entry = ch_next, d++) {
+ ch_next = SLIST_NEXT(ch_entry, src_link);
+
+ /* XXX do we want to read this again? */
+ if (sr_meta_native_read(sd, ch_entry->src_dev_mm, md,
+ NULL))
+ printf("%s: could not read native metadata\n",
+ DEVNAME(sc));
+ if (md->ssd_ondisk != version)
+ sd->sd_vol.sv_chunks[d]->src_meta.scm_status =
+ BIOC_SDOFFLINE;
+ }
+ }
+
if (expected != sr && !force && expected != -1) {
/* XXX make this smart so that we can bring up degraded disks */
printf("%s: not all chunks were provided\n", DEVNAME(sc));
@@ -1706,18 +1750,33 @@ sr_ioctl_setstate(struct sr_softc *sc, struct bioc_setstate *bs)
int rv = EINVAL;
#ifdef SR_UNIT_TEST
- int i, vol, state;
+ int i, vol, state, found, tg;
struct sr_discipline *sd;
+ struct sr_chunk *ch_entry;
+ struct sr_chunk_head *cl;
+
+ if (bs->bs_other_id_type == BIOC_SSOTHER_UNUSED)
+ goto done;
for (i = 0, vol = -1; i < SR_MAXSCSIBUS; i++) {
/* XXX this will not work when we stagger disciplines */
if (sc->sc_dis[i])
vol++;
- if (vol != bs->bs_channel)
+ if (vol != bs->bs_volid)
continue;
+ sd = sc->sc_dis[i];
- sd = sc->sc_dis[vol];
- if (bs->bs_target >= sd->sd_meta->ssdi.ssd_chunk_no)
+ found = 0;
+ tg = 0;
+ cl = &sd->sd_vol.sv_chunk_list;
+ SLIST_FOREACH(ch_entry, cl, src_link) {
+ if (ch_entry->src_dev_mm == bs->bs_other_id) {
+ found = 1;
+ break;
+ }
+ tg++;
+ }
+ if (found == 0)
goto done;
switch (bs->bs_status) {
@@ -1738,10 +1797,7 @@ sr_ioctl_setstate(struct sr_softc *sc, struct bioc_setstate *bs)
goto done;
}
- printf("status change for %u:%u -> %u %u\n",
- bs->bs_channel, bs->bs_target, bs->bs_status, state);
-
- sd->sd_set_chunk_state(sd, bs->bs_target, bs->bs_status);
+ sd->sd_set_chunk_state(sd, tg, bs->bs_status);
rv = 0;
@@ -1787,10 +1843,6 @@ sr_ioctl_createraid(struct sr_softc *sc, struct bioc_createraid *bc, int user)
cl = &sd->sd_vol.sv_chunk_list;
SLIST_INIT(cl);
- /* we have a valid list now create an array index */
- sd->sd_vol.sv_chunks = malloc(sizeof(struct sr_chunk *) * no_chunk,
- M_DEVBUF, M_WAITOK | M_ZERO);
-
sd->sd_meta_type = sr_meta_probe(sd, dt, no_chunk);
if (sd->sd_meta_type == SR_META_F_INVALID) {
printf("%s: invalid metadata format\n", DEVNAME(sc));
@@ -1972,11 +2024,11 @@ sr_ioctl_createraid(struct sr_softc *sc, struct bioc_createraid *bc, int user)
"assemble it\n");
goto unwind;
}
- printf("%s: not yet partial bringup\n", DEVNAME(sc));
- goto unwind;
+ printf("%s: trying to bring up %s degraded\n", DEVNAME(sc),
+ sd->sd_meta->ssd_devname);
}
- /* XXX metadata SHALL be fully filled in at this point */
+ /* metadata SHALL be fully filled in at this point */
switch (bc->bc_level) {
case 0:
@@ -2099,6 +2151,9 @@ sr_ioctl_createraid(struct sr_softc *sc, struct bioc_createraid *bc, int user)
goto unwind;
if (disk) {
+ /* set volume status */
+ sd->sd_set_vol_state(sd);
+
/* setup scsi midlayer */
sd->sd_link.openings = sd->sd_max_wu;
sd->sd_link.device = &sr_dev;