1 /*-
2 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3 *
4 * Copyright (c) 2004-2009 Pawel Jakub Dawidek <pjd@FreeBSD.org>
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
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.
15 *
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
26 * SUCH DAMAGE.
27 */
28
29 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
31
32 #include <sys/param.h>
33 #include <sys/systm.h>
34 #include <sys/bio.h>
35 #include <sys/kernel.h>
36 #include <sys/limits.h>
37 #include <sys/lock.h>
38 #include <sys/malloc.h>
39 #include <sys/sbuf.h>
40 #include <sys/sx.h>
41
42 #include <geom/geom.h>
43 #include <geom/geom_dbg.h>
44 #include <geom/geom_int.h>
45 #include <geom/mirror/g_mirror.h>
46
47 /*
48 * Configure, Rebuild, Remove, Deactivate, Forget, and Stop operations do not
49 * seem to depend on any particular g_mirror initialization state.
50 */
51 static struct g_mirror_softc *
52 g_mirror_find_device(struct g_class *mp, const char *name)
53 {
54 struct g_mirror_softc *sc;
55 struct g_geom *gp;
56
57 g_topology_lock();
58 LIST_FOREACH(gp, &mp->geom, geom) {
59 sc = gp->softc;
60 if (sc == NULL)
61 continue;
62 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0)
63 continue;
64 if (strcmp(gp->name, name) == 0 ||
65 strcmp(sc->sc_name, name) == 0) {
66 g_topology_unlock();
67 sx_xlock(&sc->sc_lock);
68 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0) {
69 sx_xunlock(&sc->sc_lock);
70 return (NULL);
71 }
72 return (sc);
73 }
74 }
75 g_topology_unlock();
76 return (NULL);
77 }
78
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)
83 {
84 struct g_mirror_softc *sc;
85 int error;
86
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
91
92 while (true) {
93 sc = g_mirror_find_device(mp, name);
94 if (sc == NULL)
95 return (NULL);
96 if (sc->sc_provider != NULL)
97 return (sc);
98 if (flags & M_NOWAIT) {
99 sx_xunlock(&sc->sc_lock);
100 return (NULL);
101 }
102
103 /*
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.
108 *
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.
112 *
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.
116 *
117 * For now, just dummy sleep-poll until sc_provider shows up,
118 * the user cancels, or the g_mirror is destroyed.
119 */
120 error = sx_sleep(&sc, &sc->sc_lock, PRIBIO | PCATCH | PDROP,
121 "GM:launched", 1);
122 if (error != 0 && error != EWOULDBLOCK)
123 return (NULL);
124 }
125 __unreachable();
126 }
127
128 static struct g_mirror_disk *
129 g_mirror_find_disk(struct g_mirror_softc *sc, const char *name)
130 {
131 struct g_mirror_disk *disk;
132
133 sx_assert(&sc->sc_lock, SX_XLOCKED);
134 if (strncmp(name, _PATH_DEV, 5) == 0)
135 name += 5;
136 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
137 if (disk->d_consumer == NULL)
138 continue;
139 if (disk->d_consumer->provider == NULL)
140 continue;
141 if (strcmp(disk->d_consumer->provider->name, name) == 0)
142 return (disk);
143 }
144 return (NULL);
145 }
146
147 static void
148 g_mirror_ctl_configure(struct gctl_req *req, struct g_class *mp)
149 {
150 struct g_mirror_softc *sc;
151 struct g_mirror_disk *disk;
152 const char *name, *balancep, *prov;
153 intmax_t *slicep, *priority;
154 uint32_t slice;
155 uint8_t balance;
156 int *autosync, *noautosync, *failsync, *nofailsync, *hardcode, *dynamic;
157 int *nargs, do_sync = 0, dirty = 1, do_priority = 0;
158
159 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
160 if (nargs == NULL) {
161 gctl_error(req, "No '%s' argument.", "nargs");
162 return;
163 }
164 if (*nargs != 1 && *nargs != 2) {
165 gctl_error(req, "Invalid number of arguments.");
166 return;
167 }
168 name = gctl_get_asciiparam(req, "arg0");
169 if (name == NULL) {
170 gctl_error(req, "No 'arg%u' argument.", 0);
171 return;
172 }
173 balancep = gctl_get_asciiparam(req, "balance");
174 if (balancep == NULL) {
175 gctl_error(req, "No '%s' argument.", "balance");
176 return;
177 }
178 autosync = gctl_get_paraml(req, "autosync", sizeof(*autosync));
179 if (autosync == NULL) {
180 gctl_error(req, "No '%s' argument.", "autosync");
181 return;
182 }
183 noautosync = gctl_get_paraml(req, "noautosync", sizeof(*noautosync));
184 if (noautosync == NULL) {
185 gctl_error(req, "No '%s' argument.", "noautosync");
186 return;
187 }
188 failsync = gctl_get_paraml(req, "failsync", sizeof(*failsync));
189 if (failsync == NULL) {
190 gctl_error(req, "No '%s' argument.", "failsync");
191 return;
192 }
193 nofailsync = gctl_get_paraml(req, "nofailsync", sizeof(*nofailsync));
194 if (nofailsync == NULL) {
195 gctl_error(req, "No '%s' argument.", "nofailsync");
196 return;
197 }
198 hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
199 if (hardcode == NULL) {
200 gctl_error(req, "No '%s' argument.", "hardcode");
201 return;
202 }
203 dynamic = gctl_get_paraml(req, "dynamic", sizeof(*dynamic));
204 if (dynamic == NULL) {
205 gctl_error(req, "No '%s' argument.", "dynamic");
206 return;
207 }
208 priority = gctl_get_paraml(req, "priority", sizeof(*priority));
209 if (priority == NULL) {
210 gctl_error(req, "No '%s' argument.", "priority");
211 return;
212 }
213 if (*priority < -1 || *priority > 255) {
214 gctl_error(req, "Priority range is 0 to 255, %jd given",
215 *priority);
216 return;
217 }
218 /*
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.
222 */
223 prov = gctl_get_asciiparam(req, "arg1");
224 if (*priority > -1) {
225 if (prov == NULL) {
226 gctl_error(req, "Priority needs a disk name");
227 return;
228 }
229 do_priority = 1;
230 }
231 if (*autosync && *noautosync) {
232 gctl_error(req, "'%s' and '%s' specified.", "autosync",
233 "noautosync");
234 return;
235 }
236 if (*failsync && *nofailsync) {
237 gctl_error(req, "'%s' and '%s' specified.", "failsync",
238 "nofailsync");
239 return;
240 }
241 if (*hardcode && *dynamic) {
242 gctl_error(req, "'%s' and '%s' specified.", "hardcode",
243 "dynamic");
244 return;
245 }
246 sc = g_mirror_find_device(mp, name);
247 if (sc == NULL) {
248 gctl_error(req, "No such device: %s.", name);
249 return;
250 }
251 if (*balancep == '\0')
252 balance = sc->sc_balance;
253 else {
254 if (balance_id(balancep) == -1) {
255 gctl_error(req, "Invalid balance algorithm.");
256 sx_xunlock(&sc->sc_lock);
257 return;
258 }
259 balance = balance_id(balancep);
260 }
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);
265 return;
266 }
267 if (*slicep == -1)
268 slice = sc->sc_slice;
269 else
270 slice = *slicep;
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");
277 return;
278 }
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.");
284 return;
285 }
286 if ((!do_priority && *nargs != 1) || (do_priority && *nargs != 2)) {
287 sx_xunlock(&sc->sc_lock);
288 gctl_error(req, "Invalid number of arguments.");
289 return;
290 }
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 "
294 "first.");
295 return;
296 }
297 sc->sc_balance = balance;
298 sc->sc_slice = slice;
299 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOAUTOSYNC) != 0) {
300 if (*autosync) {
301 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
302 do_sync = 1;
303 }
304 } else {
305 if (*noautosync)
306 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
307 }
308 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOFAILSYNC) != 0) {
309 if (*failsync)
310 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
311 } else {
312 if (*nofailsync) {
313 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
314 dirty = 0;
315 }
316 }
317 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
318 /*
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.
322 */
323 if (do_priority) {
324 if (strcmp(disk->d_name, prov) == 0) {
325 if (disk->d_priority == *priority)
326 gctl_error(req, "Nothing has changed.");
327 else {
328 disk->d_priority = *priority;
329 g_mirror_update_metadata(disk);
330 }
331 break;
332 }
333 continue;
334 }
335 if (do_sync) {
336 if (disk->d_state == G_MIRROR_DISK_STATE_SYNCHRONIZING)
337 disk->d_flags &= ~G_MIRROR_DISK_FLAG_FORCE_SYNC;
338 }
339 if (*hardcode)
340 disk->d_flags |= G_MIRROR_DISK_FLAG_HARDCODED;
341 else if (*dynamic)
342 disk->d_flags &= ~G_MIRROR_DISK_FLAG_HARDCODED;
343 if (!dirty)
344 disk->d_flags &= ~G_MIRROR_DISK_FLAG_DIRTY;
345 g_mirror_update_metadata(disk);
346 if (do_sync) {
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);
351 }
352 }
353 }
354 sx_xunlock(&sc->sc_lock);
355 }
356
357 static void
358 g_mirror_create_orphan(struct g_consumer *cp)
359 {
360
361 KASSERT(1 == 0, ("%s called while creating %s.", __func__,
362 cp->provider->name));
363 }
364
365 static void
366 g_mirror_ctl_create(struct gctl_req *req, struct g_class *mp)
367 {
368 struct g_mirror_metadata md;
369 struct g_geom *gp;
370 struct g_consumer *cp;
371 struct g_provider *pp;
372 struct g_mirror_softc *sc;
373 struct sbuf *sb;
374 const char *name;
375 char param[16];
376 int *nargs;
377 intmax_t *val;
378 int *ival;
379 const char *sval;
380 int bal;
381 unsigned attached, no, sectorsize;
382 off_t mediasize;
383
384 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
385 if (nargs == NULL) {
386 gctl_error(req, "No '%s' argument.", "nargs");
387 return;
388 }
389 if (*nargs <= 2) {
390 gctl_error(req, "Too few arguments.");
391 return;
392 }
393
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");
397 if (name == NULL) {
398 gctl_error(req, "No 'arg%u' argument.", 0);
399 return;
400 }
401 strlcpy(md.md_name, name, sizeof(md.md_name));
402 md.md_mid = arc4random();
403 md.md_all = *nargs - 1;
404 md.md_genid = 0;
405 md.md_syncid = 1;
406 md.md_sync_offset = 0;
407 val = gctl_get_paraml(req, "slice", sizeof(*val));
408 if (val == NULL) {
409 gctl_error(req, "No slice argument.");
410 return;
411 }
412 md.md_slice = *val;
413 sval = gctl_get_asciiparam(req, "balance");
414 if (sval == NULL) {
415 gctl_error(req, "No balance argument.");
416 return;
417 }
418 bal = balance_id(sval);
419 if (bal < 0) {
420 gctl_error(req, "Invalid balance algorithm.");
421 return;
422 }
423 md.md_balance = bal;
424 md.md_mflags = 0;
425 md.md_dflags = 0;
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));
434 md.md_provsize = 0;
435
436 g_topology_lock();
437 mediasize = OFF_MAX;
438 sectorsize = 0;
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 pp = gctl_get_provider(req, param);
445 if (pp == NULL) {
446 err:
447 g_destroy_consumer(cp);
448 g_destroy_geom(gp);
449 g_topology_unlock();
450 return;
451 }
452 if (g_attach(cp, pp) != 0) {
453 G_MIRROR_DEBUG(1, "Can't attach disk %s.", pp->name);
454 gctl_error(req, "Can't attach disk %s.", pp->name);
455 goto err;
456 }
457 if (g_access(cp, 1, 0, 0) != 0) {
458 G_MIRROR_DEBUG(1, "Can't open disk %s.", pp->name);
459 gctl_error(req, "Can't open disk %s.", pp->name);
460 err2:
461 g_detach(cp);
462 goto err;
463 }
464 if (pp->mediasize == 0 || pp->sectorsize == 0) {
465 G_MIRROR_DEBUG(1, "Disk %s has no media.", pp->name);
466 gctl_error(req, "Disk %s has no media.", pp->name);
467 g_access(cp, -1, 0, 0);
468 goto err2;
469 }
470 if (pp->mediasize < mediasize)
471 mediasize = pp->mediasize;
472 if (pp->sectorsize > sectorsize)
473 sectorsize = pp->sectorsize;
474 g_access(cp, -1, 0, 0);
475 g_detach(cp);
476 }
477 g_destroy_consumer(cp);
478 g_destroy_geom(gp);
479 md.md_mediasize = mediasize;
480 md.md_sectorsize = sectorsize;
481 md.md_mediasize -= (md.md_mediasize % md.md_sectorsize);
482
483 gp = g_mirror_create(mp, &md, G_MIRROR_TYPE_MANUAL);
484 if (gp == NULL) {
485 gctl_error(req, "Can't create %s.", md.md_name);
486 g_topology_unlock();
487 return;
488 }
489
490 sc = gp->softc;
491 g_topology_unlock();
492 sx_xlock(&sc->sc_lock);
493 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_TASTING;
494 sb = sbuf_new_auto();
495 sbuf_printf(sb, "Can't attach disk(s) to %s:", gp->name);
496 for (attached = 0, no = 1; no < *nargs; no++) {
497 snprintf(param, sizeof(param), "arg%u", no);
498 pp = gctl_get_provider(req, param);
499 if (pp == NULL) {
500 name = gctl_get_asciiparam(req, param);
501 MPASS(name != NULL);
502 sbuf_printf(sb, " %s", name);
503 continue;
504 }
505 md.md_did = arc4random();
506 md.md_priority = no - 1;
507 if (g_mirror_add_disk(sc, pp, &md) != 0) {
508 G_MIRROR_DEBUG(1, "Disk %u (%s) not attached to %s.",
509 no, pp->name, gp->name);
510 sbuf_printf(sb, " %s", pp->name);
511 continue;
512 }
513 attached++;
514 }
515 sbuf_finish(sb);
516 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_TASTING;
517 if (md.md_all != attached ||
518 (sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0) {
519 g_mirror_destroy(gp->softc, G_MIRROR_DESTROY_HARD);
520 gctl_error(req, "%s", sbuf_data(sb));
521 } else
522 sx_xunlock(&sc->sc_lock);
523 sbuf_delete(sb);
524 }
525
526 static void
527 g_mirror_ctl_rebuild(struct gctl_req *req, struct g_class *mp)
528 {
529 struct g_mirror_metadata md;
530 struct g_mirror_softc *sc;
531 struct g_mirror_disk *disk;
532 struct g_provider *pp;
533 const char *name;
534 char param[16];
535 int error, *nargs;
536 u_int i;
537
538 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
539 if (nargs == NULL) {
540 gctl_error(req, "No '%s' argument.", "nargs");
541 return;
542 }
543 if (*nargs < 2) {
544 gctl_error(req, "Too few arguments.");
545 return;
546 }
547 name = gctl_get_asciiparam(req, "arg0");
548 if (name == NULL) {
549 gctl_error(req, "No 'arg%u' argument.", 0);
550 return;
551 }
552 sc = g_mirror_find_device(mp, name);
553 if (sc == NULL) {
554 gctl_error(req, "No such device: %s.", name);
555 return;
556 }
557 for (i = 1; i < (u_int)*nargs; i++) {
558 snprintf(param, sizeof(param), "arg%u", i);
559 name = gctl_get_asciiparam(req, param);
560 if (name == NULL) {
561 gctl_error(req, "No 'arg%u' argument.", i);
562 continue;
563 }
564 disk = g_mirror_find_disk(sc, name);
565 if (disk == NULL) {
566 gctl_error(req, "No such provider: %s.", name);
567 continue;
568 }
569 if (g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE) == 1 &&
570 disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
571 /*
572 * This is the last active disk. There will be nothing
573 * to rebuild it from, so deny this request.
574 */
575 gctl_error(req,
576 "Provider %s is the last active provider in %s.",
577 name, sc->sc_geom->name);
578 break;
579 }
580 /*
581 * Do rebuild by resetting syncid, disconnecting the disk and
582 * connecting it again.
583 */
584 disk->d_sync.ds_syncid = 0;
585 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOAUTOSYNC) != 0)
586 disk->d_flags |= G_MIRROR_DISK_FLAG_FORCE_SYNC;
587 g_mirror_update_metadata(disk);
588 pp = disk->d_consumer->provider;
589 g_topology_lock();
590 error = g_mirror_read_metadata(disk->d_consumer, &md);
591 g_topology_unlock();
592 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DISCONNECTED,
593 G_MIRROR_EVENT_WAIT);
594 if (error != 0) {
595 gctl_error(req, "Cannot read metadata from %s.",
596 pp->name);
597 continue;
598 }
599 error = g_mirror_add_disk(sc, pp, &md);
600 if (error != 0) {
601 gctl_error(req, "Cannot reconnect component %s.",
602 pp->name);
603 continue;
604 }
605 }
606 sx_xunlock(&sc->sc_lock);
607 }
608
609 static void
610 g_mirror_ctl_insert(struct gctl_req *req, struct g_class *mp)
611 {
612 struct g_mirror_softc *sc;
613 struct g_mirror_disk *disk;
614 struct g_mirror_metadata md;
615 struct g_provider *pp;
616 struct g_consumer *cp;
617 intmax_t *priority;
618 const char *name;
619 char param[16];
620 u_char *sector;
621 u_int i, n;
622 int error, *nargs, *hardcode, *inactive;
623 struct {
624 struct g_provider *provider;
625 struct g_consumer *consumer;
626 } *disks;
627 off_t mdsize;
628
629 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
630 if (nargs == NULL) {
631 gctl_error(req, "No '%s' argument.", "nargs");
632 return;
633 }
634 if (*nargs < 2) {
635 gctl_error(req, "Too few arguments.");
636 return;
637 }
638 priority = gctl_get_paraml(req, "priority", sizeof(*priority));
639 if (priority == NULL) {
640 gctl_error(req, "No '%s' argument.", "priority");
641 return;
642 }
643 inactive = gctl_get_paraml(req, "inactive", sizeof(*inactive));
644 if (inactive == NULL) {
645 gctl_error(req, "No '%s' argument.", "inactive");
646 return;
647 }
648 hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
649 if (hardcode == NULL) {
650 gctl_error(req, "No '%s' argument.", "hardcode");
651 return;
652 }
653 name = gctl_get_asciiparam(req, "arg0");
654 if (name == NULL) {
655 gctl_error(req, "No 'arg%u' argument.", 0);
656 return;
657 }
658 sc = g_mirror_find_launched_device(mp, name, M_WAITOK);
659 if (sc == NULL) {
660 gctl_error(req, "No such device: %s.", name);
661 return;
662 }
663 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
664 gctl_error(req, "Not all disks connected.");
665 sx_xunlock(&sc->sc_lock);
666 return;
667 }
668
669 disks = g_malloc(sizeof(*disks) * (*nargs), M_WAITOK | M_ZERO);
670 g_topology_lock();
671 for (i = 1, n = 0; i < (u_int)*nargs; i++) {
672 snprintf(param, sizeof(param), "arg%u", i);
673 pp = gctl_get_provider(req, param);
674 if (pp == NULL)
675 continue;
676 if (g_mirror_find_disk(sc, pp->name) != NULL) {
677 gctl_error(req, "Provider %s already inserted.", pp->name);
678 continue;
679 }
680 cp = g_new_consumer(sc->sc_geom);
681 if (g_attach(cp, pp) != 0) {
682 g_destroy_consumer(cp);
683 gctl_error(req, "Cannot attach to provider %s.", pp->name);
684 continue;
685 }
686 if (g_access(cp, 0, 1, 1) != 0) {
687 gctl_error(req, "Cannot access provider %s.", pp->name);
688 err:
689 g_detach(cp);
690 g_destroy_consumer(cp);
691 continue;
692 }
693 mdsize = (sc->sc_type == G_MIRROR_TYPE_AUTOMATIC) ?
694 pp->sectorsize : 0;
695 if (sc->sc_provider->mediasize > pp->mediasize - mdsize) {
696 gctl_error(req, "Provider %s too small.", pp->name);
697 err2:
698 g_access(cp, 0, -1, -1);
699 goto err;
700 }
701 if ((sc->sc_provider->sectorsize % pp->sectorsize) != 0) {
702 gctl_error(req, "Invalid sectorsize of provider %s.",
703 pp->name);
704 goto err2;
705 }
706 if (sc->sc_type != G_MIRROR_TYPE_AUTOMATIC) {
707 g_access(cp, 0, -1, -1);
708 g_detach(cp);
709 g_destroy_consumer(cp);
710 g_topology_unlock();
711 sc->sc_ndisks++;
712 g_mirror_fill_metadata(sc, NULL, &md);
713 md.md_priority = *priority;
714 if (*inactive)
715 md.md_dflags |= G_MIRROR_DISK_FLAG_INACTIVE;
716 if (g_mirror_add_disk(sc, pp, &md) != 0) {
717 sc->sc_ndisks--;
718 gctl_error(req, "Disk %s not inserted.", pp->name);
719 }
720 g_topology_lock();
721 continue;
722 }
723 disks[n].provider = pp;
724 disks[n].consumer = cp;
725 n++;
726 }
727 if (n == 0) {
728 g_topology_unlock();
729 sx_xunlock(&sc->sc_lock);
730 g_free(disks);
731 return;
732 }
733 sc->sc_ndisks += n;
734 again:
735 for (i = 0; i < n; i++) {
736 if (disks[i].consumer == NULL)
737 continue;
738 g_mirror_fill_metadata(sc, NULL, &md);
739 md.md_priority = *priority;
740 if (*inactive)
741 md.md_dflags |= G_MIRROR_DISK_FLAG_INACTIVE;
742 pp = disks[i].provider;
743 if (*hardcode) {
744 strlcpy(md.md_provider, pp->name,
745 sizeof(md.md_provider));
746 } else {
747 bzero(md.md_provider, sizeof(md.md_provider));
748 }
749 md.md_provsize = pp->mediasize;
750 sector = g_malloc(pp->sectorsize, M_WAITOK | M_ZERO);
751 mirror_metadata_encode(&md, sector);
752 error = g_write_data(disks[i].consumer,
753 pp->mediasize - pp->sectorsize, sector, pp->sectorsize);
754 g_free(sector);
755 if (error != 0) {
756 gctl_error(req, "Cannot store metadata on %s.",
757 pp->name);
758 g_access(disks[i].consumer, 0, -1, -1);
759 g_detach(disks[i].consumer);
760 g_destroy_consumer(disks[i].consumer);
761 disks[i].consumer = NULL;
762 disks[i].provider = NULL;
763 sc->sc_ndisks--;
764 goto again;
765 }
766 }
767 g_topology_unlock();
768 if (i == 0) {
769 /* All writes failed. */
770 sx_xunlock(&sc->sc_lock);
771 g_free(disks);
772 return;
773 }
774 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
775 g_mirror_update_metadata(disk);
776 }
777 /*
778 * Release provider and wait for retaste.
779 */
780 g_topology_lock();
781 for (i = 0; i < n; i++) {
782 if (disks[i].consumer == NULL)
783 continue;
784 g_access(disks[i].consumer, 0, -1, -1);
785 g_detach(disks[i].consumer);
786 g_destroy_consumer(disks[i].consumer);
787 }
788 g_topology_unlock();
789 sx_xunlock(&sc->sc_lock);
790 g_free(disks);
791 }
792
793 static void
794 g_mirror_ctl_remove(struct gctl_req *req, struct g_class *mp)
795 {
796 struct g_mirror_softc *sc;
797 struct g_mirror_disk *disk;
798 const char *name;
799 char param[16];
800 int *nargs;
801 u_int i, active;
802
803 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
804 if (nargs == NULL) {
805 gctl_error(req, "No '%s' argument.", "nargs");
806 return;
807 }
808 if (*nargs < 2) {
809 gctl_error(req, "Too few arguments.");
810 return;
811 }
812 name = gctl_get_asciiparam(req, "arg0");
813 if (name == NULL) {
814 gctl_error(req, "No 'arg%u' argument.", 0);
815 return;
816 }
817 sc = g_mirror_find_device(mp, name);
818 if (sc == NULL) {
819 gctl_error(req, "No such device: %s.", name);
820 return;
821 }
822 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
823 sx_xunlock(&sc->sc_lock);
824 gctl_error(req, "Not all disks connected. Try 'forget' command "
825 "first.");
826 return;
827 }
828 active = g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE);
829 for (i = 1; i < (u_int)*nargs; i++) {
830 snprintf(param, sizeof(param), "arg%u", i);
831 name = gctl_get_asciiparam(req, param);
832 if (name == NULL) {
833 gctl_error(req, "No 'arg%u' argument.", i);
834 continue;
835 }
836 disk = g_mirror_find_disk(sc, name);
837 if (disk == NULL) {
838 gctl_error(req, "No such provider: %s.", name);
839 continue;
840 }
841 if (disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
842 if (active > 1)
843 active--;
844 else {
845 gctl_error(req, "%s: Can't remove the last "
846 "ACTIVE component %s.", sc->sc_geom->name,
847 name);
848 continue;
849 }
850 }
851 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DESTROY,
852 G_MIRROR_EVENT_DONTWAIT);
853 }
854 sx_xunlock(&sc->sc_lock);
855 }
856
857 static void
858 g_mirror_ctl_resize(struct gctl_req *req, struct g_class *mp)
859 {
860 struct g_mirror_softc *sc;
861 struct g_mirror_disk *disk;
862 uint64_t mediasize;
863 const char *name, *s;
864 char *x;
865 int *nargs;
866
867 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
868 if (nargs == NULL) {
869 gctl_error(req, "No '%s' argument.", "nargs");
870 return;
871 }
872 if (*nargs != 1) {
873 gctl_error(req, "Missing device.");
874 return;
875 }
876 name = gctl_get_asciiparam(req, "arg0");
877 if (name == NULL) {
878 gctl_error(req, "No 'arg%u' argument.", 0);
879 return;
880 }
881 s = gctl_get_asciiparam(req, "size");
882 if (s == NULL) {
883 gctl_error(req, "No '%s' argument.", "size");
884 return;
885 }
886 mediasize = strtouq(s, &x, 0);
887 if (*x != '\0' || mediasize == 0) {
888 gctl_error(req, "Invalid '%s' argument.", "size");
889 return;
890 }
891 sc = g_mirror_find_launched_device(mp, name, M_WAITOK);
892 if (sc == NULL) {
893 gctl_error(req, "No such device: %s.", name);
894 return;
895 }
896 /* Deny shrinking of an opened provider */
897 if ((g_debugflags & G_F_FOOTSHOOTING) == 0 && sc->sc_provider_open > 0) {
898 if (sc->sc_mediasize > mediasize) {
899 gctl_error(req, "Device %s is busy.",
900 sc->sc_provider->name);
901 sx_xunlock(&sc->sc_lock);
902 return;
903 }
904 }
905 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
906 if (mediasize > disk->d_consumer->provider->mediasize -
907 disk->d_consumer->provider->sectorsize) {
908 gctl_error(req, "Provider %s is too small.",
909 disk->d_name);
910 sx_xunlock(&sc->sc_lock);
911 return;
912 }
913 }
914 /* Update the size. */
915 sc->sc_mediasize = mediasize;
916 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
917 g_mirror_update_metadata(disk);
918 }
919 g_topology_lock();
920 g_resize_provider(sc->sc_provider, mediasize);
921 g_topology_unlock();
922 sx_xunlock(&sc->sc_lock);
923 }
924
925 static void
926 g_mirror_ctl_deactivate(struct gctl_req *req, struct g_class *mp)
927 {
928 struct g_mirror_softc *sc;
929 struct g_mirror_disk *disk;
930 const char *name;
931 char param[16];
932 int *nargs;
933 u_int i, active;
934
935 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
936 if (nargs == NULL) {
937 gctl_error(req, "No '%s' argument.", "nargs");
938 return;
939 }
940 if (*nargs < 2) {
941 gctl_error(req, "Too few arguments.");
942 return;
943 }
944 name = gctl_get_asciiparam(req, "arg0");
945 if (name == NULL) {
946 gctl_error(req, "No 'arg%u' argument.", 0);
947 return;
948 }
949 sc = g_mirror_find_device(mp, name);
950 if (sc == NULL) {
951 gctl_error(req, "No such device: %s.", name);
952 return;
953 }
954 active = g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE);
955 for (i = 1; i < (u_int)*nargs; i++) {
956 snprintf(param, sizeof(param), "arg%u", i);
957 name = gctl_get_asciiparam(req, param);
958 if (name == NULL) {
959 gctl_error(req, "No 'arg%u' argument.", i);
960 continue;
961 }
962 disk = g_mirror_find_disk(sc, name);
963 if (disk == NULL) {
964 gctl_error(req, "No such provider: %s.", name);
965 continue;
966 }
967 if (disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
968 if (active > 1)
969 active--;
970 else {
971 gctl_error(req, "%s: Can't deactivate the "
972 "last ACTIVE component %s.",
973 sc->sc_geom->name, name);
974 continue;
975 }
976 }
977 disk->d_flags |= G_MIRROR_DISK_FLAG_INACTIVE;
978 disk->d_flags &= ~G_MIRROR_DISK_FLAG_FORCE_SYNC;
979 g_mirror_update_metadata(disk);
980 sc->sc_bump_id |= G_MIRROR_BUMP_SYNCID;
981 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DISCONNECTED,
982 G_MIRROR_EVENT_DONTWAIT);
983 }
984 sx_xunlock(&sc->sc_lock);
985 }
986
987 static void
988 g_mirror_ctl_forget(struct gctl_req *req, struct g_class *mp)
989 {
990 struct g_mirror_softc *sc;
991 struct g_mirror_disk *disk;
992 const char *name;
993 char param[16];
994 int *nargs;
995 u_int i;
996
997 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
998 if (nargs == NULL) {
999 gctl_error(req, "No '%s' argument.", "nargs");
1000 return;
1001 }
1002 if (*nargs < 1) {
1003 gctl_error(req, "Missing device(s).");
1004 return;
1005 }
1006
1007 for (i = 0; i < (u_int)*nargs; i++) {
1008 snprintf(param, sizeof(param), "arg%u", i);
1009 name = gctl_get_asciiparam(req, param);
1010 if (name == NULL) {
1011 gctl_error(req, "No 'arg%u' argument.", i);
1012 return;
1013 }
1014 sc = g_mirror_find_device(mp, name);
1015 if (sc == NULL) {
1016 gctl_error(req, "No such device: %s.", name);
1017 return;
1018 }
1019 if (g_mirror_ndisks(sc, -1) == sc->sc_ndisks) {
1020 sx_xunlock(&sc->sc_lock);
1021 G_MIRROR_DEBUG(1,
1022 "All disks connected in %s, skipping.",
1023 sc->sc_name);
1024 continue;
1025 }
1026 sc->sc_ndisks = g_mirror_ndisks(sc, -1);
1027 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
1028 g_mirror_update_metadata(disk);
1029 }
1030 sx_xunlock(&sc->sc_lock);
1031 }
1032 }
1033
1034 static void
1035 g_mirror_ctl_stop(struct gctl_req *req, struct g_class *mp, int wipe)
1036 {
1037 struct g_mirror_softc *sc;
1038 int *force, *nargs, error;
1039 const char *name;
1040 char param[16];
1041 u_int i;
1042 int how;
1043
1044 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
1045 if (nargs == NULL) {
1046 gctl_error(req, "No '%s' argument.", "nargs");
1047 return;
1048 }
1049 if (*nargs < 1) {
1050 gctl_error(req, "Missing device(s).");
1051 return;
1052 }
1053 force = gctl_get_paraml(req, "force", sizeof(*force));
1054 if (force == NULL) {
1055 gctl_error(req, "No '%s' argument.", "force");
1056 return;
1057 }
1058 if (*force)
1059 how = G_MIRROR_DESTROY_HARD;
1060 else
1061 how = G_MIRROR_DESTROY_SOFT;
1062
1063 for (i = 0; i < (u_int)*nargs; i++) {
1064 snprintf(param, sizeof(param), "arg%u", i);
1065 name = gctl_get_asciiparam(req, param);
1066 if (name == NULL) {
1067 gctl_error(req, "No 'arg%u' argument.", i);
1068 return;
1069 }
1070 sc = g_mirror_find_device(mp, name);
1071 if (sc == NULL) {
1072 gctl_error(req, "No such device: %s.", name);
1073 return;
1074 }
1075 g_cancel_event(sc);
1076 if (wipe)
1077 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_WIPE;
1078 error = g_mirror_destroy(sc, how);
1079 if (error != 0) {
1080 gctl_error(req, "Cannot destroy device %s (error=%d).",
1081 sc->sc_geom->name, error);
1082 if (wipe)
1083 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_WIPE;
1084 sx_xunlock(&sc->sc_lock);
1085 return;
1086 }
1087 /* No need to unlock, because lock is already dead. */
1088 }
1089 }
1090
1091 void
1092 g_mirror_config(struct gctl_req *req, struct g_class *mp, const char *verb)
1093 {
1094 uint32_t *version;
1095
1096 g_topology_assert();
1097
1098 version = gctl_get_paraml(req, "version", sizeof(*version));
1099 if (version == NULL) {
1100 gctl_error(req, "No '%s' argument.", "version");
1101 return;
1102 }
1103 if (*version != G_MIRROR_VERSION) {
1104 gctl_error(req, "Userland and kernel parts are out of sync.");
1105 return;
1106 }
1107
1108 g_topology_unlock();
1109 if (strcmp(verb, "configure") == 0)
1110 g_mirror_ctl_configure(req, mp);
1111 else if (strcmp(verb, "create") == 0)
1112 g_mirror_ctl_create(req, mp);
1113 else if (strcmp(verb, "rebuild") == 0)
1114 g_mirror_ctl_rebuild(req, mp);
1115 else if (strcmp(verb, "insert") == 0)
1116 g_mirror_ctl_insert(req, mp);
1117 else if (strcmp(verb, "remove") == 0)
1118 g_mirror_ctl_remove(req, mp);
1119 else if (strcmp(verb, "resize") == 0)
1120 g_mirror_ctl_resize(req, mp);
1121 else if (strcmp(verb, "deactivate") == 0)
1122 g_mirror_ctl_deactivate(req, mp);
1123 else if (strcmp(verb, "forget") == 0)
1124 g_mirror_ctl_forget(req, mp);
1125 else if (strcmp(verb, "stop") == 0)
1126 g_mirror_ctl_stop(req, mp, 0);
1127 else if (strcmp(verb, "destroy") == 0)
1128 g_mirror_ctl_stop(req, mp, 1);
1129 else
1130 gctl_error(req, "Unknown verb.");
1131 g_topology_lock();
1132 }
Cache object: 971b9a6cbefb9486c8279d03ce69eb98
|