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 */
51static struct g_mirror_softc *
52g_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)
81static struct g_mirror_softc *
82g_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
128static struct g_mirror_disk *
129g_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
147static void
148g_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
357static void
358g_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
365static void
366g_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) {
446err:
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);
460err2:
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
526static void
527g_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
609static void
610g_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);
688err:
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);
697err2:
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;
734again:
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);
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
793static void
794g_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
857static void
858g_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
925static void
926g_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
987static void
988g_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
1034static void
1035g_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
1091void
1092g_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}
1133