VirtualBox

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

Last change on this file since 62877 was 62463, checked in by vboxsync, 9 years ago

Devices: scm

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 54.5 KB
Line 
1/* $Id: DrvHostALSAAudio.cpp 62463 2016-07-22 16:32:54Z vboxsync $ */
2/** @file
3 * VBox audio devices: ALSA audio driver.
4 */
5
6/*
7 * Copyright (C) 2006-2016 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#include <alsa/control.h> /* For device enumeration. */
60
61#include "DrvAudio.h"
62#include "AudioMixBuffer.h"
63
64#include "VBoxDD.h"
65
66
67/*********************************************************************************************************************************
68* Defines *
69*********************************************************************************************************************************/
70
71/** Makes DRVHOSTALSAAUDIO out of PDMIHOSTAUDIO. */
72#define PDMIHOSTAUDIO_2_DRVHOSTALSAAUDIO(pInterface) \
73 ( (PDRVHOSTALSAAUDIO)((uintptr_t)pInterface - RT_OFFSETOF(DRVHOSTALSAAUDIO, IHostAudio)) )
74
75
76/*********************************************************************************************************************************
77* Structures *
78*********************************************************************************************************************************/
79
80typedef struct ALSAAUDIOSTREAMIN
81{
82 /** Associated host input stream.
83 * Note: Always must come first! */
84 PDMAUDIOSTREAM Stream;
85 snd_pcm_t *phPCM;
86 void *pvBuf;
87 size_t cbBuf;
88} ALSAAUDIOSTREAMIN, *PALSAAUDIOSTREAMIN;
89
90typedef struct ALSAAUDIOSTREAMOUT
91{
92 /** Associated host output stream.
93 * Note: Always must come first! */
94 PDMAUDIOSTREAM Stream;
95 snd_pcm_t *phPCM;
96 void *pvBuf;
97 size_t cbBuf;
98 /** Minimum samples required for ALSA to play data. */
99 uint32_t cSamplesMin;
100} ALSAAUDIOSTREAMOUT, *PALSAAUDIOSTREAMOUT;
101
102/* latency = period_size * periods / (rate * bytes_per_frame) */
103
104typedef struct ALSAAUDIOCFG
105{
106 int size_in_usec_in;
107 int size_in_usec_out;
108 const char *pcm_name_in;
109 const char *pcm_name_out;
110 unsigned int buffer_size_in;
111 unsigned int period_size_in;
112 unsigned int buffer_size_out;
113 unsigned int period_size_out;
114 unsigned int threshold;
115
116 int buffer_size_in_overriden;
117 int period_size_in_overriden;
118
119 int buffer_size_out_overriden;
120 int period_size_out_overriden;
121
122} ALSAAUDIOCFG, *PALSAAUDIOCFG;
123
124static int alsaStreamRecover(snd_pcm_t *phPCM);
125
126static ALSAAUDIOCFG s_ALSAConf =
127{
128#ifdef HIGH_LATENCY
129 1,
130 1,
131#else
132 0,
133 0,
134#endif
135 "default",
136 "default",
137#ifdef HIGH_LATENCY
138 400000,
139 400000 / 4,
140 400000,
141 400000 / 4,
142#else
143# define DEFAULT_BUFFER_SIZE 1024
144# define DEFAULT_PERIOD_SIZE 256
145 DEFAULT_BUFFER_SIZE * 4,
146 DEFAULT_PERIOD_SIZE * 4,
147 DEFAULT_BUFFER_SIZE,
148 DEFAULT_PERIOD_SIZE,
149#endif
150 0,
151 0,
152 0,
153 0,
154 0
155};
156
157/**
158 * Host Alsa audio driver instance data.
159 * @implements PDMIAUDIOCONNECTOR
160 */
161typedef struct DRVHOSTALSAAUDIO
162{
163 /** Pointer to the driver instance structure. */
164 PPDMDRVINS pDrvIns;
165 /** Pointer to host audio interface. */
166 PDMIHOSTAUDIO IHostAudio;
167 /** Error count for not flooding the release log.
168 * UINT32_MAX for unlimited logging. */
169 uint32_t cLogErrors;
170} DRVHOSTALSAAUDIO, *PDRVHOSTALSAAUDIO;
171
172/** Maximum number of tries to recover a broken pipe. */
173#define ALSA_RECOVERY_TRIES_MAX 5
174
175typedef struct ALSAAUDIOSTREAMCFG
176{
177 unsigned int freq;
178 /** PCM sound format. */
179 snd_pcm_format_t fmt;
180 /** PCM data access type. */
181 snd_pcm_access_t access;
182 /** Whether resampling should be performed by alsalib or not. */
183 int resample;
184 int nchannels;
185 unsigned long buffer_size;
186 unsigned long period_size;
187 snd_pcm_uframes_t samples;
188} ALSAAUDIOSTREAMCFG, *PALSAAUDIOSTREAMCFG;
189
190
191
192static snd_pcm_format_t alsaAudioFmtToALSA(PDMAUDIOFMT fmt)
193{
194 switch (fmt)
195 {
196 case PDMAUDIOFMT_S8:
197 return SND_PCM_FORMAT_S8;
198
199 case PDMAUDIOFMT_U8:
200 return SND_PCM_FORMAT_U8;
201
202 case PDMAUDIOFMT_S16:
203 return SND_PCM_FORMAT_S16_LE;
204
205 case PDMAUDIOFMT_U16:
206 return SND_PCM_FORMAT_U16_LE;
207
208 case PDMAUDIOFMT_S32:
209 return SND_PCM_FORMAT_S32_LE;
210
211 case PDMAUDIOFMT_U32:
212 return SND_PCM_FORMAT_U32_LE;
213
214 default:
215 break;
216 }
217
218 AssertMsgFailed(("Format %ld not supported\n", fmt));
219 return SND_PCM_FORMAT_U8;
220}
221
222static int alsaALSAToAudioFmt(snd_pcm_format_t fmt,
223 PDMAUDIOFMT *pFmt, PDMAUDIOENDIANNESS *pEndianness)
224{
225 AssertPtrReturn(pFmt, VERR_INVALID_POINTER);
226 /* pEndianness is optional. */
227
228 switch (fmt)
229 {
230 case SND_PCM_FORMAT_S8:
231 *pFmt = PDMAUDIOFMT_S8;
232 if (pEndianness)
233 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
234 break;
235
236 case SND_PCM_FORMAT_U8:
237 *pFmt = PDMAUDIOFMT_U8;
238 if (pEndianness)
239 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
240 break;
241
242 case SND_PCM_FORMAT_S16_LE:
243 *pFmt = PDMAUDIOFMT_S16;
244 if (pEndianness)
245 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
246 break;
247
248 case SND_PCM_FORMAT_U16_LE:
249 *pFmt = PDMAUDIOFMT_U16;
250 if (pEndianness)
251 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
252 break;
253
254 case SND_PCM_FORMAT_S16_BE:
255 *pFmt = PDMAUDIOFMT_S16;
256 if (pEndianness)
257 *pEndianness = PDMAUDIOENDIANNESS_BIG;
258 break;
259
260 case SND_PCM_FORMAT_U16_BE:
261 *pFmt = PDMAUDIOFMT_U16;
262 if (pEndianness)
263 *pEndianness = PDMAUDIOENDIANNESS_BIG;
264 break;
265
266 case SND_PCM_FORMAT_S32_LE:
267 *pFmt = PDMAUDIOFMT_S32;
268 if (pEndianness)
269 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
270 break;
271
272 case SND_PCM_FORMAT_U32_LE:
273 *pFmt = PDMAUDIOFMT_U32;
274 if (pEndianness)
275 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
276 break;
277
278 case SND_PCM_FORMAT_S32_BE:
279 *pFmt = PDMAUDIOFMT_S32;
280 if (pEndianness)
281 *pEndianness = PDMAUDIOENDIANNESS_BIG;
282 break;
283
284 case SND_PCM_FORMAT_U32_BE:
285 *pFmt = PDMAUDIOFMT_U32;
286 if (pEndianness)
287 *pEndianness = PDMAUDIOENDIANNESS_BIG;
288 break;
289
290 default:
291 AssertMsgFailed(("Format %ld not supported\n", fmt));
292 return VERR_NOT_SUPPORTED;
293 }
294
295 return VINF_SUCCESS;
296}
297
298static int alsaGetSampleShift(snd_pcm_format_t fmt, unsigned *puShift)
299{
300 AssertPtrReturn(puShift, VERR_INVALID_POINTER);
301
302 switch (fmt)
303 {
304 case SND_PCM_FORMAT_S8:
305 case SND_PCM_FORMAT_U8:
306 *puShift = 0;
307 break;
308
309 case SND_PCM_FORMAT_S16_LE:
310 case SND_PCM_FORMAT_U16_LE:
311 case SND_PCM_FORMAT_S16_BE:
312 case SND_PCM_FORMAT_U16_BE:
313 *puShift = 1;
314 break;
315
316 case SND_PCM_FORMAT_S32_LE:
317 case SND_PCM_FORMAT_U32_LE:
318 case SND_PCM_FORMAT_S32_BE:
319 case SND_PCM_FORMAT_U32_BE:
320 *puShift = 2;
321 break;
322
323 default:
324 AssertMsgFailed(("Format %ld not supported\n", fmt));
325 return VERR_NOT_SUPPORTED;
326 }
327
328 return VINF_SUCCESS;
329}
330
331static int alsaStreamSetThreshold(snd_pcm_t *phPCM, snd_pcm_uframes_t threshold)
332{
333 snd_pcm_sw_params_t *pSWParms = NULL;
334 snd_pcm_sw_params_alloca(&pSWParms);
335 if (!pSWParms)
336 return VERR_NO_MEMORY;
337
338 int rc;
339 do
340 {
341 int err = snd_pcm_sw_params_current(phPCM, pSWParms);
342 if (err < 0)
343 {
344 LogRel(("ALSA: Failed to get current software parameters for threshold: %s\n",
345 snd_strerror(err)));
346 rc = VERR_ACCESS_DENIED;
347 break;
348 }
349
350 err = snd_pcm_sw_params_set_start_threshold(phPCM, pSWParms, threshold);
351 if (err < 0)
352 {
353 LogRel(("ALSA: Failed to set software threshold to %ld: %s\n",
354 threshold, snd_strerror(err)));
355 rc = VERR_ACCESS_DENIED;
356 break;
357 }
358
359 err = snd_pcm_sw_params_set_avail_min(phPCM, pSWParms, 512);
360 if (err < 0)
361 {
362 LogRel(("ALSA: Failed to set available minimum to %ld: %s\n",
363 threshold, snd_strerror(err)));
364 rc = VERR_ACCESS_DENIED;
365 break;
366 }
367
368 err = snd_pcm_sw_params(phPCM, pSWParms);
369 if (err < 0)
370 {
371 LogRel(("ALSA: Failed to set new software parameters for threshold: %s\n",
372 snd_strerror(err)));
373 rc = VERR_ACCESS_DENIED;
374 break;
375 }
376
377 LogFlowFunc(("Setting threshold to %RU32\n", threshold));
378 rc = VINF_SUCCESS;
379 }
380 while (0);
381
382 return rc;
383}
384
385static int alsaStreamClose(snd_pcm_t **pphPCM)
386{
387 if (!pphPCM || !*pphPCM)
388 return VINF_SUCCESS;
389
390 int rc;
391 int rc2 = snd_pcm_close(*pphPCM);
392 if (rc2)
393 {
394 LogRel(("ALSA: Closing PCM descriptor failed: %s\n", snd_strerror(rc2)));
395 rc = VERR_GENERAL_FAILURE; /** @todo */
396 }
397 else
398 {
399 *pphPCM = NULL;
400 rc = VINF_SUCCESS;
401 }
402
403 return rc;
404}
405
406#if 0 /* After Beta. */
407static int alsaSetHWParams(snd_pcm_t *phPCM, PALSAAUDIOSTREAMCFG pCfg)
408{
409 int rc;
410 snd_pcm_hw_params_t *pParams = NULL;
411
412 do
413 {
414 snd_pcm_hw_params_alloca(&pParams);
415 if (!pParams)
416 {
417 rc = VERR_NO_MEMORY;
418 break;
419 }
420
421 unsigned int rrate;
422 snd_pcm_uframes_t size;
423 int dir;
424
425 /* choose all parameters */
426 int err = snd_pcm_hw_params_any(phPCM, pParams);
427 if (err < 0)
428 {
429 LogRel(("ALSA: Broken configuration for playback: no configurations available: %s\n", snd_strerror(err)));
430 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
431 break;
432 }
433 /* set hardware resampling */
434 err = snd_pcm_hw_params_set_rate_resample(phPCM, pParams, pCfg->resample);
435 if (err < 0)
436 {
437 LogRel(("ALSA: Resampling setup failed for playback: %s\n", snd_strerror(err)));
438 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
439 break;
440 }
441 /* set the interleaved read/write format */
442 err = snd_pcm_hw_params_set_access(phPCM, pParams, pCfg->access);
443 if (err < 0)
444 {
445 LogRel(("ALSA: Access type not available for playback: %s\n", snd_strerror(err)));
446 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
447 break;
448 }
449 /* set the sample format */
450 err = snd_pcm_hw_params_set_format(phPCM, pParams, pCfg->fmt);
451 if (err < 0)
452 {
453 LogRel(("ALSA: Sample format not available for playback: %s\n", snd_strerror(err)));
454 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
455 break;
456 }
457 /* set the count of channels */
458 err = snd_pcm_hw_params_set_channels(phPCM, pParams, pCfg->nchannels);
459 if (err < 0)
460 {
461 LogRel(("ALSA: Channels count (%d) not available for playbacks: %s\n", pCfg->nchannels, snd_strerror(err)));
462 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
463 break;
464 }
465 /* set the stream rate */
466 rrate = pCfg->freq;
467 err = snd_pcm_hw_params_set_rate_near(phPCM, pParams, &rrate, 0);
468 if (err < 0)
469 {
470 LogRel(("ALSA: Rate %uHz not available for playback: %s\n", pCfg->freq, snd_strerror(err)));
471 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
472 break;
473 }
474 if (rrate != pCfg->freq)
475 {
476 LogRel(("ALSA: Rate doesn't match (requested %iHz, get %uHz)\n", pCfg->freq, err));
477 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
478 break;
479 }
480 /* set the buffer time */
481 err = snd_pcm_hw_params_set_buffer_time_near(phPCM, pParams, &pCfg->buffer_time, &dir);
482 if (err < 0)
483 {
484 LogRel(("ALSA: Unable to set buffer time %i for playback: %s\n", buffer_time, snd_strerror(err)));
485 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
486 break;
487 }
488 err = snd_pcm_hw_params_get_buffer_size(pParams, &size);
489 if (err < 0)
490 {
491 LogRel(("ALSA: Unable to get buffer size for playback: %s\n", snd_strerror(err)));
492 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
493 break;
494 }
495 buffer_size = size;
496 /* set the period time */
497 err = snd_pcm_hw_params_set_period_time_near(phPCM, pParams, &period_time, &dir);
498 if (err < 0)
499 {
500 LogRel(("ALSA: Unable to set period time %i for playback: %s\n", period_time, snd_strerror(err)));
501 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
502 break;
503 }
504 err = snd_pcm_hw_params_get_period_size(pParams, &size, &dir);
505 if (err < 0)
506 {
507 LogRel(("ALSA: Unable to get period size for playback: %s\n", snd_strerror(err)));
508 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
509 break;
510 }
511 period_size = size;
512 /* write the parameters to device */
513 err = snd_pcm_hw_params(phPCM, pParams);
514 if (err < 0)
515 {
516 LogRel(("ALSA: Unable to set hw params for playback: %s\n", snd_strerror(err)));
517 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
518 break;
519 }
520
521 rc = VINF_SUCCESS;
522
523 } while (0);
524
525 if (pParams)
526 {
527 snd_pcm_hw_params_free(pParams);
528 pParams = NULL;
529 }
530
531 LogFlowFuncLeaveRC(rc);
532 return rc;
533}
534static int alsaSetSWParams(snd_pcm_t *phPCM, PALSAAUDIOCFG pCfg)
535{
536 int rc;
537 snd_pcm_sw_params_t *pParams = NULL;
538
539 do
540 {
541 snd_pcm_sw_params_alloca(&pParams);
542 if (!pParams)
543 {
544 rc = VERR_NO_MEMORY;
545 break;
546 }
547 /* get the current swparams */
548 int err = snd_pcm_sw_params_current(phPCM, pParams);
549 if (err < 0)
550 {
551 LogRel(("ALSA: Unable to determine current swparams for playback: %s\n", snd_strerror(err)));
552 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
553 break;
554 }
555 /* start the transfer when the buffer is almost full: */
556 /* (buffer_size / avail_min) * avail_min */
557 err = snd_pcm_sw_params_set_start_threshold(phPCM, pParams, (buffer_size / period_size) * period_size);
558 if (err < 0)
559 {
560 LogRel(("ALSA: Unable to set start threshold mode for playback: %s\n", snd_strerror(err)));
561 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
562 break;
563 }
564 /* allow the transfer when at least period_size samples can be processed */
565 /* or disable this mechanism when period event is enabled (aka interrupt like style processing) */
566 err = snd_pcm_sw_params_set_avail_min(phPCM, pParams, period_size);
567 if (err < 0)
568 {
569 LogRel(("ALSA: Unable to set avail min for playback: %s\n", snd_strerror(err)));
570 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
571 break;
572 }
573 /* write the parameters to the playback device */
574 err = snd_pcm_sw_params(phPCM, pParams);
575 if (err < 0)
576 {
577 LogRel(("ALSA: Unable to set sw params for playback: %s\n", snd_strerror(err)));
578 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
579 break;
580 }
581
582 rc = VINF_SUCCESS;
583
584 } while (0);
585
586 if (pParams)
587 {
588 snd_pcm_sw_params_free(pParams);
589 pParams = NULL;
590 }
591
592 LogFlowFuncLeaveRC(rc);
593 return rc;
594}
595#endif
596
597static int alsaStreamOpen(bool fIn, PALSAAUDIOSTREAMCFG pCfgReq, PALSAAUDIOSTREAMCFG pCfgObt, snd_pcm_t **pphPCM)
598{
599 snd_pcm_t *phPCM = NULL;
600 int rc;
601
602 unsigned int cChannels = pCfgReq->nchannels;
603 unsigned int uFreq = pCfgReq->freq;
604 snd_pcm_uframes_t obt_buffer_size;
605
606 do
607 {
608 const char *pszDev = fIn ? s_ALSAConf.pcm_name_in : s_ALSAConf.pcm_name_out;
609 if (!pszDev)
610 {
611 LogRel(("ALSA: Invalid or no %s device name set\n", fIn ? "input" : "output"));
612 rc = VERR_INVALID_PARAMETER;
613 break;
614 }
615
616 int err = snd_pcm_open(&phPCM, pszDev,
617 fIn ? SND_PCM_STREAM_CAPTURE : SND_PCM_STREAM_PLAYBACK,
618 SND_PCM_NONBLOCK);
619 if (err < 0)
620 {
621 LogRel(("ALSA: Failed to open \"%s\" as %s device: %s\n", pszDev, fIn ? "input" : "output", snd_strerror(err)));
622 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
623 break;
624 }
625
626 LogRel(("ALSA: Using %s device \"%s\"\n", fIn ? "input" : "output", pszDev));
627
628 snd_pcm_hw_params_t *pHWParms;
629 snd_pcm_hw_params_alloca(&pHWParms); /** @todo Check for successful allocation? */
630 err = snd_pcm_hw_params_any(phPCM, pHWParms);
631 if (err < 0)
632 {
633 LogRel(("ALSA: Failed to initialize hardware parameters: %s\n", snd_strerror(err)));
634 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
635 break;
636 }
637
638 err = snd_pcm_hw_params_set_access(phPCM, pHWParms,
639 SND_PCM_ACCESS_RW_INTERLEAVED);
640 if (err < 0)
641 {
642 LogRel(("ALSA: Failed to set access type: %s\n", snd_strerror(err)));
643 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
644 break;
645 }
646
647 err = snd_pcm_hw_params_set_format(phPCM, pHWParms, pCfgReq->fmt);
648 if (err < 0)
649 {
650 LogRel(("ALSA: Failed to set audio format to %d: %s\n", pCfgReq->fmt, snd_strerror(err)));
651 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
652 break;
653 }
654
655 err = snd_pcm_hw_params_set_rate_near(phPCM, pHWParms, &uFreq, 0);
656 if (err < 0)
657 {
658 LogRel(("ALSA: Failed to set frequency to %uHz: %s\n", pCfgReq->freq, snd_strerror(err)));
659 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
660 break;
661 }
662
663 err = snd_pcm_hw_params_set_channels_near(phPCM, pHWParms, &cChannels);
664 if (err < 0)
665 {
666 LogRel(("ALSA: Failed to set number of channels to %d\n", pCfgReq->nchannels));
667 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
668 break;
669 }
670
671 if ( cChannels != 1
672 && cChannels != 2)
673 {
674 LogRel(("ALSA: Number of audio channels (%u) not supported\n", cChannels));
675 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
676 break;
677 }
678
679 unsigned int period_size = pCfgReq->period_size;
680 unsigned int buffer_size = pCfgReq->buffer_size;
681
682 if ( !((fIn && s_ALSAConf.size_in_usec_in)
683 || (!fIn && s_ALSAConf.size_in_usec_out)))
684 {
685 if (!buffer_size)
686 {
687 buffer_size = DEFAULT_BUFFER_SIZE;
688 period_size = DEFAULT_PERIOD_SIZE;
689 }
690 }
691
692 if (buffer_size)
693 {
694 if ( ( fIn && s_ALSAConf.size_in_usec_in)
695 || (!fIn && s_ALSAConf.size_in_usec_out))
696 {
697 if (period_size)
698 {
699 err = snd_pcm_hw_params_set_period_time_near(phPCM, pHWParms,
700 &period_size, 0);
701 if (err < 0)
702 {
703 LogRel(("ALSA: Failed to set period time %d\n", pCfgReq->period_size));
704 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
705 break;
706 }
707 }
708
709 err = snd_pcm_hw_params_set_buffer_time_near(phPCM, pHWParms,
710 &buffer_size, 0);
711 if (err < 0)
712 {
713 LogRel(("ALSA: Failed to set buffer time %d\n", pCfgReq->buffer_size));
714 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
715 break;
716 }
717 }
718 else
719 {
720 snd_pcm_uframes_t period_size_f = (snd_pcm_uframes_t)period_size;
721 snd_pcm_uframes_t buffer_size_f = (snd_pcm_uframes_t)buffer_size;
722
723 snd_pcm_uframes_t minval;
724
725 if (period_size_f)
726 {
727 minval = period_size_f;
728
729 int dir = 0;
730 err = snd_pcm_hw_params_get_period_size_min(pHWParms,
731 &minval, &dir);
732 if (err < 0)
733 {
734 LogRel(("ALSA: Could not determine minimal period size\n"));
735 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
736 break;
737 }
738 else
739 {
740 LogFunc(("Minimal period size is: %ld\n", minval));
741 if (period_size_f < minval)
742 {
743 if ( ( fIn && s_ALSAConf.period_size_in_overriden)
744 || (!fIn && s_ALSAConf.period_size_out_overriden))
745 {
746 LogFunc(("Period size %RU32 is less than minimal period size %RU32\n",
747 period_size_f, minval));
748 }
749
750 period_size_f = minval;
751 }
752 }
753
754 err = snd_pcm_hw_params_set_period_size_near(phPCM, pHWParms,
755 &period_size_f, 0);
756 LogFunc(("Period size is: %RU32\n", period_size_f));
757 if (err < 0)
758 {
759 LogRel(("ALSA: Failed to set period size %d (%s)\n",
760 period_size_f, snd_strerror(err)));
761 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
762 break;
763 }
764 }
765
766 /* Calculate default buffer size here since it might have been changed
767 * in the _near functions */
768 buffer_size_f = 4 * period_size_f;
769
770 minval = buffer_size_f;
771 err = snd_pcm_hw_params_get_buffer_size_min(pHWParms, &minval);
772 if (err < 0)
773 {
774 LogRel(("ALSA: Could not retrieve minimal buffer size\n"));
775 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
776 break;
777 }
778 else
779 {
780 LogFunc(("Minimal buffer size is: %RU32\n", minval));
781 if (buffer_size_f < minval)
782 {
783 if ( ( fIn && s_ALSAConf.buffer_size_in_overriden)
784 || (!fIn && s_ALSAConf.buffer_size_out_overriden))
785 {
786 LogFunc(("Buffer size %RU32 is less than minimal buffer size %RU32\n",
787 buffer_size_f, minval));
788 }
789
790 buffer_size_f = minval;
791 }
792 }
793
794 err = snd_pcm_hw_params_set_buffer_size_near(phPCM,
795 pHWParms, &buffer_size_f);
796 LogFunc(("Buffer size is: %RU32\n", buffer_size_f));
797 if (err < 0)
798 {
799 LogRel(("ALSA: Failed to set buffer size %d: %s\n",
800 buffer_size_f, snd_strerror(err)));
801 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
802 break;
803 }
804 }
805 }
806 else
807 LogFunc(("Warning: Buffer size is not set\n"));
808
809 err = snd_pcm_hw_params(phPCM, pHWParms);
810 if (err < 0)
811 {
812 LogRel(("ALSA: Failed to apply audio parameters\n"));
813 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
814 break;
815 }
816
817 err = snd_pcm_hw_params_get_buffer_size(pHWParms, &obt_buffer_size);
818 if (err < 0)
819 {
820 LogRel(("ALSA: Failed to get buffer size\n"));
821 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
822 break;
823 }
824
825 LogFunc(("Buffer sample size is: %RU32\n", obt_buffer_size));
826
827 snd_pcm_uframes_t obt_period_size;
828 int dir = 0;
829 err = snd_pcm_hw_params_get_period_size(pHWParms, &obt_period_size, &dir);
830 if (err < 0)
831 {
832 LogRel(("ALSA: Failed to get period size\n"));
833 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
834 break;
835 }
836
837 LogFunc(("Freq=%dHz, period size=%RU32, buffer size=%RU32\n",
838 pCfgReq->freq, obt_period_size, obt_buffer_size));
839
840 err = snd_pcm_prepare(phPCM);
841 if (err < 0)
842 {
843 LogRel(("ALSA: Could not prepare hPCM %p\n", (void *)phPCM));
844 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
845 break;
846 }
847
848 if ( !fIn
849 && s_ALSAConf.threshold)
850 {
851 unsigned uShift;
852 rc = alsaGetSampleShift(pCfgReq->fmt, &uShift);
853 if (RT_SUCCESS(rc))
854 {
855 int bytes_per_sec = uFreq
856 << (cChannels == 2)
857 << uShift;
858
859 snd_pcm_uframes_t threshold
860 = (s_ALSAConf.threshold * bytes_per_sec) / 1000;
861
862 rc = alsaStreamSetThreshold(phPCM, threshold);
863 }
864 }
865 else
866 rc = VINF_SUCCESS;
867 }
868 while (0);
869
870 if (RT_SUCCESS(rc))
871 {
872 pCfgObt->fmt = pCfgReq->fmt;
873 pCfgObt->nchannels = cChannels;
874 pCfgObt->freq = uFreq;
875 pCfgObt->samples = obt_buffer_size;
876
877 *pphPCM = phPCM;
878 }
879 else
880 alsaStreamClose(&phPCM);
881
882 LogFlowFuncLeaveRC(rc);
883 return rc;
884}
885
886#ifdef DEBUG
887static void alsaDbgErrorHandler(const char *file, int line, const char *function,
888 int err, const char *fmt, ...)
889{
890 /** @todo Implement me! */
891}
892#endif
893
894static int alsaStreamGetAvail(snd_pcm_t *phPCM, snd_pcm_sframes_t *pFramesAvail)
895{
896 AssertPtrReturn(phPCM, VERR_INVALID_POINTER);
897 AssertPtrReturn(pFramesAvail, VERR_INVALID_POINTER);
898
899 int rc;
900
901 snd_pcm_sframes_t framesAvail;
902 framesAvail = snd_pcm_avail_update(phPCM);
903 if (framesAvail < 0)
904 {
905 if (framesAvail == -EPIPE)
906 {
907 rc = alsaStreamRecover(phPCM);
908 if (RT_SUCCESS(rc))
909 framesAvail = snd_pcm_avail_update(phPCM);
910 }
911 else
912 rc = VERR_ACCESS_DENIED; /** @todo Find a better rc. */
913 }
914 else
915 rc = VINF_SUCCESS;
916
917 if (framesAvail >= 0)
918 *pFramesAvail = framesAvail;
919
920 return rc;
921}
922
923static int alsaStreamRecover(snd_pcm_t *phPCM)
924{
925 AssertPtrReturn(phPCM, VERR_INVALID_POINTER);
926
927 int err = snd_pcm_prepare(phPCM);
928 if (err < 0)
929 {
930 LogFunc(("Failed to recover stream %p: %s\n", phPCM, snd_strerror(err)));
931 return VERR_ACCESS_DENIED; /** @todo Find a better rc. */
932 }
933
934 return VINF_SUCCESS;
935}
936
937static int alsaStreamResume(snd_pcm_t *phPCM)
938{
939 AssertPtrReturn(phPCM, VERR_INVALID_POINTER);
940
941 int err = snd_pcm_resume(phPCM);
942 if (err < 0)
943 {
944 LogFunc(("Failed to resume stream %p: %s\n", phPCM, snd_strerror(err)));
945 return VERR_ACCESS_DENIED; /** @todo Find a better rc. */
946 }
947
948 return VINF_SUCCESS;
949}
950
951static int drvHostALSAAudioStreamCtl(snd_pcm_t *phPCM, bool fPause)
952{
953 int err;
954 if (fPause)
955 {
956 err = snd_pcm_drop(phPCM);
957 if (err < 0)
958 {
959 LogRel(("ALSA: Error stopping stream %p: %s\n", phPCM, snd_strerror(err)));
960 return VERR_ACCESS_DENIED;
961 }
962 }
963 else
964 {
965 err = snd_pcm_prepare(phPCM);
966 if (err < 0)
967 {
968 LogRel(("ALSA: Error preparing stream %p: %s\n", phPCM, snd_strerror(err)));
969 return VERR_ACCESS_DENIED;
970 }
971 }
972
973 return VINF_SUCCESS;
974}
975
976static DECLCALLBACK(int) drvHostALSAAudioInit(PPDMIHOSTAUDIO pInterface)
977{
978 NOREF(pInterface);
979
980 LogFlowFuncEnter();
981
982 int rc = audioLoadAlsaLib();
983 if (RT_FAILURE(rc))
984 LogRel(("ALSA: Failed to load the ALSA shared library, rc=%Rrc\n", rc));
985 else
986 {
987#ifdef DEBUG
988 snd_lib_error_set_handler(alsaDbgErrorHandler);
989#endif
990 }
991
992 return rc;
993}
994
995static DECLCALLBACK(int) drvHostALSAAudioStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
996 uint32_t *pcSamplesCaptured)
997{
998 NOREF(pInterface);
999 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1000
1001 PALSAAUDIOSTREAMIN pThisStream = (PALSAAUDIOSTREAMIN)pStream;
1002
1003 snd_pcm_sframes_t cAvail;
1004 int rc = alsaStreamGetAvail(pThisStream->phPCM, &cAvail);
1005 if (RT_FAILURE(rc))
1006 {
1007 LogFunc(("Error getting number of captured frames, rc=%Rrc\n", rc));
1008 return rc;
1009 }
1010
1011 if (!cAvail) /* No data yet? */
1012 {
1013 snd_pcm_state_t state = snd_pcm_state(pThisStream->phPCM);
1014 switch (state)
1015 {
1016 case SND_PCM_STATE_PREPARED:
1017 cAvail = AudioMixBufFree(&pStream->MixBuf);
1018 break;
1019
1020 case SND_PCM_STATE_SUSPENDED:
1021 {
1022 rc = alsaStreamResume(pThisStream->phPCM);
1023 if (RT_FAILURE(rc))
1024 break;
1025
1026 LogFlow(("Resuming suspended input stream\n"));
1027 break;
1028 }
1029
1030 default:
1031 LogFlow(("No frames available, state=%d\n", state));
1032 break;
1033 }
1034
1035 if (!cAvail)
1036 {
1037 if (pcSamplesCaptured)
1038 *pcSamplesCaptured = 0;
1039 return VINF_SUCCESS;
1040 }
1041 }
1042
1043 /*
1044 * Check how much we can read from the capture device without overflowing
1045 * the mixer buffer.
1046 */
1047 Assert(cAvail);
1048 size_t cbMixFree = AudioMixBufFreeBytes(&pStream->MixBuf);
1049 size_t cbToRead = RT_MIN((size_t)AUDIOMIXBUF_S2B(&pStream->MixBuf, cAvail), cbMixFree);
1050
1051 LogFlowFunc(("cbToRead=%zu, cAvail=%RI32\n", cbToRead, cAvail));
1052
1053 uint32_t cWrittenTotal = 0;
1054 snd_pcm_uframes_t cToRead;
1055 snd_pcm_sframes_t cRead;
1056
1057 while ( cbToRead
1058 && RT_SUCCESS(rc))
1059 {
1060 cToRead = RT_MIN(AUDIOMIXBUF_B2S(&pStream->MixBuf, cbToRead),
1061 AUDIOMIXBUF_B2S(&pStream->MixBuf, pThisStream->cbBuf));
1062 AssertBreakStmt(cToRead, rc = VERR_NO_DATA);
1063 cRead = snd_pcm_readi(pThisStream->phPCM, pThisStream->pvBuf, cToRead);
1064 if (cRead <= 0)
1065 {
1066 switch (cRead)
1067 {
1068 case 0:
1069 {
1070 LogFunc(("No input frames available\n"));
1071 rc = VERR_ACCESS_DENIED;
1072 break;
1073 }
1074
1075 case -EAGAIN:
1076 {
1077 /*
1078 * Don't set error here because EAGAIN means there are no further frames
1079 * available at the moment, try later. As we might have read some frames
1080 * already these need to be processed instead.
1081 */
1082 cbToRead = 0;
1083 break;
1084 }
1085
1086 case -EPIPE:
1087 {
1088 rc = alsaStreamRecover(pThisStream->phPCM);
1089 if (RT_FAILURE(rc))
1090 break;
1091
1092 LogFlowFunc(("Recovered from capturing\n"));
1093 continue;
1094 }
1095
1096 default:
1097 {
1098 LogFunc(("Failed to read input frames: %s\n", snd_strerror(cRead)));
1099 rc = VERR_GENERAL_FAILURE; /** @todo Fudge! */
1100 break;
1101 }
1102 }
1103 }
1104 else
1105 {
1106 uint32_t cWritten;
1107 rc = AudioMixBufWriteCirc(&pStream->MixBuf,
1108 pThisStream->pvBuf, AUDIOMIXBUF_S2B(&pStream->MixBuf, cRead),
1109 &cWritten);
1110 if (RT_FAILURE(rc))
1111 break;
1112
1113 /*
1114 * We should not run into a full mixer buffer or we loose samples and
1115 * run into an endless loop if ALSA keeps producing samples ("null"
1116 * capture device for example).
1117 */
1118 AssertLogRelMsgBreakStmt(cWritten > 0, ("Mixer buffer shouldn't be full at this point!\n"),
1119 rc = VERR_INTERNAL_ERROR);
1120 uint32_t cbWritten = AUDIOMIXBUF_S2B(&pStream->MixBuf, cWritten);
1121
1122 Assert(cbToRead >= cbWritten);
1123 cbToRead -= cbWritten;
1124 cWrittenTotal += cWritten;
1125 }
1126 }
1127
1128 if (RT_SUCCESS(rc))
1129 {
1130 uint32_t cProcessed = 0;
1131 if (cWrittenTotal)
1132 rc = AudioMixBufMixToParent(&pStream->MixBuf, cWrittenTotal,
1133 &cProcessed);
1134
1135 if (pcSamplesCaptured)
1136 *pcSamplesCaptured = cWrittenTotal;
1137
1138 LogFlowFunc(("cWrittenTotal=%RU32 (%RU32 processed), rc=%Rrc\n",
1139 cWrittenTotal, cProcessed, rc));
1140 }
1141
1142 LogFlowFuncLeaveRC(rc);
1143 return rc;
1144}
1145
1146static DECLCALLBACK(int) drvHostALSAAudioStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1147 uint32_t *pcSamplesPlayed)
1148{
1149 NOREF(pInterface);
1150 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1151
1152 PALSAAUDIOSTREAMOUT pThisStream = (PALSAAUDIOSTREAMOUT)pStream;
1153
1154 int rc = VINF_SUCCESS;
1155 uint32_t cbReadTotal = 0;
1156
1157 do
1158 {
1159 snd_pcm_sframes_t cAvail;
1160 rc = alsaStreamGetAvail(pThisStream->phPCM, &cAvail);
1161 if (RT_FAILURE(rc))
1162 {
1163 LogFunc(("Error getting number of playback frames, rc=%Rrc\n", rc));
1164 break;
1165 }
1166
1167 size_t cbToRead = RT_MIN(AUDIOMIXBUF_S2B(&pStream->MixBuf,
1168 (uint32_t)cAvail), /* cAvail is always >= 0 */
1169 AUDIOMIXBUF_S2B(&pStream->MixBuf,
1170 AudioMixBufLive(&pStream->MixBuf)));
1171 LogFlowFunc(("cbToRead=%zu, cbAvail=%zu\n",
1172 cbToRead, AUDIOMIXBUF_S2B(&pStream->MixBuf, cAvail)));
1173
1174 uint32_t cRead, cbRead;
1175 snd_pcm_sframes_t cWritten;
1176 while (cbToRead)
1177 {
1178 rc = AudioMixBufReadCirc(&pStream->MixBuf, pThisStream->pvBuf, cbToRead, &cRead);
1179 if (RT_FAILURE(rc))
1180 break;
1181
1182 cbRead = AUDIOMIXBUF_S2B(&pStream->MixBuf, cRead);
1183 AssertBreak(cbRead);
1184
1185 /* Don't try infinitely on recoverable errors. */
1186 unsigned iTry;
1187 for (iTry = 0; iTry < ALSA_RECOVERY_TRIES_MAX; iTry++)
1188 {
1189 cWritten = snd_pcm_writei(pThisStream->phPCM, pThisStream->pvBuf, cRead);
1190 if (cWritten <= 0)
1191 {
1192 switch (cWritten)
1193 {
1194 case 0:
1195 {
1196 LogFunc(("Failed to write %RU32 samples\n", cRead));
1197 rc = VERR_ACCESS_DENIED;
1198 break;
1199 }
1200
1201 case -EPIPE:
1202 {
1203 rc = alsaStreamRecover(pThisStream->phPCM);
1204 if (RT_FAILURE(rc))
1205 break;
1206
1207 LogFlowFunc(("Recovered from playback\n"));
1208 continue;
1209 }
1210
1211 case -ESTRPIPE:
1212 {
1213 /* Stream was suspended and waiting for a recovery. */
1214 rc = alsaStreamResume(pThisStream->phPCM);
1215 if (RT_FAILURE(rc))
1216 {
1217 LogRel(("ALSA: Failed to resume output stream\n"));
1218 break;
1219 }
1220
1221 LogFlowFunc(("Resumed suspended output stream\n"));
1222 continue;
1223 }
1224
1225 default:
1226 LogFlowFunc(("Failed to write %RI32 output frames, rc=%Rrc\n",
1227 cRead, rc));
1228 rc = VERR_GENERAL_FAILURE; /** @todo */
1229 break;
1230 }
1231 }
1232 else
1233 break;
1234 } /* For number of tries. */
1235
1236 if ( iTry == ALSA_RECOVERY_TRIES_MAX
1237 && cWritten <= 0)
1238 rc = VERR_BROKEN_PIPE;
1239
1240 if (RT_FAILURE(rc))
1241 break;
1242
1243 Assert(cbToRead >= cbRead);
1244 cbToRead -= cbRead;
1245 cbReadTotal += cbRead;
1246 }
1247 }
1248 while (0);
1249
1250 if (RT_SUCCESS(rc))
1251 {
1252 uint32_t cReadTotal = AUDIOMIXBUF_B2S(&pStream->MixBuf, cbReadTotal);
1253 if (cReadTotal)
1254 AudioMixBufFinish(&pStream->MixBuf, cReadTotal);
1255
1256 if (pcSamplesPlayed)
1257 *pcSamplesPlayed = cReadTotal;
1258
1259 LogFlowFunc(("cReadTotal=%RU32 (%RU32 bytes), rc=%Rrc\n",
1260 cReadTotal, cbReadTotal, rc));
1261 }
1262
1263 LogFlowFuncLeaveRC(rc);
1264 return rc;
1265}
1266
1267static int alsaDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1268{
1269 NOREF(pInterface);
1270 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1271
1272 PALSAAUDIOSTREAMIN pThisStream = (PALSAAUDIOSTREAMIN)pStream;
1273
1274 alsaStreamClose(&pThisStream->phPCM);
1275
1276 if (pThisStream->pvBuf)
1277 {
1278 RTMemFree(pThisStream->pvBuf);
1279 pThisStream->pvBuf = NULL;
1280 }
1281
1282 return VINF_SUCCESS;
1283}
1284
1285static int alsaDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1286{
1287 NOREF(pInterface);
1288 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1289
1290 PALSAAUDIOSTREAMOUT pThisStream = (PALSAAUDIOSTREAMOUT)pStream;
1291
1292 alsaStreamClose(&pThisStream->phPCM);
1293
1294 if (pThisStream->pvBuf)
1295 {
1296 RTMemFree(pThisStream->pvBuf);
1297 pThisStream->pvBuf = NULL;
1298 }
1299
1300 return VINF_SUCCESS;
1301}
1302
1303static int alsaCreateStreamOut(PPDMIHOSTAUDIO pInterface,
1304 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg,
1305 uint32_t *pcSamples)
1306{
1307 NOREF(pInterface);
1308 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1309 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1310
1311 PALSAAUDIOSTREAMOUT pThisStream = (PALSAAUDIOSTREAMOUT)pStream;
1312 snd_pcm_t *phPCM = NULL;
1313
1314 int rc;
1315
1316 do
1317 {
1318 ALSAAUDIOSTREAMCFG req;
1319 req.fmt = alsaAudioFmtToALSA(pCfg->enmFormat);
1320 req.freq = pCfg->uHz;
1321 req.nchannels = pCfg->cChannels;
1322 req.period_size = s_ALSAConf.period_size_out;
1323 req.buffer_size = s_ALSAConf.buffer_size_out;
1324
1325 ALSAAUDIOSTREAMCFG obt;
1326 rc = alsaStreamOpen(false /* fIn */, &req, &obt, &phPCM);
1327 if (RT_FAILURE(rc))
1328 break;
1329
1330 PDMAUDIOFMT enmFormat;
1331 PDMAUDIOENDIANNESS enmEnd;
1332 rc = alsaALSAToAudioFmt(obt.fmt, &enmFormat, &enmEnd);
1333 if (RT_FAILURE(rc))
1334 break;
1335
1336 PDMAUDIOSTREAMCFG streamCfg;
1337 streamCfg.uHz = obt.freq;
1338 streamCfg.cChannels = obt.nchannels;
1339 streamCfg.enmFormat = enmFormat;
1340 streamCfg.enmEndianness = enmEnd;
1341
1342 rc = DrvAudioHlpStreamCfgToProps(&streamCfg, &pStream->Props);
1343 if (RT_FAILURE(rc))
1344 break;
1345
1346 AssertBreakStmt(obt.samples, rc = VERR_INVALID_PARAMETER);
1347 size_t cbBuf = obt.samples * (1 << pStream->Props.cShift);
1348 AssertBreakStmt(cbBuf, rc = VERR_INVALID_PARAMETER);
1349 pThisStream->pvBuf = RTMemAlloc(cbBuf);
1350 if (!pThisStream->pvBuf)
1351 {
1352 LogRel(("ALSA: Not enough memory for output DAC buffer (%RU32 samples, each %d bytes)\n",
1353 obt.samples, 1 << pStream->Props.cShift));
1354 rc = VERR_NO_MEMORY;
1355 break;
1356 }
1357
1358 pThisStream->cbBuf = cbBuf;
1359 pThisStream->phPCM = phPCM;
1360
1361 if (pcSamples)
1362 *pcSamples = obt.samples * 4;
1363 }
1364 while (0);
1365
1366 if (RT_FAILURE(rc))
1367 alsaStreamClose(&phPCM);
1368
1369 LogFlowFuncLeaveRC(rc);
1370 return rc;
1371}
1372
1373static int alsaCreateStreamIn(PPDMIHOSTAUDIO pInterface,
1374 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
1375{
1376 NOREF(pInterface);
1377 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1378 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1379
1380 int rc;
1381
1382 PALSAAUDIOSTREAMIN pThisStream = (PALSAAUDIOSTREAMIN)pStream;
1383 snd_pcm_t *phPCM = NULL;
1384
1385 do
1386 {
1387 ALSAAUDIOSTREAMCFG req;
1388 req.fmt = alsaAudioFmtToALSA(pCfg->enmFormat);
1389 req.freq = pCfg->uHz;
1390 req.nchannels = pCfg->cChannels;
1391 req.period_size = s_ALSAConf.period_size_in;
1392 req.buffer_size = s_ALSAConf.buffer_size_in;
1393
1394 ALSAAUDIOSTREAMCFG obt;
1395 rc = alsaStreamOpen(true /* fIn */, &req, &obt, &phPCM);
1396 if (RT_FAILURE(rc))
1397 break;
1398
1399 PDMAUDIOFMT enmFormat;
1400 PDMAUDIOENDIANNESS enmEnd;
1401 rc = alsaALSAToAudioFmt(obt.fmt, &enmFormat, &enmEnd);
1402 if (RT_FAILURE(rc))
1403 break;
1404
1405 PDMAUDIOSTREAMCFG streamCfg;
1406 streamCfg.uHz = obt.freq;
1407 streamCfg.cChannels = obt.nchannels;
1408 streamCfg.enmFormat = enmFormat;
1409 streamCfg.enmEndianness = enmEnd;
1410
1411 rc = DrvAudioHlpStreamCfgToProps(&streamCfg, &pStream->Props);
1412 if (RT_FAILURE(rc))
1413 break;
1414
1415 AssertBreakStmt(obt.samples, rc = VERR_INVALID_PARAMETER);
1416 size_t cbBuf = obt.samples * (1 << pStream->Props.cShift);
1417 AssertBreakStmt(cbBuf, rc = VERR_INVALID_PARAMETER);
1418 pThisStream->pvBuf = RTMemAlloc(cbBuf);
1419 if (!pThisStream->pvBuf)
1420 {
1421 LogRel(("ALSA: Not enough memory for input ADC buffer (%RU32 samples, each %d bytes)\n",
1422 obt.samples, 1 << pStream->Props.cShift));
1423 rc = VERR_NO_MEMORY;
1424 break;
1425 }
1426
1427 pThisStream->cbBuf = cbBuf;
1428 pThisStream->phPCM = phPCM;
1429
1430 if (pcSamples)
1431 *pcSamples = obt.samples;
1432 }
1433 while (0);
1434
1435 if (RT_FAILURE(rc))
1436 alsaStreamClose(&phPCM);
1437
1438 LogFlowFuncLeaveRC(rc);
1439 return rc;
1440}
1441
1442static int alsaControlStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1443 PDMAUDIOSTREAMCMD enmStreamCmd)
1444{
1445 NOREF(pInterface);
1446 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1447 PALSAAUDIOSTREAMIN pThisStream = (PALSAAUDIOSTREAMIN)pStream;
1448
1449 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
1450
1451 int rc;
1452 switch (enmStreamCmd)
1453 {
1454 case PDMAUDIOSTREAMCMD_ENABLE:
1455 case PDMAUDIOSTREAMCMD_RESUME:
1456 rc = drvHostALSAAudioStreamCtl(pThisStream->phPCM, false /* fStop */);
1457 break;
1458
1459 case PDMAUDIOSTREAMCMD_DISABLE:
1460 case PDMAUDIOSTREAMCMD_PAUSE:
1461 rc = drvHostALSAAudioStreamCtl(pThisStream->phPCM, true /* fStop */);
1462 break;
1463
1464 default:
1465 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1466 rc = VERR_INVALID_PARAMETER;
1467 break;
1468 }
1469
1470 return rc;
1471}
1472
1473static int alsaControlStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1474 PDMAUDIOSTREAMCMD enmStreamCmd)
1475{
1476 NOREF(pInterface);
1477 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1478 PALSAAUDIOSTREAMOUT pThisStream = (PALSAAUDIOSTREAMOUT)pStream;
1479
1480 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
1481
1482 int rc;
1483 switch (enmStreamCmd)
1484 {
1485 case PDMAUDIOSTREAMCMD_ENABLE:
1486 case PDMAUDIOSTREAMCMD_RESUME:
1487 rc = drvHostALSAAudioStreamCtl(pThisStream->phPCM, false /* fStop */);
1488 break;
1489
1490 case PDMAUDIOSTREAMCMD_DISABLE:
1491 case PDMAUDIOSTREAMCMD_PAUSE:
1492 rc = drvHostALSAAudioStreamCtl(pThisStream->phPCM, true /* fStop */);
1493 break;
1494
1495 default:
1496 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1497 rc = VERR_INVALID_PARAMETER;
1498 break;
1499 }
1500
1501 return rc;
1502}
1503
1504static DECLCALLBACK(int) drvHostALSAAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
1505{
1506 NOREF(pInterface);
1507 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1508
1509 pCfg->cbStreamIn = sizeof(ALSAAUDIOSTREAMIN);
1510 pCfg->cbStreamOut = sizeof(ALSAAUDIOSTREAMOUT);
1511
1512 pCfg->cSources = 0;
1513 pCfg->cSinks = 0;
1514
1515 /* Enumerate sound devices. */
1516 char **pszHints;
1517 int err = snd_device_name_hint(-1 /* All cards */, "pcm", (void***)&pszHints);
1518 if (err == 0)
1519 {
1520 char** pszHintCur = pszHints;
1521 while (*pszHintCur != NULL)
1522 {
1523 char *pszDev = snd_device_name_get_hint(*pszHintCur, "NAME");
1524 bool fSkip = !pszDev
1525 || !RTStrICmp("null", pszDev);
1526 if (fSkip)
1527 {
1528 if (pszDev)
1529 free(pszDev);
1530 pszHintCur++;
1531 continue;
1532 }
1533
1534 char *pszIOID = snd_device_name_get_hint(*pszHintCur, "IOID");
1535 if (pszIOID)
1536 {
1537 if (!RTStrICmp("input", pszIOID))
1538 pCfg->cSources++;
1539 else if (!RTStrICmp("output", pszIOID))
1540 pCfg->cSinks++;
1541 }
1542 else /* NULL means bidirectional, input + output. */
1543 {
1544 pCfg->cSources++;
1545 pCfg->cSinks++;
1546 }
1547
1548 LogRel2(("ALSA: Found %s device: %s\n", pszIOID ? RTStrToLower(pszIOID) : "bidirectional", pszDev));
1549
1550 /* Special case for ALSAAudio. */
1551 if ( pszDev
1552 && RTStrIStr("pulse", pszDev) != NULL)
1553 LogRel2(("ALSA: ALSAAudio plugin in use\n"));
1554
1555 if (pszIOID)
1556 free(pszIOID);
1557
1558 if (pszDev)
1559 free(pszDev);
1560
1561 pszHintCur++;
1562 }
1563
1564 LogRel2(("ALSA: Found %RU8 host playback devices\n", pCfg->cSinks));
1565 LogRel2(("ALSA: Found %RU8 host capturing devices\n", pCfg->cSources));
1566
1567 snd_device_name_free_hint((void **)pszHints);
1568 pszHints = NULL;
1569 }
1570 else
1571 LogRel2(("ALSA: Error enumerating PCM devices: %Rrc (%d)\n", RTErrConvertFromErrno(err), err));
1572
1573 /* ALSA allows exactly one input and one output used at a time for the selected device(s). */
1574 pCfg->cMaxStreamsIn = 1;
1575 pCfg->cMaxStreamsOut = 1;
1576
1577 return VINF_SUCCESS;
1578}
1579
1580static DECLCALLBACK(void) drvHostALSAAudioShutdown(PPDMIHOSTAUDIO pInterface)
1581{
1582 NOREF(pInterface);
1583}
1584
1585static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostALSAAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
1586{
1587 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
1588
1589 return PDMAUDIOBACKENDSTS_RUNNING;
1590}
1591
1592static DECLCALLBACK(int) drvHostALSAAudioStreamCreate(PPDMIHOSTAUDIO pInterface,
1593 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
1594{
1595 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1596 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1597 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1598
1599 int rc;
1600 if (pCfg->enmDir == PDMAUDIODIR_IN)
1601 rc = alsaCreateStreamIn(pInterface, pStream, pCfg, pcSamples);
1602 else
1603 rc = alsaCreateStreamOut(pInterface, pStream, pCfg, pcSamples);
1604
1605 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
1606 return rc;
1607}
1608
1609static DECLCALLBACK(int) drvHostALSAAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1610{
1611 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1612 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1613
1614 int rc;
1615 if (pStream->enmDir == PDMAUDIODIR_IN)
1616 rc = alsaDestroyStreamIn(pInterface, pStream);
1617 else
1618 rc = alsaDestroyStreamOut(pInterface, pStream);
1619
1620 return rc;
1621}
1622
1623static DECLCALLBACK(int) drvHostALSAAudioStreamControl(PPDMIHOSTAUDIO pInterface,
1624 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
1625{
1626 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1627 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1628
1629 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1630
1631 int rc;
1632 if (pStream->enmDir == PDMAUDIODIR_IN)
1633 rc = alsaControlStreamIn(pInterface, pStream, enmStreamCmd);
1634 else
1635 rc = alsaControlStreamOut(pInterface, pStream, enmStreamCmd);
1636
1637 return rc;
1638}
1639
1640static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostALSAAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1641{
1642 NOREF(pInterface);
1643 NOREF(pStream);
1644
1645 PDMAUDIOSTRMSTS strmSts = PDMAUDIOSTRMSTS_FLAG_INITIALIZED
1646 | PDMAUDIOSTRMSTS_FLAG_ENABLED;
1647
1648 if (pStream->enmDir == PDMAUDIODIR_IN)
1649 {
1650
1651 }
1652 else
1653 {
1654 PALSAAUDIOSTREAMOUT pStreamOut = (PALSAAUDIOSTREAMOUT)pStream;
1655
1656 snd_pcm_sframes_t cAvail;
1657 int rc2 = alsaStreamGetAvail(pStreamOut->phPCM, &cAvail);
1658 if (RT_SUCCESS(rc2))
1659 {
1660 LogFlowFunc(("cAvail=%ld\n", cAvail));
1661 if (cAvail >= pStreamOut->cSamplesMin)
1662 strmSts |= PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE;
1663 }
1664 }
1665
1666 return strmSts;
1667}
1668
1669static DECLCALLBACK(int) drvHostALSAAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1670{
1671 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1672 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1673
1674 LogFlowFuncEnter();
1675
1676 /* Nothing to do here for ALSA. */
1677 return VINF_SUCCESS;
1678}
1679
1680/**
1681 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1682 */
1683static DECLCALLBACK(void *) drvHostALSAAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1684{
1685 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1686 PDRVHOSTALSAAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTALSAAUDIO);
1687 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1688 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
1689
1690 return NULL;
1691}
1692
1693/**
1694 * Construct a DirectSound Audio driver instance.
1695 *
1696 * @copydoc FNPDMDRVCONSTRUCT
1697 */
1698static DECLCALLBACK(int) drvHostAlsaAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1699{
1700 PDRVHOSTALSAAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTALSAAUDIO);
1701 LogRel(("Audio: Initializing ALSA driver\n"));
1702
1703 /*
1704 * Init the static parts.
1705 */
1706 pThis->pDrvIns = pDrvIns;
1707 /* IBase */
1708 pDrvIns->IBase.pfnQueryInterface = drvHostALSAAudioQueryInterface;
1709 /* IHostAudio */
1710 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostALSAAudio);
1711
1712 return VINF_SUCCESS;
1713}
1714
1715/**
1716 * Char driver registration record.
1717 */
1718const PDMDRVREG g_DrvHostALSAAudio =
1719{
1720 /* u32Version */
1721 PDM_DRVREG_VERSION,
1722 /* szName */
1723 "ALSAAudio",
1724 /* szRCMod */
1725 "",
1726 /* szR0Mod */
1727 "",
1728 /* pszDescription */
1729 "ALSA host audio driver",
1730 /* fFlags */
1731 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1732 /* fClass. */
1733 PDM_DRVREG_CLASS_AUDIO,
1734 /* cMaxInstances */
1735 ~0U,
1736 /* cbInstance */
1737 sizeof(DRVHOSTALSAAUDIO),
1738 /* pfnConstruct */
1739 drvHostAlsaAudioConstruct,
1740 /* pfnDestruct */
1741 NULL,
1742 /* pfnRelocate */
1743 NULL,
1744 /* pfnIOCtl */
1745 NULL,
1746 /* pfnPowerOn */
1747 NULL,
1748 /* pfnReset */
1749 NULL,
1750 /* pfnSuspend */
1751 NULL,
1752 /* pfnResume */
1753 NULL,
1754 /* pfnAttach */
1755 NULL,
1756 /* pfnDetach */
1757 NULL,
1758 /* pfnPowerOff */
1759 NULL,
1760 /* pfnSoftReset */
1761 NULL,
1762 /* u32EndVersion */
1763 PDM_DRVREG_VERSION
1764};
1765
1766static struct audio_option alsa_options[] =
1767{
1768 {"DACSizeInUsec", AUD_OPT_BOOL, &s_ALSAConf.size_in_usec_out,
1769 "DAC period/buffer size in microseconds (otherwise in frames)", NULL, 0},
1770 {"DACPeriodSize", AUD_OPT_INT, &s_ALSAConf.period_size_out,
1771 "DAC period size", &s_ALSAConf.period_size_out_overriden, 0},
1772 {"DACBufferSize", AUD_OPT_INT, &s_ALSAConf.buffer_size_out,
1773 "DAC buffer size", &s_ALSAConf.buffer_size_out_overriden, 0},
1774
1775 {"ADCSizeInUsec", AUD_OPT_BOOL, &s_ALSAConf.size_in_usec_in,
1776 "ADC period/buffer size in microseconds (otherwise in frames)", NULL, 0},
1777 {"ADCPeriodSize", AUD_OPT_INT, &s_ALSAConf.period_size_in,
1778 "ADC period size", &s_ALSAConf.period_size_in_overriden, 0},
1779 {"ADCBufferSize", AUD_OPT_INT, &s_ALSAConf.buffer_size_in,
1780 "ADC buffer size", &s_ALSAConf.buffer_size_in_overriden, 0},
1781
1782 {"Threshold", AUD_OPT_INT, &s_ALSAConf.threshold,
1783 "(undocumented)", NULL, 0},
1784
1785 {"DACDev", AUD_OPT_STR, &s_ALSAConf.pcm_name_out,
1786 "DAC device name (for instance dmix)", NULL, 0},
1787
1788 {"ADCDev", AUD_OPT_STR, &s_ALSAConf.pcm_name_in,
1789 "ADC device name", NULL, 0}
1790};
1791
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