2 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
4 * Copyright (c) 2004-2009 Pawel Jakub Dawidek <pjd@FreeBSD.org>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
32 #include <sys/param.h>
33 #include <sys/systm.h>
35 #include <sys/kernel.h>
36 #include <sys/limits.h>
38 #include <sys/malloc.h>
42 #include <geom/geom.h>
43 #include <geom/geom_dbg.h>
44 #include <geom/geom_int.h>
45 #include <geom/mirror/g_mirror.h>
48 * Configure, Rebuild, Remove, Deactivate, Forget, and Stop operations do not
49 * seem to depend on any particular g_mirror initialization state.
51 static struct g_mirror_softc *
52 g_mirror_find_device(struct g_class *mp, const char *name)
54 struct g_mirror_softc *sc;
58 LIST_FOREACH(gp, &mp->geom, geom) {
62 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0)
64 if (strcmp(gp->name, name) == 0 ||
65 strcmp(sc->sc_name, name) == 0) {
67 sx_xlock(&sc->sc_lock);
68 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0) {
69 sx_xunlock(&sc->sc_lock);
79 /* Insert and Resize operations depend on a launched GEOM (sc_provider). */
80 #define GMFL_VALID_FLAGS (M_WAITOK | M_NOWAIT)
81 static struct g_mirror_softc *
82 g_mirror_find_launched_device(struct g_class *mp, const char *name, int flags)
84 struct g_mirror_softc *sc;
87 KASSERT((flags & ~GMFL_VALID_FLAGS) == 0 &&
88 flags != GMFL_VALID_FLAGS && flags != 0,
89 ("%s: Invalid flags %x\n", __func__, (unsigned)flags));
90 #undef GMFL_VALID_FLAGS
93 sc = g_mirror_find_device(mp, name);
96 if (sc->sc_provider != NULL)
98 if (flags & M_NOWAIT) {
99 sx_xunlock(&sc->sc_lock);
104 * This is a dumb hack. G_mirror does not expose any real
105 * wakeup API for observing state changes, and even if it did,
106 * its "RUNNING" state does not actually reflect all softc
107 * elements being initialized.
109 * Revamping g_mirror to have a 3rd, ACTUALLY_RUNNING state and
110 * updating all assertions and sc_state checks is a large work
111 * and would be easy to introduce regressions.
113 * Revamping g_mirror to have a wakeup for state changes would
114 * be difficult if one wanted to capture more than just
115 * sc_state and sc_provider.
117 * For now, just dummy sleep-poll until sc_provider shows up,
118 * the user cancels, or the g_mirror is destroyed.
120 error = sx_sleep(&sc, &sc->sc_lock, PRIBIO | PCATCH | PDROP,
122 if (error != 0 && error != EWOULDBLOCK)
128 static struct g_mirror_disk *
129 g_mirror_find_disk(struct g_mirror_softc *sc, const char *name)
131 struct g_mirror_disk *disk;
133 sx_assert(&sc->sc_lock, SX_XLOCKED);
134 if (strncmp(name, "/dev/", 5) == 0)
136 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
137 if (disk->d_consumer == NULL)
139 if (disk->d_consumer->provider == NULL)
141 if (strcmp(disk->d_consumer->provider->name, name) == 0)
148 g_mirror_ctl_configure(struct gctl_req *req, struct g_class *mp)
150 struct g_mirror_softc *sc;
151 struct g_mirror_disk *disk;
152 const char *name, *balancep, *prov;
153 intmax_t *slicep, *priority;
156 int *autosync, *noautosync, *failsync, *nofailsync, *hardcode, *dynamic;
157 int *nargs, do_sync = 0, dirty = 1, do_priority = 0;
159 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
161 gctl_error(req, "No '%s' argument.", "nargs");
164 if (*nargs != 1 && *nargs != 2) {
165 gctl_error(req, "Invalid number of arguments.");
168 name = gctl_get_asciiparam(req, "arg0");
170 gctl_error(req, "No 'arg%u' argument.", 0);
173 balancep = gctl_get_asciiparam(req, "balance");
174 if (balancep == NULL) {
175 gctl_error(req, "No '%s' argument.", "balance");
178 autosync = gctl_get_paraml(req, "autosync", sizeof(*autosync));
179 if (autosync == NULL) {
180 gctl_error(req, "No '%s' argument.", "autosync");
183 noautosync = gctl_get_paraml(req, "noautosync", sizeof(*noautosync));
184 if (noautosync == NULL) {
185 gctl_error(req, "No '%s' argument.", "noautosync");
188 failsync = gctl_get_paraml(req, "failsync", sizeof(*failsync));
189 if (failsync == NULL) {
190 gctl_error(req, "No '%s' argument.", "failsync");
193 nofailsync = gctl_get_paraml(req, "nofailsync", sizeof(*nofailsync));
194 if (nofailsync == NULL) {
195 gctl_error(req, "No '%s' argument.", "nofailsync");
198 hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
199 if (hardcode == NULL) {
200 gctl_error(req, "No '%s' argument.", "hardcode");
203 dynamic = gctl_get_paraml(req, "dynamic", sizeof(*dynamic));
204 if (dynamic == NULL) {
205 gctl_error(req, "No '%s' argument.", "dynamic");
208 priority = gctl_get_paraml(req, "priority", sizeof(*priority));
209 if (priority == NULL) {
210 gctl_error(req, "No '%s' argument.", "priority");
213 if (*priority < -1 || *priority > 255) {
214 gctl_error(req, "Priority range is 0 to 255, %jd given",
219 * Since we have a priority, we also need a provider now.
220 * Note: be WARNS safe, by always assigning prov and only throw an
221 * error if *priority != -1.
223 prov = gctl_get_asciiparam(req, "arg1");
224 if (*priority > -1) {
226 gctl_error(req, "Priority needs a disk name");
231 if (*autosync && *noautosync) {
232 gctl_error(req, "'%s' and '%s' specified.", "autosync",
236 if (*failsync && *nofailsync) {
237 gctl_error(req, "'%s' and '%s' specified.", "failsync",
241 if (*hardcode && *dynamic) {
242 gctl_error(req, "'%s' and '%s' specified.", "hardcode",
246 sc = g_mirror_find_device(mp, name);
248 gctl_error(req, "No such device: %s.", name);
251 if (*balancep == '\0')
252 balance = sc->sc_balance;
254 if (balance_id(balancep) == -1) {
255 gctl_error(req, "Invalid balance algorithm.");
256 sx_xunlock(&sc->sc_lock);
259 balance = balance_id(balancep);
261 slicep = gctl_get_paraml(req, "slice", sizeof(*slicep));
262 if (slicep == NULL) {
263 gctl_error(req, "No '%s' argument.", "slice");
264 sx_xunlock(&sc->sc_lock);
268 slice = sc->sc_slice;
271 /* Enforce usage() of -p not allowing any other options. */
272 if (do_priority && (*autosync || *noautosync || *failsync ||
273 *nofailsync || *hardcode || *dynamic || *slicep != -1 ||
274 *balancep != '\0')) {
275 sx_xunlock(&sc->sc_lock);
276 gctl_error(req, "only -p accepted when setting priority");
279 if (sc->sc_balance == balance && sc->sc_slice == slice && !*autosync &&
280 !*noautosync && !*failsync && !*nofailsync && !*hardcode &&
281 !*dynamic && !do_priority) {
282 sx_xunlock(&sc->sc_lock);
283 gctl_error(req, "Nothing has changed.");
286 if ((!do_priority && *nargs != 1) || (do_priority && *nargs != 2)) {
287 sx_xunlock(&sc->sc_lock);
288 gctl_error(req, "Invalid number of arguments.");
291 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
292 sx_xunlock(&sc->sc_lock);
293 gctl_error(req, "Not all disks connected. Try 'forget' command "
297 sc->sc_balance = balance;
298 sc->sc_slice = slice;
299 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOAUTOSYNC) != 0) {
301 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
306 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
308 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOFAILSYNC) != 0) {
310 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
313 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
317 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
319 * Handle priority first, since we only need one disk, do one
320 * operation on it and then we're done. No need to check other
321 * flags, as usage doesn't allow it.
324 if (strcmp(disk->d_name, prov) == 0) {
325 if (disk->d_priority == *priority)
326 gctl_error(req, "Nothing has changed.");
328 disk->d_priority = *priority;
329 g_mirror_update_metadata(disk);
336 if (disk->d_state == G_MIRROR_DISK_STATE_SYNCHRONIZING)
337 disk->d_flags &= ~G_MIRROR_DISK_FLAG_FORCE_SYNC;
340 disk->d_flags |= G_MIRROR_DISK_FLAG_HARDCODED;
342 disk->d_flags &= ~G_MIRROR_DISK_FLAG_HARDCODED;
344 disk->d_flags &= ~G_MIRROR_DISK_FLAG_DIRTY;
345 g_mirror_update_metadata(disk);
347 if (disk->d_state == G_MIRROR_DISK_STATE_STALE) {
348 g_mirror_event_send(disk,
349 G_MIRROR_DISK_STATE_DISCONNECTED,
350 G_MIRROR_EVENT_DONTWAIT);
354 sx_xunlock(&sc->sc_lock);
358 g_mirror_create_orphan(struct g_consumer *cp)
361 KASSERT(1 == 0, ("%s called while creating %s.", __func__,
362 cp->provider->name));
366 g_mirror_ctl_create(struct gctl_req *req, struct g_class *mp)
368 struct g_mirror_metadata md;
370 struct g_consumer *cp;
371 struct g_provider *pp;
372 struct g_mirror_softc *sc;
381 unsigned attached, no, sectorsize;
384 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
386 gctl_error(req, "No '%s' argument.", "nargs");
390 gctl_error(req, "Too few arguments.");
394 strlcpy(md.md_magic, G_MIRROR_MAGIC, sizeof(md.md_magic));
395 md.md_version = G_MIRROR_VERSION;
396 name = gctl_get_asciiparam(req, "arg0");
398 gctl_error(req, "No 'arg%u' argument.", 0);
401 strlcpy(md.md_name, name, sizeof(md.md_name));
402 md.md_mid = arc4random();
403 md.md_all = *nargs - 1;
406 md.md_sync_offset = 0;
407 val = gctl_get_paraml(req, "slice", sizeof(*val));
409 gctl_error(req, "No slice argument.");
413 sval = gctl_get_asciiparam(req, "balance");
415 gctl_error(req, "No balance argument.");
418 bal = balance_id(sval);
420 gctl_error(req, "Invalid balance algorithm.");
426 ival = gctl_get_paraml(req, "noautosync", sizeof(*ival));
427 if (ival != NULL && *ival)
428 md.md_mflags |= G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
429 ival = gctl_get_paraml(req, "nofailsync", sizeof(*ival));
430 if (ival != NULL && *ival)
431 md.md_mflags |= G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
432 /* These fields not used in manual mode. */
433 bzero(md.md_provider, sizeof(md.md_provider));
439 gp = g_new_geomf(mp, "%s", md.md_name);
440 gp->orphan = g_mirror_create_orphan;
441 cp = g_new_consumer(gp);
442 for (no = 1; no < *nargs; no++) {
443 snprintf(param, sizeof(param), "arg%u", no);
444 name = gctl_get_asciiparam(req, param);
446 gctl_error(req, "No 'arg%u' argument.", no);
448 g_destroy_consumer(cp);
453 if (strncmp(name, "/dev/", strlen("/dev/")) == 0)
454 name += strlen("/dev/");
455 pp = g_provider_by_name(name);
457 G_MIRROR_DEBUG(1, "Disk %s is invalid.", name);
458 gctl_error(req, "Disk %s is invalid.", name);
462 if (g_access(cp, 1, 0, 0) != 0) {
463 G_MIRROR_DEBUG(1, "Can't open disk %s.", name);
464 gctl_error(req, "Can't open disk %s.", name);
469 if (pp->mediasize == 0 || pp->sectorsize == 0) {
470 G_MIRROR_DEBUG(1, "Disk %s has no media.", name);
471 gctl_error(req, "Disk %s has no media.", name);
472 g_access(cp, -1, 0, 0);
475 if (pp->mediasize < mediasize)
476 mediasize = pp->mediasize;
477 if (pp->sectorsize > sectorsize)
478 sectorsize = pp->sectorsize;
479 g_access(cp, -1, 0, 0);
482 g_destroy_consumer(cp);
484 md.md_mediasize = mediasize;
485 md.md_sectorsize = sectorsize;
486 md.md_mediasize -= (md.md_mediasize % md.md_sectorsize);
488 gp = g_mirror_create(mp, &md, G_MIRROR_TYPE_MANUAL);
490 gctl_error(req, "Can't create %s.", md.md_name);
497 sx_xlock(&sc->sc_lock);
498 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_TASTING;
499 sb = sbuf_new_auto();
500 sbuf_printf(sb, "Can't attach disk(s) to %s:", gp->name);
501 for (attached = 0, no = 1; no < *nargs; no++) {
502 snprintf(param, sizeof(param), "arg%u", no);
503 name = gctl_get_asciiparam(req, param);
504 if (strncmp(name, "/dev/", strlen("/dev/")) == 0)
505 name += strlen("/dev/");
506 pp = g_provider_by_name(name);
508 G_MIRROR_DEBUG(1, "Provider %s disappear?!", name);
509 sbuf_printf(sb, " %s", name);
512 md.md_did = arc4random();
513 md.md_priority = no - 1;
514 if (g_mirror_add_disk(sc, pp, &md) != 0) {
515 G_MIRROR_DEBUG(1, "Disk %u (%s) not attached to %s.",
516 no, pp->name, gp->name);
517 sbuf_printf(sb, " %s", pp->name);
523 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_TASTING;
524 if (md.md_all != attached ||
525 (sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0) {
526 g_mirror_destroy(gp->softc, G_MIRROR_DESTROY_HARD);
527 gctl_error(req, "%s", sbuf_data(sb));
529 sx_xunlock(&sc->sc_lock);
534 g_mirror_ctl_rebuild(struct gctl_req *req, struct g_class *mp)
536 struct g_mirror_metadata md;
537 struct g_mirror_softc *sc;
538 struct g_mirror_disk *disk;
539 struct g_provider *pp;
545 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
547 gctl_error(req, "No '%s' argument.", "nargs");
551 gctl_error(req, "Too few arguments.");
554 name = gctl_get_asciiparam(req, "arg0");
556 gctl_error(req, "No 'arg%u' argument.", 0);
559 sc = g_mirror_find_device(mp, name);
561 gctl_error(req, "No such device: %s.", name);
564 for (i = 1; i < (u_int)*nargs; i++) {
565 snprintf(param, sizeof(param), "arg%u", i);
566 name = gctl_get_asciiparam(req, param);
568 gctl_error(req, "No 'arg%u' argument.", i);
571 disk = g_mirror_find_disk(sc, name);
573 gctl_error(req, "No such provider: %s.", name);
576 if (g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE) == 1 &&
577 disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
579 * This is the last active disk. There will be nothing
580 * to rebuild it from, so deny this request.
583 "Provider %s is the last active provider in %s.",
584 name, sc->sc_geom->name);
588 * Do rebuild by resetting syncid, disconnecting the disk and
589 * connecting it again.
591 disk->d_sync.ds_syncid = 0;
592 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOAUTOSYNC) != 0)
593 disk->d_flags |= G_MIRROR_DISK_FLAG_FORCE_SYNC;
594 g_mirror_update_metadata(disk);
595 pp = disk->d_consumer->provider;
597 error = g_mirror_read_metadata(disk->d_consumer, &md);
599 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DISCONNECTED,
600 G_MIRROR_EVENT_WAIT);
602 gctl_error(req, "Cannot read metadata from %s.",
606 error = g_mirror_add_disk(sc, pp, &md);
608 gctl_error(req, "Cannot reconnect component %s.",
613 sx_xunlock(&sc->sc_lock);
617 g_mirror_ctl_insert(struct gctl_req *req, struct g_class *mp)
619 struct g_mirror_softc *sc;
620 struct g_mirror_disk *disk;
621 struct g_mirror_metadata md;
622 struct g_provider *pp;
623 struct g_consumer *cp;
629 int error, *nargs, *hardcode, *inactive;
631 struct g_provider *provider;
632 struct g_consumer *consumer;
636 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
638 gctl_error(req, "No '%s' argument.", "nargs");
642 gctl_error(req, "Too few arguments.");
645 priority = gctl_get_paraml(req, "priority", sizeof(*priority));
646 if (priority == NULL) {
647 gctl_error(req, "No '%s' argument.", "priority");
650 inactive = gctl_get_paraml(req, "inactive", sizeof(*inactive));
651 if (inactive == NULL) {
652 gctl_error(req, "No '%s' argument.", "inactive");
655 hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
656 if (hardcode == NULL) {
657 gctl_error(req, "No '%s' argument.", "hardcode");
660 name = gctl_get_asciiparam(req, "arg0");
662 gctl_error(req, "No 'arg%u' argument.", 0);
665 sc = g_mirror_find_launched_device(mp, name, M_WAITOK);
667 gctl_error(req, "No such device: %s.", name);
670 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
671 gctl_error(req, "Not all disks connected.");
672 sx_xunlock(&sc->sc_lock);
676 disks = g_malloc(sizeof(*disks) * (*nargs), M_WAITOK | M_ZERO);
678 for (i = 1, n = 0; i < (u_int)*nargs; i++) {
679 snprintf(param, sizeof(param), "arg%u", i);
680 name = gctl_get_asciiparam(req, param);
682 gctl_error(req, "No 'arg%u' argument.", i);
685 if (g_mirror_find_disk(sc, name) != NULL) {
686 gctl_error(req, "Provider %s already inserted.", name);
689 if (strncmp(name, "/dev/", 5) == 0)
691 pp = g_provider_by_name(name);
693 gctl_error(req, "Unknown provider %s.", name);
696 cp = g_new_consumer(sc->sc_geom);
697 if (g_attach(cp, pp) != 0) {
698 g_destroy_consumer(cp);
699 gctl_error(req, "Cannot attach to provider %s.", name);
702 if (g_access(cp, 0, 1, 1) != 0) {
703 gctl_error(req, "Cannot access provider %s.", name);
706 g_destroy_consumer(cp);
709 mdsize = (sc->sc_type == G_MIRROR_TYPE_AUTOMATIC) ?
711 if (sc->sc_provider->mediasize > pp->mediasize - mdsize) {
712 gctl_error(req, "Provider %s too small.", name);
714 g_access(cp, 0, -1, -1);
717 if ((sc->sc_provider->sectorsize % pp->sectorsize) != 0) {
718 gctl_error(req, "Invalid sectorsize of provider %s.",
722 if (sc->sc_type != G_MIRROR_TYPE_AUTOMATIC) {
723 g_access(cp, 0, -1, -1);
725 g_destroy_consumer(cp);
728 g_mirror_fill_metadata(sc, NULL, &md);
729 md.md_priority = *priority;
731 md.md_dflags |= G_MIRROR_DISK_FLAG_INACTIVE;
732 if (g_mirror_add_disk(sc, pp, &md) != 0) {
734 gctl_error(req, "Disk %s not inserted.", name);
739 disks[n].provider = pp;
740 disks[n].consumer = cp;
745 sx_xunlock(&sc->sc_lock);
751 for (i = 0; i < n; i++) {
752 if (disks[i].consumer == NULL)
754 g_mirror_fill_metadata(sc, NULL, &md);
755 md.md_priority = *priority;
757 md.md_dflags |= G_MIRROR_DISK_FLAG_INACTIVE;
758 pp = disks[i].provider;
760 strlcpy(md.md_provider, pp->name,
761 sizeof(md.md_provider));
763 bzero(md.md_provider, sizeof(md.md_provider));
765 md.md_provsize = pp->mediasize;
766 sector = g_malloc(pp->sectorsize, M_WAITOK);
767 mirror_metadata_encode(&md, sector);
768 error = g_write_data(disks[i].consumer,
769 pp->mediasize - pp->sectorsize, sector, pp->sectorsize);
772 gctl_error(req, "Cannot store metadata on %s.",
774 g_access(disks[i].consumer, 0, -1, -1);
775 g_detach(disks[i].consumer);
776 g_destroy_consumer(disks[i].consumer);
777 disks[i].consumer = NULL;
778 disks[i].provider = NULL;
785 /* All writes failed. */
786 sx_xunlock(&sc->sc_lock);
790 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
791 g_mirror_update_metadata(disk);
794 * Release provider and wait for retaste.
797 for (i = 0; i < n; i++) {
798 if (disks[i].consumer == NULL)
800 g_access(disks[i].consumer, 0, -1, -1);
801 g_detach(disks[i].consumer);
802 g_destroy_consumer(disks[i].consumer);
805 sx_xunlock(&sc->sc_lock);
810 g_mirror_ctl_remove(struct gctl_req *req, struct g_class *mp)
812 struct g_mirror_softc *sc;
813 struct g_mirror_disk *disk;
819 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
821 gctl_error(req, "No '%s' argument.", "nargs");
825 gctl_error(req, "Too few arguments.");
828 name = gctl_get_asciiparam(req, "arg0");
830 gctl_error(req, "No 'arg%u' argument.", 0);
833 sc = g_mirror_find_device(mp, name);
835 gctl_error(req, "No such device: %s.", name);
838 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
839 sx_xunlock(&sc->sc_lock);
840 gctl_error(req, "Not all disks connected. Try 'forget' command "
844 active = g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE);
845 for (i = 1; i < (u_int)*nargs; i++) {
846 snprintf(param, sizeof(param), "arg%u", i);
847 name = gctl_get_asciiparam(req, param);
849 gctl_error(req, "No 'arg%u' argument.", i);
852 disk = g_mirror_find_disk(sc, name);
854 gctl_error(req, "No such provider: %s.", name);
857 if (disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
861 gctl_error(req, "%s: Can't remove the last "
862 "ACTIVE component %s.", sc->sc_geom->name,
867 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DESTROY,
868 G_MIRROR_EVENT_DONTWAIT);
870 sx_xunlock(&sc->sc_lock);
874 g_mirror_ctl_resize(struct gctl_req *req, struct g_class *mp)
876 struct g_mirror_softc *sc;
877 struct g_mirror_disk *disk;
879 const char *name, *s;
883 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
885 gctl_error(req, "No '%s' argument.", "nargs");
889 gctl_error(req, "Missing device.");
892 name = gctl_get_asciiparam(req, "arg0");
894 gctl_error(req, "No 'arg%u' argument.", 0);
897 s = gctl_get_asciiparam(req, "size");
899 gctl_error(req, "No '%s' argument.", "size");
902 mediasize = strtouq(s, &x, 0);
903 if (*x != '\0' || mediasize == 0) {
904 gctl_error(req, "Invalid '%s' argument.", "size");
907 sc = g_mirror_find_launched_device(mp, name, M_WAITOK);
909 gctl_error(req, "No such device: %s.", name);
912 /* Deny shrinking of an opened provider */
913 if ((g_debugflags & G_F_FOOTSHOOTING) == 0 && sc->sc_provider_open > 0) {
914 if (sc->sc_mediasize > mediasize) {
915 gctl_error(req, "Device %s is busy.",
916 sc->sc_provider->name);
917 sx_xunlock(&sc->sc_lock);
921 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
922 if (mediasize > disk->d_consumer->provider->mediasize -
923 disk->d_consumer->provider->sectorsize) {
924 gctl_error(req, "Provider %s is too small.",
926 sx_xunlock(&sc->sc_lock);
930 /* Update the size. */
931 sc->sc_mediasize = mediasize;
932 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
933 g_mirror_update_metadata(disk);
936 g_resize_provider(sc->sc_provider, mediasize);
938 sx_xunlock(&sc->sc_lock);
942 g_mirror_ctl_deactivate(struct gctl_req *req, struct g_class *mp)
944 struct g_mirror_softc *sc;
945 struct g_mirror_disk *disk;
951 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
953 gctl_error(req, "No '%s' argument.", "nargs");
957 gctl_error(req, "Too few arguments.");
960 name = gctl_get_asciiparam(req, "arg0");
962 gctl_error(req, "No 'arg%u' argument.", 0);
965 sc = g_mirror_find_device(mp, name);
967 gctl_error(req, "No such device: %s.", name);
970 active = g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE);
971 for (i = 1; i < (u_int)*nargs; i++) {
972 snprintf(param, sizeof(param), "arg%u", i);
973 name = gctl_get_asciiparam(req, param);
975 gctl_error(req, "No 'arg%u' argument.", i);
978 disk = g_mirror_find_disk(sc, name);
980 gctl_error(req, "No such provider: %s.", name);
983 if (disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
987 gctl_error(req, "%s: Can't deactivate the "
988 "last ACTIVE component %s.",
989 sc->sc_geom->name, name);
993 disk->d_flags |= G_MIRROR_DISK_FLAG_INACTIVE;
994 disk->d_flags &= ~G_MIRROR_DISK_FLAG_FORCE_SYNC;
995 g_mirror_update_metadata(disk);
996 sc->sc_bump_id |= G_MIRROR_BUMP_SYNCID;
997 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DISCONNECTED,
998 G_MIRROR_EVENT_DONTWAIT);
1000 sx_xunlock(&sc->sc_lock);
1004 g_mirror_ctl_forget(struct gctl_req *req, struct g_class *mp)
1006 struct g_mirror_softc *sc;
1007 struct g_mirror_disk *disk;
1013 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
1014 if (nargs == NULL) {
1015 gctl_error(req, "No '%s' argument.", "nargs");
1019 gctl_error(req, "Missing device(s).");
1023 for (i = 0; i < (u_int)*nargs; i++) {
1024 snprintf(param, sizeof(param), "arg%u", i);
1025 name = gctl_get_asciiparam(req, param);
1027 gctl_error(req, "No 'arg%u' argument.", i);
1030 sc = g_mirror_find_device(mp, name);
1032 gctl_error(req, "No such device: %s.", name);
1035 if (g_mirror_ndisks(sc, -1) == sc->sc_ndisks) {
1036 sx_xunlock(&sc->sc_lock);
1038 "All disks connected in %s, skipping.",
1042 sc->sc_ndisks = g_mirror_ndisks(sc, -1);
1043 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
1044 g_mirror_update_metadata(disk);
1046 sx_xunlock(&sc->sc_lock);
1051 g_mirror_ctl_stop(struct gctl_req *req, struct g_class *mp, int wipe)
1053 struct g_mirror_softc *sc;
1054 int *force, *nargs, error;
1060 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
1061 if (nargs == NULL) {
1062 gctl_error(req, "No '%s' argument.", "nargs");
1066 gctl_error(req, "Missing device(s).");
1069 force = gctl_get_paraml(req, "force", sizeof(*force));
1070 if (force == NULL) {
1071 gctl_error(req, "No '%s' argument.", "force");
1075 how = G_MIRROR_DESTROY_HARD;
1077 how = G_MIRROR_DESTROY_SOFT;
1079 for (i = 0; i < (u_int)*nargs; i++) {
1080 snprintf(param, sizeof(param), "arg%u", i);
1081 name = gctl_get_asciiparam(req, param);
1083 gctl_error(req, "No 'arg%u' argument.", i);
1086 sc = g_mirror_find_device(mp, name);
1088 gctl_error(req, "No such device: %s.", name);
1093 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_WIPE;
1094 error = g_mirror_destroy(sc, how);
1096 gctl_error(req, "Cannot destroy device %s (error=%d).",
1097 sc->sc_geom->name, error);
1099 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_WIPE;
1100 sx_xunlock(&sc->sc_lock);
1103 /* No need to unlock, because lock is already dead. */
1108 g_mirror_config(struct gctl_req *req, struct g_class *mp, const char *verb)
1112 g_topology_assert();
1114 version = gctl_get_paraml(req, "version", sizeof(*version));
1115 if (version == NULL) {
1116 gctl_error(req, "No '%s' argument.", "version");
1119 if (*version != G_MIRROR_VERSION) {
1120 gctl_error(req, "Userland and kernel parts are out of sync.");
1124 g_topology_unlock();
1125 if (strcmp(verb, "configure") == 0)
1126 g_mirror_ctl_configure(req, mp);
1127 else if (strcmp(verb, "create") == 0)
1128 g_mirror_ctl_create(req, mp);
1129 else if (strcmp(verb, "rebuild") == 0)
1130 g_mirror_ctl_rebuild(req, mp);
1131 else if (strcmp(verb, "insert") == 0)
1132 g_mirror_ctl_insert(req, mp);
1133 else if (strcmp(verb, "remove") == 0)
1134 g_mirror_ctl_remove(req, mp);
1135 else if (strcmp(verb, "resize") == 0)
1136 g_mirror_ctl_resize(req, mp);
1137 else if (strcmp(verb, "deactivate") == 0)
1138 g_mirror_ctl_deactivate(req, mp);
1139 else if (strcmp(verb, "forget") == 0)
1140 g_mirror_ctl_forget(req, mp);
1141 else if (strcmp(verb, "stop") == 0)
1142 g_mirror_ctl_stop(req, mp, 0);
1143 else if (strcmp(verb, "destroy") == 0)
1144 g_mirror_ctl_stop(req, mp, 1);
1146 gctl_error(req, "Unknown verb.");