1 /*-
2 * Copyright (c) 2004-2006 Pawel Jakub Dawidek <pjd@FreeBSD.org>
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 *
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24 * SUCH DAMAGE.
25 */
26
27 #include <sys/cdefs.h>
28 __FBSDID("$FreeBSD: releng/7.3/sys/geom/mirror/g_mirror_ctl.c 163888 2006-11-01 22:51:49Z pjd $");
29
30 #include <sys/param.h>
31 #include <sys/systm.h>
32 #include <sys/kernel.h>
33 #include <sys/module.h>
34 #include <sys/lock.h>
35 #include <sys/mutex.h>
36 #include <sys/bio.h>
37 #include <sys/sysctl.h>
38 #include <sys/malloc.h>
39 #include <sys/bitstring.h>
40 #include <vm/uma.h>
41 #include <machine/atomic.h>
42 #include <geom/geom.h>
43 #include <sys/proc.h>
44 #include <sys/kthread.h>
45 #include <geom/mirror/g_mirror.h>
46
47
48 static struct g_mirror_softc *
49 g_mirror_find_device(struct g_class *mp, const char *name)
50 {
51 struct g_mirror_softc *sc;
52 struct g_geom *gp;
53
54 g_topology_lock();
55 LIST_FOREACH(gp, &mp->geom, geom) {
56 sc = gp->softc;
57 if (sc == NULL)
58 continue;
59 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_DESTROY) != 0)
60 continue;
61 if (strcmp(gp->name, name) == 0 ||
62 strcmp(sc->sc_name, name) == 0) {
63 g_topology_unlock();
64 sx_xlock(&sc->sc_lock);
65 return (sc);
66 }
67 }
68 g_topology_unlock();
69 return (NULL);
70 }
71
72 static struct g_mirror_disk *
73 g_mirror_find_disk(struct g_mirror_softc *sc, const char *name)
74 {
75 struct g_mirror_disk *disk;
76
77 sx_assert(&sc->sc_lock, SX_XLOCKED);
78 if (strncmp(name, "/dev/", 5) == 0)
79 name += 5;
80 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
81 if (disk->d_consumer == NULL)
82 continue;
83 if (disk->d_consumer->provider == NULL)
84 continue;
85 if (strcmp(disk->d_consumer->provider->name, name) == 0)
86 return (disk);
87 }
88 return (NULL);
89 }
90
91 static void
92 g_mirror_ctl_configure(struct gctl_req *req, struct g_class *mp)
93 {
94 struct g_mirror_softc *sc;
95 struct g_mirror_disk *disk;
96 const char *name, *balancep;
97 intmax_t *slicep;
98 uint32_t slice;
99 uint8_t balance;
100 int *autosync, *noautosync, *failsync, *nofailsync, *hardcode, *dynamic;
101 int *nargs, do_sync = 0, dirty = 1;
102
103 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
104 if (nargs == NULL) {
105 gctl_error(req, "No '%s' argument.", "nargs");
106 return;
107 }
108 if (*nargs != 1) {
109 gctl_error(req, "Invalid number of arguments.");
110 return;
111 }
112 name = gctl_get_asciiparam(req, "arg0");
113 if (name == NULL) {
114 gctl_error(req, "No 'arg%u' argument.", 0);
115 return;
116 }
117 balancep = gctl_get_asciiparam(req, "balance");
118 if (balancep == NULL) {
119 gctl_error(req, "No '%s' argument.", "balance");
120 return;
121 }
122 autosync = gctl_get_paraml(req, "autosync", sizeof(*autosync));
123 if (autosync == NULL) {
124 gctl_error(req, "No '%s' argument.", "autosync");
125 return;
126 }
127 noautosync = gctl_get_paraml(req, "noautosync", sizeof(*noautosync));
128 if (noautosync == NULL) {
129 gctl_error(req, "No '%s' argument.", "noautosync");
130 return;
131 }
132 failsync = gctl_get_paraml(req, "failsync", sizeof(*failsync));
133 if (failsync == NULL) {
134 gctl_error(req, "No '%s' argument.", "failsync");
135 return;
136 }
137 nofailsync = gctl_get_paraml(req, "nofailsync", sizeof(*nofailsync));
138 if (nofailsync == NULL) {
139 gctl_error(req, "No '%s' argument.", "nofailsync");
140 return;
141 }
142 hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
143 if (hardcode == NULL) {
144 gctl_error(req, "No '%s' argument.", "hardcode");
145 return;
146 }
147 dynamic = gctl_get_paraml(req, "dynamic", sizeof(*dynamic));
148 if (dynamic == NULL) {
149 gctl_error(req, "No '%s' argument.", "dynamic");
150 return;
151 }
152 if (*autosync && *noautosync) {
153 gctl_error(req, "'%s' and '%s' specified.", "autosync",
154 "noautosync");
155 return;
156 }
157 if (*failsync && *nofailsync) {
158 gctl_error(req, "'%s' and '%s' specified.", "failsync",
159 "nofailsync");
160 return;
161 }
162 if (*hardcode && *dynamic) {
163 gctl_error(req, "'%s' and '%s' specified.", "hardcode",
164 "dynamic");
165 return;
166 }
167 sc = g_mirror_find_device(mp, name);
168 if (sc == NULL) {
169 gctl_error(req, "No such device: %s.", name);
170 return;
171 }
172 if (strcmp(balancep, "none") == 0)
173 balance = sc->sc_balance;
174 else {
175 if (balance_id(balancep) == -1) {
176 gctl_error(req, "Invalid balance algorithm.");
177 sx_xunlock(&sc->sc_lock);
178 return;
179 }
180 balance = balance_id(balancep);
181 }
182 slicep = gctl_get_paraml(req, "slice", sizeof(*slicep));
183 if (slicep == NULL) {
184 gctl_error(req, "No '%s' argument.", "slice");
185 sx_xunlock(&sc->sc_lock);
186 return;
187 }
188 if (*slicep == -1)
189 slice = sc->sc_slice;
190 else
191 slice = *slicep;
192 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
193 sx_xunlock(&sc->sc_lock);
194 gctl_error(req, "Not all disks connected. Try 'forget' command "
195 "first.");
196 return;
197 }
198 if (sc->sc_balance == balance && sc->sc_slice == slice && !*autosync &&
199 !*noautosync && !*failsync && !*nofailsync && !*hardcode &&
200 !*dynamic) {
201 sx_xunlock(&sc->sc_lock);
202 gctl_error(req, "Nothing has changed.");
203 return;
204 }
205 sc->sc_balance = balance;
206 sc->sc_slice = slice;
207 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOAUTOSYNC) != 0) {
208 if (*autosync) {
209 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
210 do_sync = 1;
211 }
212 } else {
213 if (*noautosync)
214 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_NOAUTOSYNC;
215 }
216 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOFAILSYNC) != 0) {
217 if (*failsync)
218 sc->sc_flags &= ~G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
219 } else {
220 if (*nofailsync) {
221 sc->sc_flags |= G_MIRROR_DEVICE_FLAG_NOFAILSYNC;
222 dirty = 0;
223 }
224 }
225 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
226 if (do_sync) {
227 if (disk->d_state == G_MIRROR_DISK_STATE_SYNCHRONIZING)
228 disk->d_flags &= ~G_MIRROR_DISK_FLAG_FORCE_SYNC;
229 }
230 if (*hardcode)
231 disk->d_flags |= G_MIRROR_DISK_FLAG_HARDCODED;
232 else if (*dynamic)
233 disk->d_flags &= ~G_MIRROR_DISK_FLAG_HARDCODED;
234 if (!dirty)
235 disk->d_flags &= ~G_MIRROR_DISK_FLAG_DIRTY;
236 g_mirror_update_metadata(disk);
237 if (do_sync) {
238 if (disk->d_state == G_MIRROR_DISK_STATE_STALE) {
239 g_mirror_event_send(disk,
240 G_MIRROR_DISK_STATE_DISCONNECTED,
241 G_MIRROR_EVENT_DONTWAIT);
242 }
243 }
244 }
245 sx_xunlock(&sc->sc_lock);
246 }
247
248 static void
249 g_mirror_ctl_rebuild(struct gctl_req *req, struct g_class *mp)
250 {
251 struct g_mirror_metadata md;
252 struct g_mirror_softc *sc;
253 struct g_mirror_disk *disk;
254 struct g_provider *pp;
255 const char *name;
256 char param[16];
257 int error, *nargs;
258 u_int i;
259
260 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
261 if (nargs == NULL) {
262 gctl_error(req, "No '%s' argument.", "nargs");
263 return;
264 }
265 if (*nargs < 2) {
266 gctl_error(req, "Too few arguments.");
267 return;
268 }
269 name = gctl_get_asciiparam(req, "arg0");
270 if (name == NULL) {
271 gctl_error(req, "No 'arg%u' argument.", 0);
272 return;
273 }
274 sc = g_mirror_find_device(mp, name);
275 if (sc == NULL) {
276 gctl_error(req, "No such device: %s.", name);
277 return;
278 }
279 for (i = 1; i < (u_int)*nargs; i++) {
280 snprintf(param, sizeof(param), "arg%u", i);
281 name = gctl_get_asciiparam(req, param);
282 if (name == NULL) {
283 gctl_error(req, "No 'arg%u' argument.", i);
284 continue;
285 }
286 disk = g_mirror_find_disk(sc, name);
287 if (disk == NULL) {
288 gctl_error(req, "No such provider: %s.", name);
289 continue;
290 }
291 if (g_mirror_ndisks(sc, G_MIRROR_DISK_STATE_ACTIVE) == 1 &&
292 disk->d_state == G_MIRROR_DISK_STATE_ACTIVE) {
293 /*
294 * This is the last active disk. There will be nothing
295 * to rebuild it from, so deny this request.
296 */
297 gctl_error(req,
298 "Provider %s is the last active provider in %s.",
299 name, sc->sc_geom->name);
300 break;
301 }
302 /*
303 * Do rebuild by resetting syncid, disconnecting the disk and
304 * connecting it again.
305 */
306 disk->d_sync.ds_syncid = 0;
307 if ((sc->sc_flags & G_MIRROR_DEVICE_FLAG_NOAUTOSYNC) != 0)
308 disk->d_flags |= G_MIRROR_DISK_FLAG_FORCE_SYNC;
309 g_mirror_update_metadata(disk);
310 pp = disk->d_consumer->provider;
311 g_topology_lock();
312 error = g_mirror_read_metadata(disk->d_consumer, &md);
313 g_topology_unlock();
314 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DISCONNECTED,
315 G_MIRROR_EVENT_WAIT);
316 if (error != 0) {
317 gctl_error(req, "Cannot read metadata from %s.",
318 pp->name);
319 continue;
320 }
321 error = g_mirror_add_disk(sc, pp, &md);
322 if (error != 0) {
323 gctl_error(req, "Cannot reconnect component %s.",
324 pp->name);
325 continue;
326 }
327 }
328 sx_xunlock(&sc->sc_lock);
329 }
330
331 static void
332 g_mirror_ctl_insert(struct gctl_req *req, struct g_class *mp)
333 {
334 struct g_mirror_softc *sc;
335 struct g_mirror_disk *disk;
336 struct g_mirror_metadata md;
337 struct g_provider *pp;
338 struct g_consumer *cp;
339 intmax_t *priority;
340 const char *name;
341 char param[16];
342 u_char *sector;
343 u_int i, n;
344 int error, *nargs, *hardcode, *inactive;
345 struct {
346 struct g_provider *provider;
347 struct g_consumer *consumer;
348 } *disks;
349
350 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
351 if (nargs == NULL) {
352 gctl_error(req, "No '%s' argument.", "nargs");
353 return;
354 }
355 if (*nargs < 2) {
356 gctl_error(req, "Too few arguments.");
357 return;
358 }
359 priority = gctl_get_paraml(req, "priority", sizeof(*priority));
360 if (priority == NULL) {
361 gctl_error(req, "No '%s' argument.", "priority");
362 return;
363 }
364 inactive = gctl_get_paraml(req, "inactive", sizeof(*inactive));
365 if (inactive == NULL) {
366 gctl_error(req, "No '%s' argument.", "inactive");
367 return;
368 }
369 hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
370 if (hardcode == NULL) {
371 gctl_error(req, "No '%s' argument.", "hardcode");
372 return;
373 }
374 name = gctl_get_asciiparam(req, "arg0");
375 if (name == NULL) {
376 gctl_error(req, "No 'arg%u' argument.", 0);
377 return;
378 }
379 sc = g_mirror_find_device(mp, name);
380 if (sc == NULL) {
381 gctl_error(req, "No such device: %s.", name);
382 return;
383 }
384 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
385 gctl_error(req, "Not all disks connected.");
386 sx_xunlock(&sc->sc_lock);
387 return;
388 }
389
390 disks = g_malloc(sizeof(*disks) * (*nargs), M_WAITOK | M_ZERO);
391 g_topology_lock();
392 for (i = 1, n = 0; i < (u_int)*nargs; i++) {
393 snprintf(param, sizeof(param), "arg%u", i);
394 name = gctl_get_asciiparam(req, param);
395 if (name == NULL) {
396 gctl_error(req, "No 'arg%u' argument.", i);
397 continue;
398 }
399 if (g_mirror_find_disk(sc, name) != NULL) {
400 gctl_error(req, "Provider %s already inserted.", name);
401 continue;
402 }
403 if (strncmp(name, "/dev/", 5) == 0)
404 name += 5;
405 pp = g_provider_by_name(name);
406 if (pp == NULL) {
407 gctl_error(req, "Unknown provider %s.", name);
408 continue;
409 }
410 if (sc->sc_provider->mediasize >
411 pp->mediasize - pp->sectorsize) {
412 gctl_error(req, "Provider %s too small.", name);
413 continue;
414 }
415 if ((sc->sc_provider->sectorsize % pp->sectorsize) != 0) {
416 gctl_error(req, "Invalid sectorsize of provider %s.",
417 name);
418 continue;
419 }
420 cp = g_new_consumer(sc->sc_geom);
421 if (g_attach(cp, pp) != 0) {
422 g_destroy_consumer(cp);
423 gctl_error(req, "Cannot attach to provider %s.", name);
424 continue;
425 }
426 if (g_access(cp, 0, 1, 1) != 0) {
427 g_detach(cp);
428 g_destroy_consumer(cp);
429 gctl_error(req, "Cannot access provider %s.", name);
430 continue;
431 }
432 disks[n].provider = pp;
433 disks[n].consumer = cp;
434 n++;
435 }
436 if (n == 0) {
437 g_topology_unlock();
438 sx_xunlock(&sc->sc_lock);
439 g_free(disks);
440 return;
441 }
442 sc->sc_ndisks += n;
443 again:
444 for (i = 0; i < n; i++) {
445 if (disks[i].consumer == NULL)
446 continue;
447 g_mirror_fill_metadata(sc, NULL, &md);
448 md.md_priority = *priority;
449 if (*inactive)
450 md.md_dflags |= G_MIRROR_DISK_FLAG_INACTIVE;
451 pp = disks[i].provider;
452 if (*hardcode) {
453 strlcpy(md.md_provider, pp->name,
454 sizeof(md.md_provider));
455 } else {
456 bzero(md.md_provider, sizeof(md.md_provider));
457 }
458 md.md_provsize = pp->mediasize;
459 sector = g_malloc(pp->sectorsize, M_WAITOK);
460 mirror_metadata_encode(&md, sector);
461 error = g_write_data(disks[i].consumer,
462 pp->mediasize - pp->sectorsize, sector, pp->sectorsize);
463 g_free(sector);
464 if (error != 0) {
465 gctl_error(req, "Cannot store metadata on %s.",
466 pp->name);
467 g_access(disks[i].consumer, 0, -1, -1);
468 g_detach(disks[i].consumer);
469 g_destroy_consumer(disks[i].consumer);
470 disks[i].consumer = NULL;
471 disks[i].provider = NULL;
472 sc->sc_ndisks--;
473 goto again;
474 }
475 }
476 g_topology_unlock();
477 if (i == 0) {
478 /* All writes failed. */
479 sx_xunlock(&sc->sc_lock);
480 g_free(disks);
481 return;
482 }
483 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
484 g_mirror_update_metadata(disk);
485 }
486 /*
487 * Release provider and wait for retaste.
488 */
489 g_topology_lock();
490 for (i = 0; i < n; i++) {
491 if (disks[i].consumer == NULL)
492 continue;
493 g_access(disks[i].consumer, 0, -1, -1);
494 g_detach(disks[i].consumer);
495 g_destroy_consumer(disks[i].consumer);
496 }
497 g_topology_unlock();
498 sx_xunlock(&sc->sc_lock);
499 g_free(disks);
500 }
501
502 static void
503 g_mirror_ctl_remove(struct gctl_req *req, struct g_class *mp)
504 {
505 struct g_mirror_softc *sc;
506 struct g_mirror_disk *disk;
507 const char *name;
508 char param[16];
509 int *nargs;
510 u_int i;
511
512 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
513 if (nargs == NULL) {
514 gctl_error(req, "No '%s' argument.", "nargs");
515 return;
516 }
517 if (*nargs < 2) {
518 gctl_error(req, "Too few arguments.");
519 return;
520 }
521 name = gctl_get_asciiparam(req, "arg0");
522 if (name == NULL) {
523 gctl_error(req, "No 'arg%u' argument.", 0);
524 return;
525 }
526 sc = g_mirror_find_device(mp, name);
527 if (sc == NULL) {
528 gctl_error(req, "No such device: %s.", name);
529 return;
530 }
531 if (g_mirror_ndisks(sc, -1) < sc->sc_ndisks) {
532 sx_xunlock(&sc->sc_lock);
533 gctl_error(req, "Not all disks connected. Try 'forget' command "
534 "first.");
535 return;
536 }
537 for (i = 1; i < (u_int)*nargs; i++) {
538 snprintf(param, sizeof(param), "arg%u", i);
539 name = gctl_get_asciiparam(req, param);
540 if (name == NULL) {
541 gctl_error(req, "No 'arg%u' argument.", i);
542 continue;
543 }
544 disk = g_mirror_find_disk(sc, name);
545 if (disk == NULL) {
546 gctl_error(req, "No such provider: %s.", name);
547 continue;
548 }
549 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DESTROY,
550 G_MIRROR_EVENT_DONTWAIT);
551 }
552 sx_xunlock(&sc->sc_lock);
553 }
554
555 static void
556 g_mirror_ctl_deactivate(struct gctl_req *req, struct g_class *mp)
557 {
558 struct g_mirror_softc *sc;
559 struct g_mirror_disk *disk;
560 const char *name;
561 char param[16];
562 int *nargs;
563 u_int i;
564
565 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
566 if (nargs == NULL) {
567 gctl_error(req, "No '%s' argument.", "nargs");
568 return;
569 }
570 if (*nargs < 2) {
571 gctl_error(req, "Too few arguments.");
572 return;
573 }
574 name = gctl_get_asciiparam(req, "arg0");
575 if (name == NULL) {
576 gctl_error(req, "No 'arg%u' argument.", 0);
577 return;
578 }
579 sc = g_mirror_find_device(mp, name);
580 if (sc == NULL) {
581 gctl_error(req, "No such device: %s.", name);
582 return;
583 }
584 for (i = 1; i < (u_int)*nargs; i++) {
585 snprintf(param, sizeof(param), "arg%u", i);
586 name = gctl_get_asciiparam(req, param);
587 if (name == NULL) {
588 gctl_error(req, "No 'arg%u' argument.", i);
589 continue;
590 }
591 disk = g_mirror_find_disk(sc, name);
592 if (disk == NULL) {
593 gctl_error(req, "No such provider: %s.", name);
594 continue;
595 }
596 disk->d_flags |= G_MIRROR_DISK_FLAG_INACTIVE;
597 disk->d_flags &= ~G_MIRROR_DISK_FLAG_FORCE_SYNC;
598 g_mirror_update_metadata(disk);
599 sc->sc_bump_id |= G_MIRROR_BUMP_SYNCID;
600 g_mirror_event_send(disk, G_MIRROR_DISK_STATE_DISCONNECTED,
601 G_MIRROR_EVENT_DONTWAIT);
602 }
603 sx_xunlock(&sc->sc_lock);
604 }
605
606 static void
607 g_mirror_ctl_forget(struct gctl_req *req, struct g_class *mp)
608 {
609 struct g_mirror_softc *sc;
610 struct g_mirror_disk *disk;
611 const char *name;
612 char param[16];
613 int *nargs;
614 u_int i;
615
616 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
617 if (nargs == NULL) {
618 gctl_error(req, "No '%s' argument.", "nargs");
619 return;
620 }
621 if (*nargs < 1) {
622 gctl_error(req, "Missing device(s).");
623 return;
624 }
625
626 for (i = 0; i < (u_int)*nargs; i++) {
627 snprintf(param, sizeof(param), "arg%u", i);
628 name = gctl_get_asciiparam(req, param);
629 if (name == NULL) {
630 gctl_error(req, "No 'arg%u' argument.", i);
631 return;
632 }
633 sc = g_mirror_find_device(mp, name);
634 if (sc == NULL) {
635 gctl_error(req, "No such device: %s.", name);
636 return;
637 }
638 if (g_mirror_ndisks(sc, -1) == sc->sc_ndisks) {
639 sx_xunlock(&sc->sc_lock);
640 G_MIRROR_DEBUG(1,
641 "All disks connected in %s, skipping.",
642 sc->sc_name);
643 continue;
644 }
645 sc->sc_ndisks = g_mirror_ndisks(sc, -1);
646 LIST_FOREACH(disk, &sc->sc_disks, d_next) {
647 g_mirror_update_metadata(disk);
648 }
649 sx_xunlock(&sc->sc_lock);
650 }
651 }
652
653 static void
654 g_mirror_ctl_stop(struct gctl_req *req, struct g_class *mp)
655 {
656 struct g_mirror_softc *sc;
657 int *force, *nargs, error;
658 const char *name;
659 char param[16];
660 u_int i;
661 int how;
662
663 nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
664 if (nargs == NULL) {
665 gctl_error(req, "No '%s' argument.", "nargs");
666 return;
667 }
668 if (*nargs < 1) {
669 gctl_error(req, "Missing device(s).");
670 return;
671 }
672 force = gctl_get_paraml(req, "force", sizeof(*force));
673 if (force == NULL) {
674 gctl_error(req, "No '%s' argument.", "force");
675 return;
676 }
677 if (*force)
678 how = G_MIRROR_DESTROY_HARD;
679 else
680 how = G_MIRROR_DESTROY_SOFT;
681
682 for (i = 0; i < (u_int)*nargs; i++) {
683 snprintf(param, sizeof(param), "arg%u", i);
684 name = gctl_get_asciiparam(req, param);
685 if (name == NULL) {
686 gctl_error(req, "No 'arg%u' argument.", i);
687 return;
688 }
689 sc = g_mirror_find_device(mp, name);
690 if (sc == NULL) {
691 gctl_error(req, "No such device: %s.", name);
692 return;
693 }
694 g_cancel_event(sc);
695 error = g_mirror_destroy(sc, how);
696 if (error != 0) {
697 gctl_error(req, "Cannot destroy device %s (error=%d).",
698 sc->sc_geom->name, error);
699 sx_xunlock(&sc->sc_lock);
700 return;
701 }
702 /* No need to unlock, because lock is already dead. */
703 }
704 }
705
706 void
707 g_mirror_config(struct gctl_req *req, struct g_class *mp, const char *verb)
708 {
709 uint32_t *version;
710
711 g_topology_assert();
712
713 version = gctl_get_paraml(req, "version", sizeof(*version));
714 if (version == NULL) {
715 gctl_error(req, "No '%s' argument.", "version");
716 return;
717 }
718 if (*version != G_MIRROR_VERSION) {
719 gctl_error(req, "Userland and kernel parts are out of sync.");
720 return;
721 }
722
723 g_topology_unlock();
724 if (strcmp(verb, "configure") == 0)
725 g_mirror_ctl_configure(req, mp);
726 else if (strcmp(verb, "rebuild") == 0)
727 g_mirror_ctl_rebuild(req, mp);
728 else if (strcmp(verb, "insert") == 0)
729 g_mirror_ctl_insert(req, mp);
730 else if (strcmp(verb, "remove") == 0)
731 g_mirror_ctl_remove(req, mp);
732 else if (strcmp(verb, "deactivate") == 0)
733 g_mirror_ctl_deactivate(req, mp);
734 else if (strcmp(verb, "forget") == 0)
735 g_mirror_ctl_forget(req, mp);
736 else if (strcmp(verb, "stop") == 0)
737 g_mirror_ctl_stop(req, mp);
738 else
739 gctl_error(req, "Unknown verb.");
740 g_topology_lock();
741 }
Cache object: 6a1191fe320ccc9659efe87142207afd
|