VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostPulseAudio.cpp@ 60961

Last change on this file since 60961 was 60925, checked in by vboxsync, 9 years ago

Audio: Update on infrastructure:

  • More work on HDA stream interleaving + surround support
  • The mixer can now (optionally) act as a supplemental layer between audio connector interface and device emulation (where applicable)
  • Multiple LUN streams can be bound to a certain sink, which in turn then can be treated as separate input/output channels
  • Unified more code which was duplicated between different audio device emulations
  • Tiny bit of documentation

Work in progress.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 45.8 KB
Line 
1/* $Id: DrvHostPulseAudio.cpp 60925 2016-05-10 13:27:44Z vboxsync $ */
2/** @file
3 * VBox audio devices: Pulse Audio 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/*********************************************************************************************************************************
19* Header Files *
20*********************************************************************************************************************************/
21#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
22#include <VBox/log.h>
23
24#include <stdio.h>
25
26#include <iprt/alloc.h>
27#include <iprt/mem.h>
28#include <iprt/uuid.h>
29
30RT_C_DECLS_BEGIN
31 #include "pulse_mangling.h"
32 #include "pulse_stubs.h"
33RT_C_DECLS_END
34
35#include <pulse/pulseaudio.h>
36
37#include "DrvAudio.h"
38#include "AudioMixBuffer.h"
39
40#include "VBoxDD.h"
41
42/*********************************************************************************************************************************
43* Defines *
44*********************************************************************************************************************************/
45#define VBOX_PULSEAUDIO_MAX_LOG_REL_ERRORS 32 /** @todo Make this configurable thru driver options. */
46
47#ifndef PA_STREAM_NOFLAGS
48# define PA_STREAM_NOFLAGS (pa_context_flags_t)0x0000U /* since 0.9.19 */
49#endif
50
51#ifndef PA_CONTEXT_NOFLAGS
52# define PA_CONTEXT_NOFLAGS (pa_context_flags_t)0x0000U /* since 0.9.19 */
53#endif
54
55/** No flags specified. */
56#define PULSEAUDIOENUMCBFLAGS_NONE 0
57/** (Release) log found devices. */
58#define PULSEAUDIOENUMCBFLAGS_LOG RT_BIT(0)
59
60/** Makes DRVHOSTPULSEAUDIO out of PDMIHOSTAUDIO. */
61#define PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface) \
62 ( (PDRVHOSTPULSEAUDIO)((uintptr_t)pInterface - RT_OFFSETOF(DRVHOSTPULSEAUDIO, IHostAudio)) )
63
64/*********************************************************************************************************************************
65* Structures *
66*********************************************************************************************************************************/
67
68/**
69 * Host Pulse audio driver instance data.
70 * @implements PDMIAUDIOCONNECTOR
71 */
72typedef struct DRVHOSTPULSEAUDIO
73{
74 /** Pointer to the driver instance structure. */
75 PPDMDRVINS pDrvIns;
76 /** Pointer to PulseAudio's threaded main loop. */
77 pa_threaded_mainloop *pMainLoop;
78 /**
79 * Pointer to our PulseAudio context.
80 * Note: We use a pMainLoop in a separate thread (pContext).
81 * So either use callback functions or protect these functions
82 * by pa_threaded_mainloop_lock() / pa_threaded_mainloop_unlock().
83 */
84 pa_context *pContext;
85 /** Shutdown indicator. */
86 bool fLoopWait;
87 /** Pointer to host audio interface. */
88 PDMIHOSTAUDIO IHostAudio;
89 /** Error count for not flooding the release log.
90 * Specify UINT32_MAX for unlimited logging. */
91 uint32_t cLogErrors;
92 /** Configuration option: stream name. Optional. */
93 char *pszStreamName;
94} DRVHOSTPULSEAUDIO, *PDRVHOSTPULSEAUDIO;
95
96typedef struct PULSEAUDIOSTREAM
97{
98 /** Must come first, as this struct might be
99 * casted to one of these structs. */
100 union
101 {
102 PDMAUDIOHSTSTRMIN In;
103 PDMAUDIOHSTSTRMOUT Out;
104 };
105 /** Pointer to driver instance. */
106 PDRVHOSTPULSEAUDIO pDrv;
107 /** DAC/ADC buffer. */
108 void *pvPCMBuf;
109 /** Size (in bytes) of DAC/ADC buffer. */
110 uint32_t cbPCMBuf;
111 /** Pointer to opaque PulseAudio stream. */
112 pa_stream *pStream;
113 /** Pulse sample format and attribute specification. */
114 pa_sample_spec SampleSpec;
115 /** Pulse playback and buffer metrics. */
116 pa_buffer_attr BufAttr;
117 int fOpSuccess;
118 /** Pointer to Pulse sample peeking buffer. */
119 const uint8_t *pu8PeekBuf;
120 /** Current size (in bytes) of peeking data in
121 * buffer. */
122 size_t cbPeekBuf;
123 /** Our offset (in bytes) in peeking buffer. */
124 size_t offPeekBuf;
125 pa_operation *pDrainOp;
126} PULSEAUDIOSTREAM, *PPULSEAUDIOSTREAM;
127
128/* The desired buffer length in milliseconds. Will be the target total stream
129 * latency on newer version of pulse. Apparent latency can be less (or more.)
130 */
131typedef struct PULSEAUDIOCFG
132{
133 RTMSINTERVAL buffer_msecs_out;
134 RTMSINTERVAL buffer_msecs_in;
135} PULSEAUDIOCFG, *PPULSEAUDIOCFG;
136
137static PULSEAUDIOCFG s_pulseCfg =
138{
139 100, /* buffer_msecs_out */
140 100 /* buffer_msecs_in */
141};
142
143/**
144 * Callback context for server enumeration callbacks.
145 */
146typedef struct PULSEAUDIOENUMCBCTX
147{
148 /** Pointer to host backend driver. */
149 PDRVHOSTPULSEAUDIO pDrv;
150 /** Enumeration flags. */
151 uint32_t fFlags;
152 /** Number of found input devices. */
153 uint8_t cDevIn;
154 /** Number of found output devices. */
155 uint8_t cDevOut;
156 /** Name of default sink being used. Must be free'd using RTStrFree(). */
157 char *pszDefaultSink;
158 /** Name of default source being used. Must be free'd using RTStrFree(). */
159 char *pszDefaultSource;
160} PULSEAUDIOENUMCBCTX, *PPULSEAUDIOENUMCBCTX;
161
162/*********************************************************************************************************************************
163* Prototypes *
164*********************************************************************************************************************************/
165
166static int paEnumerate(PDRVHOSTPULSEAUDIO pThis, PPDMAUDIOBACKENDCFG pCfg, uint32_t fEnum);
167static int paError(PDRVHOSTPULSEAUDIO pThis, const char *szMsg);
168static void paStreamCbSuccess(pa_stream *pStream, int fSuccess, void *pvContext);
169
170/**
171 * Signal the main loop to abort. Just signalling isn't sufficient as the
172 * mainloop might not have been entered yet.
173 */
174static void paSignalWaiter(PDRVHOSTPULSEAUDIO pThis)
175{
176 pThis->fLoopWait = true;
177 pa_threaded_mainloop_signal(pThis->pMainLoop, 0);
178}
179
180static pa_sample_format_t paFmtToPulse(PDMAUDIOFMT fmt)
181{
182 switch (fmt)
183 {
184 case PDMAUDIOFMT_U8:
185 return PA_SAMPLE_U8;
186
187 case PDMAUDIOFMT_S16:
188 return PA_SAMPLE_S16LE;
189
190#ifdef PA_SAMPLE_S32LE
191 case PDMAUDIOFMT_S32:
192 return PA_SAMPLE_S32LE;
193#endif
194 default:
195 break;
196 }
197
198 AssertMsgFailed(("Format %ld not supported\n", fmt));
199 return PA_SAMPLE_U8;
200}
201
202static int paPulseToFmt(pa_sample_format_t pulsefmt,
203 PDMAUDIOFMT *pFmt, PDMAUDIOENDIANNESS *pEndianness)
204{
205 switch (pulsefmt)
206 {
207 case PA_SAMPLE_U8:
208 *pFmt = PDMAUDIOFMT_U8;
209 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
210 break;
211
212 case PA_SAMPLE_S16LE:
213 *pFmt = PDMAUDIOFMT_S16;
214 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
215 break;
216
217 case PA_SAMPLE_S16BE:
218 *pFmt = PDMAUDIOFMT_S16;
219 *pEndianness = PDMAUDIOENDIANNESS_BIG;
220 break;
221
222#ifdef PA_SAMPLE_S32LE
223 case PA_SAMPLE_S32LE:
224 *pFmt = PDMAUDIOFMT_S32;
225 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
226 break;
227#endif
228
229#ifdef PA_SAMPLE_S32BE
230 case PA_SAMPLE_S32BE:
231 *pFmt = PDMAUDIOFMT_S32;
232 *pEndianness = PDMAUDIOENDIANNESS_BIG;
233 break;
234#endif
235
236 default:
237 AssertMsgFailed(("Format %ld not supported\n", pulsefmt));
238 return VERR_NOT_SUPPORTED;
239 }
240
241 return VINF_SUCCESS;
242}
243
244/**
245 * Synchronously wait until an operation completed.
246 */
247static int paWaitForEx(PDRVHOSTPULSEAUDIO pThis, pa_operation *pOP, RTMSINTERVAL cMsTimeout)
248{
249 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
250 AssertPtrReturn(pOP, VERR_INVALID_POINTER);
251
252 int rc = VINF_SUCCESS;
253
254 uint64_t u64StartMs = RTTimeMilliTS();
255 while (pa_operation_get_state(pOP) == PA_OPERATION_RUNNING)
256 {
257 if (!pThis->fLoopWait)
258 {
259 AssertPtr(pThis->pMainLoop);
260 pa_threaded_mainloop_wait(pThis->pMainLoop);
261 }
262 pThis->fLoopWait = false;
263
264 uint64_t u64ElapsedMs = RTTimeMilliTS() - u64StartMs;
265 if (u64ElapsedMs >= cMsTimeout)
266 {
267 rc = VERR_TIMEOUT;
268 break;
269 }
270 }
271
272 pa_operation_unref(pOP);
273
274 return rc;
275}
276
277static int paWaitFor(PDRVHOSTPULSEAUDIO pThis, pa_operation *pOP)
278{
279 return paWaitForEx(pThis, pOP, 10 * 1000 /* 10s timeout */);
280}
281
282/**
283 * Context status changed.
284 */
285static void paContextCbStateChanged(pa_context *pCtx, void *pvUser)
286{
287 AssertPtrReturnVoid(pCtx);
288
289 PDRVHOSTPULSEAUDIO pThis = (PDRVHOSTPULSEAUDIO)pvUser;
290 AssertPtrReturnVoid(pThis);
291
292 switch (pa_context_get_state(pCtx))
293 {
294 case PA_CONTEXT_READY:
295 case PA_CONTEXT_TERMINATED:
296 paSignalWaiter(pThis);
297 break;
298
299 case PA_CONTEXT_FAILED:
300 LogRel(("PulseAudio: Audio context has failed, stopping\n"));
301 paSignalWaiter(pThis);
302 break;
303
304 default:
305 break;
306 }
307}
308
309/**
310 * Callback called when our pa_stream_drain operation was completed.
311 */
312static void paStreamCbDrain(pa_stream *pStream, int fSuccess, void *pvUser)
313{
314 AssertPtrReturnVoid(pStream);
315
316 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pvUser;
317 AssertPtrReturnVoid(pStrm);
318
319 pStrm->fOpSuccess = fSuccess;
320 if (fSuccess)
321 {
322 pa_operation_unref(pa_stream_cork(pStream, 1,
323 paStreamCbSuccess, pvUser));
324 }
325 else
326 paError(pStrm->pDrv, "Failed to drain stream");
327
328 pa_operation_unref(pStrm->pDrainOp);
329 pStrm->pDrainOp = NULL;
330}
331
332/**
333 * Stream status changed.
334 */
335static void paStreamCbStateChanged(pa_stream *pStream, void *pvUser)
336{
337 AssertPtrReturnVoid(pStream);
338
339 PDRVHOSTPULSEAUDIO pThis = (PDRVHOSTPULSEAUDIO)pvUser;
340 AssertPtrReturnVoid(pThis);
341
342 switch (pa_stream_get_state(pStream))
343 {
344 case PA_STREAM_READY:
345 case PA_STREAM_FAILED:
346 case PA_STREAM_TERMINATED:
347 paSignalWaiter(pThis);
348 break;
349
350 default:
351 break;
352 }
353}
354
355static void paStreamCbSuccess(pa_stream *pStream, int fSuccess, void *pvUser)
356{
357 AssertPtrReturnVoid(pStream);
358
359 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pvUser;
360 AssertPtrReturnVoid(pStrm);
361
362 pStrm->fOpSuccess = fSuccess;
363
364 if (fSuccess)
365 paSignalWaiter(pStrm->pDrv);
366 else
367 paError(pStrm->pDrv, "Failed to finish stream operation");
368}
369
370static int paStreamOpen(PDRVHOSTPULSEAUDIO pThis, bool fIn, const char *pszName,
371 pa_sample_spec *pSampleSpec, pa_buffer_attr *pBufAttr,
372 pa_stream **ppStream)
373{
374 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
375 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
376 AssertPtrReturn(pSampleSpec, VERR_INVALID_POINTER);
377 AssertPtrReturn(pBufAttr, VERR_INVALID_POINTER);
378 AssertPtrReturn(ppStream, VERR_INVALID_POINTER);
379
380 if (!pa_sample_spec_valid(pSampleSpec))
381 {
382 LogRel(("PulseAudio: Unsupported sample specification for stream \"%s\"\n",
383 pszName));
384 return VERR_NOT_SUPPORTED;
385 }
386
387 int rc = VINF_SUCCESS;
388
389 pa_stream *pStream = NULL;
390 uint32_t flags = PA_STREAM_NOFLAGS;
391
392 LogFunc(("Opening \"%s\", rate=%dHz, channels=%d, format=%s\n",
393 pszName, pSampleSpec->rate, pSampleSpec->channels,
394 pa_sample_format_to_string(pSampleSpec->format)));
395
396 pa_threaded_mainloop_lock(pThis->pMainLoop);
397
398 do
399 {
400 if (!(pStream = pa_stream_new(pThis->pContext, pszName, pSampleSpec,
401 NULL /* pa_channel_map */)))
402 {
403 LogRel(("PulseAudio: Could not create stream \"%s\"\n", pszName));
404 rc = VERR_NO_MEMORY;
405 break;
406 }
407
408 pa_stream_set_state_callback(pStream, paStreamCbStateChanged, pThis);
409
410#if PA_API_VERSION >= 12
411 /* XXX */
412 flags |= PA_STREAM_ADJUST_LATENCY;
413#endif
414
415#if 0
416 /* Not applicable as we don't use pa_stream_get_latency() and pa_stream_get_time(). */
417 flags |= PA_STREAM_INTERPOLATE_TIMING | PA_STREAM_AUTO_TIMING_UPDATE;
418#endif
419 /* No input/output right away after the stream was started. */
420 flags |= PA_STREAM_START_CORKED;
421
422 if (fIn)
423 {
424 LogFunc(("Input stream attributes: maxlength=%d fragsize=%d\n",
425 pBufAttr->maxlength, pBufAttr->fragsize));
426
427 if (pa_stream_connect_record(pStream, /*dev=*/NULL, pBufAttr, (pa_stream_flags_t)flags) < 0)
428 {
429 LogRel(("PulseAudio: Could not connect input stream \"%s\": %s\n",
430 pszName, pa_strerror(pa_context_errno(pThis->pContext))));
431 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
432 break;
433 }
434 }
435 else
436 {
437 LogFunc(("Output buffer attributes: maxlength=%d tlength=%d prebuf=%d minreq=%d\n",
438 pBufAttr->maxlength, pBufAttr->tlength, pBufAttr->prebuf, pBufAttr->minreq));
439
440 if (pa_stream_connect_playback(pStream, /*dev=*/NULL, pBufAttr, (pa_stream_flags_t)flags,
441 /*cvolume=*/NULL, /*sync_stream=*/NULL) < 0)
442 {
443 LogRel(("PulseAudio: Could not connect playback stream \"%s\": %s\n",
444 pszName, pa_strerror(pa_context_errno(pThis->pContext))));
445 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
446 break;
447 }
448 }
449
450 /* Wait until the stream is ready. */
451 for (;;)
452 {
453 if (!pThis->fLoopWait)
454 pa_threaded_mainloop_wait(pThis->pMainLoop);
455 pThis->fLoopWait = false;
456
457 pa_stream_state_t streamSt = pa_stream_get_state(pStream);
458 if (streamSt == PA_STREAM_READY)
459 break;
460 else if ( streamSt == PA_STREAM_FAILED
461 || streamSt == PA_STREAM_TERMINATED)
462 {
463 LogRel(("PulseAudio: Failed to initialize stream \"%s\" (state %ld)\n", pszName, streamSt));
464 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
465 break;
466 }
467 }
468
469 if (RT_FAILURE(rc))
470 break;
471
472 const pa_buffer_attr *pBufAttrObtained = pa_stream_get_buffer_attr(pStream);
473 AssertPtr(pBufAttrObtained);
474 memcpy(pBufAttr, pBufAttrObtained, sizeof(pa_buffer_attr));
475
476 if (fIn)
477 LogFunc(("Obtained record buffer attributes: maxlength=%RU32, fragsize=%RU32\n",
478 pBufAttr->maxlength, pBufAttr->fragsize));
479 else
480 LogFunc(("Obtained playback buffer attributes: maxlength=%d, tlength=%d, prebuf=%d, minreq=%d\n",
481 pBufAttr->maxlength, pBufAttr->tlength, pBufAttr->prebuf, pBufAttr->minreq));
482
483 }
484 while (0);
485
486 if ( RT_FAILURE(rc)
487 && pStream)
488 pa_stream_disconnect(pStream);
489
490 pa_threaded_mainloop_unlock(pThis->pMainLoop);
491
492 if (RT_FAILURE(rc))
493 {
494 if (pStream)
495 pa_stream_unref(pStream);
496 }
497 else
498 *ppStream = pStream;
499
500 LogFlowFuncLeaveRC(rc);
501 return rc;
502}
503
504static DECLCALLBACK(int) drvHostPulseAudioInit(PPDMIHOSTAUDIO pInterface)
505{
506 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
507
508 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
509
510 LogFlowFuncEnter();
511
512 int rc = audioLoadPulseLib();
513 if (RT_FAILURE(rc))
514 {
515 LogRel(("PulseAudio: Failed to load the PulseAudio shared library! Error %Rrc\n", rc));
516 return rc;
517 }
518
519 pThis->fLoopWait = false;
520 pThis->pMainLoop = NULL;
521
522 bool fLocked = false;
523
524 do
525 {
526 if (!(pThis->pMainLoop = pa_threaded_mainloop_new()))
527 {
528 LogRel(("PulseAudio: Failed to allocate main loop: %s\n",
529 pa_strerror(pa_context_errno(pThis->pContext))));
530 rc = VERR_NO_MEMORY;
531 break;
532 }
533
534 if (!(pThis->pContext = pa_context_new(pa_threaded_mainloop_get_api(pThis->pMainLoop), "VirtualBox")))
535 {
536 LogRel(("PulseAudio: Failed to allocate context: %s\n",
537 pa_strerror(pa_context_errno(pThis->pContext))));
538 rc = VERR_NO_MEMORY;
539 break;
540 }
541
542 if (pa_threaded_mainloop_start(pThis->pMainLoop) < 0)
543 {
544 LogRel(("PulseAudio: Failed to start threaded mainloop: %s\n",
545 pa_strerror(pa_context_errno(pThis->pContext))));
546 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
547 break;
548 }
549
550 /* Install a global callback to known if something happens to our acquired context. */
551 pa_context_set_state_callback(pThis->pContext, paContextCbStateChanged, pThis /* pvUserData */);
552
553 pa_threaded_mainloop_lock(pThis->pMainLoop);
554 fLocked = true;
555
556 if (pa_context_connect(pThis->pContext, NULL /* pszServer */,
557 PA_CONTEXT_NOFLAGS, NULL) < 0)
558 {
559 LogRel(("PulseAudio: Failed to connect to server: %s\n",
560 pa_strerror(pa_context_errno(pThis->pContext))));
561 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
562 break;
563 }
564
565 /* Wait until the pThis->pContext is ready. */
566 for (;;)
567 {
568 if (!pThis->fLoopWait)
569 pa_threaded_mainloop_wait(pThis->pMainLoop);
570 pThis->fLoopWait = false;
571
572 pa_context_state_t cstate = pa_context_get_state(pThis->pContext);
573 if (cstate == PA_CONTEXT_READY)
574 break;
575 else if ( cstate == PA_CONTEXT_TERMINATED
576 || cstate == PA_CONTEXT_FAILED)
577 {
578 LogRel(("PulseAudio: Failed to initialize context (state %d)\n", cstate));
579 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
580 break;
581 }
582 }
583 }
584 while (0);
585
586 if (fLocked)
587 pa_threaded_mainloop_unlock(pThis->pMainLoop);
588
589 if (RT_FAILURE(rc))
590 {
591 if (pThis->pMainLoop)
592 pa_threaded_mainloop_stop(pThis->pMainLoop);
593
594 if (pThis->pContext)
595 {
596 pa_context_disconnect(pThis->pContext);
597 pa_context_unref(pThis->pContext);
598 pThis->pContext = NULL;
599 }
600
601 if (pThis->pMainLoop)
602 {
603 pa_threaded_mainloop_free(pThis->pMainLoop);
604 pThis->pMainLoop = NULL;
605 }
606 }
607
608 LogFlowFuncLeaveRC(rc);
609 return rc;
610}
611
612static DECLCALLBACK(int) drvHostPulseAudioInitOut(PPDMIHOSTAUDIO pInterface,
613 PPDMAUDIOHSTSTRMOUT pHstStrmOut, PPDMAUDIOSTREAMCFG pCfg,
614 uint32_t *pcSamples)
615{
616 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
617 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
618 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
619 /* pcSamples is optional. */
620
621 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
622 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmOut;
623
624 LogFlowFuncEnter();
625
626 pStrm->pDrainOp = NULL;
627
628 pStrm->SampleSpec.format = paFmtToPulse(pCfg->enmFormat);
629 pStrm->SampleSpec.rate = pCfg->uHz;
630 pStrm->SampleSpec.channels = pCfg->cChannels;
631
632 /* Note that setting maxlength to -1 does not work on PulseAudio servers
633 * older than 0.9.10. So use the suggested value of 3/2 of tlength */
634 pStrm->BufAttr.tlength = (pa_bytes_per_second(&pStrm->SampleSpec)
635 * s_pulseCfg.buffer_msecs_out) / 1000;
636 pStrm->BufAttr.maxlength = (pStrm->BufAttr.tlength * 3) / 2;
637 pStrm->BufAttr.prebuf = -1; /* Same as tlength */
638 pStrm->BufAttr.minreq = -1; /* Pulse should set something sensible for minreq on it's own */
639
640 /* Note that the struct BufAttr is updated to the obtained values after this call! */
641 char achName[64];
642 RTStrPrintf(achName, sizeof(achName), "%.32s (out)", pThis->pszStreamName);
643
644 int rc = paStreamOpen(pThis, false /* fIn */, achName, &pStrm->SampleSpec, &pStrm->BufAttr, &pStrm->pStream);
645 if (RT_FAILURE(rc))
646 return rc;
647
648 PDMAUDIOSTREAMCFG streamCfg;
649 rc = paPulseToFmt(pStrm->SampleSpec.format,
650 &streamCfg.enmFormat, &streamCfg.enmEndianness);
651 if (RT_FAILURE(rc))
652 {
653 LogRel(("PulseAudio: Cannot find audio output format %ld\n", pStrm->SampleSpec.format));
654 return rc;
655 }
656
657 streamCfg.uHz = pStrm->SampleSpec.rate;
658 streamCfg.cChannels = pStrm->SampleSpec.channels;
659
660 rc = DrvAudioStreamCfgToProps(&streamCfg, &pHstStrmOut->Props);
661 if (RT_SUCCESS(rc))
662 {
663 uint32_t cbBuf = RT_MIN(pStrm->BufAttr.tlength * 2,
664 pStrm->BufAttr.maxlength); /** @todo Make this configurable! */
665 if (cbBuf)
666 {
667 pStrm->pvPCMBuf = RTMemAllocZ(cbBuf);
668 if (pStrm->pvPCMBuf)
669 {
670 pStrm->cbPCMBuf = cbBuf;
671
672 uint32_t cSamples = cbBuf >> pHstStrmOut->Props.cShift;
673 if (pcSamples)
674 *pcSamples = cSamples;
675
676 /* Save pointer to driver instance. */
677 pStrm->pDrv = pThis;
678
679 LogFunc(("cbBuf=%RU32, cSamples=%RU32\n", cbBuf, cSamples));
680 }
681 else
682 rc = VERR_NO_MEMORY;
683 }
684 else
685 rc = VERR_INVALID_PARAMETER;
686 }
687
688 LogFlowFuncLeaveRC(rc);
689 return rc;
690}
691
692static DECLCALLBACK(bool) drvHostPulseAudioIsEnabled(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
693{
694 NOREF(pInterface);
695 NOREF(enmDir);
696 return true; /* Always all enabled. */
697}
698
699static DECLCALLBACK(int) drvHostPulseAudioInitIn(PPDMIHOSTAUDIO pInterface,
700 PPDMAUDIOHSTSTRMIN pHstStrmIn, PPDMAUDIOSTREAMCFG pCfg,
701 PDMAUDIORECSOURCE enmRecSource,
702 uint32_t *pcSamples)
703{
704 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
705 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
706 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
707 /* pcSamples is optional. */
708
709 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
710 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmIn;
711
712 LogFunc(("enmRecSrc=%ld\n", enmRecSource));
713
714 pStrm->SampleSpec.format = paFmtToPulse(pCfg->enmFormat);
715 pStrm->SampleSpec.rate = pCfg->uHz;
716 pStrm->SampleSpec.channels = pCfg->cChannels;
717
718 /* XXX check these values */
719 pStrm->BufAttr.fragsize = (pa_bytes_per_second(&pStrm->SampleSpec)
720 * s_pulseCfg.buffer_msecs_in) / 1000;
721 pStrm->BufAttr.maxlength = (pStrm->BufAttr.fragsize * 3) / 2;
722 /* Note: Other members of pa_buffer_attr are ignored for record streams. */
723
724 char achName[64];
725 RTStrPrintf(achName, sizeof(achName), "%.32s (in)", pThis->pszStreamName);
726
727 int rc = paStreamOpen(pThis, true /* fIn */, achName, &pStrm->SampleSpec, &pStrm->BufAttr,
728 &pStrm->pStream);
729 if (RT_FAILURE(rc))
730 return rc;
731
732 PDMAUDIOSTREAMCFG streamCfg;
733 rc = paPulseToFmt(pStrm->SampleSpec.format, &streamCfg.enmFormat,
734 &streamCfg.enmEndianness);
735 if (RT_FAILURE(rc))
736 {
737 LogRel(("PulseAudio: Cannot find audio capture format %ld\n", pStrm->SampleSpec.format));
738 return rc;
739 }
740
741 streamCfg.uHz = pStrm->SampleSpec.rate;
742 streamCfg.cChannels = pStrm->SampleSpec.channels;
743
744 rc = DrvAudioStreamCfgToProps(&streamCfg, &pHstStrmIn->Props);
745 if (RT_SUCCESS(rc))
746 {
747 uint32_t cSamples = RT_MIN(pStrm->BufAttr.fragsize * 10, pStrm->BufAttr.maxlength)
748 >> pHstStrmIn->Props.cShift;
749 LogFunc(("cShift=%RU8, cSamples=%RU32\n", pHstStrmIn->Props.cShift, cSamples));
750
751 if (pcSamples)
752 *pcSamples = cSamples;
753
754 /* Save pointer to driver instance. */
755 pStrm->pDrv = pThis;
756
757 pStrm->pu8PeekBuf = NULL;
758 }
759
760 LogFlowFuncLeaveRC(rc);
761 return rc;
762}
763
764static DECLCALLBACK(int) drvHostPulseAudioCaptureIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMIN pHstStrmIn,
765 uint32_t *pcSamplesCaptured)
766{
767 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
768 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
769 /* pcSamplesPlayed is optional. */
770
771 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
772 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmIn;
773
774 /* We should only call pa_stream_readable_size() once and trust the first value. */
775 pa_threaded_mainloop_lock(pThis->pMainLoop);
776 size_t cbAvail = pa_stream_readable_size(pStrm->pStream);
777 pa_threaded_mainloop_unlock(pThis->pMainLoop);
778
779 if (cbAvail == (size_t)-1)
780 return paError(pStrm->pDrv, "Failed to determine input data size");
781
782 /* If the buffer was not dropped last call, add what remains. */
783 if (pStrm->pu8PeekBuf)
784 {
785 Assert(pStrm->cbPeekBuf >= pStrm->offPeekBuf);
786 cbAvail += (pStrm->cbPeekBuf - pStrm->offPeekBuf);
787 }
788
789 if (!cbAvail) /* No data? Bail out. */
790 {
791 if (pcSamplesCaptured)
792 *pcSamplesCaptured = 0;
793 return VINF_SUCCESS;
794 }
795
796 int rc = VINF_SUCCESS;
797
798 size_t cbToRead = RT_MIN(cbAvail, AudioMixBufFreeBytes(&pHstStrmIn->MixBuf));
799
800 LogFlowFunc(("cbToRead=%zu, cbAvail=%zu, offPeekBuf=%zu, cbPeekBuf=%zu\n",
801 cbToRead, cbAvail, pStrm->offPeekBuf, pStrm->cbPeekBuf));
802
803 size_t offWrite = 0;
804 uint32_t cWrittenTotal = 0;
805
806 while (cbToRead)
807 {
808 /* If there is no data, do another peek. */
809 if (!pStrm->pu8PeekBuf)
810 {
811 pa_threaded_mainloop_lock(pThis->pMainLoop);
812 pa_stream_peek(pStrm->pStream,
813 (const void**)&pStrm->pu8PeekBuf, &pStrm->cbPeekBuf);
814 pa_threaded_mainloop_unlock(pThis->pMainLoop);
815
816 pStrm->offPeekBuf = 0;
817
818 /* No data anymore?
819 * Note: If there's a data hole (cbPeekBuf then contains the length of the hole)
820 * we need to drop the stream lateron. */
821 if ( !pStrm->pu8PeekBuf
822 && !pStrm->cbPeekBuf)
823 {
824 break;
825 }
826 }
827
828 Assert(pStrm->cbPeekBuf >= pStrm->offPeekBuf);
829 size_t cbToWrite = RT_MIN(pStrm->cbPeekBuf - pStrm->offPeekBuf, cbToRead);
830
831 LogFlowFunc(("cbToRead=%zu, cbToWrite=%zu, offPeekBuf=%zu, cbPeekBuf=%zu, pu8PeekBuf=%p\n",
832 cbToRead, cbToWrite,
833 pStrm->offPeekBuf, pStrm->cbPeekBuf, pStrm->pu8PeekBuf));
834
835 if (cbToWrite)
836 {
837 uint32_t cWritten;
838 rc = AudioMixBufWriteCirc(&pHstStrmIn->MixBuf,
839 pStrm->pu8PeekBuf + pStrm->offPeekBuf,
840 cbToWrite, &cWritten);
841 if (RT_FAILURE(rc))
842 break;
843
844 uint32_t cbWritten = AUDIOMIXBUF_S2B(&pHstStrmIn->MixBuf, cWritten);
845
846 Assert(cbToRead >= cbWritten);
847 cbToRead -= cbWritten;
848 cWrittenTotal += cWritten;
849 pStrm->offPeekBuf += cbWritten;
850 }
851
852 if (/* Nothing to write anymore? Drop the buffer. */
853 !cbToWrite
854 /* Was there a hole in the peeking buffer? Drop it. */
855 || !pStrm->pu8PeekBuf
856 /* If the buffer is done, drop it. */
857 || pStrm->offPeekBuf == pStrm->cbPeekBuf)
858 {
859 pa_threaded_mainloop_lock(pThis->pMainLoop);
860 pa_stream_drop(pStrm->pStream);
861 pa_threaded_mainloop_unlock(pThis->pMainLoop);
862
863 pStrm->pu8PeekBuf = NULL;
864 }
865 }
866
867 if (RT_SUCCESS(rc))
868 {
869 uint32_t cProcessed = 0;
870 if (cWrittenTotal)
871 rc = AudioMixBufMixToParent(&pHstStrmIn->MixBuf, cWrittenTotal,
872 &cProcessed);
873
874 if (pcSamplesCaptured)
875 *pcSamplesCaptured = cWrittenTotal;
876
877 LogFlowFunc(("cWrittenTotal=%RU32 (%RU32 processed), rc=%Rrc\n",
878 cWrittenTotal, cProcessed, rc));
879 }
880
881 LogFlowFuncLeaveRC(rc);
882 return rc;
883}
884
885static DECLCALLBACK(int) drvHostPulseAudioPlayOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMOUT pHstStrmOut,
886 uint32_t *pcSamplesPlayed)
887{
888 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
889 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
890 /* pcSamplesPlayed is optional. */
891
892 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
893 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmOut;
894
895 int rc = VINF_SUCCESS;
896 uint32_t cbReadTotal = 0;
897
898 uint32_t cLive = AudioMixBufAvail(&pHstStrmOut->MixBuf);
899 if (!cLive)
900 {
901#if 0
902 LogFlowFunc(("No live samples, skipping\n"));
903#endif
904 if (pcSamplesPlayed)
905 *pcSamplesPlayed = 0;
906 return VINF_SUCCESS;
907 }
908
909 pa_threaded_mainloop_lock(pThis->pMainLoop);
910
911 do
912 {
913 size_t cbWriteable = pa_stream_writable_size(pStrm->pStream);
914 if (cbWriteable == (size_t)-1)
915 {
916 rc = paError(pStrm->pDrv, "Failed to determine output data size");
917 break;
918 }
919
920 size_t cbLive = AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf, cLive);
921 size_t cbToRead = RT_MIN(cbWriteable, cbLive);
922
923 LogFlowFunc(("cbToRead=%zu, cbWriteable=%zu, cbLive=%zu\n",
924 cbToRead, cbWriteable, cbLive));
925
926 uint32_t cRead, cbRead;
927 while (cbToRead)
928 {
929 rc = AudioMixBufReadCirc(&pHstStrmOut->MixBuf, pStrm->pvPCMBuf,
930 RT_MIN(cbToRead, pStrm->cbPCMBuf), &cRead);
931 if ( !cRead
932 || RT_FAILURE(rc))
933 {
934 break;
935 }
936
937 cbRead = AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf, cRead);
938 if (pa_stream_write(pStrm->pStream, pStrm->pvPCMBuf, cbRead, NULL /* Cleanup callback */,
939 0, PA_SEEK_RELATIVE) < 0)
940 {
941 rc = paError(pStrm->pDrv, "Failed to write to output stream");
942 break;
943 }
944
945 Assert(cbToRead >= cbRead);
946 cbToRead -= cbRead;
947 cbReadTotal += cbRead;
948
949 LogFlowFunc(("\tcRead=%RU32 (%zu bytes) cbReadTotal=%RU32, cbToRead=%RU32\n",
950 cRead, AUDIOMIXBUF_S2B(&pHstStrmOut->MixBuf, cRead), cbReadTotal, cbToRead));
951 }
952
953 } while (0);
954
955 pa_threaded_mainloop_unlock(pThis->pMainLoop);
956
957 if (RT_SUCCESS(rc))
958 {
959 uint32_t cReadTotal = AUDIOMIXBUF_B2S(&pHstStrmOut->MixBuf, cbReadTotal);
960 if (cReadTotal)
961 AudioMixBufFinish(&pHstStrmOut->MixBuf, cReadTotal);
962
963 if (pcSamplesPlayed)
964 *pcSamplesPlayed = cReadTotal;
965
966 LogFlowFunc(("cReadTotal=%RU32 (%RU32 bytes), rc=%Rrc\n", cReadTotal, cbReadTotal, rc));
967 }
968
969 LogFlowFuncLeaveRC(rc);
970 return rc;
971}
972
973/** @todo Implement va handling. */
974static int paError(PDRVHOSTPULSEAUDIO pThis, const char *szMsg)
975{
976 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
977 AssertPtrReturn(szMsg, VERR_INVALID_POINTER);
978
979 if (pThis->cLogErrors++ < VBOX_PULSEAUDIO_MAX_LOG_REL_ERRORS)
980 {
981 int rc2 = pa_context_errno(pThis->pContext);
982 LogRel(("PulseAudio: %s: %s\n", szMsg, pa_strerror(rc2)));
983 }
984
985 /** @todo Implement some PulseAudio -> IPRT mapping here. */
986 return VERR_GENERAL_FAILURE;
987}
988
989static void paEnumSinkCb(pa_context *pCtx, const pa_sink_info *pInfo, int eol, void *pvUserData)
990{
991 if (eol != 0)
992 return;
993
994 AssertPtrReturnVoid(pCtx);
995 AssertPtrReturnVoid(pInfo);
996
997 PPULSEAUDIOENUMCBCTX pCbCtx = (PPULSEAUDIOENUMCBCTX)pvUserData;
998 AssertPtrReturnVoid(pCbCtx);
999 AssertPtrReturnVoid(pCbCtx->pDrv);
1000
1001 LogRel2(("PulseAudio: Using output sink '%s'\n", pInfo->name));
1002
1003 /** @todo Store sinks + channel mapping in callback context as soon as we have surround support. */
1004 pCbCtx->cDevOut++;
1005
1006 pa_threaded_mainloop_signal(pCbCtx->pDrv->pMainLoop, 0);
1007}
1008
1009static void paEnumSourceCb(pa_context *pCtx, const pa_source_info *pInfo, int eol, void *pvUserData)
1010{
1011 if (eol != 0)
1012 return;
1013
1014 AssertPtrReturnVoid(pCtx);
1015 AssertPtrReturnVoid(pInfo);
1016
1017 PPULSEAUDIOENUMCBCTX pCbCtx = (PPULSEAUDIOENUMCBCTX)pvUserData;
1018 AssertPtrReturnVoid(pCbCtx);
1019 AssertPtrReturnVoid(pCbCtx->pDrv);
1020
1021 LogRel2(("PulseAudio: Using input source '%s'\n", pInfo->name));
1022
1023 /** @todo Store sources + channel mapping in callback context as soon as we have surround support. */
1024 pCbCtx->cDevIn++;
1025
1026 pa_threaded_mainloop_signal(pCbCtx->pDrv->pMainLoop, 0);
1027}
1028
1029static void paEnumServerCb(pa_context *pCtx, const pa_server_info *pInfo, void *pvUserData)
1030{
1031 AssertPtrReturnVoid(pCtx);
1032 AssertPtrReturnVoid(pInfo);
1033
1034 PPULSEAUDIOENUMCBCTX pCbCtx = (PPULSEAUDIOENUMCBCTX)pvUserData;
1035 AssertPtrReturnVoid(pCbCtx);
1036
1037 PDRVHOSTPULSEAUDIO pThis = pCbCtx->pDrv;
1038 AssertPtrReturnVoid(pThis);
1039
1040 if (pInfo->default_sink_name)
1041 {
1042 Assert(RTStrIsValidEncoding(pInfo->default_sink_name));
1043 pCbCtx->pszDefaultSink = RTStrDup(pInfo->default_sink_name);
1044 }
1045
1046 if (pInfo->default_sink_name)
1047 {
1048 Assert(RTStrIsValidEncoding(pInfo->default_source_name));
1049 pCbCtx->pszDefaultSource = RTStrDup(pInfo->default_source_name);
1050 }
1051
1052 pa_threaded_mainloop_signal(pThis->pMainLoop, 0);
1053}
1054
1055static int paEnumerate(PDRVHOSTPULSEAUDIO pThis, PPDMAUDIOBACKENDCFG pCfg, uint32_t fEnum)
1056{
1057 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1058 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1059
1060 PDMAUDIOBACKENDCFG Cfg;
1061 RT_ZERO(Cfg);
1062
1063 Cfg.cbStreamOut = sizeof(PULSEAUDIOSTREAM);
1064 Cfg.cbStreamIn = sizeof(PULSEAUDIOSTREAM);
1065 Cfg.cMaxStreamsOut = UINT32_MAX;
1066 Cfg.cMaxStreamsIn = UINT32_MAX;
1067
1068 PULSEAUDIOENUMCBCTX cbCtx;
1069 RT_ZERO(cbCtx);
1070
1071 cbCtx.pDrv = pThis;
1072 cbCtx.fFlags = fEnum;
1073
1074 bool fLog = (fEnum & PULSEAUDIOENUMCBFLAGS_LOG);
1075
1076 int rc = paWaitFor(pThis, pa_context_get_server_info(pThis->pContext, paEnumServerCb, &cbCtx));
1077 if (RT_SUCCESS(rc))
1078 {
1079 if (cbCtx.pszDefaultSink)
1080 {
1081 if (fLog)
1082 LogRel2(("PulseAudio: Default output sink is '%s'\n", cbCtx.pszDefaultSink));
1083
1084 rc = paWaitFor(pThis, pa_context_get_sink_info_by_name(pThis->pContext, cbCtx.pszDefaultSink,
1085 paEnumSinkCb, &cbCtx));
1086 if ( RT_FAILURE(rc)
1087 && fLog)
1088 {
1089 LogRel(("PulseAudio: Error enumerating properties for default output sink '%s'\n", cbCtx.pszDefaultSink));
1090 }
1091 }
1092 else if (fLog)
1093 LogRel2(("PulseAudio: No default output sink found\n"));
1094
1095 if (RT_SUCCESS(rc))
1096 {
1097 if (cbCtx.pszDefaultSource)
1098 {
1099 if (fLog)
1100 LogRel2(("PulseAudio: Default input source is '%s'\n", cbCtx.pszDefaultSource));
1101
1102 rc = paWaitFor(pThis, pa_context_get_source_info_by_name(pThis->pContext, cbCtx.pszDefaultSource,
1103 paEnumSourceCb, &cbCtx));
1104 if ( RT_FAILURE(rc)
1105 && fLog)
1106 {
1107 LogRel(("PulseAudio: Error enumerating properties for default input source '%s'\n", cbCtx.pszDefaultSource));
1108 }
1109 }
1110 else if (fLog)
1111 LogRel2(("PulseAudio: No default input source found\n"));
1112 }
1113
1114 if (RT_SUCCESS(rc))
1115 {
1116 Cfg.cSinks = cbCtx.cDevOut;
1117 Cfg.cSources = cbCtx.cDevIn;
1118
1119 if (fLog)
1120 {
1121 LogRel2(("PulseAudio: Found %RU8 host playback device(s)\n", cbCtx.cDevOut));
1122 LogRel2(("PulseAudio: Found %RU8 host capturing device(s)\n", cbCtx.cDevIn));
1123 }
1124
1125 if (pCfg)
1126 memcpy(pCfg, &Cfg, sizeof(PDMAUDIOBACKENDCFG));
1127 }
1128
1129 if (cbCtx.pszDefaultSink)
1130 {
1131 RTStrFree(cbCtx.pszDefaultSink);
1132 cbCtx.pszDefaultSink = NULL;
1133 }
1134
1135 if (cbCtx.pszDefaultSource)
1136 {
1137 RTStrFree(cbCtx.pszDefaultSource);
1138 cbCtx.pszDefaultSource = NULL;
1139 }
1140 }
1141 else if (fLog)
1142 LogRel(("PulseAudio: Error enumerating PulseAudio server properties\n"));
1143
1144 LogFlowFuncLeaveRC(rc);
1145 return rc;
1146}
1147
1148static DECLCALLBACK(int) drvHostPulseAudioFiniIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMIN pHstStrmIn)
1149{
1150 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1151 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
1152
1153 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1154 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmIn;
1155
1156 LogFlowFuncEnter();
1157
1158 if (pStrm->pStream)
1159 {
1160 pa_threaded_mainloop_lock(pThis->pMainLoop);
1161 pa_stream_disconnect(pStrm->pStream);
1162 pa_stream_unref(pStrm->pStream);
1163 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1164
1165 pStrm->pStream = NULL;
1166 }
1167
1168 return VINF_SUCCESS;
1169}
1170
1171static DECLCALLBACK(int) drvHostPulseAudioFiniOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMOUT pHstStrmOut)
1172{
1173 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1174 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
1175
1176 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1177 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmOut;
1178
1179 LogFlowFuncEnter();
1180
1181 if (pStrm->pStream)
1182 {
1183 pa_threaded_mainloop_lock(pThis->pMainLoop);
1184 pa_stream_disconnect(pStrm->pStream);
1185 pa_stream_unref(pStrm->pStream);
1186 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1187
1188 pStrm->pStream = NULL;
1189 }
1190
1191 if (pStrm->pvPCMBuf)
1192 {
1193 RTMemFree(pStrm->pvPCMBuf);
1194 pStrm->pvPCMBuf = NULL;
1195 pStrm->cbPCMBuf = 0;
1196 }
1197
1198 return VINF_SUCCESS;
1199}
1200
1201static DECLCALLBACK(int) drvHostPulseAudioControlOut(PPDMIHOSTAUDIO pInterface,
1202 PPDMAUDIOHSTSTRMOUT pHstStrmOut, PDMAUDIOSTREAMCMD enmStreamCmd)
1203{
1204 AssertPtrReturn(pInterface , VERR_INVALID_POINTER);
1205 AssertPtrReturn(pHstStrmOut, VERR_INVALID_POINTER);
1206
1207 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1208 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmOut;
1209
1210 int rc = VINF_SUCCESS;
1211
1212 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
1213
1214 switch (enmStreamCmd)
1215 {
1216 case PDMAUDIOSTREAMCMD_ENABLE:
1217 case PDMAUDIOSTREAMCMD_RESUME:
1218 {
1219 pa_threaded_mainloop_lock(pThis->pMainLoop);
1220
1221 if ( pStrm->pDrainOp
1222 && pa_operation_get_state(pStrm->pDrainOp) != PA_OPERATION_DONE)
1223 {
1224 pa_operation_cancel(pStrm->pDrainOp);
1225 pa_operation_unref(pStrm->pDrainOp);
1226
1227 pStrm->pDrainOp = NULL;
1228 }
1229 else
1230 {
1231 rc = paWaitFor(pThis, pa_stream_cork(pStrm->pStream, 0, paStreamCbSuccess, pStrm));
1232 }
1233
1234 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1235 break;
1236 }
1237
1238 case PDMAUDIOSTREAMCMD_DISABLE:
1239 case PDMAUDIOSTREAMCMD_PAUSE:
1240 {
1241 /* Pause audio output (the Pause bit of the AC97 x_CR register is set).
1242 * Note that we must return immediately from here! */
1243 pa_threaded_mainloop_lock(pThis->pMainLoop);
1244 if (!pStrm->pDrainOp)
1245 {
1246 rc = paWaitFor(pThis, pa_stream_trigger(pStrm->pStream, paStreamCbSuccess, pStrm));
1247 if (RT_LIKELY(RT_SUCCESS(rc)))
1248 pStrm->pDrainOp = pa_stream_drain(pStrm->pStream, paStreamCbDrain, pStrm);
1249 }
1250 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1251 break;
1252 }
1253
1254 default:
1255 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1256 rc = VERR_INVALID_PARAMETER;
1257 break;
1258 }
1259
1260 LogFlowFuncLeaveRC(rc);
1261 return rc;
1262}
1263
1264static DECLCALLBACK(int) drvHostPulseAudioControlIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOHSTSTRMIN pHstStrmIn,
1265 PDMAUDIOSTREAMCMD enmStreamCmd)
1266{
1267 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1268 AssertPtrReturn(pHstStrmIn, VERR_INVALID_POINTER);
1269
1270 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1271 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pHstStrmIn;
1272
1273 int rc = VINF_SUCCESS;
1274
1275 LogFlowFunc(("enmStreamCmd=%ld\n", enmStreamCmd));
1276
1277 switch (enmStreamCmd)
1278 {
1279 case PDMAUDIOSTREAMCMD_ENABLE:
1280 case PDMAUDIOSTREAMCMD_RESUME:
1281 {
1282 pa_threaded_mainloop_lock(pThis->pMainLoop);
1283 rc = paWaitFor(pThis, pa_stream_cork(pStrm->pStream, 0 /* Play / resume */, paStreamCbSuccess, pStrm));
1284 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1285 break;
1286 }
1287
1288 case PDMAUDIOSTREAMCMD_DISABLE:
1289 case PDMAUDIOSTREAMCMD_PAUSE:
1290 {
1291 pa_threaded_mainloop_lock(pThis->pMainLoop);
1292 if (pStrm->pu8PeekBuf) /* Do we need to drop the peek buffer?*/
1293 {
1294 pa_stream_drop(pStrm->pStream);
1295 pStrm->pu8PeekBuf = NULL;
1296 }
1297
1298 rc = paWaitFor(pThis, pa_stream_cork(pStrm->pStream, 1 /* Stop / pause */, paStreamCbSuccess, pStrm));
1299 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1300 break;
1301 }
1302
1303 default:
1304 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
1305 rc = VERR_INVALID_PARAMETER;
1306 break;
1307 }
1308
1309 return rc;
1310}
1311
1312static DECLCALLBACK(int) drvHostPulseAudioGetConf(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
1313{
1314 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1315 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1316
1317 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1318
1319 return paEnumerate(pThis, pCfg, PULSEAUDIOENUMCBFLAGS_LOG /* fEnum */);
1320}
1321
1322static DECLCALLBACK(void) drvHostPulseAudioShutdown(PPDMIHOSTAUDIO pInterface)
1323{
1324 AssertPtrReturnVoid(pInterface);
1325
1326 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1327
1328 LogFlowFuncEnter();
1329
1330 if (pThis->pMainLoop)
1331 pa_threaded_mainloop_stop(pThis->pMainLoop);
1332
1333 if (pThis->pContext)
1334 {
1335 pa_context_disconnect(pThis->pContext);
1336 pa_context_unref(pThis->pContext);
1337 pThis->pContext = NULL;
1338 }
1339
1340 if (pThis->pMainLoop)
1341 {
1342 pa_threaded_mainloop_free(pThis->pMainLoop);
1343 pThis->pMainLoop = NULL;
1344 }
1345
1346 LogFlowFuncLeave();
1347}
1348
1349/**
1350 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1351 */
1352static DECLCALLBACK(void *) drvHostPulseAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1353{
1354 AssertPtrReturn(pInterface, NULL);
1355 AssertPtrReturn(pszIID, NULL);
1356
1357 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1358 PDRVHOSTPULSEAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPULSEAUDIO);
1359 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1360 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
1361
1362 return NULL;
1363}
1364
1365/**
1366 * Constructs a PulseAudio Audio driver instance.
1367 *
1368 * @copydoc FNPDMDRVCONSTRUCT
1369 */
1370static DECLCALLBACK(int) drvHostPulseAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1371{
1372 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
1373
1374 PDRVHOSTPULSEAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPULSEAUDIO);
1375 LogRel(("Audio: Initializing PulseAudio driver\n"));
1376
1377 CFGMR3QueryStringAlloc(pCfg, "StreamName", &pThis->pszStreamName);
1378
1379 pThis->pDrvIns = pDrvIns;
1380 /* IBase */
1381 pDrvIns->IBase.pfnQueryInterface = drvHostPulseAudioQueryInterface;
1382 /* IHostAudio */
1383 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostPulseAudio);
1384
1385 return VINF_SUCCESS;
1386}
1387
1388/**
1389 * Destructs a PulseAudio Audio driver instance.
1390 *
1391 * @copydoc FNPDMDRVCONSTRUCT
1392 */
1393static DECLCALLBACK(void) drvHostPulseAudioDestruct(PPDMDRVINS pDrvIns)
1394{
1395 PDRVHOSTPULSEAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPULSEAUDIO);
1396 LogFlowFuncEnter();
1397 if (pThis->pszStreamName)
1398 {
1399 MMR3HeapFree(pThis->pszStreamName);
1400 pThis->pszStreamName = NULL;
1401 }
1402}
1403
1404/**
1405 * Char driver registration record.
1406 */
1407const PDMDRVREG g_DrvHostPulseAudio =
1408{
1409 /* u32Version */
1410 PDM_DRVREG_VERSION,
1411 /* szName */
1412 "PulseAudio",
1413 /* szRCMod */
1414 "",
1415 /* szR0Mod */
1416 "",
1417 /* pszDescription */
1418 "Pulse Audio host driver",
1419 /* fFlags */
1420 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1421 /* fClass. */
1422 PDM_DRVREG_CLASS_AUDIO,
1423 /* cMaxInstances */
1424 ~0U,
1425 /* cbInstance */
1426 sizeof(DRVHOSTPULSEAUDIO),
1427 /* pfnConstruct */
1428 drvHostPulseAudioConstruct,
1429 /* pfnDestruct */
1430 drvHostPulseAudioDestruct,
1431 /* pfnRelocate */
1432 NULL,
1433 /* pfnIOCtl */
1434 NULL,
1435 /* pfnPowerOn */
1436 NULL,
1437 /* pfnReset */
1438 NULL,
1439 /* pfnSuspend */
1440 NULL,
1441 /* pfnResume */
1442 NULL,
1443 /* pfnAttach */
1444 NULL,
1445 /* pfnDetach */
1446 NULL,
1447 /* pfnPowerOff */
1448 NULL,
1449 /* pfnSoftReset */
1450 NULL,
1451 /* u32EndVersion */
1452 PDM_DRVREG_VERSION
1453};
1454
1455static struct audio_option pulse_options[] =
1456{
1457 {"DAC_MS", AUD_OPT_INT, &s_pulseCfg.buffer_msecs_out,
1458 "DAC period size in milliseconds", NULL, 0},
1459 {"ADC_MS", AUD_OPT_INT, &s_pulseCfg.buffer_msecs_in,
1460 "ADC period size in milliseconds", NULL, 0}
1461};
1462
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