xref: /titanic_41/usr/src/uts/common/io/audio/impl/audio_sun.c (revision 2c30fa4582c5d6c659e059e719c5f6163f7ef1e3)
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 (c) 2009, 2010, Oracle and/or its affiliates. All rights reserved.
23  */
24 
25 /*
26  * Sun audio(7I) and mixer(7I) personality.
27  *
28  * There are some "undocumented" details of how legacy Sun audio
29  * interfaces work.  The following "rules" were derived from reading the
30  * legacy Sun mixer code, and to the best of our knowledge are not
31  * documented elsewhere.
32  *
33  * - We create a "fake" audio device, which behaves like a classic
34  *   exclusive audio device, for each PID, as determined during open(2).
35  *
36  * - Different processes don't interfere with each other.  Even though
37  *   they are running concurrently, they each think they have exclusive
38  *   control over the audio device.
39  *
40  * - Read and write directions operate independent of each other.  That
41  *   is, a device open for reading won't intefere with a future open for
42  *   writing, and vice versa.  This is true even within the same process.
43  *
44  * - Because the virtualization is by PID, strange behavior may occur
45  *   if a process tries to open an audio device at the same time it
46  *   has already received a file descriptor from another process (such
47  *   through inheritence via fork()).
48  *
49  * - The "fake" audio device has no control over physical settings.
50  *   It sees only the software attenuation-based volumes for play and
51  *   record, and has no support for alternate input or output ports or
52  *   access to the monitoring features of the hardware.
53  *
54  * - Explicit notificaton signals (SIGPOLL) are only ever sent up the
55  *   audioctl node -- never up a regular audio node.  (The stream head
56  *   may still issue SIGPOLL based on readability/writability of
57  *   course.)
58  *
59  * - Corollary: processes that want asynch. notifications will open
60  *   /dev/audioctl as well as /dev/audio.
61  *
62  * - We don't support the MIXER mode at all.
63  *
64  * - By corollary, a process is only allowed to open /dev/audio once
65  *   (in each direction.)
66  *
67  * - Attempts to open /dev/audio in duplex mode (O_RDWR) fail (EBUSY)
68  *   if the device cannot support duplex operation.
69  *
70  * - Attempts to open a device with FREAD set fail if the device is not
71  *   capable of recording.  (Likewise for FWRITE and playback.)
72  *
73  * - No data transfer is permitted for audioctl nodes.  (No actual
74  *   record or play.)
75  *
76  * - Sun audio does not support any formats other than linear and
77  *   ULAW/ALAW.  I.e. it will never support AC3 or other "opaque"
78  *   streams which require special handling.
79  *
80  * - Sun audio only supports stereo or monophonic data streams.
81  */
82 
83 #include <sys/types.h>
84 #include <sys/open.h>
85 #include <sys/errno.h>
86 #include <sys/audio.h>
87 #include <sys/mixer.h>
88 #include <sys/file.h>
89 #include <sys/stropts.h>
90 #include <sys/strsun.h>
91 #include <sys/sysmacros.h>
92 #include <sys/list.h>
93 #include <sys/note.h>
94 #include <sys/stat.h>
95 #include <sys/ddi.h>
96 #include <sys/sunddi.h>
97 #include "audio_client.h"
98 
99 typedef struct daclient daclient_t;
100 typedef struct dadev dadev_t;
101 typedef struct daproc daproc_t;
102 
103 /* common structure shared between both audioctl and audio nodes */
104 struct daclient {
105 	daproc_t		*dc_proc;
106 	dadev_t			*dc_dev;
107 	audio_client_t		*dc_client;
108 	queue_t			*dc_wq;
109 	unsigned		dc_eof;
110 	list_t			dc_eofcnt;
111 	kmutex_t		dc_lock;
112 	mblk_t			*dc_draining;
113 };
114 
115 struct eofcnt {
116 	list_node_t		linkage;
117 	uint64_t		tail;
118 };
119 
120 struct dadev {
121 	audio_dev_t		*d_dev;
122 
123 	list_t			d_procs;
124 	kmutex_t		d_mx;
125 	kcondvar_t		d_cv;
126 };
127 
128 struct daproc {
129 	pid_t			p_id;
130 	struct audio_info	p_info;
131 	int			p_refcnt;
132 	int			p_oflag;
133 	list_node_t		p_linkage;
134 	dadev_t			*p_dev;
135 	audio_client_t		*p_writer;
136 	audio_client_t		*p_reader;
137 };
138 
139 int devaudio_proc_hold(audio_client_t *, int);
140 void devaudio_proc_release(audio_client_t *);
141 static void devaudio_proc_update(daproc_t *);
142 
143 
144 static int
devaudio_compose_format(audio_prinfo_t * prinfo)145 devaudio_compose_format(audio_prinfo_t *prinfo)
146 {
147 	switch (prinfo->precision) {
148 	case 8:
149 		switch (prinfo->encoding) {
150 		case AUDIO_ENCODING_ULAW:
151 			return (AUDIO_FORMAT_ULAW);
152 		case AUDIO_ENCODING_ALAW:
153 			return (AUDIO_FORMAT_ALAW);
154 		case AUDIO_ENCODING_LINEAR8:
155 			return (AUDIO_FORMAT_U8);
156 		case AUDIO_ENCODING_LINEAR:
157 			return (AUDIO_FORMAT_S8);
158 		}
159 		break;
160 	case 16:
161 		if (prinfo->encoding == AUDIO_ENCODING_LINEAR)
162 			return (AUDIO_FORMAT_S16_NE);
163 		break;
164 	case 32:
165 		if (prinfo->encoding == AUDIO_ENCODING_LINEAR)
166 			return (AUDIO_FORMAT_S32_NE);
167 		break;
168 	}
169 	return (AUDIO_FORMAT_NONE);
170 
171 }
172 
173 static void
devaudio_decompose_format(audio_prinfo_t * prinfo,int afmt)174 devaudio_decompose_format(audio_prinfo_t *prinfo, int afmt)
175 {
176 	int	e, p;
177 
178 	/*
179 	 * N.B.: Even though some of the formats below can't be set by
180 	 * this personality, reporting them (using the closest match)
181 	 * allows this personality to roughly approximate settings for
182 	 * other streams.  It would be incredibly poor form for any
183 	 * personality to modify the format settings for a different
184 	 * personality, so we don't worry about that case.
185 	 */
186 
187 	switch (afmt) {
188 	case AUDIO_FORMAT_ULAW:
189 		e = AUDIO_ENCODING_ULAW;
190 		p = 8;
191 		break;
192 
193 	case AUDIO_FORMAT_ALAW:
194 		e = AUDIO_ENCODING_ALAW;
195 		p = 8;
196 		break;
197 
198 	case AUDIO_FORMAT_U8:
199 		e = AUDIO_ENCODING_LINEAR8;
200 		p = 8;
201 		break;
202 
203 	case AUDIO_FORMAT_S8:
204 		e = AUDIO_ENCODING_LINEAR;
205 		p = 8;
206 		break;
207 
208 	case AUDIO_FORMAT_S16_NE:
209 	case AUDIO_FORMAT_S16_OE:
210 	case AUDIO_FORMAT_U16_NE:
211 	case AUDIO_FORMAT_U16_OE:
212 		e = AUDIO_ENCODING_LINEAR;
213 		p = 16;
214 		break;
215 
216 	case AUDIO_FORMAT_S24_NE:
217 	case AUDIO_FORMAT_S24_OE:
218 	case AUDIO_FORMAT_S24_PACKED:
219 		e = AUDIO_ENCODING_LINEAR;
220 		p = 24;
221 		break;
222 
223 	case AUDIO_FORMAT_S32_NE:
224 	case AUDIO_FORMAT_S32_OE:
225 		e = AUDIO_ENCODING_LINEAR;
226 		p = 32;
227 		break;
228 
229 	default:
230 		/* all other formats (e.g. AC3) are uninterpreted */
231 		e = AUDIO_ENCODING_NONE;
232 		p = 32;
233 		break;
234 	}
235 
236 	prinfo->encoding = e;
237 	prinfo->precision = p;
238 }
239 
240 static daproc_t *
devaudio_proc_alloc(audio_client_t * c)241 devaudio_proc_alloc(audio_client_t *c)
242 {
243 	audio_info_t	*info;
244 	audio_prinfo_t	*prinfo;
245 	uint32_t	caps;
246 	daproc_t	*proc;
247 
248 	if ((proc = kmem_zalloc(sizeof (*proc), KM_NOSLEEP)) == NULL) {
249 		return (NULL);
250 	}
251 	info = &proc->p_info;
252 
253 	/*
254 	 * audio(7I) says: Upon the initial open() of the audio
255 	 * device, the driver resets the data format of the device to
256 	 * the default state of 8-bit, 8Khz, mono u-Law data.
257 	 */
258 	prinfo = &info->play;
259 	prinfo->channels =	1;
260 	prinfo->sample_rate =	8000;
261 	prinfo->encoding =	AUDIO_ENCODING_ULAW;
262 	prinfo->precision =	8;
263 	prinfo->gain =		AUDIO_MAX_GAIN;
264 	prinfo->balance =	AUDIO_MID_BALANCE;
265 	prinfo->buffer_size =	8192;
266 	prinfo->pause =		B_FALSE;
267 	prinfo->waiting =	B_FALSE;
268 	prinfo->open =		B_FALSE;
269 	prinfo->active =	B_FALSE;
270 	prinfo->samples =	0;
271 	prinfo->eof =		0;
272 	prinfo->error =		0;
273 	prinfo->minordev =	0;
274 	prinfo->port =		AUDIO_SPEAKER;
275 	prinfo->avail_ports =	AUDIO_SPEAKER;
276 	prinfo->mod_ports =	AUDIO_NONE;
277 	prinfo->_xxx =		0;
278 
279 	prinfo = &info->record;
280 	prinfo->channels =	1;
281 	prinfo->sample_rate =	8000;
282 	prinfo->encoding =	AUDIO_ENCODING_ULAW;
283 	prinfo->precision =	8;
284 	prinfo->gain =		AUDIO_MAX_GAIN;
285 	prinfo->balance =	AUDIO_MID_BALANCE;
286 	prinfo->buffer_size =	8192;
287 	prinfo->waiting =	B_FALSE;
288 	prinfo->open =  	B_FALSE;
289 	prinfo->active =	B_FALSE;
290 	prinfo->samples =	0;
291 	prinfo->eof =		0;
292 	prinfo->error =		0;
293 	prinfo->minordev =	0;
294 	prinfo->port =		AUDIO_MICROPHONE;
295 	prinfo->avail_ports =	AUDIO_MICROPHONE;
296 	prinfo->mod_ports =	AUDIO_MICROPHONE;
297 
298 	info->output_muted =	B_FALSE;
299 	/* pretend we don't have a software mixer - we don't support the API */
300 	info->hw_features =	0;
301 	info->sw_features =	0;
302 	info->sw_features_enabled = 0;
303 
304 	caps = auclnt_get_dev_capab(auclnt_get_dev(c));
305 	if (caps & AUDIO_CLIENT_CAP_PLAY)
306 		info->hw_features |= AUDIO_HWFEATURE_PLAY;
307 	if (caps & AUDIO_CLIENT_CAP_RECORD)
308 		info->hw_features |= AUDIO_HWFEATURE_RECORD;
309 	if (caps & AUDIO_CLIENT_CAP_DUPLEX)
310 		info->hw_features |= AUDIO_HWFEATURE_DUPLEX;
311 
312 	return (proc);
313 }
314 
315 static void
devaudio_proc_free(daproc_t * proc)316 devaudio_proc_free(daproc_t *proc)
317 {
318 	kmem_free(proc, sizeof (*proc));
319 }
320 
321 int
devaudio_proc_hold(audio_client_t * c,int oflag)322 devaudio_proc_hold(audio_client_t *c, int oflag)
323 {
324 	pid_t		pid;
325 	daproc_t	*proc;
326 	dadev_t		*dev;
327 	daclient_t	*dc;
328 	list_t		*l;
329 	audio_dev_t	*adev;
330 	int		rv;
331 
332 	adev = auclnt_get_dev(c);
333 
334 	/* first allocate and initialize the daclient private data */
335 	if ((dc = kmem_zalloc(sizeof (*dc), KM_NOSLEEP)) == NULL) {
336 		return (ENOMEM);
337 	}
338 
339 	mutex_init(&dc->dc_lock, NULL, MUTEX_DRIVER, NULL);
340 	list_create(&dc->dc_eofcnt, sizeof (struct eofcnt),
341 	    offsetof(struct eofcnt, linkage));
342 	auclnt_set_private(c, dc);
343 
344 	dev = auclnt_get_dev_minor_data(adev, AUDIO_MINOR_DEVAUDIO);
345 	l = &dev->d_procs;
346 	pid = auclnt_get_pid(c);
347 
348 	/* set a couple of common fields */
349 	dc->dc_client = c;
350 	dc->dc_dev = dev;
351 
352 	mutex_enter(&dev->d_mx);
353 	for (proc = list_head(l); proc != NULL; proc = list_next(l, proc)) {
354 		if (proc->p_id == pid) {
355 			proc->p_refcnt++;
356 			break;
357 		}
358 	}
359 	if (proc == NULL) {
360 		if ((proc = devaudio_proc_alloc(c)) == NULL) {
361 			rv = ENOMEM;
362 			goto failed;
363 		}
364 		proc->p_refcnt = 1;
365 		proc->p_id = pid;
366 		proc->p_dev = dev;
367 		list_insert_tail(l, proc);
368 	}
369 
370 	while (proc->p_oflag & oflag) {
371 
372 		if (oflag & (FNDELAY|FNONBLOCK)) {
373 			rv = EBUSY;
374 			goto failed;
375 		}
376 		if (oflag & FWRITE)
377 			proc->p_info.play.waiting++;
378 		if (oflag & FREAD)
379 			proc->p_info.record.waiting++;
380 		if (cv_wait_sig(&dev->d_cv, &dev->d_mx) == 0) {
381 			/* interrupted! */
382 			if (oflag & FWRITE)
383 				proc->p_info.play.waiting--;
384 			if (oflag & FREAD)
385 				proc->p_info.record.waiting--;
386 			rv = EINTR;
387 			goto failed;
388 		}
389 		if (oflag & FWRITE)
390 			proc->p_info.play.waiting--;
391 		if (oflag & FREAD)
392 			proc->p_info.record.waiting--;
393 	}
394 
395 	if (oflag & FWRITE) {
396 		audio_prinfo_t	*play = &proc->p_info.play;
397 		audio_stream_t	*sp = auclnt_output_stream(c);
398 
399 		if (((rv = auclnt_set_rate(sp, 8000)) != 0) ||
400 		    ((rv = auclnt_set_format(sp, AUDIO_FORMAT_ULAW)) != 0) ||
401 		    ((rv = auclnt_set_channels(sp, 1)) != 0)) {
402 			goto failed;
403 		}
404 
405 		auclnt_set_samples(sp, 0);
406 		auclnt_set_errors(sp, 0);
407 		play->eof = 0;
408 		play->buffer_size = 8192;
409 
410 		auclnt_set_gain(sp, ((play->gain * 100) / AUDIO_MAX_GAIN));
411 		auclnt_set_muted(sp, proc->p_info.output_muted);
412 		play->open = B_TRUE;
413 		proc->p_writer = c;
414 		proc->p_oflag |= FWRITE;
415 	}
416 
417 	if (oflag & FREAD) {
418 		audio_prinfo_t	*rec = &proc->p_info.record;
419 		audio_stream_t	*sp = auclnt_input_stream(c);
420 
421 		if (((rv = auclnt_set_rate(sp, 8000)) != 0) ||
422 		    ((rv = auclnt_set_format(sp, AUDIO_FORMAT_ULAW)) != 0) ||
423 		    ((rv = auclnt_set_channels(sp, 1)) != 0)) {
424 			goto failed;
425 		}
426 
427 		auclnt_set_samples(sp, 0);
428 		auclnt_set_errors(sp, 0);
429 		rec->eof = 0;
430 		rec->buffer_size = 8192;
431 
432 		auclnt_set_gain(sp, ((rec->gain * 100) / AUDIO_MAX_GAIN));
433 		rec->open = B_TRUE;
434 		proc->p_reader = c;
435 		proc->p_oflag |= FREAD;
436 	}
437 
438 
439 	dc->dc_wq = auclnt_get_wq(c);
440 
441 	/* we update the s_proc last to avoid a race */
442 	dc->dc_proc = proc;
443 
444 	devaudio_proc_update(proc);
445 
446 	mutex_exit(&dev->d_mx);
447 
448 	return (0);
449 
450 failed:
451 	mutex_exit(&dev->d_mx);
452 	devaudio_proc_release(c);
453 	return (rv);
454 
455 }
456 
457 static void
devaudio_clear_eof(audio_client_t * c)458 devaudio_clear_eof(audio_client_t *c)
459 {
460 	struct eofcnt	*eof;
461 	daclient_t	*dc;
462 
463 	dc = auclnt_get_private(c);
464 	mutex_enter(&dc->dc_lock);
465 	while ((eof = list_remove_head(&dc->dc_eofcnt)) != NULL) {
466 		kmem_free(eof, sizeof (*eof));
467 	}
468 	mutex_exit(&dc->dc_lock);
469 }
470 
471 void
devaudio_proc_release(audio_client_t * c)472 devaudio_proc_release(audio_client_t *c)
473 {
474 	daproc_t	*proc;
475 	dadev_t		*dev;
476 	mblk_t		*mp;
477 	daclient_t	*dc;
478 
479 	dc = auclnt_get_private(c);
480 	proc = dc->dc_proc;
481 	dev = dc->dc_dev;
482 	dc->dc_proc = NULL;
483 
484 	mutex_enter(&dev->d_mx);
485 
486 	if (proc != NULL) {
487 		proc->p_refcnt--;
488 		ASSERT(proc->p_refcnt >= 0);
489 
490 		if (c == proc->p_writer) {
491 			proc->p_oflag &= ~FWRITE;
492 			proc->p_writer = NULL;
493 		}
494 		if (c == proc->p_reader) {
495 			proc->p_oflag &= ~FREAD;
496 			proc->p_reader = NULL;
497 		}
498 		cv_broadcast(&dev->d_cv);
499 
500 		if (proc->p_refcnt == 0) {
501 			list_remove(&dev->d_procs, proc);
502 			devaudio_proc_free(proc);
503 		}
504 		dc->dc_proc = NULL;
505 	}
506 
507 	mutex_exit(&dev->d_mx);
508 
509 	devaudio_clear_eof(c);
510 
511 	while ((mp = dc->dc_draining) != NULL) {
512 		dc->dc_draining = mp->b_next;
513 		mp->b_next = NULL;
514 		freemsg(mp);
515 	}
516 
517 	mutex_destroy(&dc->dc_lock);
518 	list_destroy(&dc->dc_eofcnt);
519 	kmem_free(dc, sizeof (*dc));
520 }
521 
522 static void
devaudio_input(audio_client_t * c)523 devaudio_input(audio_client_t *c)
524 {
525 	audio_stream_t	*sp = auclnt_input_stream(c);
526 	daclient_t	*dc = auclnt_get_private(c);
527 	unsigned	framesz = auclnt_get_framesz(sp);
528 	queue_t		*rq = auclnt_get_rq(c);
529 	mblk_t		*mp;
530 	unsigned	nbytes = dc->dc_proc->p_info.record.buffer_size;
531 	unsigned	count = nbytes / framesz;
532 
533 	/*
534 	 * Potentially send a message upstream with the record data.
535 	 * We collect this up in chunks of the buffer size requested
536 	 * by the client.
537 	 */
538 
539 	while (auclnt_get_count(sp) >= count) {
540 
541 		if ((!canput(rq)) ||
542 		    ((mp = allocb(nbytes, BPRI_MED)) == NULL)) {
543 			/*
544 			 * This will apply back pressure to the
545 			 * buffer.  We haven't yet lost any data, we
546 			 * just can't send it up.  The point at which
547 			 * we have an unrecoverable overrun is in the
548 			 * buffer, not in the streams queue.  So, no
549 			 * need to do anything right now.
550 			 *
551 			 * Note that since recording is enabled, we
552 			 * expect that the callback routine will be
553 			 * called repeatedly & regularly, so we don't
554 			 * have to worry about leaving data orphaned
555 			 * in the queue.
556 			 */
557 			break;
558 		}
559 
560 		(void) auclnt_consume_data(sp, (caddr_t)mp->b_wptr, count);
561 		mp->b_wptr += nbytes;
562 		(void) putq(rq, mp);
563 	}
564 }
565 
566 static void
devaudio_proc_update(daproc_t * proc)567 devaudio_proc_update(daproc_t *proc)
568 {
569 	audio_info_t	*info;
570 	audio_stream_t	*sp;
571 	audio_client_t	*c;
572 
573 	info = &proc->p_info;
574 
575 	ASSERT(mutex_owned(&proc->p_dev->d_mx));
576 
577 	if ((c = proc->p_writer) != NULL) {
578 		sp = auclnt_output_stream(c);
579 
580 		info->play.sample_rate = auclnt_get_rate(sp);
581 		info->play.channels = auclnt_get_channels(sp);
582 		devaudio_decompose_format(&info->play, auclnt_get_format(sp));
583 
584 		info->play.gain =
585 		    (auclnt_get_gain(sp) * AUDIO_MAX_GAIN) / 100;
586 		info->play.pause = auclnt_is_paused(sp);
587 		info->play.active = auclnt_is_running(sp);
588 		info->play.samples = auclnt_get_samples(sp);
589 		info->play.error = auclnt_get_errors(sp) ? B_TRUE : B_FALSE;
590 		info->output_muted = auclnt_get_muted(sp);
591 	} else {
592 		info->play.encoding = AUDIO_ENCODING_NONE;
593 		info->play.precision = 0;
594 		info->play.sample_rate = 0;
595 		info->play.pause = B_FALSE;
596 		info->play.active = B_FALSE;
597 		info->play.error = B_FALSE;
598 		info->play.samples = 0;
599 	}
600 
601 	if ((c = proc->p_reader) != NULL) {
602 		sp = auclnt_input_stream(c);
603 
604 		info->record.sample_rate = auclnt_get_rate(sp);
605 		info->record.channels = auclnt_get_channels(sp);
606 		devaudio_decompose_format(&info->record, auclnt_get_format(sp));
607 
608 		info->record.gain =
609 		    (auclnt_get_gain(sp) * AUDIO_MAX_GAIN) / 100;
610 		info->record.pause = auclnt_is_paused(sp);
611 		info->record.active = auclnt_is_running(sp);
612 		info->record.samples = auclnt_get_samples(sp);
613 		info->record.error = auclnt_get_errors(sp) ? B_TRUE : B_FALSE;
614 	} else {
615 		info->record.encoding = AUDIO_ENCODING_NONE;
616 		info->record.precision = 0;
617 		info->record.sample_rate = 0;
618 		info->record.pause = B_FALSE;
619 		info->record.active = B_FALSE;
620 		info->record.error = B_FALSE;
621 		info->record.samples = 0;
622 	}
623 }
624 
625 static void
devaudio_ioc_getinfo(queue_t * wq,audio_client_t * c,mblk_t * mp)626 devaudio_ioc_getinfo(queue_t *wq, audio_client_t *c, mblk_t *mp)
627 {
628 	daclient_t	*dc = auclnt_get_private(c);
629 	daproc_t	*proc = dc->dc_proc;
630 	mblk_t		*bcont;
631 
632 	if ((bcont = allocb(sizeof (audio_info_t), BPRI_MED)) == NULL) {
633 		miocnak(wq, mp, 0, ENOMEM);
634 		return;
635 	}
636 
637 	mutex_enter(&dc->dc_dev->d_mx);
638 	devaudio_proc_update(proc);
639 	bcopy(&proc->p_info, bcont->b_wptr, sizeof (audio_info_t));
640 	mutex_exit(&dc->dc_dev->d_mx);
641 
642 	bcont->b_wptr += sizeof (audio_info_t);
643 
644 	mcopyout(mp, NULL, sizeof (audio_info_t), NULL, bcont);
645 	qreply(wq, mp);
646 }
647 
648 #define	CHANGED(new, old, field)			\
649 	((new->field != ((uint32_t)~0)) && (new->field != old->field))
650 #define	CHANGED8(new, old, field)			\
651 	((new->field != ((uint8_t)~0)) && (new->field != old->field))
652 
653 static void
devaudio_ioc_setinfo(queue_t * wq,audio_client_t * c,mblk_t * mp)654 devaudio_ioc_setinfo(queue_t *wq, audio_client_t *c, mblk_t *mp)
655 {
656 	daclient_t	*dc;
657 	daproc_t	*proc;
658 	audio_info_t	*oinfo;
659 	audio_info_t	*ninfo;
660 	audio_prinfo_t	*npr;
661 	audio_prinfo_t	*opr;
662 
663 	int		pfmt = AUDIO_FORMAT_NONE;
664 	int		rfmt = AUDIO_FORMAT_NONE;
665 
666 	boolean_t	reader;
667 	boolean_t	writer;
668 	boolean_t	isctl;
669 	audio_stream_t	*sp;
670 	int		rv;
671 	caddr_t		uaddr;
672 	mblk_t		*bcont;
673 
674 	struct copyresp	*csp;
675 
676 	if (DB_TYPE(mp) == M_IOCTL) {
677 		/* the special value "1" indicates that this is a copyin */
678 		uaddr = *(caddr_t *)(void *)mp->b_cont->b_rptr;
679 
680 		mcopyin(mp, uaddr, sizeof (audio_info_t), NULL);
681 		qreply(wq, mp);
682 		return;
683 	}
684 
685 	ASSERT(DB_TYPE(mp) == M_IOCDATA);
686 	if (((bcont = mp->b_cont) == NULL) ||
687 	    (MBLKL(mp->b_cont) != sizeof (audio_info_t))) {
688 		miocnak(wq, mp, 0, EINVAL);
689 		return;
690 	}
691 
692 	mp->b_cont = NULL;
693 	csp = (void *)mp->b_rptr;
694 	uaddr = (void *)csp->cp_private;
695 	dc = auclnt_get_private(c);
696 	ninfo = (void *)bcont->b_rptr;
697 
698 	mutex_enter(&dc->dc_dev->d_mx);
699 
700 	proc = dc->dc_proc;
701 	oinfo = &proc->p_info;
702 
703 	if (auclnt_get_minor_type(c) == AUDIO_MINOR_DEVAUDIOCTL) {
704 		/* control node can do both read and write fields */
705 		isctl = B_TRUE;
706 		reader = B_TRUE;
707 		writer = B_TRUE;
708 	} else {
709 		isctl = B_FALSE;
710 		writer = (c == proc->p_writer);
711 		reader = (c == proc->p_reader);
712 	}
713 
714 	/*
715 	 * Start by validating settings.
716 	 */
717 	npr = &ninfo->play;
718 	opr = &oinfo->play;
719 
720 	if (writer && CHANGED(npr, opr, sample_rate)) {
721 		if ((isctl) ||
722 		    (npr->sample_rate < 5500) || (npr->sample_rate > 48000)) {
723 			rv = EINVAL;
724 			goto err;
725 		}
726 	}
727 	if (writer && CHANGED(npr, opr, channels)) {
728 		if ((isctl) || (npr->channels < 1) || (npr->channels > 2)) {
729 			rv = EINVAL;
730 			goto err;
731 		}
732 	}
733 	if (writer &&
734 	    (CHANGED(npr, opr, encoding) || CHANGED(npr, opr, precision))) {
735 		if (npr->encoding == (uint32_t)~0)
736 			npr->encoding = opr->encoding;
737 		if (npr->precision == (uint32_t)~0)
738 			npr->precision = opr->precision;
739 		pfmt = devaudio_compose_format(npr);
740 		if ((isctl) || (pfmt == AUDIO_FORMAT_NONE)) {
741 			rv = EINVAL;
742 			goto err;
743 		}
744 	}
745 
746 	/* play fields that anyone can modify */
747 	if (CHANGED(npr, opr, gain)) {
748 		if (npr->gain > AUDIO_MAX_GAIN) {
749 			rv = EINVAL;
750 			goto err;
751 		}
752 	}
753 
754 
755 	npr = &ninfo->record;
756 	opr = &oinfo->record;
757 
758 	if (reader && CHANGED(npr, opr, sample_rate)) {
759 		if ((isctl) ||
760 		    (npr->sample_rate < 5500) || (npr->sample_rate > 48000)) {
761 			rv = EINVAL;
762 			goto err;
763 		}
764 	}
765 	if (reader && CHANGED(npr, opr, channels)) {
766 		if ((isctl) || (npr->channels < 1) || (npr->channels > 2)) {
767 			rv = EINVAL;
768 			goto err;
769 		}
770 	}
771 	if (reader &&
772 	    (CHANGED(npr, opr, encoding) || CHANGED(npr, opr, precision))) {
773 		if (npr->encoding == (uint32_t)~0)
774 			npr->encoding = opr->encoding;
775 		if (npr->precision == (uint32_t)~0)
776 			npr->precision = opr->precision;
777 		rfmt = devaudio_compose_format(npr);
778 		if ((isctl) || (rfmt == AUDIO_FORMAT_NONE)) {
779 			rv = EINVAL;
780 			goto err;
781 		}
782 	}
783 	if (reader && CHANGED(npr, opr, buffer_size)) {
784 		if (isctl) {
785 			rv = EINVAL;
786 			goto err;
787 		}
788 		/* make sure we can support 16-bit stereo samples */
789 		if ((npr->buffer_size % 4) != 0) {
790 			npr->buffer_size = (npr->buffer_size + 3) & ~3;
791 		}
792 		/* limit the maximum buffer size somewhat */
793 		if (npr->buffer_size > 16384) {
794 			npr->buffer_size = 16384;
795 		}
796 	}
797 
798 	/* record fields that anyone can modify */
799 	if (CHANGED(npr, opr, gain)) {
800 		if (npr->gain > AUDIO_MAX_GAIN) {
801 			rv = EINVAL;
802 			goto err;
803 		}
804 	}
805 
806 	/*
807 	 * Now apply the changes.
808 	 */
809 	if (proc->p_writer != NULL) {
810 		sp = auclnt_output_stream(proc->p_writer);
811 		npr = &ninfo->play;
812 		opr = &oinfo->play;
813 
814 		if (CHANGED(npr, opr, sample_rate)) {
815 			if ((rv = auclnt_set_rate(sp, npr->sample_rate)) != 0)
816 				goto err;
817 		}
818 		if (CHANGED(npr, opr, channels)) {
819 			if ((rv = auclnt_set_channels(sp, npr->channels)) != 0)
820 				goto err;
821 		}
822 		if (pfmt != AUDIO_FORMAT_NONE) {
823 			if ((rv = auclnt_set_format(sp, pfmt)) != 0)
824 				goto err;
825 		}
826 		if (CHANGED(npr, opr, samples)) {
827 			auclnt_set_samples(sp, npr->samples);
828 		}
829 		if (CHANGED(npr, opr, eof)) {
830 			/*
831 			 * This ugly special case code is required to
832 			 * prevent problems with realaudio.
833 			 */
834 			if (npr->eof == 0) {
835 				devaudio_clear_eof(proc->p_writer);
836 			}
837 			opr->eof = npr->eof;
838 		}
839 		if (CHANGED8(npr, opr, pause)) {
840 			if (npr->pause) {
841 				auclnt_set_paused(sp);
842 			} else {
843 				auclnt_clear_paused(sp);
844 
845 				/* qenable to start up the playback */
846 				qenable(auclnt_get_wq(proc->p_writer));
847 			}
848 		}
849 		if (CHANGED8(npr, opr, waiting) && (npr->waiting)) {
850 			opr->waiting = npr->waiting;
851 		}
852 		if (CHANGED8(npr, opr, error)) {
853 			auclnt_set_errors(sp, npr->error);
854 		}
855 		if (CHANGED(npr, opr, gain)) {
856 			auclnt_set_gain(sp, (npr->gain * 100) / AUDIO_MAX_GAIN);
857 		}
858 		if (CHANGED8(ninfo, oinfo, output_muted)) {
859 			auclnt_set_muted(sp, ninfo->output_muted);
860 		}
861 		if (CHANGED(npr, opr, buffer_size)) {
862 			/*
863 			 * No checks on the buffer size are performed
864 			 * for play side.  The value of the buffer size
865 			 * is meaningless for play side anyway.
866 			 */
867 			opr->buffer_size = npr->buffer_size;
868 		}
869 	} else {
870 		/* these values are preserved even if /dev/audio not open */
871 		if (CHANGED(npr, opr, gain)) {
872 			opr->gain = npr->gain;
873 		}
874 		if (CHANGED8(ninfo, oinfo, output_muted)) {
875 			oinfo->output_muted = ninfo->output_muted;
876 		}
877 	}
878 
879 	if (proc->p_reader != NULL) {
880 		sp = auclnt_input_stream(proc->p_reader);
881 		npr = &ninfo->record;
882 		opr = &oinfo->record;
883 
884 		if (CHANGED(npr, opr, sample_rate)) {
885 			if ((rv = auclnt_set_rate(sp, npr->sample_rate)) != 0)
886 				goto err;
887 		}
888 		if (CHANGED(npr, opr, channels)) {
889 			if ((rv = auclnt_set_channels(sp, npr->channels)) != 0)
890 				goto err;
891 		}
892 		if (rfmt != AUDIO_FORMAT_NONE) {
893 			if ((rv = auclnt_set_format(sp, rfmt)) != 0)
894 				goto err;
895 		}
896 		if (CHANGED(npr, opr, samples)) {
897 			auclnt_set_samples(sp, npr->samples);
898 		}
899 		if (CHANGED(npr, opr, eof)) {
900 			opr->eof = npr->eof;
901 		}
902 		if (CHANGED8(npr, opr, pause)) {
903 			if (npr->pause) {
904 				auclnt_set_paused(sp);
905 			} else {
906 				auclnt_clear_paused(sp);
907 				auclnt_start(sp);
908 			}
909 		}
910 		if (CHANGED8(npr, opr, waiting) && (npr->waiting)) {
911 			opr->waiting = npr->waiting;
912 		}
913 		if (CHANGED8(npr, opr, error)) {
914 			auclnt_set_errors(sp, npr->error);
915 		}
916 		if (CHANGED(npr, opr, buffer_size)) {
917 			opr->buffer_size = npr->buffer_size;
918 		}
919 		if (CHANGED(npr, opr, gain)) {
920 			auclnt_set_gain(sp, (npr->gain * 100) / AUDIO_MAX_GAIN);
921 		}
922 	} else {
923 		/* these values are preserved even if /dev/audio not open */
924 		if (CHANGED(npr, opr, gain)) {
925 			opr->gain = npr->gain;
926 		}
927 	}
928 
929 	devaudio_proc_update(dc->dc_proc);
930 	bcopy(&dc->dc_proc->p_info, ninfo, sizeof (*ninfo));
931 
932 	mutex_exit(&dc->dc_dev->d_mx);
933 	mcopyout(mp, NULL, sizeof (audio_info_t), uaddr, bcont);
934 	qreply(wq, mp);
935 	return;
936 
937 err:
938 	mutex_exit(&dc->dc_dev->d_mx);
939 	miocnak(wq, mp, 0, rv);
940 }
941 
942 static void
devaudio_ioc_getdev(queue_t * wq,audio_client_t * c,mblk_t * mp)943 devaudio_ioc_getdev(queue_t *wq, audio_client_t *c, mblk_t *mp)
944 {
945 	audio_dev_t	*d = auclnt_get_dev(c);
946 	mblk_t		*bcont;
947 	audio_device_t	*a;
948 
949 	if ((bcont = allocb(sizeof (*a), BPRI_MED)) == NULL) {
950 		miocnak(wq, mp, 0, ENOMEM);
951 		return;
952 	}
953 
954 	a = (void *)bcont->b_wptr;
955 	(void) snprintf(a->name, sizeof (a->name),
956 	    "SUNW,%s", auclnt_get_dev_name(d));
957 	(void) strlcpy(a->config,
958 	    auclnt_get_dev_description(d), sizeof (a->config));
959 	(void) strlcpy(a->version,
960 	    auclnt_get_dev_version(d),  sizeof (a->version));
961 	bcont->b_wptr += sizeof (*a);
962 
963 	mcopyout(mp, NULL, sizeof (*a), NULL, bcont);
964 	qreply(wq, mp);
965 }
966 
967 static int
devaudio_sigpoll(audio_client_t * c,void * arg)968 devaudio_sigpoll(audio_client_t *c, void *arg)
969 {
970 	pid_t		pid = (pid_t)(uintptr_t)arg;
971 
972 	if (auclnt_get_minor_type(c) == AUDIO_MINOR_DEVAUDIOCTL) {
973 		/* we only need to notify peers in our own process */
974 		if (auclnt_get_pid(c) == pid) {
975 			(void) putctl1(auclnt_get_rq(c), M_PCSIG, SIGPOLL);
976 		}
977 	}
978 	return (AUDIO_WALK_CONTINUE);
979 }
980 
981 static void
devaudio_drain(audio_client_t * c)982 devaudio_drain(audio_client_t *c)
983 {
984 	daclient_t	*dc = auclnt_get_private(c);
985 	mblk_t		*mplist, *mp;
986 
987 	mutex_enter(&dc->dc_lock);
988 	mplist = dc->dc_draining;
989 	dc->dc_draining = NULL;
990 	mutex_exit(&dc->dc_lock);
991 
992 	while ((mp = mplist) != NULL) {
993 		mplist = mp->b_next;
994 		mp->b_next = NULL;
995 		mioc2ack(mp, NULL, 0, 0);
996 		(void) putq(auclnt_get_rq(c), mp);
997 	}
998 }
999 
1000 static void
devaudio_output(audio_client_t * c)1001 devaudio_output(audio_client_t *c)
1002 {
1003 	daclient_t	*dc = auclnt_get_private(c);
1004 	daproc_t	*proc = dc->dc_proc;
1005 	uint64_t	tail;
1006 	struct eofcnt	*eof;
1007 	int		eofs = 0;
1008 
1009 	tail = auclnt_get_tail(auclnt_output_stream(c));
1010 
1011 	/* get more data! (do this early) */
1012 	qenable(auclnt_get_wq(c));
1013 
1014 	mutex_enter(&dc->dc_lock);
1015 	while (((eof = list_head(&dc->dc_eofcnt)) != NULL) &&
1016 	    (eof->tail <= tail)) {
1017 		list_remove(&dc->dc_eofcnt, eof);
1018 		kmem_free(eof, sizeof (*eof));
1019 		eofs++;
1020 	}
1021 	proc->p_info.play.eof += eofs;
1022 	mutex_exit(&dc->dc_lock);
1023 
1024 	if (eofs) {
1025 		auclnt_dev_walk_clients(auclnt_get_dev(c),
1026 		    devaudio_sigpoll, (void *)(uintptr_t)auclnt_get_pid(c));
1027 	}
1028 }
1029 
1030 static void *
devaudio_init(audio_dev_t * adev)1031 devaudio_init(audio_dev_t *adev)
1032 {
1033 	dadev_t		*dev;
1034 	unsigned	cap;
1035 
1036 	cap = auclnt_get_dev_capab(adev);
1037 	/* if not a play or record device, don't bother initializing it */
1038 	if ((cap & (AUDIO_CLIENT_CAP_PLAY | AUDIO_CLIENT_CAP_RECORD)) == 0) {
1039 		return (NULL);
1040 	}
1041 
1042 	dev = kmem_zalloc(sizeof (*dev), KM_SLEEP);
1043 	dev->d_dev = adev;
1044 	mutex_init(&dev->d_mx, NULL, MUTEX_DRIVER, NULL);
1045 	cv_init(&dev->d_cv, NULL, CV_DRIVER, NULL);
1046 	list_create(&dev->d_procs, sizeof (struct daproc),
1047 	    offsetof(struct daproc, p_linkage));
1048 
1049 	return (dev);
1050 }
1051 
1052 static void
devaudio_fini(void * arg)1053 devaudio_fini(void *arg)
1054 {
1055 	dadev_t	*dev = arg;
1056 
1057 	if (dev != NULL) {
1058 
1059 		mutex_destroy(&dev->d_mx);
1060 		cv_destroy(&dev->d_cv);
1061 		list_destroy(&dev->d_procs);
1062 		kmem_free(dev, sizeof (*dev));
1063 	}
1064 }
1065 
1066 static int
devaudio_open(audio_client_t * c,int oflag)1067 devaudio_open(audio_client_t *c, int oflag)
1068 {
1069 	int	rv;
1070 
1071 	if ((rv = auclnt_open(c, oflag)) != 0) {
1072 		return (rv);
1073 	}
1074 
1075 	if ((rv = devaudio_proc_hold(c, oflag)) != 0) {
1076 		auclnt_close(c);
1077 		return (rv);
1078 	}
1079 
1080 	/* start up the input */
1081 	if (oflag & FREAD) {
1082 		auclnt_start(auclnt_input_stream(c));
1083 	}
1084 
1085 	return (0);
1086 }
1087 
1088 static int
devaudioctl_open(audio_client_t * c,int oflag)1089 devaudioctl_open(audio_client_t *c, int oflag)
1090 {
1091 	int	rv;
1092 
1093 	_NOTE(ARGUNUSED(oflag));
1094 
1095 	oflag &= ~(FWRITE | FREAD);
1096 
1097 	if ((rv = auclnt_open(c, 0)) != 0) {
1098 		return (rv);
1099 	}
1100 
1101 	if ((rv = devaudio_proc_hold(c, oflag)) != 0) {
1102 		auclnt_close(c);
1103 		return (rv);
1104 	}
1105 
1106 	return (0);
1107 }
1108 
1109 static void
devaudio_close(audio_client_t * c)1110 devaudio_close(audio_client_t *c)
1111 {
1112 	auclnt_stop(auclnt_output_stream(c));
1113 	auclnt_stop(auclnt_input_stream(c));
1114 
1115 	auclnt_close(c);
1116 	devaudio_proc_release(c);
1117 }
1118 
1119 static void
devaudioctl_close(audio_client_t * c)1120 devaudioctl_close(audio_client_t *c)
1121 {
1122 	auclnt_close(c);
1123 	devaudio_proc_release(c);
1124 }
1125 
1126 static void
devaudio_miocdata(audio_client_t * c,mblk_t * mp)1127 devaudio_miocdata(audio_client_t *c, mblk_t *mp)
1128 {
1129 	struct copyresp		*csp;
1130 	queue_t			*wq;
1131 
1132 	csp = (void *)mp->b_rptr;
1133 	wq = auclnt_get_wq(c);
1134 
1135 	/*
1136 	 * If a transfer error occurred, the framework already
1137 	 * MIOCNAK'd it.
1138 	 */
1139 	if (csp->cp_rval != 0) {
1140 		freemsg(mp);
1141 		return;
1142 	}
1143 
1144 	/*
1145 	 * If no state, then this is a response to M_COPYOUT, and we
1146 	 * are done.  (Audio ioctls just copyout a single structure at
1147 	 * completion of work.)
1148 	 */
1149 	if (csp->cp_private == NULL) {
1150 		miocack(wq, mp, 0, 0);
1151 		return;
1152 	}
1153 
1154 	/* now, call the handler ioctl */
1155 	switch (csp->cp_cmd) {
1156 	case AUDIO_SETINFO:
1157 		devaudio_ioc_setinfo(wq, c, mp);
1158 		break;
1159 	default:
1160 		miocnak(wq, mp, 0, EINVAL);
1161 		break;
1162 	}
1163 }
1164 
1165 static void
devaudio_mioctl(audio_client_t * c,mblk_t * mp)1166 devaudio_mioctl(audio_client_t *c, mblk_t *mp)
1167 {
1168 	struct iocblk	*iocp = (void *)mp->b_rptr;
1169 	queue_t		*wq = auclnt_get_wq(c);
1170 
1171 	/* BSD legacy here: we only support transparent ioctls */
1172 	if (iocp->ioc_count != TRANSPARENT) {
1173 		miocnak(wq, mp, 0, EINVAL);
1174 		return;
1175 	}
1176 
1177 	switch (iocp->ioc_cmd) {
1178 	case AUDIO_GETINFO:
1179 		devaudio_ioc_getinfo(wq, c, mp);
1180 		break;
1181 
1182 	case AUDIO_SETINFO:
1183 		devaudio_ioc_setinfo(wq, c, mp);
1184 		break;
1185 
1186 	case AUDIO_GETDEV:
1187 		devaudio_ioc_getdev(wq, c, mp);
1188 		break;
1189 
1190 	case AUDIO_DIAG_LOOPBACK:
1191 		/* we don't support this one */
1192 		miocnak(wq, mp, 0, ENOTTY);
1193 		break;
1194 
1195 	case AUDIO_MIXERCTL_GET_MODE:
1196 	case AUDIO_MIXERCTL_SET_MODE:
1197 	case AUDIO_MIXERCTL_GET_CHINFO:
1198 	case AUDIO_MIXERCTL_SET_CHINFO:
1199 	case AUDIO_MIXERCTL_GETINFO:
1200 	case AUDIO_MIXERCTL_SETINFO:
1201 	case AUDIO_GET_NUM_CHS:
1202 	case AUDIO_GET_CH_NUMBER:
1203 	case AUDIO_GET_CH_TYPE:
1204 	case AUDIO_MIXER_SINGLE_OPEN:
1205 	case AUDIO_MIXER_MULTIPLE_OPEN:
1206 	case AUDIO_MIXER_GET_SAMPLE_RATES:
1207 	default:
1208 		miocnak(wq, mp, 0, EINVAL);
1209 		break;
1210 	}
1211 }
1212 
1213 static void
devaudioctl_wput(audio_client_t * c,mblk_t * mp)1214 devaudioctl_wput(audio_client_t *c, mblk_t *mp)
1215 {
1216 	queue_t		*wq = auclnt_get_wq(c);
1217 
1218 	switch (DB_TYPE(mp)) {
1219 	case M_IOCTL:
1220 		/* Drain ioctl needs to be handled on the service queue */
1221 		devaudio_mioctl(c, mp);
1222 		break;
1223 
1224 	case M_IOCDATA:
1225 		devaudio_miocdata(c, mp);
1226 		break;
1227 
1228 	case M_FLUSH:
1229 		/*
1230 		 * We don't flush the engine.  The reason is that
1231 		 * other streams might be using the engine.  This is
1232 		 * fundamentally no different from the case where the
1233 		 * engine hardware has data buffered in an
1234 		 * inaccessible FIFO.
1235 		 *
1236 		 * Clients that want to ensure no more data is coming
1237 		 * should stop the stream before flushing.
1238 		 */
1239 		if (*mp->b_rptr & FLUSHW) {
1240 			*mp->b_rptr &= ~FLUSHW;
1241 		}
1242 		if (*mp->b_rptr & FLUSHR) {
1243 			qreply(wq, mp);
1244 		} else {
1245 			freemsg(mp);
1246 		}
1247 		break;
1248 
1249 	case M_DATA:
1250 		/*
1251 		 * No audio data on control nodes!
1252 		 */
1253 		freemsg(mp);
1254 
1255 	default:
1256 		freemsg(mp);
1257 		break;
1258 	}
1259 }
1260 
1261 static void
devaudio_wput(audio_client_t * c,mblk_t * mp)1262 devaudio_wput(audio_client_t *c, mblk_t *mp)
1263 {
1264 	queue_t		*wq = auclnt_get_wq(c);
1265 
1266 	switch (DB_TYPE(mp)) {
1267 	case M_IOCTL:
1268 		/* Drain ioctl needs to be handled on the service queue */
1269 		if (*(int *)(void *)mp->b_rptr == AUDIO_DRAIN) {
1270 			(void) putq(wq, mp);
1271 		} else {
1272 			devaudio_mioctl(c, mp);
1273 		}
1274 		break;
1275 
1276 	case M_IOCDATA:
1277 		devaudio_miocdata(c, mp);
1278 		break;
1279 
1280 	case M_FLUSH:
1281 		/*
1282 		 * We don't flush the engine.  The reason is that
1283 		 * other streams might be using the engine.  This is
1284 		 * fundamentally no different from the case where the
1285 		 * engine hardware has data buffered in an
1286 		 * inaccessible FIFO.
1287 		 *
1288 		 * Clients that want to ensure no more data is coming
1289 		 * should stop the stream before flushing.
1290 		 */
1291 		if (*mp->b_rptr & FLUSHW) {
1292 			flushq(wq, FLUSHALL);
1293 			auclnt_flush(auclnt_output_stream(c));
1294 			*mp->b_rptr &= ~FLUSHW;
1295 		}
1296 		if (*mp->b_rptr & FLUSHR) {
1297 			flushq(RD(wq), FLUSHALL);
1298 			auclnt_flush(auclnt_input_stream(c));
1299 			qreply(wq, mp);
1300 		} else {
1301 			freemsg(mp);
1302 		}
1303 		break;
1304 
1305 	case M_DATA:
1306 		/*
1307 		 * Defer processing to the queue.  This keeps the data
1308 		 * ordered, and allows the wsrv routine to gather
1309 		 * multiple mblks at once.
1310 		 */
1311 		if (mp->b_cont != NULL) {
1312 
1313 			/*
1314 			 * If we need to pullup, do it here to
1315 			 * simplify the rest of the processing later.
1316 			 * This should rarely (if ever) be necessary.
1317 			 */
1318 			mblk_t	*nmp;
1319 
1320 			if ((nmp = msgpullup(mp, -1)) == NULL) {
1321 				freemsg(mp);
1322 			} else {
1323 				freemsg(mp);
1324 				(void) putq(wq, nmp);
1325 			}
1326 		} else {
1327 			(void) putq(wq, mp);
1328 		}
1329 		break;
1330 
1331 	default:
1332 		freemsg(mp);
1333 		break;
1334 	}
1335 }
1336 
1337 static void
devaudio_rsrv(audio_client_t * c)1338 devaudio_rsrv(audio_client_t *c)
1339 {
1340 	queue_t		*rq = auclnt_get_rq(c);
1341 	mblk_t		*mp;
1342 
1343 	while ((mp = getq(rq)) != NULL) {
1344 
1345 		if ((queclass(mp) != QPCTL) && (!canputnext(rq))) {
1346 			/*
1347 			 * Put it back in the queue so we can apply
1348 			 * backpressure properly.
1349 			 */
1350 			(void) putbq(rq, mp);
1351 			return;
1352 		}
1353 		putnext(rq, mp);
1354 	}
1355 }
1356 
1357 static void
devaudio_wsrv(audio_client_t * c)1358 devaudio_wsrv(audio_client_t *c)
1359 {
1360 	queue_t		*wq = auclnt_get_wq(c);
1361 	daclient_t	*dc = auclnt_get_private(c);
1362 	audio_stream_t	*sp;
1363 	mblk_t		*mp;
1364 	unsigned	framesz;
1365 
1366 	sp = auclnt_output_stream(c);
1367 
1368 	framesz = auclnt_get_framesz(sp);
1369 
1370 	while ((mp = getq(wq)) != NULL) {
1371 
1372 		unsigned	count;
1373 
1374 		/* got a message */
1375 
1376 		/* if its a drain ioctl, we need to process it here */
1377 		if (DB_TYPE(mp) == M_IOCTL) {
1378 			ASSERT((*(int *)(void *)mp->b_rptr) == AUDIO_DRAIN);
1379 			mutex_enter(&dc->dc_lock);
1380 			mp->b_next = dc->dc_draining;
1381 			dc->dc_draining = mp;
1382 			mutex_exit(&dc->dc_lock);
1383 
1384 			if (auclnt_start_drain(c) != 0) {
1385 				devaudio_drain(c);
1386 			}
1387 			continue;
1388 		}
1389 
1390 		ASSERT(DB_TYPE(mp) == M_DATA);
1391 
1392 		/*
1393 		 * Empty mblk require special handling, since they
1394 		 * indicate EOF.  We treat them separate from the main
1395 		 * processing loop.
1396 		 */
1397 		if (MBLKL(mp) == 0) {
1398 			struct eofcnt	*eof;
1399 
1400 			eof = kmem_zalloc(sizeof (*eof), KM_NOSLEEP);
1401 			if (eof != NULL) {
1402 				eof->tail = auclnt_get_head(sp);
1403 				mutex_enter(&dc->dc_lock);
1404 				list_insert_tail(&dc->dc_eofcnt, eof);
1405 				mutex_exit(&dc->dc_lock);
1406 			}
1407 			freemsg(mp);
1408 			continue;
1409 		}
1410 
1411 		count = auclnt_produce_data(sp, (caddr_t)mp->b_rptr,
1412 		    MBLKL(mp) / framesz);
1413 
1414 		mp->b_rptr += count * framesz;
1415 
1416 		if (MBLKL(mp) >= framesz) {
1417 			(void) putbq(wq, mp);
1418 			break;
1419 		} else {
1420 			freemsg(mp);
1421 		}
1422 	}
1423 
1424 	/* if the stream isn't running yet, start it up */
1425 	if (!auclnt_is_paused(sp))
1426 		auclnt_start(sp);
1427 }
1428 
1429 static struct audio_client_ops devaudio_ops = {
1430 	"sound,audio",
1431 	devaudio_init,
1432 	devaudio_fini,
1433 	devaudio_open,
1434 	devaudio_close,
1435 	NULL,	/* read */
1436 	NULL,	/* write */
1437 	NULL,	/* ioctl */
1438 	NULL,	/* chpoll */
1439 	NULL,	/* mmap */
1440 	devaudio_input,
1441 	devaudio_output,
1442 	devaudio_drain,
1443 	devaudio_wput,
1444 	devaudio_wsrv,
1445 	devaudio_rsrv
1446 };
1447 
1448 static struct audio_client_ops devaudioctl_ops = {
1449 	"sound,audioctl",
1450 	NULL,	/* dev_init */
1451 	NULL,	/* dev_fini */
1452 	devaudioctl_open,
1453 	devaudioctl_close,
1454 	NULL,	/* read */
1455 	NULL,	/* write */
1456 	NULL,	/* ioctl */
1457 	NULL,	/* chpoll */
1458 	NULL,	/* mmap */
1459 	NULL,	/* output */
1460 	NULL,	/* input */
1461 	NULL,	/* drain */
1462 	devaudioctl_wput,
1463 	NULL,
1464 	devaudio_rsrv
1465 };
1466 
1467 void
auimpl_sun_init(void)1468 auimpl_sun_init(void)
1469 {
1470 	auclnt_register_ops(AUDIO_MINOR_DEVAUDIO, &devaudio_ops);
1471 	auclnt_register_ops(AUDIO_MINOR_DEVAUDIOCTL, &devaudioctl_ops);
1472 }
1473