VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostALSAAudio.cpp@ 57146

Last change on this file since 57146 was 56649, checked in by vboxsync, 10 years ago

RT_MIN is a macro

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 42.0 KB
Line 
1/* $Id: DrvHostALSAAudio.cpp 56649 2015-06-26 06:55:41Z vboxsync $ */
2/** @file
3 * VBox audio devices: ALSA audio driver.
4 */
5
6/*
7 * Copyright (C) 2006-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 * --------------------------------------------------------------------
17 *
18 * This code is based on: alsaaudio.c
19 *
20 * QEMU ALSA audio driver
21 *
22 * Copyright (c) 2005 Vassili Karpov (malc)
23 *
24 * Permission is hereby granted, free of charge, to any person obtaining a copy
25 * of this software and associated documentation files (the "Software"), to deal
26 * in the Software without restriction, including without limitation the rights
27 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
28 * copies of the Software, and to permit persons to whom the Software is
29 * furnished to do so, subject to the following conditions:
30 *
31 * The above copyright notice and this permission notice shall be included in
32 * all copies or substantial portions of the Software.
33 *
34 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
35 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
36 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
37 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
38 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
39 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
40 * THE SOFTWARE.
41 */
42
43/*******************************************************************************
44* Header Files *
45*******************************************************************************/
46#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
47#include <VBox/log.h>
48#include <iprt/alloc.h>
49#include <iprt/uuid.h> /* For PDMIBASE_2_PDMDRV. */
50#include <VBox/vmm/pdmaudioifs.h>
51
52RT_C_DECLS_BEGIN
53 #include "alsa_stubs.h"
54 #include "alsa_mangling.h"
55RT_C_DECLS_END
56
57#include <alsa/asoundlib.h>
58
59#include "DrvAudio.h"
60#include "AudioMixBuffer.h"
61
62#include "VBoxDD.h"
63
64typedef struct ALSAAUDIOSTREAMIN
65{
66 PDMAUDIOHSTSTRMIN pStreamIn;
67 snd_pcm_t *phPCM;
68 void *pvBuf;
69 size_t cbBuf;
70} ALSAAUDIOSTREAMIN, *PALSAAUDIOSTREAMIN;
71
72typedef struct ALSAAUDIOSTREAMOUT
73{
74 PDMAUDIOHSTSTRMOUT pStreamOut;
75 snd_pcm_t *phPCM;
76 void *pvBuf;
77 size_t cbBuf;
78} ALSAAUDIOSTREAMOUT, *PALSAAUDIOSTREAMOUT;
79
80/* latency = period_size * periods / (rate * bytes_per_frame) */
81
82typedef struct ALSAAUDIOCFG
83{
84 int size_in_usec_in;
85 int size_in_usec_out;
86 const char *pcm_name_in;
87 const char *pcm_name_out;
88 unsigned int buffer_size_in;
89 unsigned int period_size_in;
90 unsigned int buffer_size_out;
91 unsigned int period_size_out;
92 unsigned int threshold;
93
94 int buffer_size_in_overriden;
95 int period_size_in_overriden;
96
97 int buffer_size_out_overriden;
98 int period_size_out_overriden;
99
100} ALSAAUDIOCFG, *PALSAAUDIOCFG;
101
102static int drvHostALSAAudioRecover(snd_pcm_t *phPCM);
103
104static ALSAAUDIOCFG s_ALSAConf =
105{
106#ifdef HIGH_LATENCY
107 1,
108 1,
109#else
110 0,
111 0,
112#endif
113 "default",
114 "default",
115#ifdef HIGH_LATENCY
116 400000,
117 400000 / 4,
118 400000,
119 400000 / 4,
120#else
121# define DEFAULT_BUFFER_SIZE 1024
122# define DEFAULT_PERIOD_SIZE 256
123 DEFAULT_BUFFER_SIZE * 4,
124 DEFAULT_PERIOD_SIZE * 4,
125 DEFAULT_BUFFER_SIZE,
126 DEFAULT_PERIOD_SIZE,
127#endif
128 0,
129 0,
130 0,
131 0,
132 0
133};
134
135/**
136 * Host Alsa audio driver instance data.
137 * @implements PDMIAUDIOCONNECTOR
138 */
139typedef struct DRVHOSTALSAAUDIO
140{
141 /** Pointer to the driver instance structure. */
142 PPDMDRVINS pDrvIns;
143 /** Pointer to host audio interface. */
144 PDMIHOSTAUDIO IHostAudio;
145 /** Error count for not flooding the release log.
146 * UINT32_MAX for unlimited logging. */
147 uint32_t cLogErrors;
148} DRVHOSTALSAAUDIO, *PDRVHOSTALSAAUDIO;
149
150typedef struct ALSAAUDIOSTREAMCFG
151{
152 unsigned int freq;
153 snd_pcm_format_t fmt;
154 int nchannels;
155 unsigned long buffer_size;
156 unsigned long period_size;
157 snd_pcm_uframes_t samples;
158} ALSAAUDIOSTREAMCFG, *PALSAAUDIOSTREAMCFG;
159
160static int drvHostALSAAudioClose(snd_pcm_t **pphPCM)
161{
162 if (!pphPCM || !*pphPCM)
163 return VINF_SUCCESS;
164
165 int rc;
166 int rc2 = snd_pcm_close(*pphPCM);
167 if (rc2)
168 {
169 LogRel(("ALSA: Closing PCM descriptor failed: %s\n",
170 snd_strerror(rc2)));
171 rc = VERR_GENERAL_FAILURE; /** @todo */
172 }
173 else
174 {
175 *pphPCM = NULL;
176 rc = VINF_SUCCESS;
177 }
178
179 return rc;
180}
181
182static snd_pcm_format_t drvHostALSAAudioFmtToALSA(PDMAUDIOFMT fmt)
183{
184 switch (fmt)
185 {
186 case AUD_FMT_S8:
187 return SND_PCM_FORMAT_S8;
188
189 case AUD_FMT_U8:
190 return SND_PCM_FORMAT_U8;
191
192 case AUD_FMT_S16:
193 return SND_PCM_FORMAT_S16_LE;
194
195 case AUD_FMT_U16:
196 return SND_PCM_FORMAT_U16_LE;
197
198 case AUD_FMT_S32:
199 return SND_PCM_FORMAT_S32_LE;
200
201 case AUD_FMT_U32:
202 return SND_PCM_FORMAT_U32_LE;
203
204 default:
205 break;
206 }
207
208 AssertMsgFailed(("Format %ld not supported\n", fmt));
209 return SND_PCM_FORMAT_U8;
210}
211
212static int drvHostALSAAudioALSAToFmt(snd_pcm_format_t fmt,
213 PDMAUDIOFMT *pFmt, PDMAUDIOENDIANNESS *pEndianness)
214{
215 AssertPtrReturn(pFmt, VERR_INVALID_POINTER);
216 /* pEndianness is optional. */
217
218 switch (fmt)
219 {
220 case SND_PCM_FORMAT_S8:
221 *pFmt = AUD_FMT_S8;
222 if (pEndianness)
223 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
224 break;
225
226 case SND_PCM_FORMAT_U8:
227 *pFmt = AUD_FMT_U8;
228 if (pEndianness)
229 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
230 break;
231
232 case SND_PCM_FORMAT_S16_LE:
233 *pFmt = AUD_FMT_S16;
234 if (pEndianness)
235 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
236 break;
237
238 case SND_PCM_FORMAT_U16_LE:
239 *pFmt = AUD_FMT_U16;
240 if (pEndianness)
241 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
242 break;
243
244 case SND_PCM_FORMAT_S16_BE:
245 *pFmt = AUD_FMT_S16;
246 if (pEndianness)
247 *pEndianness = PDMAUDIOENDIANNESS_BIG;
248 break;
249
250 case SND_PCM_FORMAT_U16_BE:
251 *pFmt = AUD_FMT_U16;
252 if (pEndianness)
253 *pEndianness = PDMAUDIOENDIANNESS_BIG;
254 break;
255
256 case SND_PCM_FORMAT_S32_LE:
257 *pFmt = AUD_FMT_S32;
258 if (pEndianness)
259 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
260 break;
261
262 case SND_PCM_FORMAT_U32_LE:
263 *pFmt = AUD_FMT_U32;
264 if (pEndianness)
265 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
266 break;
267
268 case SND_PCM_FORMAT_S32_BE:
269 *pFmt = AUD_FMT_S32;
270 if (pEndianness)
271 *pEndianness = PDMAUDIOENDIANNESS_BIG;
272 break;
273
274 case SND_PCM_FORMAT_U32_BE:
275 *pFmt = AUD_FMT_U32;
276 if (pEndianness)
277 *pEndianness = PDMAUDIOENDIANNESS_BIG;
278 break;
279
280 default:
281 AssertMsgFailed(("Format %ld not supported\n", fmt));
282 return VERR_NOT_SUPPORTED;
283 }
284
285 return VINF_SUCCESS;
286}
287
288static int drvHostALSAAudioALSAGetShift(snd_pcm_format_t fmt, unsigned *puShift)
289{
290 AssertPtrReturn(puShift, VERR_INVALID_POINTER);
291
292 switch (fmt)
293 {
294 case SND_PCM_FORMAT_S8:
295 case SND_PCM_FORMAT_U8:
296 *puShift = 0;
297 break;
298
299 case SND_PCM_FORMAT_S16_LE:
300 case SND_PCM_FORMAT_U16_LE:
301 case SND_PCM_FORMAT_S16_BE:
302 case SND_PCM_FORMAT_U16_BE:
303 *puShift = 1;
304 break;
305
306 case SND_PCM_FORMAT_S32_LE:
307 case SND_PCM_FORMAT_U32_LE:
308 case SND_PCM_FORMAT_S32_BE:
309 case SND_PCM_FORMAT_U32_BE:
310 *puShift = 2;
311 break;
312
313 default:
314 AssertMsgFailed(("Format %ld not supported\n", fmt));
315 return VERR_NOT_SUPPORTED;
316 }
317
318 return VINF_SUCCESS;
319}
320
321static int drvHostALSAAudioSetThreshold(snd_pcm_t *phPCM,
322 snd_pcm_uframes_t threshold)
323{
324 snd_pcm_sw_params_t *pSWParms = NULL;
325 snd_pcm_sw_params_alloca(&pSWParms);
326 if (!pSWParms)
327 return VERR_NO_MEMORY;
328
329 int rc;
330 do
331 {
332 int err = snd_pcm_sw_params_current(phPCM, pSWParms);
333 if (err < 0)
334 {
335 LogRel(("ALSA: Failed to get current software parameters for threshold: %s\n",
336 snd_strerror(err)));
337 rc = VERR_ACCESS_DENIED;
338 break;
339 }
340
341 err = snd_pcm_sw_params_set_start_threshold(phPCM, pSWParms, threshold);
342 if (err < 0)
343 {
344 LogRel(("ALSA: Failed to set software threshold to %ld: %s\n",
345 threshold, snd_strerror(err)));
346 rc = VERR_ACCESS_DENIED;
347 break;
348 }
349
350 err = snd_pcm_sw_params(phPCM, pSWParms);
351 if (err < 0)
352 {
353 LogRel(("ALSA: Failed to set new software parameters for threshold: %s\n",
354 snd_strerror(err)));
355 rc = VERR_ACCESS_DENIED;
356 break;
357 }
358
359 LogFlowFunc(("Setting threshold to %RU32\n", threshold));
360 rc = VINF_SUCCESS;
361 }
362 while (0);
363
364 return rc;
365}
366
367static int drvHostALSAAudioOpen(bool fIn,
368 PALSAAUDIOSTREAMCFG pCfgReq,
369 PALSAAUDIOSTREAMCFG pCfgObt,
370 snd_pcm_t **pphPCM)
371{
372 snd_pcm_t *phPCM = NULL;
373 int rc;
374
375 unsigned int cChannels = pCfgReq->nchannels;
376 unsigned int uFreq = pCfgReq->freq;
377 snd_pcm_uframes_t obt_buffer_size;
378
379 do
380 {
381 const char *pszDev = fIn ? s_ALSAConf.pcm_name_in : s_ALSAConf.pcm_name_out;
382 if (!pszDev)
383 {
384 LogRel(("ALSA: Invalid or no %s device name set\n",
385 fIn ? "input" : "output"));
386 rc = VERR_INVALID_PARAMETER;
387 break;
388 }
389
390 int err = snd_pcm_open(&phPCM, pszDev,
391 fIn ? SND_PCM_STREAM_CAPTURE : SND_PCM_STREAM_PLAYBACK,
392 SND_PCM_NONBLOCK);
393 if (err < 0)
394 {
395 LogRel(("ALSA: Failed to open \"%s\" as %s: %s\n", pszDev,
396 fIn ? "ADC" : "DAC", snd_strerror(err)));
397 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
398 break;
399 }
400
401 snd_pcm_hw_params_t *pHWParms;
402 snd_pcm_hw_params_alloca(&pHWParms); /** @todo Check for successful allocation? */
403 err = snd_pcm_hw_params_any(phPCM, pHWParms);
404 if (err < 0)
405 {
406 LogRel(("ALSA: Failed to initialize hardware parameters: %s\n",
407 snd_strerror(err)));
408 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
409 break;
410 }
411
412 err = snd_pcm_hw_params_set_access(phPCM, pHWParms,
413 SND_PCM_ACCESS_RW_INTERLEAVED);
414 if (err < 0)
415 {
416 LogRel(("ALSA: Failed to set access type: %s\n", snd_strerror(err)));
417 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
418 break;
419 }
420
421 err = snd_pcm_hw_params_set_format(phPCM, pHWParms, pCfgReq->fmt);
422 if (err < 0)
423 {
424 LogRel(("ALSA: Failed to set audio format to %d: %s\n",
425 pCfgReq->fmt, snd_strerror(err)));
426 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
427 break;
428 }
429
430 err = snd_pcm_hw_params_set_rate_near(phPCM, pHWParms, &uFreq, 0);
431 if (err < 0)
432 {
433 LogRel(("ALSA: Failed to set frequency to %dHz: %s\n",
434 pCfgReq->freq, snd_strerror(err)));
435 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
436 break;
437 }
438
439 err = snd_pcm_hw_params_set_channels_near(phPCM, pHWParms, &cChannels);
440 if (err < 0)
441 {
442 LogRel(("ALSA: Failed to set number of channels to %d\n", pCfgReq->nchannels));
443 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
444 break;
445 }
446
447 if ( cChannels != 1
448 && cChannels != 2)
449 {
450 LogRel(("ALSA: Number of audio channels (%u) not supported\n", cChannels));
451 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
452 break;
453 }
454
455 unsigned int period_size = pCfgReq->period_size;
456 unsigned int buffer_size = pCfgReq->buffer_size;
457
458 if ( !((fIn && s_ALSAConf.size_in_usec_in)
459 || (!fIn && s_ALSAConf.size_in_usec_out)))
460 {
461 if (!buffer_size)
462 {
463 buffer_size = DEFAULT_BUFFER_SIZE;
464 period_size = DEFAULT_PERIOD_SIZE;
465 }
466 }
467
468 if (buffer_size)
469 {
470 if ( ( fIn && s_ALSAConf.size_in_usec_in)
471 || (!fIn && s_ALSAConf.size_in_usec_out))
472 {
473 if (period_size)
474 {
475 err = snd_pcm_hw_params_set_period_time_near(phPCM, pHWParms,
476 &period_size, 0);
477 if (err < 0)
478 {
479 LogRel(("ALSA: Failed to set period time %d\n", pCfgReq->period_size));
480 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
481 break;
482 }
483 }
484
485 err = snd_pcm_hw_params_set_buffer_time_near(phPCM, pHWParms,
486 &buffer_size, 0);
487 if (err < 0)
488 {
489 LogRel(("ALSA: Failed to set buffer time %d\n", pCfgReq->buffer_size));
490 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
491 break;
492 }
493 }
494 else
495 {
496 snd_pcm_uframes_t period_size_f = (snd_pcm_uframes_t)period_size;
497 snd_pcm_uframes_t buffer_size_f = (snd_pcm_uframes_t)buffer_size;
498
499 snd_pcm_uframes_t minval;
500
501 if (period_size_f)
502 {
503 minval = period_size_f;
504
505 int dir = 0;
506 err = snd_pcm_hw_params_get_period_size_min(pHWParms,
507 &minval, &dir);
508 if (err < 0)
509 {
510 LogRel(("ALSA: Could not determine minimal period size\n"));
511 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
512 break;
513 }
514 else
515 {
516 LogFunc(("Minimal period size is: %ld\n", minval));
517 if (period_size_f < minval)
518 {
519 if ( ( fIn && s_ALSAConf.period_size_in_overriden)
520 || (!fIn && s_ALSAConf.period_size_out_overriden))
521 {
522 LogFunc(("Period size %RU32 is less than minimal period size %RU32\n",
523 period_size_f, minval));
524 }
525
526 period_size_f = minval;
527 }
528 }
529
530 err = snd_pcm_hw_params_set_period_size_near(phPCM, pHWParms,
531 &period_size_f, 0);
532 LogFunc(("Period size is: %RU32\n", period_size_f));
533 if (err < 0)
534 {
535 LogRel(("ALSA: Failed to set period size %d (%s)\n",
536 period_size_f, snd_strerror(err)));
537 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
538 break;
539 }
540 }
541
542 /* Calculate default buffer size here since it might have been changed
543 * in the _near functions */
544 buffer_size_f = 4 * period_size_f;
545
546 minval = buffer_size_f;
547 err = snd_pcm_hw_params_get_buffer_size_min(pHWParms, &minval);
548 if (err < 0)
549 {
550 LogRel(("ALSA: Could not retrieve minimal buffer size\n"));
551 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
552 break;
553 }
554 else
555 {
556 LogFunc(("Minimal buffer size is: %RU32\n", minval));
557 if (buffer_size_f < minval)
558 {
559 if ( ( fIn && s_ALSAConf.buffer_size_in_overriden)
560 || (!fIn && s_ALSAConf.buffer_size_out_overriden))
561 {
562 LogFunc(("Buffer size %RU32 is less than minimal buffer size %RU32\n",
563 buffer_size_f, minval));
564 }
565
566 buffer_size_f = minval;
567 }
568 }
569
570 err = snd_pcm_hw_params_set_buffer_size_near(phPCM,
571 pHWParms, &buffer_size_f);
572 LogFunc(("Buffer size is: %RU32\n", buffer_size_f));
573 if (err < 0)
574 {
575 LogRel(("ALSA: Failed to set buffer size %d: %s\n",
576 buffer_size_f, snd_strerror(err)));
577 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
578 break;
579 }
580 }
581 }
582 else
583 LogFunc(("Warning: Buffer size is not set\n"));
584
585 err = snd_pcm_hw_params(phPCM, pHWParms);
586 if (err < 0)
587 {
588 LogRel(("ALSA: Failed to apply audio parameters\n"));
589 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
590 break;
591 }
592
593 err = snd_pcm_hw_params_get_buffer_size(pHWParms, &obt_buffer_size);
594 if (err < 0)
595 {
596 LogRel(("ALSA: Failed to get buffer size\n"));
597 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
598 break;
599 }
600
601 snd_pcm_uframes_t obt_period_size;
602 int dir = 0;
603 err = snd_pcm_hw_params_get_period_size(pHWParms, &obt_period_size, &dir);
604 if (err < 0)
605 {
606 LogRel(("ALSA: Failed to get period size\n"));
607 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
608 break;
609 }
610
611 LogFunc(("Freq=%dHz, period size=%RU32, buffer size=%RU32\n",
612 pCfgReq->freq, obt_period_size, obt_buffer_size));
613
614 err = snd_pcm_prepare(phPCM);
615 if (err < 0)
616 {
617 LogRel(("ALSA: Could not prepare hPCM %p\n", (void *)phPCM));
618 rc = VERR_GENERAL_FAILURE; /** @todo Find a better rc. */
619 break;
620 }
621
622 if ( !fIn
623 && s_ALSAConf.threshold)
624 {
625 unsigned uShift;
626 rc = drvHostALSAAudioALSAGetShift(pCfgReq->fmt, &uShift);
627 if (RT_SUCCESS(rc))
628 {
629 int bytes_per_sec = uFreq
630 << (cChannels == 2)
631 << uShift;
632
633 snd_pcm_uframes_t threshold
634 = (s_ALSAConf.threshold * bytes_per_sec) / 1000;
635
636 rc = drvHostALSAAudioSetThreshold(phPCM, threshold);
637 }
638 }
639 else
640 rc = VINF_SUCCESS;
641 }
642 while (0);
643
644 if (RT_SUCCESS(rc))
645 {
646 pCfgObt->fmt = pCfgReq->fmt;
647 pCfgObt->nchannels = cChannels;
648 pCfgObt->freq = uFreq;
649 pCfgObt->samples = obt_buffer_size;
650
651 *pphPCM = phPCM;
652 }
653 else
654 drvHostALSAAudioClose(&phPCM);
655
656 LogFlowFuncLeaveRC(rc);
657 return rc;
658}
659
660#ifdef DEBUG
661static void drvHostALSAAudioErrorHandler(const char *file, int line, const char *function,
662 int err, const char *fmt, ...)
663{
664 /** @todo Implement me! */
665}
666#endif
667
668static int drvHostALSAAudioGetAvail(snd_pcm_t *phPCM, snd_pcm_sframes_t *pFramesAvail)
669{
670 AssertPtrReturn(phPCM, VERR_INVALID_POINTER);
671 AssertPtrReturn(pFramesAvail, VERR_INVALID_POINTER);
672
673 int rc;
674
675 snd_pcm_sframes_t framesAvail;
676 framesAvail = snd_pcm_avail_update(phPCM);
677 if (framesAvail < 0)
678 {
679 if (framesAvail == -EPIPE)
680 {
681 rc = drvHostALSAAudioRecover(phPCM);
682 if (RT_SUCCESS(rc))
683 framesAvail = snd_pcm_avail_update(phPCM);
684 }
685 else
686 rc = VERR_ACCESS_DENIED; /** @todo Find a better rc. */
687 }
688 else
689 rc = VINF_SUCCESS;
690
691 if (framesAvail >= 0)
692 *pFramesAvail = framesAvail;
693
694 return rc;
695}
696
697static int drvHostALSAAudioRecover(snd_pcm_t *phPCM)
698{
699 AssertPtrReturn(phPCM, VERR_INVALID_POINTER);
700
701 int err = snd_pcm_prepare(phPCM);
702 if (err < 0)
703 {
704 LogFunc(("Failed to recover stream %p: %s\n",
705 phPCM, snd_strerror(err)));
706 return VERR_ACCESS_DENIED; /** @todo Find a better rc. */
707 }
708
709 return VINF_SUCCESS;
710}
711
712static int drvHostALSAAudioResume(snd_pcm_t *phPCM)
713{
714 AssertPtrReturn(phPCM, VERR_INVALID_POINTER);
715
716 int err = snd_pcm_resume(phPCM);
717 if (err < 0)
718 {
719 LogFunc(("Failed to resume stream %p: %s\n",
720 phPCM, snd_strerror(err)));
721 return VERR_ACCESS_DENIED; /** @todo Find a better rc. */
722 }
723
724 return VINF_SUCCESS;
725}
726
727static int drvHostALSAAudioStreamCtl(snd_pcm_t *phPCM, bool fPause)
728{
729 int err;
730 if (fPause)
731 {
732 err = snd_pcm_drop(phPCM);
733 if (err < 0)
734 {
735 LogFlow(("Error stopping stream %p: %s\n",
736 phPCM, snd_strerror(err)));
737 return VERR_ACCESS_DENIED;
738 }
739 }
740 else
741 {
742 err = snd_pcm_prepare (phPCM);
743 if (err < 0)
744 {
745 LogFlow(("Error preparing stream %p: %s\n",
746 phPCM, snd_strerror(err)));
747 return VERR_ACCESS_DENIED;
748 }
749 }
750
751 return VINF_SUCCESS;
752}
753
754static DECLCALLBACK(int) drvHostALSAAudioInit(PPDMIHOSTAUDIO pInterface)
755{
756 NOREF(pInterface);
757
758 LogFlowFuncEnter();
759
760 int rc = audioLoadAlsaLib();
761 if (RT_FAILURE(rc))
762 LogRel(("ALSA: Failed to load the ALSA shared library, rc=%Rrc\n", rc));
763 else
764 {
765#ifdef DEBUG
766 snd_lib_error_set_handler(drvHostALSAAudioErrorHandler);
767#endif
768 }
769
770 return rc;
771}
772
773static DECLCALLBACK(int) drvHostALSAAudioCaptureIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMIN pHstStrmIn,
774 uint32_t *pcSamplesCaptured)
775{
776 NOREF(pInterface);
777 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
778
779 PALSAAUDIOSTREAMIN pThisStrmIn = (PALSAAUDIOSTREAMIN)pHstStrmIn;
780
781 snd_pcm_sframes_t cAvail;
782 int rc = drvHostALSAAudioGetAvail(pThisStrmIn->phPCM, &cAvail);
783 if (RT_FAILURE(rc))
784 {
785 LogFunc(("Error getting number of captured frames, rc=%Rrc\n", rc));
786 return rc;
787 }
788
789 if (!cAvail) /* No data yet? */
790 {
791 snd_pcm_state_t state = snd_pcm_state(pThisStrmIn->phPCM);
792 switch (state)
793 {
794 case SND_PCM_STATE_PREPARED:
795 cAvail = AudioMixBufFree(&pHstStrmIn->MixBuf);
796 break;
797
798 case SND_PCM_STATE_SUSPENDED:
799 {
800 rc = drvHostALSAAudioResume(pThisStrmIn->phPCM);
801 if (RT_FAILURE(rc))
802 break;
803
804 LogFlow(("Resuming suspended input stream\n"));
805 break;
806 }
807
808 default:
809 LogFlow(("No frames available, state=%d\n", state));
810 break;
811 }
812
813 if (!cAvail)
814 {
815 if (pcSamplesCaptured)
816 *pcSamplesCaptured = 0;
817 return VINF_SUCCESS;
818 }
819 }
820
821 /*
822 * Check how much we can read from the capture device without overflowing
823 * the mixer buffer.
824 */
825 Assert(cAvail);
826 size_t cbMixFree = AudioMixBufFreeBytes(&pHstStrmIn->MixBuf);
827 size_t cbToRead = RT_MIN((size_t)AUDIOMIXBUF_S2B(&pHstStrmIn->MixBuf, cAvail), cbMixFree);
828
829 LogFlowFunc(("cbToRead=%zu, cAvail=%RI32\n", cbToRead, cAvail));
830
831 uint32_t cWrittenTotal = 0;
832 snd_pcm_uframes_t cToRead;
833 snd_pcm_sframes_t cRead;
834
835 while ( cbToRead
836 && RT_SUCCESS(rc))
837 {
838 cToRead = RT_MIN(AUDIOMIXBUF_B2S(&pHstStrmIn->MixBuf, cbToRead),
839 AUDIOMIXBUF_B2S(&pHstStrmIn->MixBuf, pThisStrmIn->cbBuf));
840 AssertBreakStmt(cToRead, rc = VERR_NO_DATA);
841 cRead = snd_pcm_readi(pThisStrmIn->phPCM, pThisStrmIn->pvBuf, cToRead);
842 if (cRead <= 0)
843 {
844 switch (cRead)
845 {
846 case 0:
847 {
848 LogFunc(("No input frames available\n"));
849 rc = VERR_ACCESS_DENIED;
850 break;
851 }
852
853 case -EAGAIN:
854 /*
855 * Don't set error here because EAGAIN means there are no further frames
856 * available at the moment, try later. As we might have read some frames
857 * already these need to be processed instead.
858 */
859 cbToRead = 0;
860 break;
861
862 case -EPIPE:
863 {
864 rc = drvHostALSAAudioRecover(pThisStrmIn->phPCM);
865 if (RT_FAILURE(rc))
866 break;
867
868 LogFlowFunc(("Recovered from capturing\n"));
869 continue;
870 }
871
872 default:
873 LogFunc(("Failed to read input frames: %s\n", snd_strerror(cRead)));
874 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
875 break;
876 }
877 }
878 else
879 {
880 uint32_t cWritten;
881 rc = AudioMixBufWriteCirc(&pHstStrmIn->MixBuf,
882 pThisStrmIn->pvBuf, AUDIOMIXBUF_S2B(&pHstStrmIn->MixBuf, cRead),
883 &cWritten);
884 if (RT_FAILURE(rc))
885 break;
886
887 /*
888 * We should not run into a full mixer buffer or we loose samples and
889 * run into an endless loop if ALSA keeps producing samples ("null"
890 * capture device for example).
891 */
892 AssertLogRelMsgBreakStmt(cWritten > 0, ("Mixer buffer shouldn't be full at this point!\n"),
893 rc = VERR_INTERNAL_ERROR);
894 uint32_t cbWritten = AUDIOMIXBUF_S2B(&pHstStrmIn->MixBuf, cWritten);
895
896 Assert(cbToRead >= cbWritten);
897 cbToRead -= cbWritten;
898 cWrittenTotal += cWritten;
899 }
900 }
901
902 if (RT_SUCCESS(rc))
903 {
904 uint32_t cProcessed = 0;
905 if (cWrittenTotal)
906 rc = AudioMixBufMixToParent(&pHstStrmIn->MixBuf, cWrittenTotal,
907 &cProcessed);
908
909 if (pcSamplesCaptured)
910 *pcSamplesCaptured = cWrittenTotal;
911
912 LogFlowFunc(("cWrittenTotal=%RU32 (%RU32 processed), rc=%Rrc\n",
913 cWrittenTotal, cProcessed, rc));
914 }
915
916 LogFlowFuncLeaveRC(rc);
917 return rc;
918}
919
920static DECLCALLBACK(int) drvHostALSAAudioPlayOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMOUT pHstStrmOut,
921 uint32_t *pcSamplesPlayed)
922{
923 NOREF(pInterface);
924 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
925
926 PALSAAUDIOSTREAMOUT pThisStrmOut = (PALSAAUDIOSTREAMOUT)pHstStrmOut;
927
928 int rc = VINF_SUCCESS;
929 uint32_t cbReadTotal = 0;
930
931 do
932 {
933 snd_pcm_sframes_t cAvail;
934 rc = drvHostALSAAudioGetAvail(pThisStrmOut->phPCM, &cAvail);
935 if (RT_FAILURE(rc))
936 {
937 LogFunc(("Error getting number of playback frames, rc=%Rrc\n", rc));
938 break;
939 }
940
941 size_t cbToRead = RT_MIN(AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf,
942 cAvail),
943 AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf,
944 drvAudioHstOutSamplesLive(pHstStrmOut, NULL /* pcStreamsLive */)));
945 LogFlowFunc(("cbToRead=%zu, cbAvail=%zu\n",
946 cbToRead, AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf, cAvail)));
947
948 uint32_t cRead, cbRead;
949 snd_pcm_sframes_t cWritten;
950 while (cbToRead)
951 {
952 rc = AudioMixBufReadCirc(&pHstStrmOut->MixBuf, pThisStrmOut->pvBuf, cbToRead, &cRead);
953 if (RT_FAILURE(rc))
954 break;
955
956 cbRead = AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf, cRead);
957 AssertBreak(cbRead);
958
959 cWritten = snd_pcm_writei(pThisStrmOut->phPCM, pThisStrmOut->pvBuf, cRead);
960 if (cWritten <= 0)
961 {
962 switch (cWritten)
963 {
964 case 0:
965 {
966 LogFunc(("Failed to write %RI32 frames\n", cRead));
967 rc = VERR_ACCESS_DENIED;
968 break;
969 }
970
971 case -EPIPE:
972 {
973 rc = drvHostALSAAudioRecover(pThisStrmOut->phPCM);
974 if (RT_FAILURE(rc))
975 break;
976
977 LogFlowFunc(("Recovered from playback\n"));
978 continue;
979 }
980
981 case -ESTRPIPE:
982 {
983 /* Stream was suspended and waiting for a recovery. */
984 rc = drvHostALSAAudioResume(pThisStrmOut->phPCM);
985 if (RT_FAILURE(rc))
986 {
987 LogRel(("ALSA: Failed to resume output stream\n"));
988 break;
989 }
990
991 LogFlowFunc(("Resumed suspended output stream\n"));
992 continue;
993 }
994
995 default:
996 LogFlowFunc(("Failed to write %RI32 output frames, rc=%Rrc\n",
997 cRead, rc));
998 rc = VERR_GENERAL_FAILURE; /** @todo */
999 break;
1000 }
1001 }
1002
1003 if (RT_FAILURE(rc))
1004 break;
1005
1006 Assert(cbToRead >= cbRead);
1007 cbToRead -= cbRead;
1008 cbReadTotal += cbRead;
1009 }
1010 }
1011 while (0);
1012
1013 if (RT_SUCCESS(rc))
1014 {
1015 uint32_t cReadTotal = AUDIOMIXBUF_B2S(&pHstStrmOut->MixBuf, cbReadTotal);
1016 if (cReadTotal)
1017 AudioMixBufFinish(&pHstStrmOut->MixBuf, cReadTotal);
1018
1019 if (pcSamplesPlayed)
1020 *pcSamplesPlayed = cReadTotal;
1021
1022 LogFlowFunc(("cReadTotal=%RU32 (%RU32 bytes), rc=%Rrc\n",
1023 cReadTotal, cbReadTotal, rc));
1024 }
1025
1026 LogFlowFuncLeaveRC(rc);
1027 return rc;
1028}
1029
1030static DECLCALLBACK(int) drvHostALSAAudioFiniIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMIN pHstStrmIn)
1031{
1032 NOREF(pInterface);
1033 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
1034
1035 PALSAAUDIOSTREAMIN pThisStrmIn = (PALSAAUDIOSTREAMIN)pHstStrmIn;
1036
1037 drvHostALSAAudioClose(&pThisStrmIn->phPCM);
1038
1039 if (pThisStrmIn->pvBuf)
1040 {
1041 RTMemFree(pThisStrmIn->pvBuf);
1042 pThisStrmIn->pvBuf = NULL;
1043 }
1044
1045 return VINF_SUCCESS;
1046}
1047
1048static DECLCALLBACK(int) drvHostALSAAudioFiniOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMOUT pHstStrmOut)
1049{
1050 NOREF(pInterface);
1051 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
1052
1053 PALSAAUDIOSTREAMOUT pThisStrmOut = (PALSAAUDIOSTREAMOUT)pHstStrmOut;
1054
1055 drvHostALSAAudioClose(&pThisStrmOut->phPCM);
1056
1057 if (pThisStrmOut->pvBuf)
1058 {
1059 RTMemFree(pThisStrmOut->pvBuf);
1060 pThisStrmOut->pvBuf = NULL;
1061 }
1062
1063 return VINF_SUCCESS;
1064}
1065
1066static DECLCALLBACK(int) drvHostALSAAudioInitOut(PPDMIHOSTAUDIO pInterface,
1067 PPDMAUDIOHSTSTRMOUT pHstStrmOut, PPDMAUDIOSTREAMCFG pCfg,
1068 uint32_t *pcSamples)
1069{
1070 NOREF(pInterface);
1071 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
1072 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1073
1074 PALSAAUDIOSTREAMOUT pThisStrmOut = (PALSAAUDIOSTREAMOUT)pHstStrmOut;
1075 snd_pcm_t *phPCM = NULL;
1076
1077 int rc;
1078
1079 do
1080 {
1081 ALSAAUDIOSTREAMCFG req;
1082 req.fmt = drvHostALSAAudioFmtToALSA(pCfg->enmFormat);
1083 req.freq = pCfg->uHz;
1084 req.nchannels = pCfg->cChannels;
1085 req.period_size = s_ALSAConf.period_size_out;
1086 req.buffer_size = s_ALSAConf.buffer_size_out;
1087
1088 ALSAAUDIOSTREAMCFG obt;
1089 rc = drvHostALSAAudioOpen(false /* false */, &req, &obt, &phPCM);
1090 if (RT_FAILURE(rc))
1091 break;
1092
1093 PDMAUDIOFMT enmFormat;
1094 PDMAUDIOENDIANNESS enmEnd;
1095 rc = drvHostALSAAudioALSAToFmt(obt.fmt, &enmFormat, &enmEnd);
1096 if (RT_FAILURE(rc))
1097 break;
1098
1099 PDMAUDIOSTREAMCFG streamCfg;
1100 streamCfg.uHz = obt.freq;
1101 streamCfg.cChannels = obt.nchannels;
1102 streamCfg.enmFormat = enmFormat;
1103 streamCfg.enmEndianness = enmEnd;
1104
1105 rc = drvAudioStreamCfgToProps(&streamCfg, &pHstStrmOut->Props);
1106 if (RT_FAILURE(rc))
1107 break;
1108
1109 AssertBreakStmt(obt.samples, rc = VERR_INVALID_PARAMETER);
1110 size_t cbBuf = obt.samples * (1 << pHstStrmOut->Props.cShift);
1111 AssertBreakStmt(cbBuf, rc = VERR_INVALID_PARAMETER);
1112 pThisStrmOut->pvBuf = RTMemAlloc(cbBuf);
1113 if (!pThisStrmOut->pvBuf)
1114 {
1115 LogRel(("ALSA: Not enough memory for output DAC buffer (%RU32 samples, each %d bytes)\n",
1116 obt.samples, 1 << pHstStrmOut->Props.cShift));
1117 rc = VERR_NO_MEMORY;
1118 break;
1119 }
1120
1121 pThisStrmOut->cbBuf = cbBuf;
1122 pThisStrmOut->phPCM = phPCM;
1123
1124 if (pcSamples)
1125 *pcSamples = obt.samples;
1126 }
1127 while (0);
1128
1129 if (RT_FAILURE(rc))
1130 drvHostALSAAudioClose(&phPCM);
1131
1132 LogFlowFuncLeaveRC(rc);
1133 return rc;
1134}
1135
1136static DECLCALLBACK(int) drvHostALSAAudioInitIn(PPDMIHOSTAUDIO pInterface,
1137 PPDMAUDIOHSTSTRMIN pHstStrmIn, PPDMAUDIOSTREAMCFG pCfg,
1138 PDMAUDIORECSOURCE enmRecSource,
1139 uint32_t *pcSamples)
1140{
1141 NOREF(pInterface);
1142 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
1143 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1144
1145 int rc;
1146
1147 PALSAAUDIOSTREAMIN pThisStrmIn = (PALSAAUDIOSTREAMIN)pHstStrmIn;
1148 snd_pcm_t *phPCM = NULL;
1149
1150 do
1151 {
1152 ALSAAUDIOSTREAMCFG req;
1153 req.fmt = drvHostALSAAudioFmtToALSA(pCfg->enmFormat);
1154 req.freq = pCfg->uHz;
1155 req.nchannels = pCfg->cChannels;
1156 req.period_size = s_ALSAConf.period_size_in;
1157 req.buffer_size = s_ALSAConf.buffer_size_in;
1158
1159 ALSAAUDIOSTREAMCFG obt;
1160 rc = drvHostALSAAudioOpen(true /* fIn */, &req, &obt, &phPCM);
1161 if (RT_FAILURE(rc))
1162 break;
1163
1164 PDMAUDIOFMT enmFormat;
1165 PDMAUDIOENDIANNESS enmEnd;
1166 rc = drvHostALSAAudioALSAToFmt(obt.fmt, &enmFormat, &enmEnd);
1167 if (RT_FAILURE(rc))
1168 break;
1169
1170 PDMAUDIOSTREAMCFG streamCfg;
1171 streamCfg.uHz = obt.freq;
1172 streamCfg.cChannels = obt.nchannels;
1173 streamCfg.enmFormat = enmFormat;
1174 streamCfg.enmEndianness = enmEnd;
1175
1176 rc = drvAudioStreamCfgToProps(&streamCfg, &pHstStrmIn->Props);
1177 if (RT_FAILURE(rc))
1178 break;
1179
1180 AssertBreakStmt(obt.samples, rc = VERR_INVALID_PARAMETER);
1181 size_t cbBuf = obt.samples * (1 << pHstStrmIn->Props.cShift);
1182 AssertBreakStmt(cbBuf, rc = VERR_INVALID_PARAMETER);
1183 pThisStrmIn->pvBuf = RTMemAlloc(cbBuf);
1184 if (!pThisStrmIn->pvBuf)
1185 {
1186 LogRel(("ALSA: Not enough memory for input ADC buffer (%RU32 samples, each %d bytes)\n",
1187 obt.samples, 1 << pHstStrmIn->Props.cShift));
1188 rc = VERR_NO_MEMORY;
1189 break;
1190 }
1191
1192 pThisStrmIn->cbBuf = cbBuf;
1193 pThisStrmIn->phPCM = phPCM;
1194
1195 if (pcSamples)
1196 *pcSamples = obt.samples;
1197 }
1198 while (0);
1199
1200 if (RT_FAILURE(rc))
1201 drvHostALSAAudioClose(&phPCM);
1202
1203 LogFlowFuncLeaveRC(rc);
1204 return rc;
1205}
1206
1207static DECLCALLBACK(bool) drvHostALSAAudioIsEnabled(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
1208{
1209 NOREF(pInterface);
1210 NOREF(enmDir);
1211 return true; /* Always all enabled. */
1212}
1213
1214static DECLCALLBACK(int) drvHostALSAAudioControlIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMIN pHstStrmIn,
1215 PDMAUDIOSTREAMCMD enmStreamCmd)
1216{
1217 NOREF(pInterface);
1218 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
1219 PALSAAUDIOSTREAMIN pThisStrmIn = (PALSAAUDIOSTREAMIN)pHstStrmIn;
1220
1221 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
1222
1223 int rc;
1224 switch (enmStreamCmd)
1225 {
1226 case PDMAUDIOSTREAMCMD_ENABLE:
1227 rc = drvHostALSAAudioStreamCtl(pThisStrmIn->phPCM, false /* fStop */);
1228 break;
1229
1230 case PDMAUDIOSTREAMCMD_DISABLE:
1231 rc = drvHostALSAAudioStreamCtl(pThisStrmIn->phPCM, true /* fStop */);
1232 break;
1233
1234 default:
1235 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1236 rc = VERR_INVALID_PARAMETER;
1237 break;
1238 }
1239
1240 return rc;
1241}
1242
1243static DECLCALLBACK(int) drvHostALSAAudioControlOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMOUT pHstStrmOut,
1244 PDMAUDIOSTREAMCMD enmStreamCmd)
1245{
1246 NOREF(pInterface);
1247 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
1248 PALSAAUDIOSTREAMOUT pThisStrmOut = (PALSAAUDIOSTREAMOUT)pHstStrmOut;
1249
1250 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
1251
1252 int rc;
1253 switch (enmStreamCmd)
1254 {
1255 case PDMAUDIOSTREAMCMD_ENABLE:
1256 rc = drvHostALSAAudioStreamCtl(pThisStrmOut->phPCM, false /* fStop */);
1257 break;
1258
1259 case PDMAUDIOSTREAMCMD_DISABLE:
1260 rc = drvHostALSAAudioStreamCtl(pThisStrmOut->phPCM, true /* fStop */);
1261 break;
1262
1263 default:
1264 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1265 rc = VERR_INVALID_PARAMETER;
1266 break;
1267 }
1268
1269 return rc;
1270}
1271
1272static DECLCALLBACK(int) drvHostALSAAudioGetConf(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
1273{
1274 NOREF(pInterface);
1275 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1276
1277 pCfg->cbStreamOut = sizeof(ALSAAUDIOSTREAMOUT);
1278 pCfg->cbStreamIn = sizeof(ALSAAUDIOSTREAMIN);
1279 pCfg->cMaxHstStrmsOut = INT_MAX;
1280 pCfg->cMaxHstStrmsIn = INT_MAX;
1281
1282 return VINF_SUCCESS;
1283}
1284
1285static DECLCALLBACK(void) drvHostALSAAudioShutdown(PPDMIHOSTAUDIO pInterface)
1286{
1287 NOREF(pInterface);
1288}
1289
1290/**
1291 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1292 */
1293static DECLCALLBACK(void *) drvHostALSAAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1294{
1295 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1296 PDRVHOSTALSAAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTALSAAUDIO);
1297 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1298 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
1299
1300 return NULL;
1301}
1302
1303/**
1304 * Construct a DirectSound Audio driver instance.
1305 *
1306 * @copydoc FNPDMDRVCONSTRUCT
1307 */
1308static DECLCALLBACK(int) drvHostAlsaAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1309{
1310 PDRVHOSTALSAAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTALSAAUDIO);
1311 LogRel(("Audio: Initializing ALSA driver\n"));
1312
1313 /*
1314 * Init the static parts.
1315 */
1316 pThis->pDrvIns = pDrvIns;
1317 /* IBase */
1318 pDrvIns->IBase.pfnQueryInterface = drvHostALSAAudioQueryInterface;
1319 /* IHostAudio */
1320 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostALSAAudio);
1321
1322 return VINF_SUCCESS;
1323}
1324
1325/**
1326 * Char driver registration record.
1327 */
1328const PDMDRVREG g_DrvHostALSAAudio =
1329{
1330 /* u32Version */
1331 PDM_DRVREG_VERSION,
1332 /* szName */
1333 "ALSAAudio",
1334 /* szRCMod */
1335 "",
1336 /* szR0Mod */
1337 "",
1338 /* pszDescription */
1339 "ALSA host audio driver",
1340 /* fFlags */
1341 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1342 /* fClass. */
1343 PDM_DRVREG_CLASS_AUDIO,
1344 /* cMaxInstances */
1345 ~0U,
1346 /* cbInstance */
1347 sizeof(DRVHOSTALSAAUDIO),
1348 /* pfnConstruct */
1349 drvHostAlsaAudioConstruct,
1350 /* pfnDestruct */
1351 NULL,
1352 /* pfnRelocate */
1353 NULL,
1354 /* pfnIOCtl */
1355 NULL,
1356 /* pfnPowerOn */
1357 NULL,
1358 /* pfnReset */
1359 NULL,
1360 /* pfnSuspend */
1361 NULL,
1362 /* pfnResume */
1363 NULL,
1364 /* pfnAttach */
1365 NULL,
1366 /* pfnDetach */
1367 NULL,
1368 /* pfnPowerOff */
1369 NULL,
1370 /* pfnSoftReset */
1371 NULL,
1372 /* u32EndVersion */
1373 PDM_DRVREG_VERSION
1374};
1375
1376static struct audio_option alsa_options[] =
1377{
1378 {"DACSizeInUsec", AUD_OPT_BOOL, &s_ALSAConf.size_in_usec_out,
1379 "DAC period/buffer size in microseconds (otherwise in frames)", NULL, 0},
1380 {"DACPeriodSize", AUD_OPT_INT, &s_ALSAConf.period_size_out,
1381 "DAC period size", &s_ALSAConf.period_size_out_overriden, 0},
1382 {"DACBufferSize", AUD_OPT_INT, &s_ALSAConf.buffer_size_out,
1383 "DAC buffer size", &s_ALSAConf.buffer_size_out_overriden, 0},
1384
1385 {"ADCSizeInUsec", AUD_OPT_BOOL, &s_ALSAConf.size_in_usec_in,
1386 "ADC period/buffer size in microseconds (otherwise in frames)", NULL, 0},
1387 {"ADCPeriodSize", AUD_OPT_INT, &s_ALSAConf.period_size_in,
1388 "ADC period size", &s_ALSAConf.period_size_in_overriden, 0},
1389 {"ADCBufferSize", AUD_OPT_INT, &s_ALSAConf.buffer_size_in,
1390 "ADC buffer size", &s_ALSAConf.buffer_size_in_overriden, 0},
1391
1392 {"Threshold", AUD_OPT_INT, &s_ALSAConf.threshold,
1393 "(undocumented)", NULL, 0},
1394
1395 {"DACDev", AUD_OPT_STR, &s_ALSAConf.pcm_name_out,
1396 "DAC device name (for instance dmix)", NULL, 0},
1397
1398 {"ADCDev", AUD_OPT_STR, &s_ALSAConf.pcm_name_in,
1399 "ADC device name", NULL, 0},
1400
1401 NULL
1402};
1403
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette