xref: /freebsd/sys/geom/raid3/g_raid3_ctl.c (revision d3d381b2b194b4d24853e92eecef55f262688d1a)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2004-2006 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/kernel.h>
35 #include <sys/module.h>
36 #include <sys/lock.h>
37 #include <sys/mutex.h>
38 #include <sys/bio.h>
39 #include <sys/sysctl.h>
40 #include <sys/malloc.h>
41 #include <sys/bitstring.h>
42 #include <vm/uma.h>
43 #include <machine/atomic.h>
44 #include <geom/geom.h>
45 #include <sys/proc.h>
46 #include <sys/kthread.h>
47 #include <geom/raid3/g_raid3.h>
48 
49 
50 static struct g_raid3_softc *
51 g_raid3_find_device(struct g_class *mp, const char *name)
52 {
53 	struct g_raid3_softc *sc;
54 	struct g_geom *gp;
55 
56 	g_topology_lock();
57 	LIST_FOREACH(gp, &mp->geom, geom) {
58 		sc = gp->softc;
59 		if (sc == NULL)
60 			continue;
61 		if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_DESTROY) != 0)
62 			continue;
63 		if (strcmp(gp->name, name) == 0 ||
64 		    strcmp(sc->sc_name, name) == 0) {
65 			g_topology_unlock();
66 			sx_xlock(&sc->sc_lock);
67 			return (sc);
68 		}
69 	}
70 	g_topology_unlock();
71 	return (NULL);
72 }
73 
74 static struct g_raid3_disk *
75 g_raid3_find_disk(struct g_raid3_softc *sc, const char *name)
76 {
77 	struct g_raid3_disk *disk;
78 	u_int n;
79 
80 	sx_assert(&sc->sc_lock, SX_XLOCKED);
81 	if (strncmp(name, "/dev/", 5) == 0)
82 		name += 5;
83 	for (n = 0; n < sc->sc_ndisks; n++) {
84 		disk = &sc->sc_disks[n];
85 		if (disk->d_state == G_RAID3_DISK_STATE_NODISK)
86 			continue;
87 		if (disk->d_consumer == NULL)
88 			continue;
89 		if (disk->d_consumer->provider == NULL)
90 			continue;
91 		if (strcmp(disk->d_consumer->provider->name, name) == 0)
92 			return (disk);
93 	}
94 	return (NULL);
95 }
96 
97 static void
98 g_raid3_ctl_configure(struct gctl_req *req, struct g_class *mp)
99 {
100 	struct g_raid3_softc *sc;
101 	struct g_raid3_disk *disk;
102 	const char *name;
103 	int *nargs, do_sync = 0, dirty = 1;
104 	int *autosync, *noautosync;
105 	int *failsync, *nofailsync;
106 	int *round_robin, *noround_robin;
107 	int *verify, *noverify;
108 	u_int n;
109 
110 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
111 	if (nargs == NULL) {
112 		gctl_error(req, "No '%s' argument.", "nargs");
113 		return;
114 	}
115 	if (*nargs != 1) {
116 		gctl_error(req, "Invalid number of arguments.");
117 		return;
118 	}
119 	autosync = gctl_get_paraml(req, "autosync", sizeof(*autosync));
120 	if (autosync == NULL) {
121 		gctl_error(req, "No '%s' argument.", "autosync");
122 		return;
123 	}
124 	noautosync = gctl_get_paraml(req, "noautosync", sizeof(*noautosync));
125 	if (noautosync == NULL) {
126 		gctl_error(req, "No '%s' argument.", "noautosync");
127 		return;
128 	}
129 	if (*autosync && *noautosync) {
130 		gctl_error(req, "'%s' and '%s' specified.", "autosync",
131 		    "noautosync");
132 		return;
133 	}
134 	failsync = gctl_get_paraml(req, "failsync", sizeof(*failsync));
135 	if (failsync == NULL) {
136 		gctl_error(req, "No '%s' argument.", "failsync");
137 		return;
138 	}
139 	nofailsync = gctl_get_paraml(req, "nofailsync", sizeof(*nofailsync));
140 	if (nofailsync == NULL) {
141 		gctl_error(req, "No '%s' argument.", "nofailsync");
142 		return;
143 	}
144 	if (*failsync && *nofailsync) {
145 		gctl_error(req, "'%s' and '%s' specified.", "failsync",
146 		    "nofailsync");
147 		return;
148 	}
149 	round_robin = gctl_get_paraml(req, "round_robin", sizeof(*round_robin));
150 	if (round_robin == NULL) {
151 		gctl_error(req, "No '%s' argument.", "round_robin");
152 		return;
153 	}
154 	noround_robin = gctl_get_paraml(req, "noround_robin",
155 	    sizeof(*noround_robin));
156 	if (noround_robin == NULL) {
157 		gctl_error(req, "No '%s' argument.", "noround_robin");
158 		return;
159 	}
160 	if (*round_robin && *noround_robin) {
161 		gctl_error(req, "'%s' and '%s' specified.", "round_robin",
162 		    "noround_robin");
163 		return;
164 	}
165 	verify = gctl_get_paraml(req, "verify", sizeof(*verify));
166 	if (verify == NULL) {
167 		gctl_error(req, "No '%s' argument.", "verify");
168 		return;
169 	}
170 	noverify = gctl_get_paraml(req, "noverify", sizeof(*noverify));
171 	if (noverify == NULL) {
172 		gctl_error(req, "No '%s' argument.", "noverify");
173 		return;
174 	}
175 	if (*verify && *noverify) {
176 		gctl_error(req, "'%s' and '%s' specified.", "verify",
177 		    "noverify");
178 		return;
179 	}
180 	if (!*autosync && !*noautosync && !*failsync && !*nofailsync &&
181 	    !*round_robin && !*noround_robin && !*verify && !*noverify) {
182 		gctl_error(req, "Nothing has changed.");
183 		return;
184 	}
185 	name = gctl_get_asciiparam(req, "arg0");
186 	if (name == NULL) {
187 		gctl_error(req, "No 'arg%u' argument.", 0);
188 		return;
189 	}
190 	sc = g_raid3_find_device(mp, name);
191 	if (sc == NULL) {
192 		gctl_error(req, "No such device: %s.", name);
193 		return;
194 	}
195 	if (g_raid3_ndisks(sc, -1) < sc->sc_ndisks) {
196 		gctl_error(req, "Not all disks connected.");
197 		sx_xunlock(&sc->sc_lock);
198 		return;
199 	}
200 	if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_NOAUTOSYNC) != 0) {
201 		if (*autosync) {
202 			sc->sc_flags &= ~G_RAID3_DEVICE_FLAG_NOAUTOSYNC;
203 			do_sync = 1;
204 		}
205 	} else {
206 		if (*noautosync)
207 			sc->sc_flags |= G_RAID3_DEVICE_FLAG_NOAUTOSYNC;
208 	}
209 	if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_NOFAILSYNC) != 0) {
210 		if (*failsync)
211 			sc->sc_flags &= ~G_RAID3_DEVICE_FLAG_NOFAILSYNC;
212 	} else {
213 		if (*nofailsync) {
214 			sc->sc_flags |= G_RAID3_DEVICE_FLAG_NOFAILSYNC;
215 			dirty = 0;
216 		}
217 	}
218 	if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_VERIFY) != 0) {
219 		if (*noverify)
220 			sc->sc_flags &= ~G_RAID3_DEVICE_FLAG_VERIFY;
221 	} else {
222 		if (*verify)
223 			sc->sc_flags |= G_RAID3_DEVICE_FLAG_VERIFY;
224 	}
225 	if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_ROUND_ROBIN) != 0) {
226 		if (*noround_robin)
227 			sc->sc_flags &= ~G_RAID3_DEVICE_FLAG_ROUND_ROBIN;
228 	} else {
229 		if (*round_robin)
230 			sc->sc_flags |= G_RAID3_DEVICE_FLAG_ROUND_ROBIN;
231 	}
232 	if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_VERIFY) != 0 &&
233 	    (sc->sc_flags & G_RAID3_DEVICE_FLAG_ROUND_ROBIN) != 0) {
234 		/*
235 		 * VERIFY and ROUND-ROBIN options are mutally exclusive.
236 		 */
237 		sc->sc_flags &= ~G_RAID3_DEVICE_FLAG_ROUND_ROBIN;
238 	}
239 	for (n = 0; n < sc->sc_ndisks; n++) {
240 		disk = &sc->sc_disks[n];
241 		if (do_sync) {
242 			if (disk->d_state == G_RAID3_DISK_STATE_SYNCHRONIZING)
243 				disk->d_flags &= ~G_RAID3_DISK_FLAG_FORCE_SYNC;
244 		}
245 		if (!dirty)
246 			disk->d_flags &= ~G_RAID3_DISK_FLAG_DIRTY;
247 		g_raid3_update_metadata(disk);
248 		if (do_sync) {
249 			if (disk->d_state == G_RAID3_DISK_STATE_STALE) {
250 				/*
251 				 * XXX: This is probably possible that this
252 				 *      component will not be retasted.
253 				 */
254 				g_raid3_event_send(disk,
255 				    G_RAID3_DISK_STATE_DISCONNECTED,
256 				    G_RAID3_EVENT_DONTWAIT);
257 			}
258 		}
259 	}
260 	sx_xunlock(&sc->sc_lock);
261 }
262 
263 static void
264 g_raid3_ctl_rebuild(struct gctl_req *req, struct g_class *mp)
265 {
266 	struct g_raid3_metadata md;
267 	struct g_raid3_softc *sc;
268 	struct g_raid3_disk *disk;
269 	struct g_provider *pp;
270 	const char *name;
271 	int error, *nargs;
272 
273 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
274 	if (nargs == NULL) {
275 		gctl_error(req, "No '%s' argument.", "nargs");
276 		return;
277 	}
278 	if (*nargs != 2) {
279 		gctl_error(req, "Invalid number of arguments.");
280 		return;
281 	}
282 	name = gctl_get_asciiparam(req, "arg0");
283 	if (name == NULL) {
284 		gctl_error(req, "No 'arg%u' argument.", 0);
285 		return;
286 	}
287 	sc = g_raid3_find_device(mp, name);
288 	if (sc == NULL) {
289 		gctl_error(req, "No such device: %s.", name);
290 		return;
291 	}
292 	name = gctl_get_asciiparam(req, "arg1");
293 	if (name == NULL) {
294 		gctl_error(req, "No 'arg%u' argument.", 1);
295 		sx_xunlock(&sc->sc_lock);
296 		return;
297 	}
298 	disk = g_raid3_find_disk(sc, name);
299 	if (disk == NULL) {
300 		gctl_error(req, "No such provider: %s.", name);
301 		sx_xunlock(&sc->sc_lock);
302 		return;
303 	}
304 	if (disk->d_state == G_RAID3_DISK_STATE_ACTIVE &&
305 	    g_raid3_ndisks(sc, G_RAID3_DISK_STATE_ACTIVE) < sc->sc_ndisks) {
306 		gctl_error(req, "There is one stale disk already.");
307 		sx_xunlock(&sc->sc_lock);
308 		return;
309 	}
310 	/*
311 	 * Do rebuild by resetting syncid and disconnecting disk.
312 	 * It'll be retasted, connected to the device and synchronized.
313 	 */
314 	disk->d_sync.ds_syncid = 0;
315 	if ((sc->sc_flags & G_RAID3_DEVICE_FLAG_NOAUTOSYNC) != 0)
316 		disk->d_flags |= G_RAID3_DISK_FLAG_FORCE_SYNC;
317 	g_raid3_update_metadata(disk);
318 	pp = disk->d_consumer->provider;
319 	g_topology_lock();
320 	error = g_raid3_read_metadata(disk->d_consumer, &md);
321 	g_topology_unlock();
322 	g_raid3_event_send(disk, G_RAID3_DISK_STATE_DISCONNECTED,
323 	    G_RAID3_EVENT_WAIT);
324 	if (error != 0) {
325 		gctl_error(req, "Cannot read metadata from %s.", pp->name);
326 		sx_xunlock(&sc->sc_lock);
327 		return;
328 	}
329 	error = g_raid3_add_disk(sc, pp, &md);
330 	if (error != 0)
331 		gctl_error(req, "Cannot reconnect component %s.", pp->name);
332 	sx_xunlock(&sc->sc_lock);
333 }
334 
335 static void
336 g_raid3_ctl_stop(struct gctl_req *req, struct g_class *mp)
337 {
338 	struct g_raid3_softc *sc;
339 	int *force, *nargs, error;
340 	const char *name;
341 	char param[16];
342 	u_int i;
343 	int how;
344 
345 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
346 	if (nargs == NULL) {
347 		gctl_error(req, "No '%s' argument.", "nargs");
348 		return;
349 	}
350 	if (*nargs < 1) {
351 		gctl_error(req, "Missing device(s).");
352 		return;
353 	}
354 	force = gctl_get_paraml(req, "force", sizeof(*force));
355 	if (force == NULL) {
356 		gctl_error(req, "No '%s' argument.", "force");
357 		return;
358 	}
359 	if (*force)
360 		how = G_RAID3_DESTROY_HARD;
361 	else
362 		how = G_RAID3_DESTROY_SOFT;
363 
364 	for (i = 0; i < (u_int)*nargs; i++) {
365 		snprintf(param, sizeof(param), "arg%u", i);
366 		name = gctl_get_asciiparam(req, param);
367 		if (name == NULL) {
368 			gctl_error(req, "No 'arg%u' argument.", i);
369 			return;
370 		}
371 		sc = g_raid3_find_device(mp, name);
372 		if (sc == NULL) {
373 			gctl_error(req, "No such device: %s.", name);
374 			return;
375 		}
376 		g_cancel_event(sc);
377 		error = g_raid3_destroy(sc, how);
378 		if (error != 0) {
379 			gctl_error(req, "Cannot destroy device %s (error=%d).",
380 			    sc->sc_geom->name, error);
381 			sx_xunlock(&sc->sc_lock);
382 			return;
383 		}
384 		/* No need to unlock, because lock is already dead. */
385 	}
386 }
387 
388 static void
389 g_raid3_ctl_insert_orphan(struct g_consumer *cp)
390 {
391 
392 	KASSERT(1 == 0, ("%s called while inserting %s.", __func__,
393 	    cp->provider->name));
394 }
395 
396 static void
397 g_raid3_ctl_insert(struct gctl_req *req, struct g_class *mp)
398 {
399 	struct g_raid3_metadata md;
400 	struct g_raid3_softc *sc;
401 	struct g_raid3_disk *disk;
402 	struct g_geom *gp;
403 	struct g_provider *pp;
404 	struct g_consumer *cp;
405 	const char *name;
406 	u_char *sector;
407 	off_t compsize;
408 	intmax_t *no;
409 	int *hardcode, *nargs, error, autono;
410 
411 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
412 	if (nargs == NULL) {
413 		gctl_error(req, "No '%s' argument.", "nargs");
414 		return;
415 	}
416 	if (*nargs != 2) {
417 		gctl_error(req, "Invalid number of arguments.");
418 		return;
419 	}
420 	hardcode = gctl_get_paraml(req, "hardcode", sizeof(*hardcode));
421 	if (hardcode == NULL) {
422 		gctl_error(req, "No '%s' argument.", "hardcode");
423 		return;
424 	}
425 	name = gctl_get_asciiparam(req, "arg1");
426 	if (name == NULL) {
427 		gctl_error(req, "No 'arg%u' argument.", 1);
428 		return;
429 	}
430 	if (gctl_get_param(req, "number", NULL) != NULL)
431 		no = gctl_get_paraml(req, "number", sizeof(*no));
432 	else
433 		no = NULL;
434 	if (strncmp(name, "/dev/", 5) == 0)
435 		name += 5;
436 	g_topology_lock();
437 	pp = g_provider_by_name(name);
438 	if (pp == NULL) {
439 		g_topology_unlock();
440 		gctl_error(req, "Invalid provider.");
441 		return;
442 	}
443 	gp = g_new_geomf(mp, "raid3:insert");
444 	gp->orphan = g_raid3_ctl_insert_orphan;
445 	cp = g_new_consumer(gp);
446 	error = g_attach(cp, pp);
447 	if (error != 0) {
448 		g_topology_unlock();
449 		gctl_error(req, "Cannot attach to %s.", pp->name);
450 		goto end;
451 	}
452 	error = g_access(cp, 0, 1, 1);
453 	if (error != 0) {
454 		g_topology_unlock();
455 		gctl_error(req, "Cannot access %s.", pp->name);
456 		goto end;
457 	}
458 	g_topology_unlock();
459 	name = gctl_get_asciiparam(req, "arg0");
460 	if (name == NULL) {
461 		gctl_error(req, "No 'arg%u' argument.", 0);
462 		goto end;
463 	}
464 	sc = g_raid3_find_device(mp, name);
465 	if (sc == NULL) {
466 		gctl_error(req, "No such device: %s.", name);
467 		goto end;
468 	}
469 	if (no != NULL) {
470 		if (*no < 0 || *no >= sc->sc_ndisks) {
471 			sx_xunlock(&sc->sc_lock);
472 			gctl_error(req, "Invalid component number.");
473 			goto end;
474 		}
475 		disk = &sc->sc_disks[*no];
476 		if (disk->d_state != G_RAID3_DISK_STATE_NODISK) {
477 			sx_xunlock(&sc->sc_lock);
478 			gctl_error(req, "Component %jd is already connected.",
479 			    *no);
480 			goto end;
481 		}
482 	} else {
483 		disk = NULL;
484 		for (autono = 0; autono < sc->sc_ndisks && disk == NULL; autono++)
485 			if (sc->sc_disks[autono].d_state ==
486 			    G_RAID3_DISK_STATE_NODISK)
487 				disk = &sc->sc_disks[autono];
488 		if (disk == NULL) {
489 			sx_xunlock(&sc->sc_lock);
490 			gctl_error(req, "No disconnected components.");
491 			goto end;
492 		}
493 	}
494 	if (((sc->sc_sectorsize / (sc->sc_ndisks - 1)) % pp->sectorsize) != 0) {
495 		sx_xunlock(&sc->sc_lock);
496 		gctl_error(req,
497 		    "Cannot insert provider %s, because of its sector size.",
498 		    pp->name);
499 		goto end;
500 	}
501 	compsize = sc->sc_mediasize / (sc->sc_ndisks - 1);
502 	if (compsize > pp->mediasize - pp->sectorsize) {
503 		sx_xunlock(&sc->sc_lock);
504 		gctl_error(req, "Provider %s too small.", pp->name);
505 		goto end;
506 	}
507 	if (compsize < pp->mediasize - pp->sectorsize) {
508 		gctl_error(req,
509 		    "warning: %s: only %jd bytes from %jd bytes used.",
510 		    pp->name, (intmax_t)compsize,
511 		    (intmax_t)(pp->mediasize - pp->sectorsize));
512 	}
513 	g_raid3_fill_metadata(disk, &md);
514 	sx_xunlock(&sc->sc_lock);
515 	md.md_syncid = 0;
516 	md.md_dflags = 0;
517 	if (*hardcode)
518 		strlcpy(md.md_provider, pp->name, sizeof(md.md_provider));
519 	else
520 		bzero(md.md_provider, sizeof(md.md_provider));
521 	md.md_provsize = pp->mediasize;
522 	sector = g_malloc(pp->sectorsize, M_WAITOK);
523 	raid3_metadata_encode(&md, sector);
524 	error = g_write_data(cp, pp->mediasize - pp->sectorsize, sector,
525 	    pp->sectorsize);
526 	g_free(sector);
527 	if (error != 0)
528 		gctl_error(req, "Cannot store metadata on %s.", pp->name);
529 end:
530 	g_topology_lock();
531 	if (cp->acw > 0)
532 		g_access(cp, 0, -1, -1);
533 	if (cp->provider != NULL)
534 		g_detach(cp);
535 	g_destroy_consumer(cp);
536 	g_destroy_geom(gp);
537 	g_topology_unlock();
538 }
539 
540 static void
541 g_raid3_ctl_remove(struct gctl_req *req, struct g_class *mp)
542 {
543 	struct g_raid3_softc *sc;
544 	struct g_raid3_disk *disk;
545 	const char *name;
546 	intmax_t *no;
547 	int *nargs;
548 
549 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
550 	if (nargs == NULL) {
551 		gctl_error(req, "No '%s' argument.", "nargs");
552 		return;
553 	}
554 	if (*nargs != 1) {
555 		gctl_error(req, "Invalid number of arguments.");
556 		return;
557 	}
558 	no = gctl_get_paraml(req, "number", sizeof(*no));
559 	if (no == NULL) {
560 		gctl_error(req, "No '%s' argument.", "no");
561 		return;
562 	}
563 	name = gctl_get_asciiparam(req, "arg0");
564 	if (name == NULL) {
565 		gctl_error(req, "No 'arg%u' argument.", 0);
566 		return;
567 	}
568 	sc = g_raid3_find_device(mp, name);
569 	if (sc == NULL) {
570 		gctl_error(req, "No such device: %s.", name);
571 		return;
572 	}
573 	if (*no >= sc->sc_ndisks) {
574 		sx_xunlock(&sc->sc_lock);
575 		gctl_error(req, "Invalid component number.");
576 		return;
577 	}
578 	disk = &sc->sc_disks[*no];
579 	switch (disk->d_state) {
580 	case G_RAID3_DISK_STATE_ACTIVE:
581 		/*
582 		 * When replacing ACTIVE component, all the rest has to be also
583 		 * ACTIVE.
584 		 */
585 		if (g_raid3_ndisks(sc, G_RAID3_DISK_STATE_ACTIVE) <
586 		    sc->sc_ndisks) {
587 			gctl_error(req, "Cannot replace component number %jd.",
588 			    *no);
589 			break;
590 		}
591 		/* FALLTHROUGH */
592 	case G_RAID3_DISK_STATE_STALE:
593 	case G_RAID3_DISK_STATE_SYNCHRONIZING:
594 		if (g_raid3_clear_metadata(disk) != 0) {
595 			gctl_error(req, "Cannot clear metadata on %s.",
596 			    g_raid3_get_diskname(disk));
597 		} else {
598 			g_raid3_event_send(disk,
599 			    G_RAID3_DISK_STATE_DISCONNECTED,
600 			    G_RAID3_EVENT_DONTWAIT);
601 		}
602 		break;
603 	case G_RAID3_DISK_STATE_NODISK:
604 		break;
605 	default:
606 		gctl_error(req, "Cannot replace component number %jd.", *no);
607 		break;
608 	}
609 	sx_xunlock(&sc->sc_lock);
610 }
611 
612 void
613 g_raid3_config(struct gctl_req *req, struct g_class *mp, const char *verb)
614 {
615 	uint32_t *version;
616 
617 	g_topology_assert();
618 
619 	version = gctl_get_paraml(req, "version", sizeof(*version));
620 	if (version == NULL) {
621 		gctl_error(req, "No '%s' argument.", "version");
622 		return;
623 	}
624 	if (*version != G_RAID3_VERSION) {
625 		gctl_error(req, "Userland and kernel parts are out of sync.");
626 		return;
627 	}
628 
629 	g_topology_unlock();
630 	if (strcmp(verb, "configure") == 0)
631 		g_raid3_ctl_configure(req, mp);
632 	else if (strcmp(verb, "insert") == 0)
633 		g_raid3_ctl_insert(req, mp);
634 	else if (strcmp(verb, "rebuild") == 0)
635 		g_raid3_ctl_rebuild(req, mp);
636 	else if (strcmp(verb, "remove") == 0)
637 		g_raid3_ctl_remove(req, mp);
638 	else if (strcmp(verb, "stop") == 0)
639 		g_raid3_ctl_stop(req, mp);
640 	else
641 		gctl_error(req, "Unknown verb.");
642 	g_topology_lock();
643 }
644