xref: /illumos-gate/usr/src/uts/common/io/audio/impl/audio_oss.c (revision 5ad1f010a7b934be6e0dd6c13198af62791824be)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #include <sys/types.h>
27 #include <sys/open.h>
28 #include <sys/errno.h>
29 #include <sys/ddi.h>
30 #include <sys/sunddi.h>
31 #include <sys/audio/audio_oss.h>
32 #include <sys/file.h>
33 #include <sys/note.h>
34 #include <sys/sysmacros.h>
35 #include <sys/list.h>
36 #include "audio_client.h"
37 
38 #define	OSS_FMT		AFMT_S16_LE
39 #define	OSS_RATE	48000
40 #define	OSS_CHANNELS	2
41 
42 typedef struct ossclient ossclient_t;
43 typedef struct ossdev ossdev_t;
44 
45 static const struct {
46 	int	oss;
47 	int	fmt;
48 } oss_formats[] = {
49 	{ AFMT_MU_LAW,		AUDIO_FORMAT_ULAW },
50 	{ AFMT_A_LAW,		AUDIO_FORMAT_ALAW },
51 	{ AFMT_U8,		AUDIO_FORMAT_U8 },
52 	{ AFMT_S8,		AUDIO_FORMAT_S8 },
53 	{ AFMT_S16_BE,		AUDIO_FORMAT_S16_BE },
54 	{ AFMT_S16_LE,		AUDIO_FORMAT_S16_LE },
55 	{ AFMT_U16_BE,		AUDIO_FORMAT_U16_BE },
56 	{ AFMT_U16_LE,		AUDIO_FORMAT_U16_LE },
57 	{ AFMT_S24_BE,		AUDIO_FORMAT_S24_BE },
58 	{ AFMT_S24_LE,		AUDIO_FORMAT_S24_LE },
59 	{ AFMT_S32_BE,		AUDIO_FORMAT_S32_BE },
60 	{ AFMT_S32_LE,		AUDIO_FORMAT_S32_LE },
61 	{ AFMT_S24_PACKED,	AUDIO_FORMAT_S24_PACKED },
62 	{ AFMT_AC3,		AUDIO_FORMAT_AC3 },
63 	{ AFMT_QUERY,		AUDIO_FORMAT_NONE }
64 };
65 
66 /* common structure shared between both mixer and dsp nodes */
67 struct ossclient {
68 	ossdev_t		*o_ossdev;
69 	audio_client_t		*o_client;
70 	/* sndstat */
71 	kmutex_t		o_ss_lock;
72 	char			*o_ss_buf;
73 	size_t			o_ss_len;
74 	size_t			o_ss_sz;
75 	size_t			o_ss_off;
76 };
77 
78 struct ossdev {
79 	audio_dev_t		*d_dev;
80 
81 	uint_t			d_modify_cnt;	/* flag apps of ctrl changes */
82 	uint_t			d_nctrl;	/* num actual controls */
83 	uint_t			d_nalloc;	/* num allocated controls */
84 	audio_ctrl_t		**d_ctrls;	/* array of control handles */
85 	oss_mixext		*d_exts;	/* array of mixer descs */
86 
87 	int			d_play_grp;
88 	int			d_rec_grp;
89 	int			d_mon_grp;
90 	int			d_misc_grp;
91 
92 	kmutex_t		d_mx;
93 	kcondvar_t		d_cv;
94 };
95 
96 static int
97 oss_cnt_controls(audio_ctrl_t *ctrl, void *arg)
98 {
99 	int			*pint = (int *)arg;
100 	int			cnt;
101 	audio_ctrl_desc_t	desc;
102 
103 	cnt = *pint;
104 	cnt++;
105 	*pint = cnt;
106 
107 	if (auclnt_control_describe(ctrl, &desc) != 0)
108 		return (AUDIO_WALK_CONTINUE);
109 
110 	if (desc.acd_flags & AUDIO_CTRL_FLAG_MULTI) {
111 		for (uint64_t mask = desc.acd_maxvalue; mask; mask >>= 1) {
112 			if (mask & 1) {
113 				cnt++;
114 			}
115 		}
116 		*pint = cnt;
117 	}
118 
119 	return (AUDIO_WALK_CONTINUE);
120 }
121 
122 /*
123  * Add one entry to the OSS user control table to internal control
124  * helper table.
125  *
126  * This is used with auimpl_walk_controls. The table must be pre-
127  * allocated before it is walk'd. This includes the root and
128  * extended control markers!
129  */
130 static int
131 oss_add_control(audio_ctrl_t *ctrl, void *arg)
132 {
133 	ossdev_t		*odev = arg;
134 	audio_ctrl_desc_t	desc;
135 	oss_mixext		*ext;
136 	int			bit;
137 	uint64_t		mask;
138 	const char		*name;
139 	int			parent;
140 	int			flags;
141 	unsigned		scope;
142 
143 	if (auclnt_control_describe(ctrl, &desc))
144 		return (AUDIO_WALK_CONTINUE);
145 
146 	parent = 0;
147 
148 	/*
149 	 * Add appropriate group if not already done so.
150 	 */
151 	if (desc.acd_flags & AUDIO_CTRL_FLAG_PLAY) {
152 		if (!odev->d_play_grp) {
153 			ext = &odev->d_exts[odev->d_nctrl];
154 			ext->ctrl = odev->d_nctrl;
155 			ext->control_no = -1;
156 			ext->type = MIXT_GROUP;
157 			ext->desc = MIXEXT_SCOPE_OUTPUT;
158 			ext->timestamp = gethrtime();
159 			(void) snprintf(ext->id, sizeof (ext->id), "PLAYBACK");
160 			odev->d_play_grp = odev->d_nctrl;
161 			odev->d_nctrl++;
162 		}
163 		scope = MIXEXT_SCOPE_OUTPUT;
164 		parent = odev->d_play_grp;
165 	} else if (desc.acd_flags & AUDIO_CTRL_FLAG_REC) {
166 		if (!odev->d_rec_grp) {
167 			ext = &odev->d_exts[odev->d_nctrl];
168 			ext->ctrl = odev->d_nctrl;
169 			ext->control_no = -1;
170 			ext->type = MIXT_GROUP;
171 			ext->desc = MIXEXT_SCOPE_INPUT;
172 			ext->timestamp = gethrtime();
173 			(void) snprintf(ext->id, sizeof (ext->id), "RECORD");
174 			odev->d_rec_grp = odev->d_nctrl;
175 			odev->d_nctrl++;
176 		}
177 		scope = MIXEXT_SCOPE_INPUT;
178 		parent = odev->d_rec_grp;
179 	} else if (desc.acd_flags & AUDIO_CTRL_FLAG_MONITOR) {
180 		if (!odev->d_mon_grp) {
181 			ext = &odev->d_exts[odev->d_nctrl];
182 			ext->ctrl = odev->d_nctrl;
183 			ext->control_no = -1;
184 			ext->type = MIXT_GROUP;
185 			ext->desc = MIXEXT_SCOPE_MONITOR;
186 			ext->timestamp = gethrtime();
187 			(void) snprintf(ext->id, sizeof (ext->id), "MONITOR");
188 			odev->d_mon_grp = odev->d_nctrl;
189 			odev->d_nctrl++;
190 		}
191 		scope = MIXEXT_SCOPE_MONITOR;
192 		parent = odev->d_mon_grp;
193 	} else {
194 		if (!odev->d_misc_grp) {
195 			ext = &odev->d_exts[odev->d_nctrl];
196 			ext->ctrl = odev->d_nctrl;
197 			ext->control_no = -1;
198 			ext->type = MIXT_GROUP;
199 			ext->desc = MIXEXT_SCOPE_OTHER;
200 			ext->timestamp = gethrtime();
201 			(void) snprintf(ext->id, sizeof (ext->id), "MISC");
202 			odev->d_misc_grp = odev->d_nctrl;
203 			odev->d_nctrl++;
204 		}
205 		scope = MIXEXT_SCOPE_OTHER;
206 		parent = odev->d_misc_grp;
207 	}
208 
209 	name = desc.acd_name ? desc.acd_name : "";
210 
211 	if (desc.acd_flags & AUDIO_CTRL_FLAG_MULTI) {
212 		ext = &odev->d_exts[odev->d_nctrl];
213 		ext->ctrl = odev->d_nctrl;
214 		ext->control_no = -1;
215 		ext->type = MIXT_GROUP;
216 		ext->timestamp = gethrtime();
217 		ext->parent = parent;
218 		ext->desc = scope;
219 		(void) snprintf(ext->id, sizeof (ext->id), "%s", name);
220 		(void) snprintf(ext->extname, sizeof (ext->extname),
221 		    "%s", name);
222 		parent = odev->d_nctrl++;
223 	}
224 
225 	/* Next available open entry */
226 	ext = &odev->d_exts[odev->d_nctrl];
227 
228 	/* Record the underlying control handle */
229 	odev->d_ctrls[odev->d_nctrl] = ctrl;
230 
231 	/*
232 	 * Now setup the oss entry
233 	 */
234 
235 	ext->ctrl = odev->d_nctrl;
236 	ext->control_no = -1;
237 	ext->maxvalue = (int)desc.acd_maxvalue;
238 	ext->minvalue = (int)desc.acd_minvalue;
239 	ext->timestamp = gethrtime();
240 	ext->parent = parent;
241 	ext->desc = scope;
242 	/* all controls should be pollable for now */
243 	flags = MIXF_POLL;
244 
245 	/*
246 	 * The following flags are intended to help out applications
247 	 * which need to figure out where to place certain controls.
248 	 * A few further words of guidance:
249 	 *
250 	 * Apps that just want a single master volume control should
251 	 * adjust the control(s) that are labelled with MIXF_PCMVOL if
252 	 * present.  They can fall back to adjusting all MAINVOL
253 	 * levels instead, if no PCMVOL is present.
254 	 *
255 	 * Controls that are one type on a certain device might be a
256 	 * different type on another device.  For example,
257 	 * audiopci/ak4531 can adjust input gains for individual
258 	 * levels, but lacks a master record gain.  AC'97, on the
259 	 * other hand, has individual monitor gains for inputs, but
260 	 * only a single master recording gain.
261 	 */
262 	if (desc.acd_flags & AUDIO_CTRL_FLAG_READABLE)
263 		flags |= MIXF_READABLE;
264 	if (desc.acd_flags & AUDIO_CTRL_FLAG_WRITEABLE)
265 		flags |= MIXF_WRITEABLE;
266 	if (desc.acd_flags & AUDIO_CTRL_FLAG_CENTIBEL)
267 		flags |= MIXF_CENTIBEL;
268 	if (desc.acd_flags & AUDIO_CTRL_FLAG_DECIBEL)
269 		flags |= MIXF_DECIBEL;
270 	if (desc.acd_flags & AUDIO_CTRL_FLAG_MAINVOL)
271 		flags |= MIXF_MAINVOL;
272 	if (desc.acd_flags & AUDIO_CTRL_FLAG_PCMVOL)
273 		flags |= MIXF_PCMVOL;
274 	if (desc.acd_flags & AUDIO_CTRL_FLAG_RECVOL)
275 		flags |= MIXF_RECVOL;
276 	if (desc.acd_flags & AUDIO_CTRL_FLAG_MONVOL)
277 		flags |= MIXF_MONVOL;
278 	ext->flags = flags;
279 
280 	(void) snprintf(ext->id, sizeof (ext->id), "%s", name);
281 
282 	/*
283 	 * For now just use the same extname as the real name.
284 	 */
285 	(void) snprintf(ext->extname, sizeof (ext->extname), name);
286 
287 	/*
288 	 * Now we deal with various control types.
289 	 */
290 	switch (desc.acd_type) {
291 	case AUDIO_CTRL_TYPE_BOOLEAN:
292 		ext->type = MIXT_ONOFF;
293 		ext->enumbit = -1;
294 		break;
295 	case AUDIO_CTRL_TYPE_STEREO:
296 		ext->type = MIXT_STEREOSLIDER;
297 		break;
298 	case AUDIO_CTRL_TYPE_MONO:
299 		ext->type = MIXT_MONOSLIDER;
300 		break;
301 	case AUDIO_CTRL_TYPE_ENUM:
302 
303 		if (desc.acd_flags & AUDIO_CTRL_FLAG_MULTI) {
304 			/*
305 			 * We turn AUDIO_CTRL_FLAG_MULTI into a group
306 			 * of checkboxes, since OSS can't represent it
307 			 * natively.
308 			 */
309 			mask = desc.acd_maxvalue;
310 			bit = 0;
311 			while (mask) {
312 				if (mask & 1) {
313 					ext = &odev->d_exts[odev->d_nctrl];
314 					(void) snprintf(ext->extname,
315 					    sizeof (ext->extname), "%s.%s",
316 					    name, desc.acd_enum[bit]);
317 					(void) snprintf(ext->id,
318 					    sizeof (ext->id), "%s",
319 					    desc.acd_enum[bit]);
320 					ext->ctrl = odev->d_nctrl;
321 					ext->control_no = -1;
322 					ext->parent = parent;
323 					ext->timestamp = gethrtime();
324 					ext->type = MIXT_ONOFF;
325 					ext->minvalue = 0;
326 					ext->maxvalue = 1;
327 					ext->enumbit = bit;
328 					ext->flags = flags;
329 					odev->d_ctrls[odev->d_nctrl] = ctrl;
330 					odev->d_nctrl++;
331 				}
332 				bit++;
333 				mask >>= 1;
334 			}
335 			return (AUDIO_WALK_CONTINUE);
336 		} else {
337 			/*
338 			 * NB: This is sufficient only for controls
339 			 * with a single value.  It cannot express the
340 			 * richer bitmask capabilities.
341 			 */
342 			ext->type = MIXT_ENUM;
343 			ext->minvalue = 0;
344 
345 			/*
346 			 * For an enumaration, we need to figure out
347 			 * which values are present, and set the
348 			 * appropriate mask and max value.
349 			 */
350 			bzero(ext->enum_present, sizeof (ext->enum_present));
351 			mask = desc.acd_maxvalue;
352 			bit = 0;
353 			while (mask) {
354 				if (mask & 1) {
355 					ext->enum_present[bit / 8] |=
356 					    (1 << (bit % 8));
357 				}
358 				mask >>= 1;
359 				bit++;
360 			}
361 			ext->maxvalue = bit;
362 		}
363 		break;
364 
365 	case AUDIO_CTRL_TYPE_METER:
366 	default:
367 		/* Its an unknown or unsupported (for now) control, skip */
368 		return (AUDIO_WALK_CONTINUE);
369 	}
370 
371 	odev->d_nctrl++;
372 
373 	return (AUDIO_WALK_CONTINUE);
374 }
375 
376 /*
377  * Free up an OSS user land control to internal control,
378  * helper table.
379  */
380 static void
381 oss_free_controls(ossdev_t *odev)
382 {
383 	kmem_free(odev->d_ctrls, sizeof (audio_ctrl_t *) * odev->d_nalloc);
384 	kmem_free(odev->d_exts, sizeof (oss_mixext) * odev->d_nalloc);
385 	odev->d_nctrl = 0;
386 	odev->d_nalloc = 0;
387 }
388 
389 /*
390  * Allocate and fill in an OSS user land controls to internal controls
391  * helper table. This is done on one audio_dev device.
392  */
393 static void
394 oss_alloc_controls(ossdev_t *odev)
395 {
396 	audio_dev_t		*d = odev->d_dev;
397 	int			nctrl = 0;
398 	oss_mixext		*ext;
399 	oss_mixext_root		*root_data;
400 
401 	/* Find out who many entries we need */
402 	auclnt_walk_controls(d, oss_cnt_controls, &nctrl);
403 	nctrl++;		/* Needs space for the device root node */
404 	nctrl++;		/* Needs space for the device ext marker */
405 	nctrl++;		/* Needs space for the play group */
406 	nctrl++;		/* Needs space for the record group */
407 	nctrl++;		/* Needs space for the monitor group */
408 	nctrl++;		/* Needs space for the tone group */
409 	nctrl++;		/* Needs space for the 3D group */
410 	nctrl++;		/* Needs space for the misc group */
411 
412 	/* Allocate the OSS to boomer helper table */
413 	odev->d_nalloc = nctrl;
414 	odev->d_ctrls = kmem_zalloc(sizeof (audio_ctrl_t *) * nctrl, KM_SLEEP);
415 	odev->d_exts = kmem_zalloc(sizeof (oss_mixext) * nctrl, KM_SLEEP);
416 
417 	/*
418 	 * Setup special case outputs to output OSS routes helper tables
419 	 */
420 
421 	/*
422 	 * Root node is first, that way all others parent is this one
423 	 */
424 	ext = &odev->d_exts[odev->d_nctrl];
425 	ext->ctrl = 0;
426 	ext->parent = -1;
427 	ext->type = MIXT_DEVROOT;
428 	ext->timestamp = gethrtime();
429 	(void) snprintf(ext->id, sizeof (ext->id), "DEVROOT");
430 	/*
431 	 * Root data... nobody should be using this though.
432 	 */
433 	root_data = (oss_mixext_root *)&ext->data;
434 	(void) snprintf(root_data->name, sizeof (root_data->name), "%s",
435 	    auclnt_get_dev_name(d));
436 	(void) snprintf(root_data->id, sizeof (root_data->id), "%s",
437 	    auclnt_get_dev_name(d));
438 
439 	odev->d_nctrl++;
440 
441 	/*
442 	 * Insert an extra marker -- needed to keep layout apps hapy.
443 	 * This prevents some apps from assuming we are in "LEGACY" mode.
444 	 */
445 	ext = &odev->d_exts[odev->d_nctrl];
446 	ext->ctrl = odev->d_nctrl;
447 	ext->control_no = -1;
448 	ext->type = MIXT_MARKER;
449 	ext->timestamp = gethrtime();
450 	ext->parent = 0;
451 	odev->d_nctrl++;
452 
453 	/* Fill in the complete table now */
454 	auclnt_walk_controls(d, oss_add_control, odev);
455 
456 	/* Update the update_counter reference counter for groups */
457 	for (nctrl = 0; nctrl < odev->d_nctrl; nctrl++) {
458 		int i;
459 
460 		ext = &odev->d_exts[nctrl];
461 		i = ext->parent;
462 		while ((i >= 0) && (i < odev->d_nctrl)) {
463 
464 			ext = &odev->d_exts[i];
465 			ASSERT(ext->parent < i);
466 			ASSERT((ext->type == MIXT_GROUP) ||
467 			    (ext->type == MIXT_DEVROOT));
468 			ext->update_counter++;
469 			i = ext->parent;
470 		}
471 	}
472 
473 	ASSERT(odev->d_nctrl <= odev->d_nalloc);
474 }
475 
476 static int
477 oss_open(audio_client_t *c, int oflag)
478 {
479 	int		rv;
480 	ossdev_t	*odev;
481 	ossclient_t	*sc;
482 	audio_stream_t	*isp, *osp;
483 
484 	isp = auclnt_input_stream(c);
485 	osp = auclnt_output_stream(c);
486 
487 	/* note that OSS always uses nonblocking open() semantics */
488 	if ((rv = auclnt_open(c, AUDIO_FORMAT_PCM, oflag | FNDELAY)) != 0) {
489 		return (rv);
490 	}
491 
492 	if ((sc = kmem_zalloc(sizeof (*sc), KM_NOSLEEP)) == NULL) {
493 		auclnt_close(c);
494 		return (ENOMEM);
495 	}
496 	auclnt_set_private(c, sc);
497 
498 	odev = auclnt_get_minor_data(c, AUDIO_MINOR_DSP);
499 
500 	/* set a couple of common fields */
501 	sc->o_client = c;
502 	sc->o_ossdev = odev;
503 
504 	/* set all default parameters */
505 	if (oflag & FWRITE) {
506 		if (((rv = auclnt_set_format(osp, OSS_FMT)) != 0) ||
507 		    ((rv = auclnt_set_rate(osp, OSS_RATE)) != 0) ||
508 		    ((rv = auclnt_set_channels(osp, OSS_CHANNELS)) != 0)) {
509 			goto failed;
510 		}
511 		/* default to 5 fragments to provide reasonable latency */
512 		auclnt_set_latency(osp, 5, 0);
513 	}
514 
515 	if (oflag & FREAD) {
516 		if (((rv = auclnt_set_format(isp, OSS_FMT)) != 0) ||
517 		    ((rv = auclnt_set_rate(isp, OSS_RATE)) != 0) ||
518 		    ((rv = auclnt_set_channels(isp, OSS_CHANNELS)) != 0)) {
519 			goto failed;
520 		}
521 		/* default to 5 fragments to provide reasonable latency */
522 		auclnt_set_latency(isp, 5, 0);
523 	}
524 
525 	return (0);
526 
527 failed:
528 	auclnt_close(c);
529 	return (rv);
530 }
531 
532 static void
533 oss_close(audio_client_t *c)
534 {
535 	ossclient_t	*sc;
536 
537 	sc = auclnt_get_private(c);
538 
539 	if (ddi_can_receive_sig() || (ddi_get_pid() == 0)) {
540 		(void) auclnt_drain(c);
541 	}
542 
543 	kmem_free(sc, sizeof (*sc));
544 
545 	auclnt_close(c);
546 }
547 
548 /*
549  * This is used to generate an array of names for an enumeration
550  */
551 static ushort_t
552 oss_set_enum(oss_mixer_enuminfo *ei, ushort_t nxt, const char *name)
553 {
554 	uint32_t	n;
555 
556 	/* Get current entry to fill in */
557 	n = ei->nvalues;
558 	(void) snprintf(&ei->strings[nxt], ((sizeof (ei->strings) - nxt) - 1),
559 	    "%s", name);
560 	ei->strindex[n] = nxt;
561 
562 	/* Adjust everything for next entry */
563 	nxt += strnlen(name, ((sizeof (ei->strings) - nxt) - 1));
564 	ei->strings[nxt++] = '\0';
565 
566 	ei->nvalues++;
567 	return (nxt);
568 }
569 
570 /*
571  * The following two functions are used to count the number of devices
572  * in under the boomer framework.
573  *
574  * We actually report the highest "index", and then if an audio device
575  * is not found, we report a bogus removed device for it in the actual
576  * ioctls.  This goofiness is required to make the OSS API happy.
577  */
578 int
579 oss_dev_walker(audio_dev_t *d, void *arg)
580 {
581 	int		*pcnt = arg;
582 	int		cnt;
583 	int		index;
584 
585 	cnt = *pcnt;
586 	index = auclnt_get_dev_index(d);
587 	if ((index + 1) > cnt) {
588 		cnt = index + 1;
589 		*pcnt = cnt;
590 	}
591 
592 	return (AUDIO_WALK_CONTINUE);
593 }
594 
595 static int
596 oss_cnt_devs(void)
597 {
598 	int cnt = 0;
599 
600 	auclnt_walk_devs(oss_dev_walker, &cnt);
601 	return (cnt);
602 }
603 
604 static int
605 sndctl_dsp_speed(audio_client_t *c, int *ratep)
606 {
607 	int		rv;
608 	int		rate;
609 	int		oflag;
610 
611 	rate = *ratep;
612 
613 	oflag = auclnt_get_oflag(c);
614 	if (oflag & FREAD) {
615 		if ((rv = auclnt_set_rate(auclnt_input_stream(c), rate)) != 0)
616 			return (rv);
617 	}
618 
619 	if (oflag & FWRITE) {
620 		if ((rv = auclnt_set_rate(auclnt_output_stream(c), rate)) != 0)
621 			return (rv);
622 	}
623 
624 	return (0);
625 }
626 
627 static int
628 sndctl_dsp_setfmt(audio_client_t *c, int *fmtp)
629 {
630 	int		rv;
631 	int		fmt;
632 	int		i;
633 	int		oflag;
634 
635 	oflag = auclnt_get_oflag(c);
636 
637 	if (*fmtp != AFMT_QUERY) {
638 		/* convert from OSS */
639 		for (i = 0; oss_formats[i].fmt != AUDIO_FORMAT_NONE; i++) {
640 			if (oss_formats[i].oss == *fmtp) {
641 				fmt = oss_formats[i].fmt;
642 				break;
643 			}
644 		}
645 		if (fmt == AUDIO_FORMAT_NONE) {
646 			/* if format not known, return ENOTSUP */
647 			return (ENOTSUP);
648 		}
649 
650 		if (oflag & FWRITE) {
651 			rv = auclnt_set_format(auclnt_output_stream(c), fmt);
652 			if (rv != 0)
653 				return (rv);
654 		}
655 
656 		if (oflag & FREAD) {
657 			rv = auclnt_set_format(auclnt_input_stream(c), fmt);
658 			if (rv != 0)
659 				return (rv);
660 		}
661 	}
662 
663 	if (oflag & FWRITE) {
664 		fmt = auclnt_get_format(auclnt_output_stream(c));
665 	} else if (oflag & FREAD) {
666 		fmt = auclnt_get_format(auclnt_input_stream(c));
667 	}
668 
669 	/* convert back to OSS */
670 	*(int *)fmtp = AFMT_QUERY;
671 	for (i = 0; oss_formats[i].fmt != AUDIO_FORMAT_NONE; i++) {
672 		if (oss_formats[i].fmt == fmt) {
673 			*(int *)fmtp = oss_formats[i].oss;
674 		}
675 	}
676 
677 	return (0);
678 }
679 
680 static int
681 sndctl_dsp_getfmts(audio_client_t *c, int *fmtsp)
682 {
683 	_NOTE(ARGUNUSED(c));
684 
685 	/*
686 	 * For now, we support all the standard ones.  Later we might
687 	 * add in conditional support for AC3.
688 	 */
689 	*fmtsp = (AFMT_MU_LAW | AFMT_A_LAW |
690 	    AFMT_U8 | AFMT_S8 |
691 	    AFMT_S16_LE |AFMT_S16_BE |
692 	    AFMT_S24_LE | AFMT_S24_BE |
693 	    AFMT_S32_LE | AFMT_S32_BE |
694 	    AFMT_S24_PACKED);
695 
696 	return (0);
697 }
698 
699 static int
700 sndctl_dsp_channels(audio_client_t *c, int *chanp)
701 {
702 	int		rv;
703 	int		nchan;
704 	int		oflag;
705 
706 	oflag = auclnt_get_oflag(c);
707 
708 	nchan = *chanp;
709 	if (nchan != 0) {
710 		if (oflag & FWRITE) {
711 			rv = auclnt_set_channels(auclnt_output_stream(c),
712 			    nchan);
713 			if (rv != 0)
714 				return (rv);
715 		}
716 
717 		if (oflag & FREAD) {
718 			rv = auclnt_set_channels(auclnt_input_stream(c), nchan);
719 			if (rv != 0)
720 				return (rv);
721 		}
722 	}
723 
724 	if (oflag & FWRITE) {
725 		nchan = auclnt_get_channels(auclnt_output_stream(c));
726 	} else if (oflag & FREAD) {
727 		nchan = auclnt_get_channels(auclnt_input_stream(c));
728 	}
729 	*chanp = nchan;
730 	return (0);
731 }
732 
733 static int
734 sndctl_dsp_stereo(audio_client_t *c, int *onoff)
735 {
736 	int	nchan;
737 
738 	switch (*onoff) {
739 	case 0:
740 		nchan = 1;
741 		break;
742 	case 1:
743 		nchan = 2;
744 		break;
745 	default:
746 		return (EINVAL);
747 	}
748 
749 	return (sndctl_dsp_channels(c, &nchan));
750 }
751 
752 static int
753 sndctl_dsp_post(audio_client_t *c)
754 {
755 	if (auclnt_get_oflag(c) & FWRITE) {
756 		audio_stream_t	*sp = auclnt_output_stream(c);
757 		auclnt_flush(sp);
758 		auclnt_clear_paused(sp);
759 	}
760 	return (0);
761 }
762 
763 static int
764 sndctl_dsp_getcaps(audio_client_t *c, int *capsp)
765 {
766 	int		ncaps;
767 	int		osscaps = 0;
768 
769 	ncaps = auclnt_get_dev_capab(auclnt_get_dev(c));
770 
771 	if (ncaps & AUDIO_CLIENT_CAP_PLAY)
772 		osscaps |= PCM_CAP_OUTPUT;
773 	if (ncaps & AUDIO_CLIENT_CAP_RECORD)
774 		osscaps |= PCM_CAP_INPUT;
775 	if (ncaps & AUDIO_CLIENT_CAP_DUPLEX)
776 		osscaps |= PCM_CAP_DUPLEX;
777 
778 	*capsp = osscaps;
779 	return (0);
780 }
781 
782 static int
783 sndctl_dsp_gettrigger(audio_client_t *c, int *trigp)
784 {
785 	int		triggers = 0;
786 	int		oflag;
787 
788 	oflag = auclnt_get_oflag(c);
789 
790 	if (oflag & FWRITE) {
791 		if (!auclnt_is_paused(auclnt_output_stream(c))) {
792 			triggers |= PCM_ENABLE_OUTPUT;
793 		}
794 	}
795 
796 	if (oflag & FREAD) {
797 		if (!auclnt_is_paused(auclnt_input_stream(c))) {
798 			triggers |= PCM_ENABLE_INPUT;
799 		}
800 	}
801 	*trigp = triggers;
802 
803 	return (0);
804 }
805 
806 static int
807 sndctl_dsp_settrigger(audio_client_t *c, int *trigp)
808 {
809 	int		triggers;
810 	int		oflag;
811 	audio_stream_t	*sp;
812 
813 	oflag = auclnt_get_oflag(c);
814 	triggers = *trigp;
815 
816 	if ((oflag & FWRITE) && (triggers & PCM_ENABLE_OUTPUT)) {
817 		sp = auclnt_output_stream(c);
818 		auclnt_clear_paused(sp);
819 		auclnt_start(sp);
820 	}
821 
822 	if ((oflag & FREAD) && (triggers & PCM_ENABLE_INPUT)) {
823 		sp = auclnt_input_stream(c);
824 		auclnt_clear_paused(sp);
825 		auclnt_start(sp);
826 	}
827 
828 	return (0);
829 }
830 
831 struct oss_legacy_volume {
832 	pid_t		pid;
833 	uint8_t		ogain;
834 	uint8_t		igain;
835 };
836 
837 static int
838 oss_legacy_volume_walker(audio_client_t *c, void *arg)
839 {
840 	struct oss_legacy_volume	 *olv = arg;
841 
842 	if (auclnt_get_pid(c) == olv->pid) {
843 		if (olv->ogain <= 100) {
844 			auclnt_set_gain(auclnt_output_stream(c), olv->ogain);
845 		}
846 		if (olv->igain <= 100) {
847 			auclnt_set_gain(auclnt_input_stream(c), olv->igain);
848 		}
849 	}
850 	return (AUDIO_WALK_CONTINUE);
851 }
852 
853 static void
854 oss_set_legacy_volume(audio_client_t *c, uint8_t ogain, uint8_t igain)
855 {
856 	struct oss_legacy_volume olv;
857 
858 	olv.pid = auclnt_get_pid(c);
859 	olv.ogain = ogain;
860 	olv.igain = igain;
861 	auclnt_dev_walk_clients(auclnt_get_dev(c),
862 	    oss_legacy_volume_walker, &olv);
863 }
864 
865 static int
866 sndctl_dsp_getplayvol(audio_client_t *c, int *volp)
867 {
868 	int	vol;
869 
870 	/* convert monophonic soft value to OSS stereo value */
871 	vol = auclnt_get_gain(auclnt_output_stream(c));
872 	*volp = vol | (vol << 8);
873 	return (0);
874 }
875 
876 static int
877 sndctl_dsp_setplayvol(audio_client_t *c, int *volp)
878 {
879 	uint8_t		vol;
880 
881 	vol = *volp & 0xff;
882 	if (vol > 100) {
883 		return (EINVAL);
884 	}
885 
886 	auclnt_set_gain(auclnt_output_stream(c), vol);
887 	*volp = (vol | (vol << 8));
888 
889 	return (0);
890 }
891 
892 static int
893 sndctl_dsp_getrecvol(audio_client_t *c, int *volp)
894 {
895 	int	vol;
896 
897 	vol = auclnt_get_gain(auclnt_input_stream(c));
898 	*volp = (vol | (vol << 8));
899 	return (0);
900 }
901 
902 static int
903 sndctl_dsp_setrecvol(audio_client_t *c, int *volp)
904 {
905 	uint8_t		vol;
906 
907 	vol = *volp & 0xff;
908 	if (vol > 100) {
909 		return (EINVAL);
910 	}
911 
912 	auclnt_set_gain(auclnt_input_stream(c), vol);
913 	*volp = (vol | (vol << 8));
914 
915 	return (0);
916 }
917 
918 static int
919 sound_mixer_write_ogain(audio_client_t *c, int *volp)
920 {
921 	uint8_t		vol;
922 
923 	vol = *volp & 0xff;
924 	if (vol > 100) {
925 		return (EINVAL);
926 	}
927 	oss_set_legacy_volume(c, vol, 255);
928 	*volp = (vol | (vol << 8));
929 	return (0);
930 }
931 
932 static int
933 sound_mixer_write_igain(audio_client_t *c, int *volp)
934 {
935 	uint8_t		vol;
936 
937 	vol = *volp & 0xff;
938 	if (vol > 100) {
939 		return (EINVAL);
940 	}
941 	oss_set_legacy_volume(c, 255, vol);
942 	*volp = (vol | (vol << 8));
943 	return (0);
944 }
945 
946 static int
947 sndctl_dsp_readctl(audio_client_t *c, oss_digital_control *ctl)
948 {
949 	/* SPDIF: need to add support with spdif */
950 	_NOTE(ARGUNUSED(c));
951 	_NOTE(ARGUNUSED(ctl));
952 	return (ENOTSUP);
953 }
954 
955 static int
956 sndctl_dsp_writectl(audio_client_t *c, oss_digital_control *ctl)
957 {
958 	/* SPDIF: need to add support with spdif */
959 	_NOTE(ARGUNUSED(c));
960 	_NOTE(ARGUNUSED(ctl));
961 	return (ENOTSUP);
962 }
963 
964 static int
965 sndctl_dsp_cookedmode(audio_client_t *c, int *rvp)
966 {
967 	_NOTE(ARGUNUSED(c));
968 
969 	/* We are *always* in cooked mode -- at least until we have AC3. */
970 	if (*rvp == 0) {
971 		return (ENOTSUP);
972 	} else {
973 		return (0);
974 	}
975 }
976 
977 static int
978 sndctl_dsp_silence(audio_client_t *c)
979 {
980 	if (auclnt_get_oflag(c) & FWRITE) {
981 		audio_stream_t	*sp = auclnt_output_stream(c);
982 		auclnt_set_paused(sp);
983 		auclnt_flush(sp);
984 	}
985 	return (0);
986 }
987 
988 static int
989 sndctl_dsp_skip(audio_client_t *c)
990 {
991 	if (auclnt_get_oflag(c) & FWRITE) {
992 		audio_stream_t	*sp = auclnt_output_stream(c);
993 		auclnt_set_paused(sp);
994 		auclnt_flush(sp);
995 		auclnt_clear_paused(sp);
996 	}
997 	return (0);
998 }
999 
1000 static int
1001 sndctl_dsp_halt_input(audio_client_t *c)
1002 {
1003 	if (auclnt_get_oflag(c) & FREAD) {
1004 		audio_stream_t	*sp = auclnt_input_stream(c);
1005 		auclnt_set_paused(sp);
1006 		auclnt_flush(sp);
1007 	}
1008 	return (0);
1009 }
1010 
1011 static int
1012 sndctl_dsp_halt_output(audio_client_t *c)
1013 {
1014 	if (auclnt_get_oflag(c) & FWRITE) {
1015 		audio_stream_t	*sp = auclnt_output_stream(c);
1016 		auclnt_set_paused(sp);
1017 		auclnt_flush(sp);
1018 	}
1019 	return (0);
1020 }
1021 
1022 static int
1023 sndctl_dsp_halt(audio_client_t *c)
1024 {
1025 	(void) sndctl_dsp_halt_input(c);
1026 	(void) sndctl_dsp_halt_output(c);
1027 	return (0);
1028 }
1029 
1030 static int
1031 sndctl_dsp_sync(audio_client_t *c)
1032 {
1033 	return (auclnt_drain(c));
1034 }
1035 
1036 static int
1037 sndctl_dsp_setfragment(audio_client_t *c, int *fragp)
1038 {
1039 	int	bufsz;
1040 	int	nfrags;
1041 	int	fragsz;
1042 
1043 	nfrags = (*fragp) >> 16;
1044 	if ((nfrags >= 0x7fffU) || (nfrags < 2)) {
1045 		/* use infinite setting... no change */
1046 		return (0);
1047 	}
1048 
1049 	fragsz = (*fragp) & 0xffff;
1050 	if (fragsz > 16) {
1051 		/* basically too big, so, no change */
1052 		return (0);
1053 	}
1054 	bufsz = (1U << fragsz) * nfrags;
1055 
1056 	/*
1057 	 * Now we have our desired buffer size, but we have to
1058 	 * make sure we have a whole number of fragments >= 2, and
1059 	 * less than the maximum.
1060 	 */
1061 	bufsz = ((*fragp) >> 16) * (1U << (*fragp));
1062 	if (bufsz >= 65536) {
1063 		return (0);
1064 	}
1065 
1066 	/*
1067 	 * We set the latency hints in terms of bytes, not fragments.
1068 	 */
1069 	auclnt_set_latency(auclnt_output_stream(c), 0, bufsz);
1070 	auclnt_set_latency(auclnt_input_stream(c), 0, bufsz);
1071 
1072 	/*
1073 	 * According to the OSS API documentation, the values provided
1074 	 * are nothing more than a "hint" and not to be relied upon
1075 	 * anyway.  And we aren't obligated to report the actual
1076 	 * values back!
1077 	 */
1078 	return (0);
1079 }
1080 
1081 static int
1082 sndctl_dsp_policy(audio_client_t *c, int *policy)
1083 {
1084 	int	hint = *policy;
1085 	if ((hint >= 2) && (hint <= 10)) {
1086 		auclnt_set_latency(auclnt_input_stream(c), hint, 0);
1087 		auclnt_set_latency(auclnt_output_stream(c), hint, 0);
1088 	}
1089 	return (0);
1090 }
1091 
1092 /*
1093  * A word about recsrc, and playtgt ioctls: We don't allow ordinary DSP
1094  * applications to change port configurations, because these could have a
1095  * bad effect for other applications.  Instead, these settings have to
1096  * be changed using the master mixer panel.  In order to make applications
1097  * happy, we just present a single "default" source/target.
1098  */
1099 static int
1100 sndctl_dsp_get_recsrc_names(audio_client_t *c, oss_mixer_enuminfo *ei)
1101 {
1102 	_NOTE(ARGUNUSED(c));
1103 
1104 	ei->nvalues = 1;
1105 	(void) snprintf(ei->strings, sizeof (ei->strings), "default");
1106 	ei->strindex[0] = 0;
1107 
1108 	return (0);
1109 }
1110 
1111 static int
1112 sndctl_dsp_get_recsrc(audio_client_t *c, int *srcp)
1113 {
1114 	_NOTE(ARGUNUSED(c));
1115 	*srcp = 0;
1116 	return (0);
1117 }
1118 
1119 static int
1120 sndctl_dsp_set_recsrc(audio_client_t *c, int *srcp)
1121 {
1122 	_NOTE(ARGUNUSED(c));
1123 	*srcp = 0;
1124 	return (0);
1125 }
1126 
1127 static int
1128 sndctl_dsp_get_playtgt_names(audio_client_t *c, oss_mixer_enuminfo *ei)
1129 {
1130 	_NOTE(ARGUNUSED(c));
1131 
1132 	ei->nvalues = 1;
1133 	(void) snprintf(ei->strings, sizeof (ei->strings), "default");
1134 	ei->strindex[0] = 0;
1135 
1136 	return (0);
1137 }
1138 
1139 static int
1140 sndctl_dsp_get_playtgt(audio_client_t *c, int *tgtp)
1141 {
1142 	_NOTE(ARGUNUSED(c));
1143 	*tgtp = 0;
1144 	return (0);
1145 }
1146 
1147 static int
1148 sndctl_dsp_set_playtgt(audio_client_t *c, int *tgtp)
1149 {
1150 	_NOTE(ARGUNUSED(c));
1151 	*tgtp = 0;
1152 	return (0);
1153 }
1154 
1155 static int
1156 sndctl_sysinfo(oss_sysinfo *si)
1157 {
1158 	bzero(si, sizeof (*si));
1159 	(void) snprintf(si->product, sizeof (si->product), "SunOS Audio");
1160 	(void) snprintf(si->version, sizeof (si->version), "4.0");
1161 	si->versionnum = OSS_VERSION;
1162 	si->numcards = oss_cnt_devs();
1163 	si->nummixers = si->numcards - 1;
1164 	si->numaudios = si->numcards - 1;
1165 	si->numaudioengines = si->numaudios;
1166 	(void) snprintf(si->license, sizeof (si->license), "CDDL");
1167 	return (0);
1168 }
1169 
1170 static int
1171 sndctl_cardinfo(audio_client_t *c, oss_card_info *ci)
1172 {
1173 	audio_dev_t	*d;
1174 	void		*iter;
1175 	const char 	*info;
1176 	int		n;
1177 	boolean_t	release;
1178 
1179 	if ((n = ci->card) == -1) {
1180 		release = B_FALSE;
1181 		d = auclnt_get_dev(c);
1182 		n = auclnt_get_dev_index(d);
1183 	} else {
1184 		release = B_TRUE;
1185 		d = auclnt_hold_dev_by_index(n);
1186 	}
1187 
1188 	bzero(ci, sizeof (*ci));
1189 	ci->card = n;
1190 
1191 	if (d == NULL) {
1192 		/*
1193 		 * If device removed (e.g. for DR), then
1194 		 * report a bogus removed entry.
1195 		 */
1196 		(void) snprintf(ci->shortname, sizeof (ci->shortname),
1197 		    "<removed>");
1198 		(void) snprintf(ci->longname, sizeof (ci->longname),
1199 		    "<removed>");
1200 		return (0);
1201 	}
1202 
1203 	(void) snprintf(ci->shortname, sizeof (ci->shortname),
1204 	    "%s", auclnt_get_dev_name(d));
1205 	(void) snprintf(ci->longname, sizeof (ci->longname),
1206 	    "%s (%s)", auclnt_get_dev_description(d),
1207 	    auclnt_get_dev_version(d));
1208 
1209 	iter = NULL;
1210 	while ((info = auclnt_get_dev_hw_info(d, &iter)) != NULL) {
1211 		(void) strlcat(ci->hw_info, info, sizeof (ci->hw_info));
1212 		(void) strlcat(ci->hw_info, "\n", sizeof (ci->hw_info));
1213 	}
1214 
1215 	/*
1216 	 * We don't report interrupt counts, ack counts (which are
1217 	 * just "read" interrupts, not spurious), or any other flags.
1218 	 * Nothing should be using any of this data anyway ... these
1219 	 * values were intended for 4Front's debugging purposes.  In
1220 	 * Solaris, drivers should use interrupt kstats to report
1221 	 * interrupt related statistics.
1222 	 */
1223 	if (release)
1224 		auclnt_release_dev(d);
1225 	return (0);
1226 }
1227 
1228 static int
1229 audioinfo_walker(audio_engine_t *e, void *a)
1230 {
1231 	oss_audioinfo *si = a;
1232 	int fmt, nchan, rate, cap;
1233 
1234 	fmt = auclnt_engine_get_format(e);
1235 	nchan = auclnt_engine_get_channels(e);
1236 	rate = auclnt_engine_get_rate(e);
1237 	cap = auclnt_engine_get_capab(e);
1238 
1239 	for (int i = 0; oss_formats[i].fmt != AUDIO_FORMAT_NONE; i++) {
1240 		if (fmt == oss_formats[i].fmt) {
1241 			if (cap & AUDIO_CLIENT_CAP_PLAY) {
1242 				si->oformats |= oss_formats[i].oss;
1243 			}
1244 			if (cap & AUDIO_CLIENT_CAP_RECORD) {
1245 				si->iformats |= oss_formats[i].oss;
1246 			}
1247 			break;
1248 		}
1249 	}
1250 	si->max_channels = max(nchan, si->max_channels);
1251 	si->max_rate = max(rate, si->max_rate);
1252 
1253 	return (AUDIO_WALK_CONTINUE);
1254 }
1255 
1256 static int
1257 sndctl_audioinfo(audio_client_t *c, oss_audioinfo *si)
1258 {
1259 	audio_dev_t		*d;
1260 	const char		*name;
1261 	int			n;
1262 	boolean_t		release;
1263 	unsigned		cap;
1264 
1265 	if ((n = si->dev) == -1) {
1266 		release = B_FALSE;
1267 		d = auclnt_get_dev(c);
1268 		n = auclnt_get_dev_index(d);
1269 	} else {
1270 		release = B_TRUE;
1271 		n++;	/* skip pseudo device */
1272 		d = auclnt_hold_dev_by_index(n);
1273 	}
1274 
1275 	bzero(si, sizeof (*si));
1276 	si->dev = n - 1;
1277 
1278 	if (d == NULL) {
1279 		/* if device not present, forge a false entry */
1280 		si->card_number = n;
1281 		si->mixer_dev = n - 1;
1282 		si->legacy_device = -1;
1283 		si->enabled = 0;
1284 		(void) snprintf(si->name, sizeof (si->name), "<removed>");
1285 		return (0);
1286 	}
1287 
1288 	name = auclnt_get_dev_name(d);
1289 	(void) snprintf(si->name, sizeof (si->name), "%s", name);
1290 
1291 	si->legacy_device = auclnt_get_dev_number(d);
1292 	si->caps = 0;
1293 
1294 	auclnt_dev_walk_engines(d, audioinfo_walker, si);
1295 
1296 	cap = auclnt_get_dev_capab(d);
1297 
1298 	if (cap	& AUDIO_CLIENT_CAP_DUPLEX) {
1299 		si->caps |= PCM_CAP_DUPLEX;
1300 	}
1301 	if (cap & AUDIO_CLIENT_CAP_PLAY) {
1302 		si->caps |= PCM_CAP_OUTPUT;
1303 	}
1304 	if (cap & AUDIO_CLIENT_CAP_RECORD) {
1305 		si->caps |= PCM_CAP_INPUT;
1306 	}
1307 
1308 	if (si->caps != 0) {
1309 		/* AC3: PCM_CAP_MULTI would be wrong for an AC3 only device */
1310 		si->caps |= PCM_CAP_BATCH | PCM_CAP_TRIGGER | PCM_CAP_MULTI;
1311 		/* MMAP: we add PCM_CAP_MMAP when we we support it */
1312 		si->enabled = 1;
1313 		si->rate_source = si->dev;
1314 
1315 		/* we can convert PCM formats */
1316 		if ((si->iformats | si->oformats) &
1317 		    AUDIO_FORMAT_PCM) {
1318 			si->min_channels = min(2, si->max_channels);
1319 			si->min_rate = min(5000, si->max_rate);
1320 			si->caps |= PCM_CAP_FREERATE;
1321 		}
1322 		(void) snprintf(si->devnode, sizeof (si->devnode),
1323 		    "/dev/sound/%s:%ddsp",
1324 		    auclnt_get_dev_driver(d), auclnt_get_dev_instance(d));
1325 	} else {
1326 		si->enabled = 0;	/* stops apps from using us directly */
1327 		si->caps = PCM_CAP_VIRTUAL;
1328 		(void) snprintf(si->devnode, sizeof (si->devnode),
1329 		    "/dev/sndstat");
1330 	}
1331 
1332 	si->pid = -1;
1333 	(void) snprintf(si->handle, sizeof (si->handle), "%s", name);
1334 	(void) snprintf(si->label, sizeof (si->label), "%s", name);
1335 	si->latency = -1;
1336 	si->card_number = n;
1337 	si->mixer_dev = n - 1;
1338 
1339 	if (release)
1340 		auclnt_release_dev(d);
1341 
1342 	return (0);
1343 }
1344 
1345 static int
1346 sound_mixer_info(audio_client_t *c, mixer_info *mi)
1347 {
1348 	audio_dev_t	*d;
1349 	ossdev_t	*odev;
1350 	ossclient_t	*sc;
1351 	const char	*name;
1352 
1353 	sc = auclnt_get_private(c);
1354 	odev = sc->o_ossdev;
1355 
1356 	d = auclnt_get_dev(c);
1357 
1358 	name = auclnt_get_dev_name(d);
1359 	(void) snprintf(mi->id, sizeof (mi->id), "%s", name);
1360 	(void) snprintf(mi->name, sizeof (mi->name), "%s", name);
1361 	(void) snprintf(mi->handle, sizeof (mi->handle), "%s", name);
1362 	mi->modify_counter = odev->d_modify_cnt;
1363 	mi->card_number = auclnt_get_dev_index(d);
1364 	mi->port_number = 0;
1365 	return (0);
1366 }
1367 
1368 static int
1369 sound_mixer_read_devmask(audio_client_t *c, int *devmask)
1370 {
1371 	_NOTE(ARGUNUSED(c));
1372 	*devmask = SOUND_MASK_VOLUME | SOUND_MASK_PCM | SOUND_MASK_IGAIN;
1373 	return (0);
1374 }
1375 
1376 static int
1377 sound_mixer_read_recmask(audio_client_t *c, int *recmask)
1378 {
1379 	_NOTE(ARGUNUSED(c));
1380 	*recmask = 0;
1381 	return (0);
1382 }
1383 
1384 static int
1385 sound_mixer_read_recsrc(audio_client_t *c, int *recsrc)
1386 {
1387 	_NOTE(ARGUNUSED(c));
1388 	*recsrc = 0;
1389 	return (0);
1390 }
1391 
1392 static int
1393 sound_mixer_read_caps(audio_client_t *c, int *caps)
1394 {
1395 	_NOTE(ARGUNUSED(c));
1396 	/* single recording source... sort of */
1397 	*caps = SOUND_CAP_EXCL_INPUT;
1398 	return (0);
1399 }
1400 
1401 static int
1402 sndctl_mixerinfo(audio_client_t *c, oss_mixerinfo *mi)
1403 {
1404 	audio_dev_t		*d;
1405 	ossdev_t 		*odev;
1406 	const char		*name;
1407 	int			n;
1408 	boolean_t		release = B_FALSE;
1409 
1410 	if ((n = mi->dev) == -1) {
1411 		release = B_FALSE;
1412 		d = auclnt_get_dev(c);
1413 		n = auclnt_get_dev_index(d);
1414 	} else {
1415 		release = B_TRUE;
1416 		n++;
1417 		d = auclnt_hold_dev_by_index(n);
1418 	}
1419 
1420 	bzero(mi, sizeof (*mi));
1421 	mi->dev = n - 1;
1422 
1423 	if (d == NULL) {
1424 		mi->card_number = n;
1425 		mi->enabled = 0;
1426 		mi->legacy_device = -1;
1427 		(void) snprintf(mi->name, sizeof (mi->name), "<removed>");
1428 		(void) snprintf(mi->id, sizeof (mi->id), "<removed>");
1429 		return (0);
1430 	}
1431 
1432 	if ((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) {
1433 		if (release)
1434 			auclnt_release_dev(d);
1435 		return (EINVAL);
1436 	}
1437 
1438 	name = auclnt_get_dev_name(d);
1439 	(void) snprintf(mi->name, sizeof (mi->name), "%s", name);
1440 	(void) snprintf(mi->id, sizeof (mi->id), "%s", name);
1441 	(void) snprintf(mi->handle, sizeof (mi->handle), "%s", name);
1442 	mi->modify_counter = odev->d_modify_cnt;
1443 	mi->card_number = auclnt_get_dev_index(d);
1444 	mi->legacy_device = auclnt_get_dev_number(d);
1445 	if (mi->legacy_device >= 0) {
1446 		(void) snprintf(mi->devnode, sizeof (mi->devnode),
1447 		    "/dev/sound/%s:%dmixer",
1448 		    auclnt_get_dev_driver(d), auclnt_get_dev_instance(d));
1449 		mi->enabled = 1;
1450 	} else {
1451 		/* special nodes use generic sndstat node */
1452 		(void) snprintf(mi->devnode, sizeof (mi->devnode),
1453 		    "/dev/sndstat");
1454 		mi->enabled = 0;
1455 	}
1456 	mi->nrext = odev->d_nctrl;
1457 
1458 	if (release)
1459 		auclnt_release_dev(d);
1460 
1461 	return (0);
1462 }
1463 
1464 static int
1465 sndctl_dsp_getblksize(audio_client_t *c, int *fragsz)
1466 {
1467 	int	oflag = auclnt_get_oflag(c);
1468 
1469 	if (oflag & FWRITE)
1470 		*fragsz  = auclnt_get_fragsz(auclnt_output_stream(c));
1471 	else if (oflag & FREAD)
1472 		*fragsz  = auclnt_get_fragsz(auclnt_input_stream(c));
1473 
1474 	return (0);
1475 }
1476 
1477 static int
1478 sndctl_dsp_getospace(audio_client_t *c, audio_buf_info *bi)
1479 {
1480 	audio_stream_t	*sp;
1481 	unsigned	n;
1482 
1483 	if ((auclnt_get_oflag(c) & FWRITE) == 0) {
1484 		return (EACCES);
1485 	}
1486 
1487 	sp = auclnt_output_stream(c);
1488 	n = auclnt_get_nframes(sp) - auclnt_get_count(sp);
1489 
1490 	bi->fragsize  = auclnt_get_fragsz(sp);
1491 	bi->fragstotal = auclnt_get_nfrags(sp);
1492 	bi->bytes = (n * auclnt_get_framesz(sp));
1493 	bi->fragments = bi->bytes / bi->fragsize;
1494 
1495 	return (0);
1496 }
1497 
1498 static int
1499 sndctl_dsp_getispace(audio_client_t *c, audio_buf_info *bi)
1500 {
1501 	audio_stream_t	*sp;
1502 	unsigned	n;
1503 
1504 	if ((auclnt_get_oflag(c) & FREAD) == 0) {
1505 		return (EACCES);
1506 	}
1507 
1508 	sp = auclnt_input_stream(c);
1509 	n = auclnt_get_count(sp);
1510 
1511 	bi->fragsize  = auclnt_get_fragsz(sp);
1512 	bi->fragstotal = auclnt_get_nfrags(sp);
1513 	bi->bytes = (n * auclnt_get_framesz(sp));
1514 	bi->fragments = bi->bytes / bi->fragsize;
1515 
1516 	return (0);
1517 }
1518 
1519 static int
1520 sndctl_dsp_getodelay(audio_client_t *c, int *bytes)
1521 {
1522 	unsigned	framesz;
1523 	unsigned	slen, flen;
1524 
1525 	if (auclnt_get_oflag(c) & FWRITE) {
1526 		audio_stream_t	*sp = auclnt_output_stream(c);
1527 		framesz = auclnt_get_framesz(sp);
1528 		auclnt_get_output_qlen(c, &slen, &flen);
1529 		*bytes = (slen + flen) * framesz;
1530 	} else {
1531 		*bytes = 0;
1532 	}
1533 	return (0);
1534 }
1535 
1536 static int
1537 sndctl_dsp_current_iptr(audio_client_t *c, oss_count_t *count)
1538 {
1539 	if (auclnt_get_oflag(c) & FREAD) {
1540 		count->samples = auclnt_get_samples(auclnt_input_stream(c));
1541 		count->fifo_samples = 0;	/* not quite accurate */
1542 	} else {
1543 		count->samples = 0;
1544 		count->fifo_samples = 0;
1545 	}
1546 	return (0);
1547 }
1548 
1549 static int
1550 sndctl_dsp_current_optr(audio_client_t *c, oss_count_t *count)
1551 {
1552 	unsigned samples, fifo;
1553 
1554 	if (auclnt_get_oflag(c) & FWRITE) {
1555 		auclnt_get_output_qlen(c, &samples, &fifo);
1556 		count->samples = samples;
1557 		count->fifo_samples = fifo;
1558 	} else {
1559 		count->samples = 0;
1560 		count->fifo_samples = 0;
1561 	}
1562 	return (0);
1563 }
1564 
1565 static int
1566 sndctl_dsp_getoptr(audio_client_t *c, count_info *ci)
1567 {
1568 	audio_stream_t	*sp;
1569 	unsigned	framesz;
1570 	unsigned	fragsz;
1571 
1572 	bzero(ci, sizeof (*ci));
1573 	if ((auclnt_get_oflag(c) & FWRITE) == 0) {
1574 		return (0);
1575 	}
1576 	sp = auclnt_output_stream(c);
1577 	framesz = auclnt_get_framesz(sp);
1578 	fragsz = auclnt_get_fragsz(sp);
1579 	ci->blocks = auclnt_get_samples(sp) * framesz / fragsz;
1580 	auclnt_set_samples(sp, 0);
1581 	ci->bytes = auclnt_get_tail(sp) * framesz;
1582 	ci->ptr = auclnt_get_tidx(sp) * framesz;
1583 	return (0);
1584 }
1585 
1586 static int
1587 sndctl_dsp_getiptr(audio_client_t *c, count_info *ci)
1588 {
1589 	audio_stream_t	*sp;
1590 	unsigned	framesz;
1591 	unsigned	fragsz;
1592 
1593 	bzero(ci, sizeof (*ci));
1594 	if ((auclnt_get_oflag(c) & FREAD) == 0) {
1595 		return (0);
1596 	}
1597 	sp = auclnt_input_stream(c);
1598 	framesz = auclnt_get_framesz(sp);
1599 	fragsz = auclnt_get_fragsz(sp);
1600 	ci->blocks = auclnt_get_samples(sp) * framesz / fragsz;
1601 	auclnt_set_samples(sp, 0);
1602 	ci->bytes = auclnt_get_head(sp) * framesz;
1603 	ci->ptr = auclnt_get_hidx(sp) * framesz;
1604 	return (0);
1605 }
1606 
1607 static int
1608 sndctl_dsp_geterror(audio_client_t *c, audio_errinfo *bi)
1609 {
1610 	audio_stream_t	*sp;
1611 	unsigned	fragsz;
1612 	/*
1613 	 * Note: The use of this structure is unsafe... different
1614 	 * meanings for error codes are used by different implementations,
1615 	 * according to the spec.  (Even different versions of the same
1616 	 * implementation could have different values.)
1617 	 *
1618 	 * Rather than try to come up with a reliable solution here, we
1619 	 * don't use it.  If you want to report errors, or see the result
1620 	 * of errors, use syslog.
1621 	 */
1622 	bzero(bi, sizeof (*bi));
1623 
1624 	sp = auclnt_output_stream(c);
1625 	fragsz = max(auclnt_get_fragsz(sp), 1);
1626 	bi->play_underruns = (int)((auclnt_get_errors(sp) + (fragsz - 1)) /
1627 	    fragsz);
1628 	auclnt_set_errors(sp, 0);
1629 
1630 	sp = auclnt_input_stream(c);
1631 	fragsz = max(auclnt_get_fragsz(sp), 1);
1632 	bi->rec_overruns = (int)((auclnt_get_errors(sp) + (fragsz - 1)) /
1633 	    fragsz);
1634 	auclnt_set_errors(sp, 0);
1635 
1636 	return (0);
1637 }
1638 
1639 static int
1640 sndctl_sun_send_number(audio_client_t *c, int *num, cred_t *cr)
1641 {
1642 	audio_dev_t	*dev;
1643 	int		rv;
1644 
1645 	if ((rv = drv_priv(cr)) != 0) {
1646 		return (rv);
1647 	}
1648 
1649 	dev = auclnt_get_dev(c);
1650 	auclnt_set_dev_number(dev, *num);
1651 	return (0);
1652 }
1653 
1654 static int
1655 oss_getversion(int *versp)
1656 {
1657 	*versp = OSS_VERSION;
1658 	return (0);
1659 }
1660 
1661 static int
1662 oss_ioctl(audio_client_t *c, int cmd, intptr_t arg, int mode, cred_t *credp,
1663     int *rvalp)
1664 {
1665 	int	sz;
1666 	void	*data;
1667 	int	rv = 0;
1668 
1669 	_NOTE(ARGUNUSED(credp));
1670 
1671 	sz = OSSIOC_GETSZ(cmd);
1672 
1673 	if ((cmd & (OSSIOC_IN | OSSIOC_OUT)) && sz) {
1674 		if ((data = kmem_zalloc(sz, KM_NOSLEEP)) == NULL) {
1675 			return (ENOMEM);
1676 		}
1677 	} else {
1678 		sz = 0;
1679 	}
1680 
1681 	if (cmd & OSSIOC_IN) {
1682 		if ((rv = ddi_copyin((void *)arg, data, sz, mode)) != 0) {
1683 			goto done;
1684 		}
1685 	}
1686 
1687 	switch (cmd) {
1688 		/*
1689 		 * DSP specific ioctls
1690 		 */
1691 	case SNDCTL_DSP_HALT:
1692 		rv = sndctl_dsp_halt(c);
1693 		break;
1694 
1695 	case SNDCTL_DSP_SYNC:
1696 		rv = sndctl_dsp_sync(c);
1697 		break;
1698 
1699 	case SNDCTL_DSP_SPEED:
1700 		rv = sndctl_dsp_speed(c, (int *)data);
1701 		break;
1702 	case SNDCTL_DSP_SETFMT:
1703 		rv = sndctl_dsp_setfmt(c, (int *)data);
1704 		break;
1705 	case SNDCTL_DSP_GETFMTS:
1706 		rv = sndctl_dsp_getfmts(c, (int *)data);
1707 		break;
1708 	case SNDCTL_DSP_STEREO:
1709 		rv = sndctl_dsp_stereo(c, (int *)data);
1710 		break;
1711 	case SNDCTL_DSP_CHANNELS:
1712 		rv = sndctl_dsp_channels(c, (int *)data);
1713 		break;
1714 	case SNDCTL_DSP_POST:
1715 		rv = sndctl_dsp_post(c);
1716 		break;
1717 	case SNDCTL_DSP_GETCAPS:
1718 		rv = sndctl_dsp_getcaps(c, (int *)data);
1719 		break;
1720 	case SNDCTL_DSP_GETTRIGGER:
1721 		rv = sndctl_dsp_gettrigger(c, (int *)data);
1722 		break;
1723 	case SNDCTL_DSP_SETTRIGGER:
1724 		rv = sndctl_dsp_settrigger(c, (int *)data);
1725 		break;
1726 	case SNDCTL_DSP_GETPLAYVOL:
1727 	case SOUND_MIXER_READ_VOLUME:	/* legacy mixer on dsp */
1728 	case SOUND_MIXER_READ_PCM:	/* legacy mixer on dsp */
1729 	case SOUND_MIXER_READ_OGAIN:	/* legacy mixer on dsp */
1730 		rv = sndctl_dsp_getplayvol(c, (int *)data);
1731 		break;
1732 	case SOUND_MIXER_WRITE_VOLUME:	/* legacy mixer on dsp */
1733 	case SOUND_MIXER_WRITE_PCM:	/* legacy mixer on dsp */
1734 	case SOUND_MIXER_WRITE_OGAIN:	/* legacy mixer on dsp */
1735 		rv = sound_mixer_write_ogain(c, (int *)data);
1736 		break;
1737 	case SNDCTL_DSP_SETPLAYVOL:
1738 		rv = sndctl_dsp_setplayvol(c, (int *)data);
1739 		break;
1740 	case SNDCTL_DSP_READCTL:
1741 		rv = sndctl_dsp_readctl(c, (oss_digital_control *)data);
1742 		break;
1743 	case SNDCTL_DSP_WRITECTL:
1744 		rv = sndctl_dsp_writectl(c, (oss_digital_control *)data);
1745 		break;
1746 	case SNDCTL_DSP_COOKEDMODE:
1747 		rv = sndctl_dsp_cookedmode(c, (int *)data);
1748 		break;
1749 	case SNDCTL_DSP_SILENCE:
1750 		rv = sndctl_dsp_silence(c);
1751 		break;
1752 	case SNDCTL_DSP_SKIP:
1753 		rv = sndctl_dsp_skip(c);
1754 		break;
1755 	case SNDCTL_DSP_HALT_INPUT:
1756 		rv = sndctl_dsp_halt_input(c);
1757 		break;
1758 	case SNDCTL_DSP_HALT_OUTPUT:
1759 		rv = sndctl_dsp_halt_output(c);
1760 		break;
1761 	case SNDCTL_DSP_GET_RECSRC_NAMES:
1762 		rv = sndctl_dsp_get_recsrc_names(c, (oss_mixer_enuminfo *)data);
1763 		break;
1764 	case SNDCTL_DSP_SETFRAGMENT:
1765 		rv = sndctl_dsp_setfragment(c, (int *)data);
1766 		break;
1767 	case SNDCTL_DSP_GET_RECSRC:
1768 		rv = sndctl_dsp_get_recsrc(c, (int *)data);
1769 		break;
1770 	case SNDCTL_DSP_SET_RECSRC:
1771 		rv = sndctl_dsp_set_recsrc(c, (int *)data);
1772 		break;
1773 	case SNDCTL_DSP_GET_PLAYTGT_NAMES:
1774 		rv = sndctl_dsp_get_playtgt_names(c,
1775 		    (oss_mixer_enuminfo *)data);
1776 		break;
1777 	case SNDCTL_DSP_GET_PLAYTGT:
1778 		rv = sndctl_dsp_get_playtgt(c, (int *)data);
1779 		break;
1780 	case SNDCTL_DSP_SET_PLAYTGT:
1781 		rv = sndctl_dsp_set_playtgt(c, (int *)data);
1782 		break;
1783 	case SNDCTL_DSP_GETRECVOL:
1784 	case SOUND_MIXER_READ_RECGAIN:	/* legacy mixer on dsp */
1785 	case SOUND_MIXER_READ_RECLEV:	/* legacy mixer on dsp */
1786 	case SOUND_MIXER_READ_IGAIN:	/* legacy mixer on dsp */
1787 		rv = sndctl_dsp_getrecvol(c, (int *)data);
1788 		break;
1789 	case SOUND_MIXER_WRITE_RECGAIN:	/* legacy mixer on dsp */
1790 	case SOUND_MIXER_WRITE_RECLEV:	/* legacy mixer on dsp */
1791 	case SOUND_MIXER_WRITE_IGAIN:	/* legacy mixer on dsp */
1792 		rv = sound_mixer_write_igain(c, (int *)data);
1793 		break;
1794 	case SNDCTL_DSP_SETRECVOL:
1795 		rv = sndctl_dsp_setrecvol(c, (int *)data);
1796 		break;
1797 	case SNDCTL_DSP_SUBDIVIDE:	/* Ignored */
1798 	case SNDCTL_DSP_SETDUPLEX:	/* Ignored */
1799 	case SNDCTL_DSP_LOW_WATER:	/* Ignored */
1800 	case SNDCTL_DSP_PROFILE:	/* Ignored */
1801 		rv = 0;
1802 		break;
1803 	case SNDCTL_DSP_POLICY:
1804 		rv = sndctl_dsp_policy(c, (int *)data);
1805 		break;
1806 	case SNDCTL_DSP_GETBLKSIZE:
1807 		rv = sndctl_dsp_getblksize(c, (int *)data);
1808 		break;
1809 	case SNDCTL_DSP_GETOSPACE:
1810 		rv = sndctl_dsp_getospace(c, (audio_buf_info *)data);
1811 		break;
1812 	case SNDCTL_DSP_GETISPACE:
1813 		rv = sndctl_dsp_getispace(c, (audio_buf_info *)data);
1814 		break;
1815 	case SNDCTL_DSP_GETODELAY:
1816 		rv = sndctl_dsp_getodelay(c, (int *)data);
1817 		break;
1818 	case SNDCTL_DSP_GETOPTR:
1819 		rv = sndctl_dsp_getoptr(c, (count_info *)data);
1820 		break;
1821 	case SNDCTL_DSP_GETIPTR:
1822 		rv = sndctl_dsp_getiptr(c, (count_info *)data);
1823 		break;
1824 	case SNDCTL_DSP_GETERROR:
1825 		rv = sndctl_dsp_geterror(c, (audio_errinfo *)data);
1826 		break;
1827 	case SNDCTL_DSP_CURRENT_IPTR:
1828 		rv = sndctl_dsp_current_iptr(c, (oss_count_t *)data);
1829 		break;
1830 	case SNDCTL_DSP_CURRENT_OPTR:
1831 		rv = sndctl_dsp_current_optr(c, (oss_count_t *)data);
1832 		break;
1833 
1834 		/*
1835 		 * Shared ioctls with /dev/mixer.
1836 		 */
1837 	case OSS_GETVERSION:
1838 		rv = oss_getversion((int *)data);
1839 		break;
1840 	case SNDCTL_CARDINFO:
1841 		rv = sndctl_cardinfo(c, (oss_card_info *)data);
1842 		break;
1843 	case SNDCTL_ENGINEINFO:
1844 	case SNDCTL_AUDIOINFO:
1845 	case SNDCTL_AUDIOINFO_EX:
1846 		rv = sndctl_audioinfo(c, (oss_audioinfo *)data);
1847 		break;
1848 	case SNDCTL_SYSINFO:
1849 		rv = sndctl_sysinfo((oss_sysinfo *)data);
1850 		break;
1851 	case SNDCTL_MIXERINFO:
1852 		rv = sndctl_mixerinfo(c, (oss_mixerinfo *)data);
1853 		break;
1854 	case SOUND_MIXER_INFO:
1855 		rv = sound_mixer_info(c, (mixer_info *)data);
1856 		break;
1857 
1858 		/*
1859 		 * These are mixer ioctls that are virtualized for the DSP
1860 		 * device.  They are accessible via either /dev/mixer or
1861 		 * /dev/dsp.
1862 		 */
1863 	case SOUND_MIXER_READ_RECSRC:
1864 	case SOUND_MIXER_WRITE_RECSRC:
1865 		rv = sound_mixer_read_recsrc(c, (int *)data);
1866 		break;
1867 
1868 	case SOUND_MIXER_READ_DEVMASK:
1869 	case SOUND_MIXER_READ_STEREODEVS:
1870 		rv = sound_mixer_read_devmask(c, (int *)data);
1871 		break;
1872 
1873 	case SOUND_MIXER_READ_RECMASK:
1874 		rv = sound_mixer_read_recmask(c, (int *)data);
1875 		break;
1876 
1877 	case SOUND_MIXER_READ_CAPS:
1878 		rv = sound_mixer_read_caps(c, (int *)data);
1879 		break;
1880 
1881 		/*
1882 		 * Ioctls we have chosen not to support for now.  Some
1883 		 * of these are of legacy interest only.
1884 		 */
1885 	case SNDCTL_SETSONG:
1886 	case SNDCTL_GETSONG:
1887 	case SNDCTL_DSP_SYNCGROUP:
1888 	case SNDCTL_DSP_SYNCSTART:
1889 	case SNDCTL_DSP_GET_CHNORDER:
1890 	case SNDCTL_DSP_SET_CHNORDER:
1891 	case SNDCTL_DSP_GETIPEAKS:
1892 	case SNDCTL_DSP_GETOPEAKS:
1893 	case SNDCTL_DSP_GETCHANNELMASK:
1894 	case SNDCTL_DSP_BIND_CHANNEL:
1895 	case SNDCTL_DSP_SETSYNCRO:
1896 	default:
1897 		rv = EINVAL;
1898 		break;
1899 	}
1900 
1901 	if ((rv == 0) && (cmd & OSSIOC_OUT)) {
1902 		rv = ddi_copyout(data, (void *)arg, sz, mode);
1903 	}
1904 	if (rv == 0) {
1905 		*rvalp = 0;
1906 	}
1907 
1908 done:
1909 	if (sz) {
1910 		kmem_free(data, sz);
1911 	}
1912 	return (rv);
1913 }
1914 
1915 static void
1916 oss_output(audio_client_t *c)
1917 {
1918 	auclnt_pollwakeup(c, POLLOUT);
1919 }
1920 
1921 static void
1922 oss_input(audio_client_t *c)
1923 {
1924 	auclnt_pollwakeup(c, POLLIN | POLLRDNORM);
1925 }
1926 
1927 static void
1928 oss_notify(audio_client_t *c)
1929 {
1930 	audio_dev_t	*d;
1931 	ossdev_t	*odev;
1932 
1933 	d = auclnt_get_dev(c);
1934 	if ((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) {
1935 		return;
1936 	}
1937 	odev->d_modify_cnt++;
1938 }
1939 
1940 static int
1941 ossmix_open(audio_client_t *c, int oflag)
1942 {
1943 	int		rv;
1944 	ossclient_t	*sc;
1945 	ossdev_t	*odev;
1946 
1947 	_NOTE(ARGUNUSED(oflag));
1948 
1949 	if ((rv = auclnt_open(c, AUDIO_FORMAT_NONE, 0)) != 0) {
1950 		return (rv);
1951 	}
1952 
1953 	if ((sc = kmem_zalloc(sizeof (*sc), KM_NOSLEEP)) == NULL) {
1954 		return (ENOMEM);
1955 	}
1956 	sc->o_ss_sz = 8192;
1957 	if ((sc->o_ss_buf = kmem_zalloc(sc->o_ss_sz, KM_NOSLEEP)) == NULL) {
1958 		kmem_free(sc, sizeof (*sc));
1959 		return (ENOMEM);
1960 	}
1961 	auclnt_set_private(c, sc);
1962 
1963 	odev = auclnt_get_minor_data(c, AUDIO_MINOR_DSP);
1964 
1965 	/* set a couple of common fields */
1966 	sc->o_client = c;
1967 	sc->o_ossdev = odev;
1968 
1969 	return (rv);
1970 }
1971 
1972 static void
1973 ossmix_close(audio_client_t *c)
1974 {
1975 	ossclient_t	*sc;
1976 
1977 	sc = auclnt_get_private(c);
1978 
1979 	kmem_free(sc->o_ss_buf, sc->o_ss_sz);
1980 	kmem_free(sc, sizeof (*sc));
1981 
1982 	auclnt_close(c);
1983 }
1984 
1985 static int
1986 sndctl_mix_nrext(audio_client_t *c, int *ncp)
1987 {
1988 	audio_dev_t	*d;
1989 	ossdev_t	*odev;
1990 
1991 	d = auclnt_get_dev(c);
1992 
1993 	if ((*ncp != -1) && (*ncp != (auclnt_get_dev_index(d) - 1))) {
1994 		return (ENXIO);
1995 	}
1996 
1997 	if ((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) {
1998 		return (EINVAL);
1999 	}
2000 
2001 	*ncp = odev->d_nctrl;
2002 
2003 	return (0);
2004 }
2005 
2006 static int
2007 sndctl_mix_extinfo(audio_client_t *c, oss_mixext *pext)
2008 {
2009 	audio_dev_t		*d;
2010 	ossdev_t		*odev;
2011 	int			rv = 0;
2012 	int			dev;
2013 
2014 	d = auclnt_get_dev(c);
2015 
2016 	if (((dev = pext->dev) != -1) && (dev != (auclnt_get_dev_index(d) - 1)))
2017 		return (ENXIO);
2018 
2019 	if (((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) ||
2020 	    (pext->ctrl >= odev->d_nctrl)) {
2021 		return (EINVAL);
2022 	}
2023 
2024 	bcopy(&odev->d_exts[pext->ctrl], pext, sizeof (*pext));
2025 	pext->enumbit = 0;
2026 	pext->dev = dev;
2027 
2028 	return (rv);
2029 }
2030 
2031 static int
2032 sndctl_mix_enuminfo(audio_client_t *c, oss_mixer_enuminfo *ei)
2033 {
2034 	audio_dev_t		*d;
2035 	audio_ctrl_desc_t	desc;
2036 	audio_ctrl_t		*ctrl;
2037 	ossdev_t		*odev;
2038 	uint64_t		mask;
2039 	int			bit;
2040 	ushort_t		nxt;
2041 
2042 	d = auclnt_get_dev(c);
2043 
2044 	if ((ei->dev != -1) && (ei->dev != (auclnt_get_dev_index(d) - 1)))
2045 		return (ENXIO);
2046 
2047 	if (((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) ||
2048 	    (ei->ctrl >= odev->d_nctrl) ||
2049 	    (odev->d_exts[ei->ctrl].type != MIXT_ENUM) ||
2050 	    ((ctrl = odev->d_ctrls[ei->ctrl]) == NULL) ||
2051 	    (auclnt_control_describe(ctrl, &desc) != 0)) {
2052 		return (EINVAL);
2053 	}
2054 
2055 	mask = desc.acd_maxvalue;
2056 	bit = 0;
2057 	nxt = 0;
2058 	ei->nvalues = 0;
2059 	bzero(ei->strings, sizeof (ei->strings));
2060 	bzero(ei->strindex, sizeof (ei->strindex));
2061 
2062 	while (mask) {
2063 		const char *name = desc.acd_enum[bit];
2064 		nxt = oss_set_enum(ei, nxt, name ? name : "");
2065 		mask >>= 1;
2066 		bit++;
2067 	}
2068 
2069 	return (0);
2070 }
2071 
2072 static int
2073 sndctl_mix_read(audio_client_t *c, oss_mixer_value *vr)
2074 {
2075 	int			rv;
2076 	uint64_t		v;
2077 	audio_dev_t		*d;
2078 	audio_ctrl_t		*ctrl;
2079 	ossdev_t		*odev;
2080 
2081 	d = auclnt_get_dev(c);
2082 
2083 	if ((vr->dev != -1) && (vr->dev != (auclnt_get_dev_index(d) - 1)))
2084 		return (ENXIO);
2085 
2086 	if (((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) ||
2087 	    (vr->ctrl >= odev->d_nctrl) ||
2088 	    ((ctrl = odev->d_ctrls[vr->ctrl]) == NULL)) {
2089 		return (EINVAL);
2090 	}
2091 	if ((rv = auclnt_control_read(ctrl, &v)) == 0) {
2092 		switch (odev->d_exts[vr->ctrl].type) {
2093 		case MIXT_ENUM:
2094 			/* translate this from an enum style bit mask */
2095 			vr->value = ddi_ffs((unsigned long)v) - 1;
2096 			break;
2097 		case MIXT_STEREOSLIDER:
2098 			vr->value = (int)ddi_swap16(v & 0xffff);
2099 			break;
2100 		case MIXT_MONOSLIDER:
2101 			vr->value = (int)(v | (v << 8));
2102 			break;
2103 		case MIXT_ONOFF:
2104 			/* this could be simple, or could be part of a multi */
2105 			if (odev->d_exts[vr->ctrl].enumbit >= 0) {
2106 				uint64_t mask;
2107 				mask = 1;
2108 				mask <<= (odev->d_exts[vr->ctrl].enumbit);
2109 				vr->value = (v & mask) ? 1 : 0;
2110 			} else {
2111 				vr->value = v ? 1 : 0;
2112 			}
2113 			break;
2114 
2115 		default:
2116 			vr->value = (int)v;
2117 			break;
2118 		}
2119 	}
2120 
2121 	return (rv);
2122 }
2123 
2124 static int
2125 sndctl_mix_write(audio_client_t *c, oss_mixer_value *vr)
2126 {
2127 	int			rv;
2128 	uint64_t		v;
2129 	audio_dev_t		*d;
2130 	audio_ctrl_t		*ctrl;
2131 	ossdev_t		*odev;
2132 
2133 	d = auclnt_get_dev(c);
2134 
2135 	if ((vr->dev != -1) && (vr->dev != (auclnt_get_dev_index(d) - 1)))
2136 		return (ENXIO);
2137 
2138 	if (((odev = auclnt_get_dev_minor_data(d, AUDIO_MINOR_DSP)) == NULL) ||
2139 	    (vr->ctrl >= odev->d_nctrl) ||
2140 	    ((ctrl = odev->d_ctrls[vr->ctrl]) == NULL)) {
2141 		return (EINVAL);
2142 	}
2143 
2144 	switch (odev->d_exts[vr->ctrl].type) {
2145 	case MIXT_ONOFF:
2146 		/* this could be standalone, or it could be part of a multi */
2147 		if (odev->d_exts[vr->ctrl].enumbit >= 0) {
2148 			uint64_t mask;
2149 			if ((rv = auclnt_control_read(ctrl, &v)) != 0) {
2150 				return (EINVAL);
2151 			}
2152 			mask = 1;
2153 			mask <<= (odev->d_exts[vr->ctrl].enumbit);
2154 			if (vr->value) {
2155 				v |= mask;
2156 			} else {
2157 				v &= ~mask;
2158 			}
2159 		} else {
2160 			v = vr->value;
2161 		}
2162 		break;
2163 	case MIXT_ENUM:
2164 		/* translate this to an enum style bit mask */
2165 		v = 1U << vr->value;
2166 		break;
2167 	case MIXT_MONOSLIDER:
2168 		/* mask off high order bits */
2169 		v = vr->value & 0xff;
2170 		break;
2171 	case MIXT_STEREOSLIDER:
2172 		/* OSS uses reverse byte ordering */
2173 		v = vr->value;
2174 		v = ddi_swap16(vr->value & 0xffff);
2175 		break;
2176 	default:
2177 		v = vr->value;
2178 	}
2179 	rv = auclnt_control_write(ctrl, v);
2180 
2181 	return (rv);
2182 }
2183 
2184 static int
2185 sndctl_mix_nrmix(audio_client_t *c, int *nmixp)
2186 {
2187 	_NOTE(ARGUNUSED(c));
2188 	*nmixp = oss_cnt_devs() - 1;
2189 	return (0);
2190 }
2191 
2192 static int
2193 ossmix_ioctl(audio_client_t *c, int cmd, intptr_t arg, int mode, cred_t *credp,
2194     int *rvalp)
2195 {
2196 	int	sz;
2197 	void	*data;
2198 	int	rv = 0;
2199 
2200 	sz = OSSIOC_GETSZ(cmd);
2201 
2202 	if ((cmd & (OSSIOC_IN | OSSIOC_OUT)) && sz) {
2203 		if ((data = kmem_zalloc(sz, KM_NOSLEEP)) == NULL) {
2204 			return (ENOMEM);
2205 		}
2206 	} else {
2207 		sz = 0;
2208 	}
2209 
2210 	if (cmd & OSSIOC_IN) {
2211 		if ((rv = ddi_copyin((void *)arg, data, sz, mode)) != 0) {
2212 			goto done;
2213 		}
2214 	}
2215 
2216 	switch (cmd) {
2217 		/*
2218 		 * Mixer specific ioctls
2219 		 */
2220 	case SNDCTL_MIX_NREXT:
2221 		rv = sndctl_mix_nrext(c, (int *)data);
2222 		break;
2223 	case SNDCTL_MIX_EXTINFO:
2224 		rv = sndctl_mix_extinfo(c, (oss_mixext *)data);
2225 		break;
2226 	case SNDCTL_MIX_ENUMINFO:
2227 		rv = sndctl_mix_enuminfo(c, (oss_mixer_enuminfo *)data);
2228 		break;
2229 	case SNDCTL_MIX_READ:
2230 		rv = sndctl_mix_read(c, (oss_mixer_value *)data);
2231 		break;
2232 	case SNDCTL_MIX_WRITE:
2233 		rv = sndctl_mix_write(c, (oss_mixer_value *)data);
2234 		break;
2235 	case SNDCTL_MIX_NRMIX:
2236 		rv = sndctl_mix_nrmix(c, (int *)data);
2237 		break;
2238 
2239 		/*
2240 		 * Legacy ioctls.  These are treated as soft values only,
2241 		 * and do not affect global hardware state.  For use by
2242 		 * legacy DSP applications.
2243 		 */
2244 	case SOUND_MIXER_READ_VOLUME:
2245 	case SOUND_MIXER_READ_PCM:
2246 	case SOUND_MIXER_READ_OGAIN:
2247 		rv = sndctl_dsp_getplayvol(c, (int *)data);
2248 		break;
2249 
2250 	case SOUND_MIXER_WRITE_VOLUME:
2251 	case SOUND_MIXER_WRITE_PCM:
2252 	case SOUND_MIXER_WRITE_OGAIN:
2253 		rv = sound_mixer_write_ogain(c, (int *)data);
2254 		break;
2255 
2256 	case SOUND_MIXER_READ_RECGAIN:
2257 	case SOUND_MIXER_READ_RECLEV:
2258 	case SOUND_MIXER_READ_IGAIN:
2259 		rv = sndctl_dsp_getrecvol(c, (int *)data);
2260 		break;
2261 
2262 	case SOUND_MIXER_WRITE_RECGAIN:
2263 	case SOUND_MIXER_WRITE_RECLEV:
2264 	case SOUND_MIXER_WRITE_IGAIN:
2265 		rv = sound_mixer_write_igain(c, (int *)data);
2266 		break;
2267 
2268 	case SOUND_MIXER_READ_RECSRC:
2269 	case SOUND_MIXER_WRITE_RECSRC:
2270 		rv = sound_mixer_read_recsrc(c, (int *)data);
2271 		break;
2272 
2273 	case SOUND_MIXER_READ_DEVMASK:
2274 	case SOUND_MIXER_READ_STEREODEVS:
2275 		rv = sound_mixer_read_devmask(c, (int *)data);
2276 		break;
2277 
2278 	case SOUND_MIXER_READ_RECMASK:
2279 		rv = sound_mixer_read_recmask(c, (int *)data);
2280 		break;
2281 
2282 	case SOUND_MIXER_READ_CAPS:
2283 		rv = sound_mixer_read_caps(c, (int *)data);
2284 		break;
2285 
2286 		/*
2287 		 * Common ioctls shared with DSP
2288 		 */
2289 	case OSS_GETVERSION:
2290 		rv = oss_getversion((int *)data);
2291 		break;
2292 
2293 	case SNDCTL_CARDINFO:
2294 		rv = sndctl_cardinfo(c, (oss_card_info *)data);
2295 		break;
2296 
2297 	case SNDCTL_ENGINEINFO:
2298 	case SNDCTL_AUDIOINFO:
2299 	case SNDCTL_AUDIOINFO_EX:
2300 		rv = sndctl_audioinfo(c, (oss_audioinfo *)data);
2301 		break;
2302 
2303 	case SNDCTL_SYSINFO:
2304 		rv = sndctl_sysinfo((oss_sysinfo *)data);
2305 		break;
2306 
2307 	case SNDCTL_MIXERINFO:
2308 		rv = sndctl_mixerinfo(c, (oss_mixerinfo *)data);
2309 		break;
2310 
2311 	case SOUND_MIXER_INFO:
2312 		rv = sound_mixer_info(c, (mixer_info *)data);
2313 		break;
2314 
2315 	case SNDCTL_MIX_DESCRIPTION:	/* NOT SUPPORTED: tooltip */
2316 		rv = EIO;	/* OSS returns EIO for this one */
2317 		break;
2318 
2319 		/*
2320 		 * Special implementation-private ioctls.
2321 		 */
2322 	case SNDCTL_SUN_SEND_NUMBER:
2323 		rv = sndctl_sun_send_number(c, (int *)data, credp);
2324 		break;
2325 
2326 		/*
2327 		 * Legacy ioctls we don't support.
2328 		 */
2329 	case SOUND_MIXER_WRITE_MONGAIN:
2330 	case SOUND_MIXER_READ_MONGAIN:
2331 	case SOUND_MIXER_READ_BASS:
2332 	case SOUND_MIXER_READ_TREBLE:
2333 	case SOUND_MIXER_READ_SPEAKER:
2334 	case SOUND_MIXER_READ_LINE:
2335 	case SOUND_MIXER_READ_MIC:
2336 	case SOUND_MIXER_READ_CD:
2337 	case SOUND_MIXER_READ_IMIX:
2338 	case SOUND_MIXER_READ_ALTPCM:
2339 	case SOUND_MIXER_READ_SYNTH:
2340 	case SOUND_MIXER_READ_LINE1:
2341 	case SOUND_MIXER_READ_LINE2:
2342 	case SOUND_MIXER_READ_LINE3:
2343 	case SOUND_MIXER_WRITE_BASS:
2344 	case SOUND_MIXER_WRITE_TREBLE:
2345 	case SOUND_MIXER_WRITE_SPEAKER:
2346 	case SOUND_MIXER_WRITE_LINE:
2347 	case SOUND_MIXER_WRITE_MIC:
2348 	case SOUND_MIXER_WRITE_CD:
2349 	case SOUND_MIXER_WRITE_IMIX:
2350 	case SOUND_MIXER_WRITE_ALTPCM:
2351 	case SOUND_MIXER_WRITE_SYNTH:
2352 	case SOUND_MIXER_WRITE_LINE1:
2353 	case SOUND_MIXER_WRITE_LINE2:
2354 	case SOUND_MIXER_WRITE_LINE3:
2355 		/*
2356 		 * Additional ioctls we *could* support, but don't.
2357 		 */
2358 	case SNDCTL_SETSONG:
2359 	case SNDCTL_SETLABEL:
2360 	case SNDCTL_GETSONG:
2361 	case SNDCTL_GETLABEL:
2362 	case SNDCTL_MIDIINFO:
2363 	case SNDCTL_SETNAME:
2364 	default:
2365 		rv = EINVAL;
2366 		break;
2367 	}
2368 
2369 	if ((rv == 0) && (cmd & OSSIOC_OUT)) {
2370 		rv = ddi_copyout(data, (void *)arg, sz, mode);
2371 	}
2372 	if (rv == 0) {
2373 		*rvalp = 0;
2374 	}
2375 
2376 done:
2377 	if (sz) {
2378 		kmem_free(data, sz);
2379 	}
2380 	return (rv);
2381 }
2382 
2383 static void *
2384 oss_dev_init(audio_dev_t *dev)
2385 {
2386 	ossdev_t	*odev;
2387 
2388 	odev = kmem_zalloc(sizeof (*odev), KM_SLEEP);
2389 	odev->d_dev = dev;
2390 
2391 	mutex_init(&odev->d_mx, NULL, MUTEX_DRIVER, NULL);
2392 	cv_init(&odev->d_cv, NULL, CV_DRIVER, NULL);
2393 	oss_alloc_controls(odev);
2394 
2395 	return (odev);
2396 }
2397 
2398 static void
2399 oss_dev_fini(void *arg)
2400 {
2401 	ossdev_t	*odev = arg;
2402 
2403 	if (odev != NULL) {
2404 		oss_free_controls(odev);
2405 		mutex_destroy(&odev->d_mx);
2406 		cv_destroy(&odev->d_cv);
2407 		kmem_free(odev, sizeof (*odev));
2408 	}
2409 }
2410 
2411 static void
2412 sndstat_printf(ossclient_t *oc, const char *fmt, ...)
2413 {
2414 	va_list	va;
2415 
2416 	va_start(va, fmt);
2417 	(void) vsnprintf(oc->o_ss_buf + oc->o_ss_len,
2418 	    oc->o_ss_sz - oc->o_ss_len, fmt, va);
2419 	va_end(va);
2420 	oc->o_ss_len = strlen(oc->o_ss_buf);
2421 }
2422 
2423 static int
2424 sndstat_dev_walker(audio_dev_t *d, void *arg)
2425 {
2426 	ossclient_t	*oc = arg;
2427 	const char	*capstr;
2428 	unsigned	cap;
2429 
2430 	cap = auclnt_get_dev_capab(d);
2431 
2432 	if (cap	& AUDIO_CLIENT_CAP_DUPLEX) {
2433 		capstr = "DUPLEX";
2434 	} else if ((cap & AUDIO_CLIENT_CAP_PLAY) &&
2435 	    (cap & AUDIO_CLIENT_CAP_RECORD)) {
2436 		capstr = "INPUT,OUTPUT";
2437 	} else if (cap & AUDIO_CLIENT_CAP_PLAY) {
2438 		capstr = "OUTPUT";
2439 	} else if (cap & AUDIO_CLIENT_CAP_RECORD) {
2440 		capstr = "INPUT";
2441 	} else {
2442 		capstr = NULL;
2443 	}
2444 
2445 	if (capstr == NULL)
2446 		return (AUDIO_WALK_CONTINUE);
2447 
2448 	sndstat_printf(oc, "%d: %s %s, %s (%s)\n",
2449 	    auclnt_get_dev_number(d), auclnt_get_dev_name(d),
2450 	    auclnt_get_dev_description(d), auclnt_get_dev_version(d), capstr);
2451 
2452 	return (AUDIO_WALK_CONTINUE);
2453 }
2454 
2455 static int
2456 sndstat_mixer_walker(audio_dev_t *d, void *arg)
2457 {
2458 	ossclient_t	*oc = arg;
2459 	unsigned	cap;
2460 	void		*iter;
2461 	const char	*info;
2462 
2463 	cap = auclnt_get_dev_capab(d);
2464 
2465 	if ((cap & (AUDIO_CLIENT_CAP_PLAY|AUDIO_CLIENT_CAP_RECORD)) == 0)
2466 		return (AUDIO_WALK_CONTINUE);
2467 
2468 	sndstat_printf(oc, "%d: %s %s, %s\n",
2469 	    auclnt_get_dev_number(d), auclnt_get_dev_name(d),
2470 	    auclnt_get_dev_description(d), auclnt_get_dev_version(d));
2471 	iter = NULL;
2472 	while ((info = auclnt_get_dev_hw_info(d, &iter)) != NULL) {
2473 		sndstat_printf(oc, "\t%s\n", info);
2474 	}
2475 	return (AUDIO_WALK_CONTINUE);
2476 }
2477 
2478 static int
2479 ossmix_write(audio_client_t *c, struct uio *uio, cred_t *cr)
2480 {
2481 	/* write on sndstat is a no-op */
2482 	_NOTE(ARGUNUSED(c));
2483 	_NOTE(ARGUNUSED(uio));
2484 	_NOTE(ARGUNUSED(cr));
2485 
2486 	return (0);
2487 }
2488 
2489 static int
2490 ossmix_read(audio_client_t *c, struct uio *uio, cred_t *cr)
2491 {
2492 	ossclient_t	*oc;
2493 	unsigned	n;
2494 	int		rv;
2495 
2496 	_NOTE(ARGUNUSED(cr));
2497 
2498 	if (uio->uio_resid == 0) {
2499 		return (0);
2500 	}
2501 
2502 	oc = auclnt_get_private(c);
2503 
2504 	mutex_enter(&oc->o_ss_lock);
2505 
2506 	if (oc->o_ss_off == 0) {
2507 
2508 		sndstat_printf(oc, "SunOS Audio Framework\n");
2509 
2510 		sndstat_printf(oc, "\nAudio Devices:\n");
2511 		auclnt_walk_devs_by_number(sndstat_dev_walker, oc);
2512 
2513 		sndstat_printf(oc, "\nMixers:\n");
2514 		auclnt_walk_devs_by_number(sndstat_mixer_walker, oc);
2515 	}
2516 
2517 	/*
2518 	 * For simplicity's sake, we implement a non-seekable device.  We could
2519 	 * support seekability, but offsets would be rather meaningless between
2520 	 * changes.
2521 	 */
2522 	n = min(uio->uio_resid, (oc->o_ss_len - oc->o_ss_off));
2523 
2524 	rv = uiomove(oc->o_ss_buf + oc->o_ss_off, n, UIO_READ, uio);
2525 	if (rv != 0) {
2526 		n = 0;
2527 	}
2528 	oc->o_ss_off += n;
2529 
2530 	if (n == 0) {
2531 		/*
2532 		 * end-of-file reached... clear the sndstat buffer so that
2533 		 * subsequent reads will get the latest data.
2534 		 */
2535 		oc->o_ss_off = oc->o_ss_len = 0;
2536 	}
2537 	mutex_exit(&oc->o_ss_lock);
2538 	return (rv);
2539 }
2540 
2541 int
2542 oss_read(audio_client_t *c, struct uio *uio, cred_t *cr)
2543 {
2544 	_NOTE(ARGUNUSED(cr));
2545 
2546 	auclnt_clear_paused(auclnt_input_stream(c));
2547 
2548 	return (auclnt_read(c, uio));
2549 }
2550 
2551 int
2552 oss_write(audio_client_t *c, struct uio *uio, cred_t *cr)
2553 {
2554 	_NOTE(ARGUNUSED(cr));
2555 
2556 	auclnt_clear_paused(auclnt_output_stream(c));
2557 
2558 	return (auclnt_write(c, uio));
2559 }
2560 
2561 int
2562 oss_chpoll(audio_client_t *c, short events, int anyyet, short *reventsp,
2563     struct pollhead **phpp)
2564 {
2565 	return (auclnt_chpoll(c, events, anyyet, reventsp, phpp));
2566 }
2567 
2568 static struct audio_client_ops oss_ops = {
2569 	"sound,dsp",
2570 	oss_dev_init,
2571 	oss_dev_fini,
2572 	oss_open,
2573 	oss_close,
2574 	oss_read,
2575 	oss_write,
2576 	oss_ioctl,
2577 	oss_chpoll,
2578 	NULL,		/* mmap */
2579 	oss_input,
2580 	oss_output,
2581 	NULL,		/* notify */
2582 	NULL,		/* drain */
2583 };
2584 
2585 static struct audio_client_ops ossmix_ops = {
2586 	"sound,mixer",
2587 	NULL,
2588 	NULL,
2589 	ossmix_open,
2590 	ossmix_close,
2591 	ossmix_read,
2592 	ossmix_write,
2593 	ossmix_ioctl,
2594 	NULL,	/* chpoll */
2595 	NULL,   /* mmap */
2596 	NULL,	/* input */
2597 	NULL,   /* output */
2598 	oss_notify,
2599 	NULL,	/* drain */
2600 	NULL,	/* wput */
2601 	NULL,	/* wsrv */
2602 };
2603 
2604 /* nearly the same as ossxmix; different minor name helps devfsadm */
2605 static struct audio_client_ops sndstat_ops = {
2606 	"sound,sndstat",
2607 	NULL,	/* dev_init */
2608 	NULL,	/* dev_fini */
2609 	ossmix_open,
2610 	ossmix_close,
2611 	ossmix_read,
2612 	ossmix_write,
2613 	ossmix_ioctl,
2614 	NULL,	/* chpoll */
2615 	NULL,	/* mmap */
2616 	NULL,	/* input */
2617 	NULL,	/* output */
2618 	NULL,	/* notify */
2619 	NULL,	/* drain */
2620 	NULL,	/* wput */
2621 	NULL,	/* wsrv */
2622 };
2623 
2624 void
2625 auimpl_oss_init(void)
2626 {
2627 	auclnt_register_ops(AUDIO_MINOR_DSP, &oss_ops);
2628 	auclnt_register_ops(AUDIO_MINOR_MIXER, &ossmix_ops);
2629 	auclnt_register_ops(AUDIO_MINOR_SNDSTAT, &sndstat_ops);
2630 }
2631