xref: /freebsd/sys/dev/sound/pcm/sndstat.c (revision 214e3e09b3381e44bf5d9c1dcd19c4b1b923a796)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2005-2009 Ariff Abdullah <ariff@FreeBSD.org>
5  * Copyright (c) 2001 Cameron Grant <cg@FreeBSD.org>
6  * Copyright (c) 2020 The FreeBSD Foundation
7  * All rights reserved.
8  *
9  * Portions of this software were developed by Ka Ho Ng
10  * under sponsorship from the FreeBSD Foundation.
11  *
12  * Redistribution and use in source and binary forms, with or without
13  * modification, are permitted provided that the following conditions
14  * are met:
15  * 1. Redistributions of source code must retain the above copyright
16  *    notice, this list of conditions and the following disclaimer.
17  * 2. Redistributions in binary form must reproduce the above copyright
18  *    notice, this list of conditions and the following disclaimer in the
19  *    documentation and/or other materials provided with the distribution.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
22  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
25  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31  * SUCH DAMAGE.
32  */
33 
34 #ifdef HAVE_KERNEL_OPTION_HEADERS
35 #include "opt_snd.h"
36 #endif
37 
38 #include <sys/param.h>
39 #include <sys/lock.h>
40 #include <sys/malloc.h>
41 #include <sys/nv.h>
42 #include <sys/dnv.h>
43 #include <sys/sx.h>
44 #ifdef COMPAT_FREEBSD32
45 #include <sys/sysent.h>
46 #endif
47 
48 #include <dev/sound/pcm/sound.h>
49 #include <dev/sound/pcm/pcm.h>
50 #include <dev/sound/version.h>
51 
52 
53 SND_DECLARE_FILE("");
54 
55 #define	SS_TYPE_MODULE		0
56 #define	SS_TYPE_PCM		1
57 #define	SS_TYPE_MIDI		2
58 #define	SS_TYPE_SEQUENCER	3
59 
60 static d_open_t sndstat_open;
61 static void sndstat_close(void *);
62 static d_read_t sndstat_read;
63 static d_write_t sndstat_write;
64 static d_ioctl_t sndstat_ioctl;
65 
66 static struct cdevsw sndstat_cdevsw = {
67 	.d_version =	D_VERSION,
68 	.d_open =	sndstat_open,
69 	.d_read =	sndstat_read,
70 	.d_write =	sndstat_write,
71 	.d_ioctl =	sndstat_ioctl,
72 	.d_name =	"sndstat",
73 	.d_flags =	D_TRACKCLOSE,
74 };
75 
76 struct sndstat_entry {
77 	TAILQ_ENTRY(sndstat_entry) link;
78 	device_t dev;
79 	char *str;
80 	sndstat_handler handler;
81 	int type, unit;
82 };
83 
84 struct sndstat_userdev {
85 	TAILQ_ENTRY(sndstat_userdev) link;
86 	char *provider;
87 	char *nameunit;
88 	char *devnode;
89 	char *desc;
90 	unsigned int pchan;
91 	unsigned int rchan;
92 	struct {
93 		uint32_t min_rate;
94 		uint32_t max_rate;
95 		uint32_t formats;
96 		uint32_t min_chn;
97 		uint32_t max_chn;
98 	} info_play, info_rec;
99 	nvlist_t *provider_nvl;
100 };
101 
102 struct sndstat_file {
103 	TAILQ_ENTRY(sndstat_file) entry;
104 	struct sbuf sbuf;
105 	struct sx lock;
106 	void *devs_nvlbuf;	/* (l) */
107 	size_t devs_nbytes;	/* (l) */
108 	TAILQ_HEAD(, sndstat_userdev) userdev_list;	/* (l) */
109 	int out_offset;
110   	int in_offset;
111 	int fflags;
112 };
113 
114 static struct sx sndstat_lock;
115 static struct cdev *sndstat_dev;
116 
117 #define	SNDSTAT_LOCK() sx_xlock(&sndstat_lock)
118 #define	SNDSTAT_UNLOCK() sx_xunlock(&sndstat_lock)
119 
120 static TAILQ_HEAD(, sndstat_entry) sndstat_devlist = TAILQ_HEAD_INITIALIZER(sndstat_devlist);
121 static TAILQ_HEAD(, sndstat_file) sndstat_filelist = TAILQ_HEAD_INITIALIZER(sndstat_filelist);
122 
123 int snd_verbose = 0;
124 
125 static int sndstat_prepare(struct sndstat_file *);
126 static struct sndstat_userdev *
127 sndstat_line2userdev(struct sndstat_file *, const char *, int);
128 
129 static int
130 sysctl_hw_sndverbose(SYSCTL_HANDLER_ARGS)
131 {
132 	int error, verbose;
133 
134 	verbose = snd_verbose;
135 	error = sysctl_handle_int(oidp, &verbose, 0, req);
136 	if (error == 0 && req->newptr != NULL) {
137 		if (verbose < 0 || verbose > 4)
138 			error = EINVAL;
139 		else
140 			snd_verbose = verbose;
141 	}
142 	return (error);
143 }
144 SYSCTL_PROC(_hw_snd, OID_AUTO, verbose,
145     CTLTYPE_INT | CTLFLAG_RWTUN | CTLFLAG_MPSAFE, 0, sizeof(int),
146     sysctl_hw_sndverbose, "I",
147     "verbosity level");
148 
149 static int
150 sndstat_open(struct cdev *i_dev, int flags, int mode, struct thread *td)
151 {
152 	struct sndstat_file *pf;
153 
154 	pf = malloc(sizeof(*pf), M_DEVBUF, M_WAITOK | M_ZERO);
155 
156 	if (sbuf_new(&pf->sbuf, NULL, 4096, SBUF_AUTOEXTEND) == NULL) {
157 		free(pf, M_DEVBUF);
158 		return (ENOMEM);
159 	}
160 
161 	pf->fflags = flags;
162 	TAILQ_INIT(&pf->userdev_list);
163 	sx_init(&pf->lock, "sndstat_file");
164 
165 	SNDSTAT_LOCK();
166 	TAILQ_INSERT_TAIL(&sndstat_filelist, pf, entry);
167 	SNDSTAT_UNLOCK();
168 
169 	devfs_set_cdevpriv(pf, &sndstat_close);
170 
171 	return (0);
172 }
173 
174 /*
175  * Should only be called either when:
176  * * Closing
177  * * pf->lock held
178  */
179 static void
180 sndstat_remove_all_userdevs(struct sndstat_file *pf)
181 {
182 	struct sndstat_userdev *ud;
183 
184 	KASSERT(
185 	    sx_xlocked(&pf->lock), ("%s: Called without pf->lock", __func__));
186 	while ((ud = TAILQ_FIRST(&pf->userdev_list)) != NULL) {
187 		TAILQ_REMOVE(&pf->userdev_list, ud, link);
188 		free(ud->provider, M_DEVBUF);
189 		free(ud->desc, M_DEVBUF);
190 		free(ud->devnode, M_DEVBUF);
191 		free(ud->nameunit, M_DEVBUF);
192 		nvlist_destroy(ud->provider_nvl);
193 		free(ud, M_DEVBUF);
194 	}
195 }
196 
197 static void
198 sndstat_close(void *sndstat_file)
199 {
200 	struct sndstat_file *pf = (struct sndstat_file *)sndstat_file;
201 
202 	SNDSTAT_LOCK();
203 	sbuf_delete(&pf->sbuf);
204 	TAILQ_REMOVE(&sndstat_filelist, pf, entry);
205 	SNDSTAT_UNLOCK();
206 
207 	free(pf->devs_nvlbuf, M_NVLIST);
208 	sx_xlock(&pf->lock);
209 	sndstat_remove_all_userdevs(pf);
210 	sx_xunlock(&pf->lock);
211 	sx_destroy(&pf->lock);
212 
213 	free(pf, M_DEVBUF);
214 }
215 
216 static int
217 sndstat_read(struct cdev *i_dev, struct uio *buf, int flag)
218 {
219 	struct sndstat_file *pf;
220 	int err;
221 	int len;
222 
223 	err = devfs_get_cdevpriv((void **)&pf);
224 	if (err != 0)
225 		return (err);
226 
227 	/* skip zero-length reads */
228 	if (buf->uio_resid == 0)
229 		return (0);
230 
231 	SNDSTAT_LOCK();
232 	if (pf->out_offset != 0) {
233 		/* don't allow both reading and writing */
234 		err = EINVAL;
235 		goto done;
236 	} else if (pf->in_offset == 0) {
237 		err = sndstat_prepare(pf);
238 		if (err <= 0) {
239 			err = ENOMEM;
240 			goto done;
241 		}
242 	}
243 	len = sbuf_len(&pf->sbuf) - pf->in_offset;
244 	if (len > buf->uio_resid)
245 		len = buf->uio_resid;
246 	if (len > 0)
247 		err = uiomove(sbuf_data(&pf->sbuf) + pf->in_offset, len, buf);
248 	pf->in_offset += len;
249 done:
250 	SNDSTAT_UNLOCK();
251 	return (err);
252 }
253 
254 static int
255 sndstat_write(struct cdev *i_dev, struct uio *buf, int flag)
256 {
257 	struct sndstat_file *pf;
258 	uint8_t temp[64];
259 	int err;
260 	int len;
261 
262 	err = devfs_get_cdevpriv((void **)&pf);
263 	if (err != 0)
264 		return (err);
265 
266 	/* skip zero-length writes */
267 	if (buf->uio_resid == 0)
268 		return (0);
269 
270 	/* don't allow writing more than 64Kbytes */
271 	if (buf->uio_resid > 65536)
272 		return (ENOMEM);
273 
274 	SNDSTAT_LOCK();
275 	if (pf->in_offset != 0) {
276 		/* don't allow both reading and writing */
277 		err = EINVAL;
278 	} else {
279 		/* only remember the last write - allows for updates */
280 		sx_xlock(&pf->lock);
281 		sndstat_remove_all_userdevs(pf);
282 		sx_xunlock(&pf->lock);
283 
284 		while (1) {
285 			len = sizeof(temp);
286 			if (len > buf->uio_resid)
287 				len = buf->uio_resid;
288 			if (len > 0) {
289 				err = uiomove(temp, len, buf);
290 				if (err)
291 					break;
292 			} else {
293 				break;
294 			}
295 			if (sbuf_bcat(&pf->sbuf, temp, len) < 0) {
296 				err = ENOMEM;
297 				break;
298 			}
299 		}
300 		sbuf_finish(&pf->sbuf);
301 
302 		if (err == 0) {
303 			char *line, *str;
304 
305 			str = sbuf_data(&pf->sbuf);
306 			while ((line = strsep(&str, "\n")) != NULL) {
307 				struct sndstat_userdev *ud;
308 
309 				ud = sndstat_line2userdev(pf, line, strlen(line));
310 				if (ud == NULL)
311 					continue;
312 
313 				sx_xlock(&pf->lock);
314 				TAILQ_INSERT_TAIL(&pf->userdev_list, ud, link);
315 				sx_xunlock(&pf->lock);
316 			}
317 
318 			pf->out_offset = sbuf_len(&pf->sbuf);
319 		} else
320 			pf->out_offset = 0;
321 
322 		sbuf_clear(&pf->sbuf);
323 	}
324 	SNDSTAT_UNLOCK();
325 	return (err);
326 }
327 
328 static void
329 sndstat_get_caps(struct snddev_info *d, bool play, uint32_t *min_rate,
330     uint32_t *max_rate, uint32_t *fmts, uint32_t *minchn, uint32_t *maxchn)
331 {
332 	struct pcm_channel *c;
333 	unsigned int encoding;
334 	int dir;
335 
336 	dir = play ? PCMDIR_PLAY : PCMDIR_REC;
337 
338 	if (play && d->pvchancount > 0) {
339 		*min_rate = *max_rate = d->pvchanrate;
340 		*fmts = AFMT_ENCODING(d->pvchanformat);
341 		*minchn = *maxchn = AFMT_CHANNEL(d->pvchanformat);
342 		return;
343 	} else if (!play && d->rvchancount > 0) {
344 		*min_rate = *max_rate = d->rvchanrate;
345 		*fmts = AFMT_ENCODING(d->rvchanformat);
346 		*minchn = *maxchn = AFMT_CHANNEL(d->rvchanformat);
347 		return;
348 	}
349 
350 	*min_rate = UINT32_MAX;
351 	*max_rate = 0;
352 	*minchn = UINT32_MAX;
353 	*maxchn = 0;
354 	encoding = 0;
355 	CHN_FOREACH(c, d, channels.pcm) {
356 		struct pcmchan_caps *caps;
357 		int i;
358 
359 		if (c->direction != dir || (c->flags & CHN_F_VIRTUAL) != 0)
360 			continue;
361 
362 		CHN_LOCK(c);
363 		caps = chn_getcaps(c);
364 		*min_rate = min(caps->minspeed, *min_rate);
365 		*max_rate = max(caps->maxspeed, *max_rate);
366 		for (i = 0; caps->fmtlist[i]; i++) {
367 			encoding |= AFMT_ENCODING(caps->fmtlist[i]);
368 			*minchn = min(AFMT_CHANNEL(encoding), *minchn);
369 			*maxchn = max(AFMT_CHANNEL(encoding), *maxchn);
370 		}
371 		CHN_UNLOCK(c);
372 	}
373 	if (*min_rate == UINT32_MAX)
374 		*min_rate = 0;
375 	if (*minchn == UINT32_MAX)
376 		*minchn = 0;
377 }
378 
379 static nvlist_t *
380 sndstat_create_diinfo_nv(uint32_t min_rate, uint32_t max_rate, uint32_t formats,
381 	    uint32_t min_chn, uint32_t max_chn)
382 {
383 	nvlist_t *nv;
384 
385 	nv = nvlist_create(0);
386 	if (nv == NULL)
387 		return (NULL);
388 	nvlist_add_number(nv, SNDST_DSPS_INFO_MIN_RATE, min_rate);
389 	nvlist_add_number(nv, SNDST_DSPS_INFO_MAX_RATE, max_rate);
390 	nvlist_add_number(nv, SNDST_DSPS_INFO_FORMATS, formats);
391 	nvlist_add_number(nv, SNDST_DSPS_INFO_MIN_CHN, min_chn);
392 	nvlist_add_number(nv, SNDST_DSPS_INFO_MAX_CHN, max_chn);
393 	return (nv);
394 }
395 
396 static int
397 sndstat_build_sound4_nvlist(struct snddev_info *d, nvlist_t **dip)
398 {
399 	uint32_t maxrate, minrate, fmts, minchn, maxchn;
400 	nvlist_t *di = NULL, *sound4di = NULL, *diinfo = NULL;
401 	int err;
402 
403 	di = nvlist_create(0);
404 	if (di == NULL) {
405 		err = ENOMEM;
406 		goto done;
407 	}
408 	sound4di = nvlist_create(0);
409 	if (sound4di == NULL) {
410 		err = ENOMEM;
411 		goto done;
412 	}
413 
414 	nvlist_add_bool(di, SNDST_DSPS_FROM_USER, false);
415 	nvlist_add_stringf(di, SNDST_DSPS_NAMEUNIT, "%s",
416 			device_get_nameunit(d->dev));
417 	nvlist_add_stringf(di, SNDST_DSPS_DEVNODE, "dsp%d",
418 			device_get_unit(d->dev));
419 	nvlist_add_string(
420 			di, SNDST_DSPS_DESC, device_get_desc(d->dev));
421 
422 	PCM_ACQUIRE_QUICK(d);
423 	nvlist_add_number(di, SNDST_DSPS_PCHAN, d->playcount);
424 	nvlist_add_number(di, SNDST_DSPS_RCHAN, d->reccount);
425 	if (d->playcount > 0) {
426 		sndstat_get_caps(d, true, &minrate, &maxrate, &fmts, &minchn,
427 		    &maxchn);
428 		nvlist_add_number(di, "pminrate", minrate);
429 		nvlist_add_number(di, "pmaxrate", maxrate);
430 		nvlist_add_number(di, "pfmts", fmts);
431 		diinfo = sndstat_create_diinfo_nv(minrate, maxrate, fmts,
432 		    minchn, maxchn);
433 		if (diinfo == NULL)
434 			nvlist_set_error(di, ENOMEM);
435 		else
436 			nvlist_move_nvlist(di, SNDST_DSPS_INFO_PLAY, diinfo);
437 	}
438 	if (d->reccount > 0) {
439 		sndstat_get_caps(d, false, &minrate, &maxrate, &fmts, &minchn,
440 		    &maxchn);
441 		nvlist_add_number(di, "rminrate", minrate);
442 		nvlist_add_number(di, "rmaxrate", maxrate);
443 		nvlist_add_number(di, "rfmts", fmts);
444 		diinfo = sndstat_create_diinfo_nv(minrate, maxrate, fmts,
445 		    minchn, maxchn);
446 		if (diinfo == NULL)
447 			nvlist_set_error(di, ENOMEM);
448 		else
449 			nvlist_move_nvlist(di, SNDST_DSPS_INFO_REC, diinfo);
450 	}
451 
452 	nvlist_add_number(sound4di, SNDST_DSPS_SOUND4_UNIT,
453 			device_get_unit(d->dev)); // XXX: I want signed integer here
454 	nvlist_add_bool(
455 	    sound4di, SNDST_DSPS_SOUND4_BITPERFECT, d->flags & SD_F_BITPERFECT);
456 	nvlist_add_number(sound4di, SNDST_DSPS_SOUND4_PVCHAN, d->pvchancount);
457 	nvlist_add_number(sound4di, SNDST_DSPS_SOUND4_RVCHAN, d->rvchancount);
458 	nvlist_move_nvlist(di, SNDST_DSPS_PROVIDER_INFO, sound4di);
459 	sound4di = NULL;
460 	PCM_RELEASE_QUICK(d);
461 	nvlist_add_string(di, SNDST_DSPS_PROVIDER, SNDST_DSPS_SOUND4_PROVIDER);
462 
463 	err = nvlist_error(di);
464 	if (err)
465 		goto done;
466 
467 	*dip = di;
468 
469 done:
470 	if (err) {
471 		nvlist_destroy(sound4di);
472 		nvlist_destroy(di);
473 	}
474 	return (err);
475 }
476 
477 static int
478 sndstat_build_userland_nvlist(struct sndstat_userdev *ud, nvlist_t **dip)
479 {
480 	nvlist_t *di, *diinfo;
481 	int err;
482 
483 	di = nvlist_create(0);
484 	if (di == NULL) {
485 		err = ENOMEM;
486 		goto done;
487 	}
488 
489 	nvlist_add_bool(di, SNDST_DSPS_FROM_USER, true);
490 	nvlist_add_number(di, SNDST_DSPS_PCHAN, ud->pchan);
491 	nvlist_add_number(di, SNDST_DSPS_RCHAN, ud->rchan);
492 	nvlist_add_string(di, SNDST_DSPS_NAMEUNIT, ud->nameunit);
493 	nvlist_add_string(
494 			di, SNDST_DSPS_DEVNODE, ud->devnode);
495 	nvlist_add_string(di, SNDST_DSPS_DESC, ud->desc);
496 	if (ud->pchan != 0) {
497 		nvlist_add_number(di, "pminrate",
498 		    ud->info_play.min_rate);
499 		nvlist_add_number(di, "pmaxrate",
500 		    ud->info_play.max_rate);
501 		nvlist_add_number(di, "pfmts",
502 		    ud->info_play.formats);
503 		diinfo = sndstat_create_diinfo_nv(ud->info_play.min_rate,
504 		    ud->info_play.max_rate, ud->info_play.formats,
505 		    ud->info_play.min_chn, ud->info_play.max_chn);
506 		if (diinfo == NULL)
507 			nvlist_set_error(di, ENOMEM);
508 		else
509 			nvlist_move_nvlist(di, SNDST_DSPS_INFO_PLAY, diinfo);
510 	}
511 	if (ud->rchan != 0) {
512 		nvlist_add_number(di, "rminrate",
513 		    ud->info_rec.min_rate);
514 		nvlist_add_number(di, "rmaxrate",
515 		    ud->info_rec.max_rate);
516 		nvlist_add_number(di, "rfmts",
517 		    ud->info_rec.formats);
518 		diinfo = sndstat_create_diinfo_nv(ud->info_rec.min_rate,
519 		    ud->info_rec.max_rate, ud->info_rec.formats,
520 		    ud->info_rec.min_chn, ud->info_rec.max_chn);
521 		if (diinfo == NULL)
522 			nvlist_set_error(di, ENOMEM);
523 		else
524 			nvlist_move_nvlist(di, SNDST_DSPS_INFO_REC, diinfo);
525 	}
526 	nvlist_add_string(di, SNDST_DSPS_PROVIDER,
527 	    (ud->provider != NULL) ? ud->provider : "");
528 	if (ud->provider_nvl != NULL)
529 		nvlist_add_nvlist(
530 		    di, SNDST_DSPS_PROVIDER_INFO, ud->provider_nvl);
531 
532 	err = nvlist_error(di);
533 	if (err)
534 		goto done;
535 
536 	*dip = di;
537 
538 done:
539 	if (err)
540 		nvlist_destroy(di);
541 	return (err);
542 }
543 
544 /*
545  * Should only be called with the following locks held:
546  * * sndstat_lock
547  */
548 static int
549 sndstat_create_devs_nvlist(nvlist_t **nvlp)
550 {
551 	int err;
552 	nvlist_t *nvl;
553 	struct sndstat_entry *ent;
554 	struct sndstat_file *pf;
555 
556 	nvl = nvlist_create(0);
557 	if (nvl == NULL)
558 		return (ENOMEM);
559 
560 	TAILQ_FOREACH(ent, &sndstat_devlist, link) {
561 		struct snddev_info *d;
562 		nvlist_t *di;
563 
564 		if (ent->dev == NULL)
565 			continue;
566 		d = device_get_softc(ent->dev);
567 		if (!PCM_REGISTERED(d))
568 			continue;
569 
570 		err = sndstat_build_sound4_nvlist(d, &di);
571 		if (err)
572 			goto done;
573 
574 		nvlist_append_nvlist_array(nvl, SNDST_DSPS, di);
575 		nvlist_destroy(di);
576 		err = nvlist_error(nvl);
577 		if (err)
578 			goto done;
579 	}
580 
581 	TAILQ_FOREACH(pf, &sndstat_filelist, entry) {
582 		struct sndstat_userdev *ud;
583 
584 		sx_xlock(&pf->lock);
585 
586 		TAILQ_FOREACH(ud, &pf->userdev_list, link) {
587 			nvlist_t *di;
588 
589 			err = sndstat_build_userland_nvlist(ud, &di);
590 			if (err != 0) {
591 				sx_xunlock(&pf->lock);
592 				goto done;
593 			}
594 			nvlist_append_nvlist_array(nvl, SNDST_DSPS, di);
595 			nvlist_destroy(di);
596 
597 			err = nvlist_error(nvl);
598 			if (err != 0) {
599 				sx_xunlock(&pf->lock);
600 				goto done;
601 			}
602 		}
603 
604 		sx_xunlock(&pf->lock);
605 	}
606 
607 	*nvlp = nvl;
608 
609 done:
610 	if (err != 0)
611 		nvlist_destroy(nvl);
612 	return (err);
613 }
614 
615 static int
616 sndstat_refresh_devs(struct sndstat_file *pf)
617 {
618 	sx_xlock(&pf->lock);
619 	free(pf->devs_nvlbuf, M_NVLIST);
620 	pf->devs_nvlbuf = NULL;
621 	pf->devs_nbytes = 0;
622 	sx_unlock(&pf->lock);
623 
624 	return (0);
625 }
626 
627 static int
628 sndstat_get_devs(struct sndstat_file *pf, caddr_t data)
629 {
630 	int err;
631 	struct sndstioc_nv_arg *arg = (struct sndstioc_nv_arg *)data;
632 
633 	SNDSTAT_LOCK();
634 	sx_xlock(&pf->lock);
635 
636 	if (pf->devs_nvlbuf == NULL) {
637 		nvlist_t *nvl;
638 		void *nvlbuf;
639 		size_t nbytes;
640 		int err;
641 
642 		sx_xunlock(&pf->lock);
643 
644 		err = sndstat_create_devs_nvlist(&nvl);
645 		if (err) {
646 			SNDSTAT_UNLOCK();
647 			return (err);
648 		}
649 
650 		sx_xlock(&pf->lock);
651 
652 		nvlbuf = nvlist_pack(nvl, &nbytes);
653 		err = nvlist_error(nvl);
654 		nvlist_destroy(nvl);
655 		if (nvlbuf == NULL || err != 0) {
656 			SNDSTAT_UNLOCK();
657 			sx_xunlock(&pf->lock);
658 			if (err == 0)
659 				return (ENOMEM);
660 			return (err);
661 		}
662 
663 		free(pf->devs_nvlbuf, M_NVLIST);
664 		pf->devs_nvlbuf = nvlbuf;
665 		pf->devs_nbytes = nbytes;
666 	}
667 
668 	SNDSTAT_UNLOCK();
669 
670 	if (!arg->nbytes) {
671 		arg->nbytes = pf->devs_nbytes;
672 		err = 0;
673 		goto done;
674 	}
675 	if (arg->nbytes < pf->devs_nbytes) {
676 		arg->nbytes = 0;
677 		err = 0;
678 		goto done;
679 	}
680 
681 	err = copyout(pf->devs_nvlbuf, arg->buf, pf->devs_nbytes);
682 	if (err)
683 		goto done;
684 
685 	arg->nbytes = pf->devs_nbytes;
686 
687 	free(pf->devs_nvlbuf, M_NVLIST);
688 	pf->devs_nvlbuf = NULL;
689 	pf->devs_nbytes = 0;
690 
691 done:
692 	sx_unlock(&pf->lock);
693 	return (err);
694 }
695 
696 static int
697 sndstat_unpack_user_nvlbuf(const void *unvlbuf, size_t nbytes, nvlist_t **nvl)
698 {
699 	void *nvlbuf;
700 	int err;
701 
702 	nvlbuf = malloc(nbytes, M_DEVBUF, M_WAITOK);
703 	err = copyin(unvlbuf, nvlbuf, nbytes);
704 	if (err != 0) {
705 		free(nvlbuf, M_DEVBUF);
706 		return (err);
707 	}
708 	*nvl = nvlist_unpack(nvlbuf, nbytes, 0);
709 	free(nvlbuf, M_DEVBUF);
710 	if (nvl == NULL) {
711 		return (EINVAL);
712 	}
713 
714 	return (0);
715 }
716 
717 static bool
718 sndstat_diinfo_is_sane(const nvlist_t *diinfo)
719 {
720 	if (!(nvlist_exists_number(diinfo, SNDST_DSPS_INFO_MIN_RATE) &&
721 	    nvlist_exists_number(diinfo, SNDST_DSPS_INFO_MAX_RATE) &&
722 	    nvlist_exists_number(diinfo, SNDST_DSPS_INFO_FORMATS) &&
723 	    nvlist_exists_number(diinfo, SNDST_DSPS_INFO_MIN_CHN) &&
724 	    nvlist_exists_number(diinfo, SNDST_DSPS_INFO_MAX_CHN)))
725 		return (false);
726 	return (true);
727 }
728 
729 static bool
730 sndstat_dsp_nvlist_is_sane(const nvlist_t *nvlist)
731 {
732 	if (!(nvlist_exists_string(nvlist, SNDST_DSPS_DEVNODE) &&
733 	    nvlist_exists_string(nvlist, SNDST_DSPS_DESC) &&
734 	    nvlist_exists_number(nvlist, SNDST_DSPS_PCHAN) &&
735 	    nvlist_exists_number(nvlist, SNDST_DSPS_RCHAN)))
736 		return (false);
737 
738 	if (nvlist_get_number(nvlist, SNDST_DSPS_PCHAN) > 0) {
739 		if (nvlist_exists_nvlist(nvlist, SNDST_DSPS_INFO_PLAY)) {
740 			if (!sndstat_diinfo_is_sane(nvlist_get_nvlist(nvlist,
741 			    SNDST_DSPS_INFO_PLAY)))
742 				return (false);
743 		} else if (!(nvlist_exists_number(nvlist, "pminrate") &&
744 		    nvlist_exists_number(nvlist, "pmaxrate") &&
745 		    nvlist_exists_number(nvlist, "pfmts")))
746 			return (false);
747 	}
748 
749 	if (nvlist_get_number(nvlist, SNDST_DSPS_RCHAN) > 0) {
750 		if (nvlist_exists_nvlist(nvlist, SNDST_DSPS_INFO_REC)) {
751 			if (!sndstat_diinfo_is_sane(nvlist_get_nvlist(nvlist,
752 			    SNDST_DSPS_INFO_REC)))
753 				return (false);
754 		} else if (!(nvlist_exists_number(nvlist, "rminrate") &&
755 		    nvlist_exists_number(nvlist, "rmaxrate") &&
756 		    nvlist_exists_number(nvlist, "rfmts")))
757 			return (false);
758 	}
759 
760 	return (true);
761 
762 }
763 
764 static void
765 sndstat_get_diinfo_nv(const nvlist_t *nv, uint32_t *min_rate,
766 	    uint32_t *max_rate, uint32_t *formats, uint32_t *min_chn,
767 	    uint32_t *max_chn)
768 {
769 	*min_rate = nvlist_get_number(nv, SNDST_DSPS_INFO_MIN_RATE);
770 	*max_rate = nvlist_get_number(nv, SNDST_DSPS_INFO_MAX_RATE);
771 	*formats = nvlist_get_number(nv, SNDST_DSPS_INFO_FORMATS);
772 	*min_chn = nvlist_get_number(nv, SNDST_DSPS_INFO_MIN_CHN);
773 	*max_chn = nvlist_get_number(nv, SNDST_DSPS_INFO_MAX_CHN);
774 }
775 
776 static int
777 sndstat_dsp_unpack_nvlist(const nvlist_t *nvlist, struct sndstat_userdev *ud)
778 {
779 	const char *nameunit, *devnode, *desc;
780 	unsigned int pchan, rchan;
781 	uint32_t pminrate = 0, pmaxrate = 0;
782 	uint32_t rminrate = 0, rmaxrate = 0;
783 	uint32_t pfmts = 0, rfmts = 0;
784 	uint32_t pminchn = 0, pmaxchn = 0;
785 	uint32_t rminchn = 0, rmaxchn = 0;
786 	nvlist_t *provider_nvl = NULL;
787 	const nvlist_t *diinfo;
788 	const char *provider;
789 
790 	devnode = nvlist_get_string(nvlist, SNDST_DSPS_DEVNODE);
791 	if (nvlist_exists_string(nvlist, SNDST_DSPS_NAMEUNIT))
792 		nameunit = nvlist_get_string(nvlist, SNDST_DSPS_NAMEUNIT);
793 	else
794 		nameunit = devnode;
795 	desc = nvlist_get_string(nvlist, SNDST_DSPS_DESC);
796 	pchan = nvlist_get_number(nvlist, SNDST_DSPS_PCHAN);
797 	rchan = nvlist_get_number(nvlist, SNDST_DSPS_RCHAN);
798 	if (pchan != 0) {
799 		if (nvlist_exists_nvlist(nvlist, SNDST_DSPS_INFO_PLAY)) {
800 			diinfo = nvlist_get_nvlist(nvlist,
801 			    SNDST_DSPS_INFO_PLAY);
802 			sndstat_get_diinfo_nv(diinfo, &pminrate, &pmaxrate,
803 			    &pfmts, &pminchn, &pmaxchn);
804 		} else {
805 			pminrate = nvlist_get_number(nvlist, "pminrate");
806 			pmaxrate = nvlist_get_number(nvlist, "pmaxrate");
807 			pfmts = nvlist_get_number(nvlist, "pfmts");
808 		}
809 	}
810 	if (rchan != 0) {
811 		if (nvlist_exists_nvlist(nvlist, SNDST_DSPS_INFO_REC)) {
812 			diinfo = nvlist_get_nvlist(nvlist,
813 			    SNDST_DSPS_INFO_REC);
814 			sndstat_get_diinfo_nv(diinfo, &rminrate, &rmaxrate,
815 			    &rfmts, &rminchn, &rmaxchn);
816 		} else {
817 			rminrate = nvlist_get_number(nvlist, "rminrate");
818 			rmaxrate = nvlist_get_number(nvlist, "rmaxrate");
819 			rfmts = nvlist_get_number(nvlist, "rfmts");
820 		}
821 	}
822 
823 	provider = dnvlist_get_string(nvlist, SNDST_DSPS_PROVIDER, "");
824 	if (provider[0] == '\0')
825 		provider = NULL;
826 
827 	if (provider != NULL &&
828 	    nvlist_exists_nvlist(nvlist, SNDST_DSPS_PROVIDER_INFO)) {
829 		provider_nvl = nvlist_clone(
830 		    nvlist_get_nvlist(nvlist, SNDST_DSPS_PROVIDER_INFO));
831 		if (provider_nvl == NULL)
832 			return (ENOMEM);
833 	}
834 
835 	ud->provider = (provider != NULL) ? strdup(provider, M_DEVBUF) : NULL;
836 	ud->devnode = strdup(devnode, M_DEVBUF);
837 	ud->nameunit = strdup(nameunit, M_DEVBUF);
838 	ud->desc = strdup(desc, M_DEVBUF);
839 	ud->pchan = pchan;
840 	ud->rchan = rchan;
841 	ud->info_play.min_rate = pminrate;
842 	ud->info_play.max_rate = pmaxrate;
843 	ud->info_play.formats = pfmts;
844 	ud->info_play.min_chn = pminchn;
845 	ud->info_play.max_chn = pmaxchn;
846 	ud->info_rec.min_rate = rminrate;
847 	ud->info_rec.max_rate = rmaxrate;
848 	ud->info_rec.formats = rfmts;
849 	ud->info_rec.min_chn = rminchn;
850 	ud->info_rec.max_chn = rmaxchn;
851 	ud->provider_nvl = provider_nvl;
852 	return (0);
853 }
854 
855 static int
856 sndstat_add_user_devs(struct sndstat_file *pf, caddr_t data)
857 {
858 	int err;
859 	nvlist_t *nvl = NULL;
860 	const nvlist_t * const *dsps;
861 	size_t i, ndsps;
862 	struct sndstioc_nv_arg *arg = (struct sndstioc_nv_arg *)data;
863 
864 	if ((pf->fflags & FWRITE) == 0) {
865 		err = EPERM;
866 		goto done;
867 	}
868 
869 	err = sndstat_unpack_user_nvlbuf(arg->buf, arg->nbytes, &nvl);
870 	if (err != 0)
871 		goto done;
872 
873 	if (!nvlist_exists_nvlist_array(nvl, SNDST_DSPS)) {
874 		err = EINVAL;
875 		goto done;
876 	}
877 	dsps = nvlist_get_nvlist_array(nvl, SNDST_DSPS, &ndsps);
878 	for (i = 0; i < ndsps; i++) {
879 		if (!sndstat_dsp_nvlist_is_sane(dsps[i])) {
880 			err = EINVAL;
881 			goto done;
882 		}
883 	}
884 	sx_xlock(&pf->lock);
885 	for (i = 0; i < ndsps; i++) {
886 		struct sndstat_userdev *ud =
887 		    malloc(sizeof(*ud), M_DEVBUF, M_WAITOK);
888 		err = sndstat_dsp_unpack_nvlist(dsps[i], ud);
889 		if (err) {
890 			sx_unlock(&pf->lock);
891 			goto done;
892 		}
893 		TAILQ_INSERT_TAIL(&pf->userdev_list, ud, link);
894 	}
895 	sx_unlock(&pf->lock);
896 
897 done:
898 	nvlist_destroy(nvl);
899 	return (err);
900 }
901 
902 static int
903 sndstat_flush_user_devs(struct sndstat_file *pf)
904 {
905 	if ((pf->fflags & FWRITE) == 0)
906 		return (EPERM);
907 
908 	sx_xlock(&pf->lock);
909 	sndstat_remove_all_userdevs(pf);
910 	sx_xunlock(&pf->lock);
911 
912 	return (0);
913 }
914 
915 #ifdef COMPAT_FREEBSD32
916 static int
917 compat_sndstat_get_devs32(struct sndstat_file *pf, caddr_t data)
918 {
919 	struct sndstioc_nv_arg32 *arg32 = (struct sndstioc_nv_arg32 *)data;
920 	struct sndstioc_nv_arg arg;
921 	int err;
922 
923 	arg.buf = (void *)(uintptr_t)arg32->buf;
924 	arg.nbytes = arg32->nbytes;
925 
926 	err = sndstat_get_devs(pf, (caddr_t)&arg);
927 	if (err == 0) {
928 		arg32->buf = (uint32_t)(uintptr_t)arg.buf;
929 		arg32->nbytes = arg.nbytes;
930 	}
931 
932 	return (err);
933 }
934 
935 static int
936 compat_sndstat_add_user_devs32(struct sndstat_file *pf, caddr_t data)
937 {
938 	struct sndstioc_nv_arg32 *arg32 = (struct sndstioc_nv_arg32 *)data;
939 	struct sndstioc_nv_arg arg;
940 	int err;
941 
942 	arg.buf = (void *)(uintptr_t)arg32->buf;
943 	arg.nbytes = arg32->nbytes;
944 
945 	err = sndstat_add_user_devs(pf, (caddr_t)&arg);
946 	if (err == 0) {
947 		arg32->buf = (uint32_t)(uintptr_t)arg.buf;
948 		arg32->nbytes = arg.nbytes;
949 	}
950 
951 	return (err);
952 }
953 #endif
954 
955 static int
956 sndstat_ioctl(
957     struct cdev *dev, u_long cmd, caddr_t data, int fflag, struct thread *td)
958 {
959 	int err;
960 	struct sndstat_file *pf;
961 
962 	err = devfs_get_cdevpriv((void **)&pf);
963 	if (err != 0)
964 		return (err);
965 
966 	switch (cmd) {
967 	case SNDSTIOC_GET_DEVS:
968 		err = sndstat_get_devs(pf, data);
969 		break;
970 #ifdef COMPAT_FREEBSD32
971 	case SNDSTIOC_GET_DEVS32:
972 		if (!SV_CURPROC_FLAG(SV_ILP32)) {
973 			err = ENODEV;
974 			break;
975 		}
976 		err = compat_sndstat_get_devs32(pf, data);
977 		break;
978 #endif
979 	case SNDSTIOC_ADD_USER_DEVS:
980 		err = sndstat_add_user_devs(pf, data);
981 		break;
982 #ifdef COMPAT_FREEBSD32
983 	case SNDSTIOC_ADD_USER_DEVS32:
984 		if (!SV_CURPROC_FLAG(SV_ILP32)) {
985 			err = ENODEV;
986 			break;
987 		}
988 		err = compat_sndstat_add_user_devs32(pf, data);
989 		break;
990 #endif
991 	case SNDSTIOC_REFRESH_DEVS:
992 		err = sndstat_refresh_devs(pf);
993 		break;
994 	case SNDSTIOC_FLUSH_USER_DEVS:
995 		err = sndstat_flush_user_devs(pf);
996 		break;
997 	default:
998 		err = ENODEV;
999 	}
1000 
1001 	return (err);
1002 }
1003 
1004 static struct sndstat_userdev *
1005 sndstat_line2userdev(struct sndstat_file *pf, const char *line, int n)
1006 {
1007 	struct sndstat_userdev *ud;
1008 	const char *e, *m;
1009 
1010 	ud = malloc(sizeof(*ud), M_DEVBUF, M_WAITOK|M_ZERO);
1011 
1012 	ud->provider = NULL;
1013 	ud->provider_nvl = NULL;
1014 	e = strchr(line, ':');
1015 	if (e == NULL)
1016 		goto fail;
1017 	ud->nameunit = strndup(line, e - line, M_DEVBUF);
1018 	ud->devnode = (char *)malloc(e - line + 1, M_DEVBUF, M_WAITOK | M_ZERO);
1019 	strlcat(ud->devnode, ud->nameunit, e - line + 1);
1020 	line = e + 1;
1021 
1022 	e = strchr(line, '<');
1023 	if (e == NULL)
1024 		goto fail;
1025 	line = e + 1;
1026 	e = strrchr(line, '>');
1027 	if (e == NULL)
1028 		goto fail;
1029 	ud->desc = strndup(line, e - line, M_DEVBUF);
1030 	line = e + 1;
1031 
1032 	e = strchr(line, '(');
1033 	if (e == NULL)
1034 		goto fail;
1035 	line = e + 1;
1036 	e = strrchr(line, ')');
1037 	if (e == NULL)
1038 		goto fail;
1039 	m = strstr(line, "play");
1040 	if (m != NULL && m < e)
1041 		ud->pchan = 1;
1042 	m = strstr(line, "rec");
1043 	if (m != NULL && m < e)
1044 		ud->rchan = 1;
1045 
1046 	return (ud);
1047 
1048 fail:
1049 	free(ud->nameunit, M_DEVBUF);
1050 	free(ud->devnode, M_DEVBUF);
1051 	free(ud->desc, M_DEVBUF);
1052 	free(ud, M_DEVBUF);
1053 	return (NULL);
1054 }
1055 
1056 /************************************************************************/
1057 
1058 int
1059 sndstat_register(device_t dev, char *str, sndstat_handler handler)
1060 {
1061 	struct sndstat_entry *ent;
1062 	struct sndstat_entry *pre;
1063 	const char *devtype;
1064 	int type, unit;
1065 
1066 	if (dev) {
1067 		unit = device_get_unit(dev);
1068 		devtype = device_get_name(dev);
1069 		if (!strcmp(devtype, "pcm"))
1070 			type = SS_TYPE_PCM;
1071 		else if (!strcmp(devtype, "midi"))
1072 			type = SS_TYPE_MIDI;
1073 		else if (!strcmp(devtype, "sequencer"))
1074 			type = SS_TYPE_SEQUENCER;
1075 		else
1076 			return (EINVAL);
1077 	} else {
1078 		type = SS_TYPE_MODULE;
1079 		unit = -1;
1080 	}
1081 
1082 	ent = malloc(sizeof *ent, M_DEVBUF, M_WAITOK | M_ZERO);
1083 	ent->dev = dev;
1084 	ent->str = str;
1085 	ent->type = type;
1086 	ent->unit = unit;
1087 	ent->handler = handler;
1088 
1089 	SNDSTAT_LOCK();
1090 	/* sorted list insertion */
1091 	TAILQ_FOREACH(pre, &sndstat_devlist, link) {
1092 		if (pre->unit > unit)
1093 			break;
1094 		else if (pre->unit < unit)
1095 			continue;
1096 		if (pre->type > type)
1097 			break;
1098 		else if (pre->type < unit)
1099 			continue;
1100 	}
1101 	if (pre == NULL) {
1102 		TAILQ_INSERT_TAIL(&sndstat_devlist, ent, link);
1103 	} else {
1104 		TAILQ_INSERT_BEFORE(pre, ent, link);
1105 	}
1106 	SNDSTAT_UNLOCK();
1107 
1108 	return (0);
1109 }
1110 
1111 int
1112 sndstat_registerfile(char *str)
1113 {
1114 	return (sndstat_register(NULL, str, NULL));
1115 }
1116 
1117 int
1118 sndstat_unregister(device_t dev)
1119 {
1120 	struct sndstat_entry *ent;
1121 	int error = ENXIO;
1122 
1123 	SNDSTAT_LOCK();
1124 	TAILQ_FOREACH(ent, &sndstat_devlist, link) {
1125 		if (ent->dev == dev) {
1126 			TAILQ_REMOVE(&sndstat_devlist, ent, link);
1127 			free(ent, M_DEVBUF);
1128 			error = 0;
1129 			break;
1130 		}
1131 	}
1132 	SNDSTAT_UNLOCK();
1133 
1134 	return (error);
1135 }
1136 
1137 int
1138 sndstat_unregisterfile(char *str)
1139 {
1140 	struct sndstat_entry *ent;
1141 	int error = ENXIO;
1142 
1143 	SNDSTAT_LOCK();
1144 	TAILQ_FOREACH(ent, &sndstat_devlist, link) {
1145 		if (ent->dev == NULL && ent->str == str) {
1146 			TAILQ_REMOVE(&sndstat_devlist, ent, link);
1147 			free(ent, M_DEVBUF);
1148 			error = 0;
1149 			break;
1150 		}
1151 	}
1152 	SNDSTAT_UNLOCK();
1153 
1154 	return (error);
1155 }
1156 
1157 /************************************************************************/
1158 
1159 static int
1160 sndstat_prepare(struct sndstat_file *pf_self)
1161 {
1162 	struct sbuf *s = &pf_self->sbuf;
1163 	struct sndstat_entry *ent;
1164 	struct snddev_info *d;
1165 	struct sndstat_file *pf;
1166     	int k;
1167 
1168 	/* make sure buffer is reset */
1169 	sbuf_clear(s);
1170 
1171 	if (snd_verbose > 0) {
1172 		sbuf_printf(s, "FreeBSD Audio Driver (%ubit %d/%s)\n",
1173 		    (u_int)sizeof(intpcm32_t) << 3, SND_DRV_VERSION,
1174 		    MACHINE_ARCH);
1175 	}
1176 
1177 	/* generate list of installed devices */
1178 	k = 0;
1179 	TAILQ_FOREACH(ent, &sndstat_devlist, link) {
1180 		if (ent->dev == NULL)
1181 			continue;
1182 		d = device_get_softc(ent->dev);
1183 		if (!PCM_REGISTERED(d))
1184 			continue;
1185 		if (!k++)
1186 			sbuf_printf(s, "Installed devices:\n");
1187 		sbuf_printf(s, "%s:", device_get_nameunit(ent->dev));
1188 		sbuf_printf(s, " <%s>", device_get_desc(ent->dev));
1189 		if (snd_verbose > 0)
1190 			sbuf_printf(s, " %s", ent->str);
1191 		if (ent->handler) {
1192 			/* XXX Need Giant magic entry ??? */
1193 			PCM_ACQUIRE_QUICK(d);
1194 			ent->handler(s, ent->dev, snd_verbose);
1195 			PCM_RELEASE_QUICK(d);
1196 		}
1197 		sbuf_printf(s, "\n");
1198 	}
1199 	if (k == 0)
1200 		sbuf_printf(s, "No devices installed.\n");
1201 
1202 	/* append any input from userspace */
1203 	k = 0;
1204 	TAILQ_FOREACH(pf, &sndstat_filelist, entry) {
1205 		struct sndstat_userdev *ud;
1206 
1207 		if (pf == pf_self)
1208 			continue;
1209 		sx_xlock(&pf->lock);
1210 		if (TAILQ_EMPTY(&pf->userdev_list)) {
1211 			sx_unlock(&pf->lock);
1212 			continue;
1213 		}
1214 		if (!k++)
1215 			sbuf_printf(s, "Installed devices from userspace:\n");
1216 		TAILQ_FOREACH(ud, &pf->userdev_list, link) {
1217 			const char *caps = (ud->pchan && ud->rchan) ?
1218 			    "play/rec" :
1219 			    (ud->pchan ? "play" : (ud->rchan ? "rec" : ""));
1220 			sbuf_printf(s, "%s: <%s>", ud->nameunit, ud->desc);
1221 			sbuf_printf(s, " (%s)", caps);
1222 			sbuf_printf(s, "\n");
1223 		}
1224 		sx_unlock(&pf->lock);
1225 	}
1226 	if (k == 0)
1227 		sbuf_printf(s, "No devices installed from userspace.\n");
1228 
1229 	/* append any file versions */
1230 	if (snd_verbose >= 3) {
1231 		k = 0;
1232 		TAILQ_FOREACH(ent, &sndstat_devlist, link) {
1233 			if (ent->dev == NULL && ent->str != NULL) {
1234 				if (!k++)
1235 					sbuf_printf(s, "\nFile Versions:\n");
1236 				sbuf_printf(s, "%s\n", ent->str);
1237 			}
1238 		}
1239 		if (k == 0)
1240 			sbuf_printf(s, "\nNo file versions.\n");
1241 	}
1242 	sbuf_finish(s);
1243     	return (sbuf_len(s));
1244 }
1245 
1246 static void
1247 sndstat_sysinit(void *p)
1248 {
1249 	sx_init(&sndstat_lock, "sndstat lock");
1250 	sndstat_dev = make_dev(&sndstat_cdevsw, SND_DEV_STATUS,
1251 	    UID_ROOT, GID_WHEEL, 0644, "sndstat");
1252 }
1253 SYSINIT(sndstat_sysinit, SI_SUB_DRIVERS, SI_ORDER_FIRST, sndstat_sysinit, NULL);
1254 
1255 static void
1256 sndstat_sysuninit(void *p)
1257 {
1258 	if (sndstat_dev != NULL) {
1259 		/* destroy_dev() will wait for all references to go away */
1260 		destroy_dev(sndstat_dev);
1261 	}
1262 	sx_destroy(&sndstat_lock);
1263 }
1264 SYSUNINIT(sndstat_sysuninit, SI_SUB_DRIVERS, SI_ORDER_FIRST, sndstat_sysuninit, NULL);
1265