VirtualBox

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

Last change on this file since 59381 was 59375, checked in by vboxsync, 9 years ago

Audio: introduced VERR_AUDIO_BACKEND_INIT_FAILED

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