VirtualBox

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

Last change on this file since 63547 was 63534, checked in by vboxsync, 8 years ago

Audio: Renaming.

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