VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DrvHostPulseAudio.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: 48.9 KB
Line 
1/* $Id: DrvHostPulseAudio.cpp 62463 2016-07-22 16:32:54Z 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 * Signal the main loop to abort. Just signalling isn't sufficient as the
171 * mainloop might not have been entered yet.
172 */
173static void paSignalWaiter(PDRVHOSTPULSEAUDIO pThis)
174{
175 if (!pThis)
176 return;
177
178 pThis->fLoopWait = true;
179 pa_threaded_mainloop_signal(pThis->pMainLoop, 0);
180}
181
182static pa_sample_format_t paFmtToPulse(PDMAUDIOFMT fmt)
183{
184 switch (fmt)
185 {
186 case PDMAUDIOFMT_U8:
187 return PA_SAMPLE_U8;
188
189 case PDMAUDIOFMT_S16:
190 return PA_SAMPLE_S16LE;
191
192#ifdef PA_SAMPLE_S32LE
193 case PDMAUDIOFMT_S32:
194 return PA_SAMPLE_S32LE;
195#endif
196 default:
197 break;
198 }
199
200 AssertMsgFailed(("Format %ld not supported\n", fmt));
201 return PA_SAMPLE_U8;
202}
203
204static int paPulseToFmt(pa_sample_format_t pulsefmt,
205 PDMAUDIOFMT *pFmt, PDMAUDIOENDIANNESS *pEndianness)
206{
207 switch (pulsefmt)
208 {
209 case PA_SAMPLE_U8:
210 *pFmt = PDMAUDIOFMT_U8;
211 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
212 break;
213
214 case PA_SAMPLE_S16LE:
215 *pFmt = PDMAUDIOFMT_S16;
216 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
217 break;
218
219 case PA_SAMPLE_S16BE:
220 *pFmt = PDMAUDIOFMT_S16;
221 *pEndianness = PDMAUDIOENDIANNESS_BIG;
222 break;
223
224#ifdef PA_SAMPLE_S32LE
225 case PA_SAMPLE_S32LE:
226 *pFmt = PDMAUDIOFMT_S32;
227 *pEndianness = PDMAUDIOENDIANNESS_LITTLE;
228 break;
229#endif
230
231#ifdef PA_SAMPLE_S32BE
232 case PA_SAMPLE_S32BE:
233 *pFmt = PDMAUDIOFMT_S32;
234 *pEndianness = PDMAUDIOENDIANNESS_BIG;
235 break;
236#endif
237
238 default:
239 AssertMsgFailed(("Format %ld not supported\n", pulsefmt));
240 return VERR_NOT_SUPPORTED;
241 }
242
243 return VINF_SUCCESS;
244}
245
246/**
247 * Synchronously wait until an operation completed.
248 */
249static int paWaitForEx(PDRVHOSTPULSEAUDIO pThis, pa_operation *pOP, RTMSINTERVAL cMsTimeout)
250{
251 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
252 AssertPtrReturn(pOP, VERR_INVALID_POINTER);
253
254 int rc = VINF_SUCCESS;
255
256 uint64_t u64StartMs = RTTimeMilliTS();
257 while (pa_operation_get_state(pOP) == PA_OPERATION_RUNNING)
258 {
259 if (!pThis->fLoopWait)
260 {
261 AssertPtr(pThis->pMainLoop);
262 pa_threaded_mainloop_wait(pThis->pMainLoop);
263 }
264 pThis->fLoopWait = false;
265
266 uint64_t u64ElapsedMs = RTTimeMilliTS() - u64StartMs;
267 if (u64ElapsedMs >= cMsTimeout)
268 {
269 rc = VERR_TIMEOUT;
270 break;
271 }
272 }
273
274 pa_operation_unref(pOP);
275
276 return rc;
277}
278
279static int paWaitFor(PDRVHOSTPULSEAUDIO pThis, pa_operation *pOP)
280{
281 return paWaitForEx(pThis, pOP, 10 * 1000 /* 10s timeout */);
282}
283
284/**
285 * Context status changed.
286 */
287static void paContextCbStateChanged(pa_context *pCtx, void *pvUser)
288{
289 AssertPtrReturnVoid(pCtx);
290
291 PDRVHOSTPULSEAUDIO pThis = (PDRVHOSTPULSEAUDIO)pvUser;
292 AssertPtrReturnVoid(pThis);
293
294 switch (pa_context_get_state(pCtx))
295 {
296 case PA_CONTEXT_READY:
297 case PA_CONTEXT_TERMINATED:
298 paSignalWaiter(pThis);
299 break;
300
301 case PA_CONTEXT_FAILED:
302 LogRel(("PulseAudio: Audio context has failed, stopping\n"));
303 paSignalWaiter(pThis);
304 break;
305
306 default:
307 break;
308 }
309}
310
311/**
312 * Callback called when our pa_stream_drain operation was completed.
313 */
314static void paStreamCbDrain(pa_stream *pStream, int fSuccess, void *pvUser)
315{
316 AssertPtrReturnVoid(pStream);
317
318 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pvUser;
319 AssertPtrReturnVoid(pStrm);
320
321 pStrm->fOpSuccess = fSuccess;
322 if (fSuccess)
323 {
324 pa_operation_unref(pa_stream_cork(pStream, 1,
325 paStreamCbSuccess, pvUser));
326 }
327 else
328 paError(pStrm->pDrv, "Failed to drain stream");
329
330 if (pStrm->pDrainOp)
331 {
332 pa_operation_unref(pStrm->pDrainOp);
333 pStrm->pDrainOp = NULL;
334 }
335}
336
337/**
338 * Stream status changed.
339 */
340static void paStreamCbStateChanged(pa_stream *pStream, void *pvUser)
341{
342 AssertPtrReturnVoid(pStream);
343
344 PDRVHOSTPULSEAUDIO pThis = (PDRVHOSTPULSEAUDIO)pvUser;
345 AssertPtrReturnVoid(pThis);
346
347 switch (pa_stream_get_state(pStream))
348 {
349 case PA_STREAM_READY:
350 case PA_STREAM_FAILED:
351 case PA_STREAM_TERMINATED:
352 paSignalWaiter(pThis);
353 break;
354
355 default:
356 break;
357 }
358}
359
360static void paStreamCbSuccess(pa_stream *pStream, int fSuccess, void *pvUser)
361{
362 AssertPtrReturnVoid(pStream);
363
364 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pvUser;
365 AssertPtrReturnVoid(pStrm);
366
367 pStrm->fOpSuccess = fSuccess;
368
369 if (fSuccess)
370 paSignalWaiter(pStrm->pDrv);
371 else
372 paError(pStrm->pDrv, "Failed to finish stream operation");
373}
374
375static int paStreamOpen(PDRVHOSTPULSEAUDIO pThis, bool fIn, const char *pszName,
376 pa_sample_spec *pSampleSpec, pa_buffer_attr *pBufAttr,
377 pa_stream **ppStream)
378{
379 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
380 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
381 AssertPtrReturn(pSampleSpec, VERR_INVALID_POINTER);
382 AssertPtrReturn(pBufAttr, VERR_INVALID_POINTER);
383 AssertPtrReturn(ppStream, VERR_INVALID_POINTER);
384
385 if (!pa_sample_spec_valid(pSampleSpec))
386 {
387 LogRel(("PulseAudio: Unsupported sample specification for stream \"%s\"\n",
388 pszName));
389 return VERR_NOT_SUPPORTED;
390 }
391
392 int rc = VINF_SUCCESS;
393
394 pa_stream *pStream = NULL;
395 uint32_t flags = PA_STREAM_NOFLAGS;
396
397 LogFunc(("Opening \"%s\", rate=%dHz, channels=%d, format=%s\n",
398 pszName, pSampleSpec->rate, pSampleSpec->channels,
399 pa_sample_format_to_string(pSampleSpec->format)));
400
401 pa_threaded_mainloop_lock(pThis->pMainLoop);
402
403 do
404 {
405 /** @todo r=andy Use pa_stream_new_with_proplist instead. */
406 if (!(pStream = pa_stream_new(pThis->pContext, pszName, pSampleSpec,
407 NULL /* pa_channel_map */)))
408 {
409 LogRel(("PulseAudio: Could not create stream \"%s\"\n", pszName));
410 rc = VERR_NO_MEMORY;
411 break;
412 }
413
414 pa_stream_set_state_callback(pStream, paStreamCbStateChanged, pThis);
415
416#if PA_API_VERSION >= 12
417 /* XXX */
418 flags |= PA_STREAM_ADJUST_LATENCY;
419#endif
420
421#if 0
422 /* Not applicable as we don't use pa_stream_get_latency() and pa_stream_get_time(). */
423 flags |= PA_STREAM_INTERPOLATE_TIMING | PA_STREAM_AUTO_TIMING_UPDATE;
424#endif
425 /* No input/output right away after the stream was started. */
426 flags |= PA_STREAM_START_CORKED;
427
428 if (fIn)
429 {
430 LogFunc(("Input stream attributes: maxlength=%d fragsize=%d\n",
431 pBufAttr->maxlength, pBufAttr->fragsize));
432
433 if (pa_stream_connect_record(pStream, /*dev=*/NULL, pBufAttr, (pa_stream_flags_t)flags) < 0)
434 {
435 LogRel(("PulseAudio: Could not connect input stream \"%s\": %s\n",
436 pszName, pa_strerror(pa_context_errno(pThis->pContext))));
437 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
438 break;
439 }
440 }
441 else
442 {
443 LogFunc(("Output buffer attributes: maxlength=%d tlength=%d prebuf=%d minreq=%d\n",
444 pBufAttr->maxlength, pBufAttr->tlength, pBufAttr->prebuf, pBufAttr->minreq));
445
446 if (pa_stream_connect_playback(pStream, /*dev=*/NULL, pBufAttr, (pa_stream_flags_t)flags,
447 /*cvolume=*/NULL, /*sync_stream=*/NULL) < 0)
448 {
449 LogRel(("PulseAudio: Could not connect playback stream \"%s\": %s\n",
450 pszName, pa_strerror(pa_context_errno(pThis->pContext))));
451 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
452 break;
453 }
454 }
455
456 /* Wait until the stream is ready. */
457 for (;;)
458 {
459 if (!pThis->fLoopWait)
460 pa_threaded_mainloop_wait(pThis->pMainLoop);
461 pThis->fLoopWait = false;
462
463 pa_stream_state_t streamSt = pa_stream_get_state(pStream);
464 if (streamSt == PA_STREAM_READY)
465 break;
466 else if ( streamSt == PA_STREAM_FAILED
467 || streamSt == PA_STREAM_TERMINATED)
468 {
469 LogRel(("PulseAudio: Failed to initialize stream \"%s\" (state %ld)\n", pszName, streamSt));
470 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
471 break;
472 }
473 }
474
475 if (RT_FAILURE(rc))
476 break;
477
478 const pa_buffer_attr *pBufAttrObtained = pa_stream_get_buffer_attr(pStream);
479 AssertPtr(pBufAttrObtained);
480 memcpy(pBufAttr, pBufAttrObtained, sizeof(pa_buffer_attr));
481
482 if (fIn)
483 LogFunc(("Obtained record buffer attributes: maxlength=%RU32, fragsize=%RU32\n",
484 pBufAttr->maxlength, pBufAttr->fragsize));
485 else
486 LogFunc(("Obtained playback buffer attributes: maxlength=%d, tlength=%d, prebuf=%d, minreq=%d\n",
487 pBufAttr->maxlength, pBufAttr->tlength, pBufAttr->prebuf, pBufAttr->minreq));
488
489 }
490 while (0);
491
492 if ( RT_FAILURE(rc)
493 && pStream)
494 pa_stream_disconnect(pStream);
495
496 pa_threaded_mainloop_unlock(pThis->pMainLoop);
497
498 if (RT_FAILURE(rc))
499 {
500 if (pStream)
501 pa_stream_unref(pStream);
502 }
503 else
504 *ppStream = pStream;
505
506 LogFlowFuncLeaveRC(rc);
507 return rc;
508}
509
510static DECLCALLBACK(int) drvHostPulseAudioInit(PPDMIHOSTAUDIO pInterface)
511{
512 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
513
514 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
515
516 LogFlowFuncEnter();
517
518 int rc = audioLoadPulseLib();
519 if (RT_FAILURE(rc))
520 {
521 LogRel(("PulseAudio: Failed to load the PulseAudio shared library! Error %Rrc\n", rc));
522 return rc;
523 }
524
525 pThis->fLoopWait = false;
526 pThis->pMainLoop = NULL;
527
528 bool fLocked = false;
529
530 do
531 {
532 if (!(pThis->pMainLoop = pa_threaded_mainloop_new()))
533 {
534 LogRel(("PulseAudio: Failed to allocate main loop: %s\n",
535 pa_strerror(pa_context_errno(pThis->pContext))));
536 rc = VERR_NO_MEMORY;
537 break;
538 }
539
540 if (!(pThis->pContext = pa_context_new(pa_threaded_mainloop_get_api(pThis->pMainLoop), "VirtualBox")))
541 {
542 LogRel(("PulseAudio: Failed to allocate context: %s\n",
543 pa_strerror(pa_context_errno(pThis->pContext))));
544 rc = VERR_NO_MEMORY;
545 break;
546 }
547
548 if (pa_threaded_mainloop_start(pThis->pMainLoop) < 0)
549 {
550 LogRel(("PulseAudio: Failed to start threaded mainloop: %s\n",
551 pa_strerror(pa_context_errno(pThis->pContext))));
552 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
553 break;
554 }
555
556 /* Install a global callback to known if something happens to our acquired context. */
557 pa_context_set_state_callback(pThis->pContext, paContextCbStateChanged, pThis /* pvUserData */);
558
559 pa_threaded_mainloop_lock(pThis->pMainLoop);
560 fLocked = true;
561
562 if (pa_context_connect(pThis->pContext, NULL /* pszServer */,
563 PA_CONTEXT_NOFLAGS, NULL) < 0)
564 {
565 LogRel(("PulseAudio: Failed to connect to server: %s\n",
566 pa_strerror(pa_context_errno(pThis->pContext))));
567 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
568 break;
569 }
570
571 /* Wait until the pThis->pContext is ready. */
572 for (;;)
573 {
574 if (!pThis->fLoopWait)
575 pa_threaded_mainloop_wait(pThis->pMainLoop);
576 pThis->fLoopWait = false;
577
578 pa_context_state_t cstate = pa_context_get_state(pThis->pContext);
579 if (cstate == PA_CONTEXT_READY)
580 break;
581 else if ( cstate == PA_CONTEXT_TERMINATED
582 || cstate == PA_CONTEXT_FAILED)
583 {
584 LogRel(("PulseAudio: Failed to initialize context (state %d)\n", cstate));
585 rc = VERR_AUDIO_BACKEND_INIT_FAILED;
586 break;
587 }
588 }
589 }
590 while (0);
591
592 if (fLocked)
593 pa_threaded_mainloop_unlock(pThis->pMainLoop);
594
595 if (RT_FAILURE(rc))
596 {
597 if (pThis->pMainLoop)
598 pa_threaded_mainloop_stop(pThis->pMainLoop);
599
600 if (pThis->pContext)
601 {
602 pa_context_disconnect(pThis->pContext);
603 pa_context_unref(pThis->pContext);
604 pThis->pContext = NULL;
605 }
606
607 if (pThis->pMainLoop)
608 {
609 pa_threaded_mainloop_free(pThis->pMainLoop);
610 pThis->pMainLoop = NULL;
611 }
612 }
613
614 LogFlowFuncLeaveRC(rc);
615 return rc;
616}
617
618static int paCreateStreamOut(PPDMIHOSTAUDIO pInterface,
619 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
620{
621 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
622 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
623 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
624 /* pcSamples is optional. */
625
626 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
627 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
628
629 LogFlowFuncEnter();
630
631 pStrm->pDrainOp = NULL;
632
633 pStrm->SampleSpec.format = paFmtToPulse(pCfg->enmFormat);
634 pStrm->SampleSpec.rate = pCfg->uHz;
635 pStrm->SampleSpec.channels = pCfg->cChannels;
636
637 /* Note that setting maxlength to -1 does not work on PulseAudio servers
638 * older than 0.9.10. So use the suggested value of 3/2 of tlength */
639 pStrm->BufAttr.tlength = (pa_bytes_per_second(&pStrm->SampleSpec)
640 * s_pulseCfg.buffer_msecs_out) / 1000;
641 pStrm->BufAttr.maxlength = (pStrm->BufAttr.tlength * 3) / 2;
642 pStrm->BufAttr.prebuf = -1; /* Same as tlength */
643 pStrm->BufAttr.minreq = -1;
644
645 /* Note that the struct BufAttr is updated to the obtained values after this call! */
646 int rc = paStreamOpen(pThis, false /* fIn */, "PulseAudio (Out)", &pStrm->SampleSpec, &pStrm->BufAttr, &pStrm->pPAStream);
647 if (RT_FAILURE(rc))
648 return rc;
649
650 PDMAUDIOSTREAMCFG streamCfg;
651 rc = paPulseToFmt(pStrm->SampleSpec.format,
652 &streamCfg.enmFormat, &streamCfg.enmEndianness);
653 if (RT_FAILURE(rc))
654 {
655 LogRel(("PulseAudio: Cannot find audio output format %ld\n", pStrm->SampleSpec.format));
656 return rc;
657 }
658
659 streamCfg.uHz = pStrm->SampleSpec.rate;
660 streamCfg.cChannels = pStrm->SampleSpec.channels;
661
662 rc = DrvAudioHlpStreamCfgToProps(&streamCfg, &pStream->Props);
663 if (RT_SUCCESS(rc))
664 {
665 uint32_t cbBuf = RT_MIN(pStrm->BufAttr.tlength * 2,
666 pStrm->BufAttr.maxlength); /** @todo Make this configurable! */
667 if (cbBuf)
668 {
669 pStrm->pvPCMBuf = RTMemAllocZ(cbBuf);
670 if (pStrm->pvPCMBuf)
671 {
672 pStrm->cbPCMBuf = cbBuf;
673
674 uint32_t cSamples = cbBuf >> pStream->Props.cShift;
675 if (pcSamples)
676 *pcSamples = cSamples;
677
678 /* Save pointer to driver instance. */
679 pStrm->pDrv = pThis;
680
681 LogFunc(("cbBuf=%RU32, cSamples=%RU32\n", cbBuf, cSamples));
682 }
683 else
684 rc = VERR_NO_MEMORY;
685 }
686 else
687 rc = VERR_INVALID_PARAMETER;
688 }
689
690 LogFlowFuncLeaveRC(rc);
691 return rc;
692}
693
694static int paCreateStreamIn(PPDMIHOSTAUDIO pInterface,
695 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
696{
697 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
698 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
699 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
700 /* pcSamples is optional. */
701
702 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
703 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
704
705 pStrm->SampleSpec.format = paFmtToPulse(pCfg->enmFormat);
706 pStrm->SampleSpec.rate = pCfg->uHz;
707 pStrm->SampleSpec.channels = pCfg->cChannels;
708
709 /* XXX check these values */
710 pStrm->BufAttr.fragsize = (pa_bytes_per_second(&pStrm->SampleSpec)
711 * s_pulseCfg.buffer_msecs_in) / 1000;
712 pStrm->BufAttr.maxlength = (pStrm->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)", &pStrm->SampleSpec, &pStrm->BufAttr,
716 &pStrm->pPAStream);
717 if (RT_FAILURE(rc))
718 return rc;
719
720 PDMAUDIOSTREAMCFG streamCfg;
721 rc = paPulseToFmt(pStrm->SampleSpec.format, &streamCfg.enmFormat,
722 &streamCfg.enmEndianness);
723 if (RT_FAILURE(rc))
724 {
725 LogRel(("PulseAudio: Cannot find audio capture format %ld\n", pStrm->SampleSpec.format));
726 return rc;
727 }
728
729 streamCfg.uHz = pStrm->SampleSpec.rate;
730 streamCfg.cChannels = pStrm->SampleSpec.channels;
731
732 rc = DrvAudioHlpStreamCfgToProps(&streamCfg, &pStream->Props);
733 if (RT_SUCCESS(rc))
734 {
735 uint32_t cSamples = RT_MIN(pStrm->BufAttr.fragsize * 10, pStrm->BufAttr.maxlength)
736 >> pStream->Props.cShift;
737
738 LogFunc(("uHz=%RU32, cChannels=%RU8, cShift=%RU8, cSamples=%RU32\n",
739 pStream->Props.uHz, pStream->Props.cChannels, pStream->Props.cShift, cSamples));
740
741 /* Save pointer to driver instance. */
742 pStrm->pDrv = pThis;
743 pStrm->pu8PeekBuf = NULL;
744
745 if (pcSamples)
746 *pcSamples = cSamples;
747 }
748
749 LogFlowFuncLeaveRC(rc);
750 return rc;
751}
752
753static DECLCALLBACK(int) drvHostPulseAudioStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
754 uint32_t *pcSamplesCaptured)
755{
756 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
757 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
758 /* pcSamplesPlayed is optional. */
759
760 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
761 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
762
763 /* We should only call pa_stream_readable_size() once and trust the first value. */
764 pa_threaded_mainloop_lock(pThis->pMainLoop);
765 size_t cbAvail = pa_stream_readable_size(pStrm->pPAStream);
766 pa_threaded_mainloop_unlock(pThis->pMainLoop);
767
768 if (cbAvail == (size_t)-1)
769 return paError(pStrm->pDrv, "Failed to determine input data size");
770
771 /* If the buffer was not dropped last call, add what remains. */
772 if (pStrm->pu8PeekBuf)
773 {
774 Assert(pStrm->cbPeekBuf >= pStrm->offPeekBuf);
775 cbAvail += (pStrm->cbPeekBuf - pStrm->offPeekBuf);
776 }
777
778 LogFlowFunc(("cbAvail=%zu\n", cbAvail));
779
780 if (!cbAvail) /* No data? Bail out. */
781 {
782 if (pcSamplesCaptured)
783 *pcSamplesCaptured = 0;
784 return VINF_SUCCESS;
785 }
786
787 int rc = VINF_SUCCESS;
788
789 size_t cbToRead = RT_MIN(cbAvail, AudioMixBufFreeBytes(&pStream->MixBuf));
790
791 LogFlowFunc(("cbToRead=%zu, cbAvail=%zu, offPeekBuf=%zu, cbPeekBuf=%zu\n",
792 cbToRead, cbAvail, pStrm->offPeekBuf, pStrm->cbPeekBuf));
793
794 size_t offWrite = 0;
795 uint32_t cWrittenTotal = 0;
796
797 while (cbToRead)
798 {
799 /* If there is no data, do another peek. */
800 if (!pStrm->pu8PeekBuf)
801 {
802 pa_threaded_mainloop_lock(pThis->pMainLoop);
803 pa_stream_peek(pStrm->pPAStream,
804 (const void**)&pStrm->pu8PeekBuf, &pStrm->cbPeekBuf);
805 pa_threaded_mainloop_unlock(pThis->pMainLoop);
806
807 pStrm->offPeekBuf = 0;
808
809 /* No data anymore?
810 * Note: If there's a data hole (cbPeekBuf then contains the length of the hole)
811 * we need to drop the stream lateron. */
812 if ( !pStrm->pu8PeekBuf
813 && !pStrm->cbPeekBuf)
814 {
815 break;
816 }
817 }
818
819 Assert(pStrm->cbPeekBuf >= pStrm->offPeekBuf);
820 size_t cbToWrite = RT_MIN(pStrm->cbPeekBuf - pStrm->offPeekBuf, cbToRead);
821
822 LogFlowFunc(("cbToRead=%zu, cbToWrite=%zu, offPeekBuf=%zu, cbPeekBuf=%zu, pu8PeekBuf=%p\n",
823 cbToRead, cbToWrite,
824 pStrm->offPeekBuf, pStrm->cbPeekBuf, pStrm->pu8PeekBuf));
825
826 if (cbToWrite)
827 {
828 uint32_t cWritten;
829 rc = AudioMixBufWriteCirc(&pStream->MixBuf,
830 pStrm->pu8PeekBuf + pStrm->offPeekBuf,
831 cbToWrite, &cWritten);
832 if (RT_FAILURE(rc))
833 break;
834
835 uint32_t cbWritten = AUDIOMIXBUF_S2B(&pStream->MixBuf, cWritten);
836
837 Assert(cbToRead >= cbWritten);
838 cbToRead -= cbWritten;
839 cWrittenTotal += cWritten;
840 pStrm->offPeekBuf += cbWritten;
841 }
842
843 if (/* Nothing to write anymore? Drop the buffer. */
844 !cbToWrite
845 /* Was there a hole in the peeking buffer? Drop it. */
846 || !pStrm->pu8PeekBuf
847 /* If the buffer is done, drop it. */
848 || pStrm->offPeekBuf == pStrm->cbPeekBuf)
849 {
850 pa_threaded_mainloop_lock(pThis->pMainLoop);
851 pa_stream_drop(pStrm->pPAStream);
852 pa_threaded_mainloop_unlock(pThis->pMainLoop);
853
854 pStrm->pu8PeekBuf = NULL;
855 }
856 }
857
858 if (RT_SUCCESS(rc))
859 {
860 uint32_t cProcessed = 0;
861 /* if (cWrittenTotal)
862 rc = AudioMixBufMixToParent(&pStream->MixBuf, cWrittenTotal,
863 &cProcessed);*/
864
865 if (pcSamplesCaptured)
866 *pcSamplesCaptured = cWrittenTotal;
867
868 LogFlowFunc(("cWrittenTotal=%RU32 (%RU32 processed), rc=%Rrc\n",
869 cWrittenTotal, cProcessed, rc));
870 }
871
872 LogFlowFuncLeaveRC(rc);
873 return rc;
874}
875
876static DECLCALLBACK(int) drvHostPulseAudioStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
877 uint32_t *pcSamplesPlayed)
878{
879 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
880 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
881 /* pcSamplesPlayed is optional. */
882
883 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
884 PPULSEAUDIOSTREAM pPAStream = (PPULSEAUDIOSTREAM)pStream;
885
886 int rc = VINF_SUCCESS;
887 uint32_t cbReadTotal = 0;
888
889 uint32_t cLive = AudioMixBufUsed(&pStream->MixBuf);
890 if (!cLive)
891 {
892 LogFlowFunc(("No live samples, skipping\n"));
893 if (pcSamplesPlayed)
894 *pcSamplesPlayed = 0;
895 return VINF_SUCCESS;
896 }
897
898 pa_threaded_mainloop_lock(pThis->pMainLoop);
899
900 do
901 {
902 size_t cbWriteable = pa_stream_writable_size(pPAStream->pPAStream);
903 if (cbWriteable == (size_t)-1)
904 {
905 rc = paError(pPAStream->pDrv, "Failed to determine output data size");
906 break;
907 }
908
909 size_t cbLive = AUDIOMIXBUF_S2B(&pStream->MixBuf, cLive);
910 size_t cbToRead = RT_MIN(cbWriteable, cbLive);
911
912 LogFlowFunc(("cbToRead=%zu, cbWriteable=%zu, cbLive=%zu\n",
913 cbToRead, cbWriteable, cbLive));
914
915 uint32_t cRead, cbRead;
916 while (cbToRead)
917 {
918 rc = AudioMixBufReadCirc(&pStream->MixBuf, pPAStream->pvPCMBuf,
919 RT_MIN(cbToRead, pPAStream->cbPCMBuf), &cRead);
920 if ( !cRead
921 || RT_FAILURE(rc))
922 {
923 break;
924 }
925
926 cbRead = AUDIOMIXBUF_S2B(&pStream->MixBuf, cRead);
927 if (pa_stream_write(pPAStream->pPAStream, pPAStream->pvPCMBuf, cbRead, NULL /* Cleanup callback */,
928 0, PA_SEEK_RELATIVE) < 0)
929 {
930 rc = paError(pPAStream->pDrv, "Failed to write to output stream");
931 break;
932 }
933
934 Assert(cbToRead >= cbRead);
935 cbToRead -= cbRead;
936 cbReadTotal += cbRead;
937
938 LogFlowFunc(("\tcRead=%RU32 (%zu bytes) cbReadTotal=%RU32, cbToRead=%RU32\n",
939 cRead, AUDIOMIXBUF_S2B(&pStream->MixBuf, cRead), cbReadTotal, cbToRead));
940 }
941
942 } while (0);
943
944 pa_threaded_mainloop_unlock(pThis->pMainLoop);
945
946 if (RT_SUCCESS(rc))
947 {
948 uint32_t cReadTotal = AUDIOMIXBUF_B2S(&pStream->MixBuf, cbReadTotal);
949 if (cReadTotal)
950 AudioMixBufFinish(&pStream->MixBuf, cReadTotal);
951
952 if (pcSamplesPlayed)
953 *pcSamplesPlayed = cReadTotal;
954
955 LogFlowFunc(("cReadTotal=%RU32 (%RU32 bytes), rc=%Rrc\n", cReadTotal, cbReadTotal, rc));
956 }
957
958 LogFlowFuncLeaveRC(rc);
959 return rc;
960}
961
962/** @todo Implement va handling. */
963static int paError(PDRVHOSTPULSEAUDIO pThis, const char *szMsg)
964{
965 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
966 AssertPtrReturn(szMsg, VERR_INVALID_POINTER);
967
968 if (pThis->cLogErrors++ < VBOX_PULSEAUDIO_MAX_LOG_REL_ERRORS)
969 {
970 int rc2 = pa_context_errno(pThis->pContext);
971 LogRel2(("PulseAudio: %s: %s\n", szMsg, pa_strerror(rc2)));
972 }
973
974 /** @todo Implement some PulseAudio -> IPRT mapping here. */
975 return VERR_GENERAL_FAILURE;
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
998static void paEnumSourceCb(pa_context *pCtx, const pa_source_info *pInfo, int eol, void *pvUserData)
999{
1000 if (eol != 0)
1001 return;
1002
1003 AssertPtrReturnVoid(pCtx);
1004 AssertPtrReturnVoid(pInfo);
1005
1006 PPULSEAUDIOENUMCBCTX pCbCtx = (PPULSEAUDIOENUMCBCTX)pvUserData;
1007 AssertPtrReturnVoid(pCbCtx);
1008 AssertPtrReturnVoid(pCbCtx->pDrv);
1009
1010 LogRel2(("PulseAudio: Using input source '%s'\n", pInfo->name));
1011
1012 /** @todo Store sources + channel mapping in callback context as soon as we have surround support. */
1013 pCbCtx->cDevIn++;
1014
1015 pa_threaded_mainloop_signal(pCbCtx->pDrv->pMainLoop, 0);
1016}
1017
1018static void paEnumServerCb(pa_context *pCtx, const pa_server_info *pInfo, void *pvUserData)
1019{
1020 AssertPtrReturnVoid(pCtx);
1021 AssertPtrReturnVoid(pInfo);
1022
1023 PPULSEAUDIOENUMCBCTX pCbCtx = (PPULSEAUDIOENUMCBCTX)pvUserData;
1024 AssertPtrReturnVoid(pCbCtx);
1025
1026 PDRVHOSTPULSEAUDIO pThis = pCbCtx->pDrv;
1027 AssertPtrReturnVoid(pThis);
1028
1029 if (pInfo->default_sink_name)
1030 {
1031 Assert(RTStrIsValidEncoding(pInfo->default_sink_name));
1032 pCbCtx->pszDefaultSink = RTStrDup(pInfo->default_sink_name);
1033 }
1034
1035 if (pInfo->default_sink_name)
1036 {
1037 Assert(RTStrIsValidEncoding(pInfo->default_source_name));
1038 pCbCtx->pszDefaultSource = RTStrDup(pInfo->default_source_name);
1039 }
1040
1041 pa_threaded_mainloop_signal(pThis->pMainLoop, 0);
1042}
1043
1044static int paEnumerate(PDRVHOSTPULSEAUDIO pThis, PPDMAUDIOBACKENDCFG pCfg, uint32_t fEnum)
1045{
1046 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1047 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1048
1049 PDMAUDIOBACKENDCFG Cfg;
1050 RT_ZERO(Cfg);
1051
1052 Cfg.cbStreamOut = sizeof(PULSEAUDIOSTREAM);
1053 Cfg.cbStreamIn = sizeof(PULSEAUDIOSTREAM);
1054 Cfg.cMaxStreamsOut = UINT32_MAX;
1055 Cfg.cMaxStreamsIn = UINT32_MAX;
1056
1057 PULSEAUDIOENUMCBCTX cbCtx;
1058 RT_ZERO(cbCtx);
1059
1060 cbCtx.pDrv = pThis;
1061 cbCtx.fFlags = fEnum;
1062
1063 bool fLog = (fEnum & PULSEAUDIOENUMCBFLAGS_LOG);
1064
1065 int rc = paWaitFor(pThis, pa_context_get_server_info(pThis->pContext, paEnumServerCb, &cbCtx));
1066 if (RT_SUCCESS(rc))
1067 {
1068 if (cbCtx.pszDefaultSink)
1069 {
1070 if (fLog)
1071 LogRel2(("PulseAudio: Default output sink is '%s'\n", cbCtx.pszDefaultSink));
1072
1073 rc = paWaitFor(pThis, pa_context_get_sink_info_by_name(pThis->pContext, cbCtx.pszDefaultSink,
1074 paEnumSinkCb, &cbCtx));
1075 if ( RT_FAILURE(rc)
1076 && fLog)
1077 {
1078 LogRel(("PulseAudio: Error enumerating properties for default output sink '%s'\n", cbCtx.pszDefaultSink));
1079 }
1080 }
1081 else if (fLog)
1082 LogRel2(("PulseAudio: No default output sink found\n"));
1083
1084 if (RT_SUCCESS(rc))
1085 {
1086 if (cbCtx.pszDefaultSource)
1087 {
1088 if (fLog)
1089 LogRel2(("PulseAudio: Default input source is '%s'\n", cbCtx.pszDefaultSource));
1090
1091 rc = paWaitFor(pThis, pa_context_get_source_info_by_name(pThis->pContext, cbCtx.pszDefaultSource,
1092 paEnumSourceCb, &cbCtx));
1093 if ( RT_FAILURE(rc)
1094 && fLog)
1095 {
1096 LogRel(("PulseAudio: Error enumerating properties for default input source '%s'\n", cbCtx.pszDefaultSource));
1097 }
1098 }
1099 else if (fLog)
1100 LogRel2(("PulseAudio: No default input source found\n"));
1101 }
1102
1103 if (RT_SUCCESS(rc))
1104 {
1105 Cfg.cSinks = cbCtx.cDevOut;
1106 Cfg.cSources = cbCtx.cDevIn;
1107
1108 if (fLog)
1109 {
1110 LogRel2(("PulseAudio: Found %RU8 host playback device(s)\n", cbCtx.cDevOut));
1111 LogRel2(("PulseAudio: Found %RU8 host capturing device(s)\n", cbCtx.cDevIn));
1112 }
1113
1114 if (pCfg)
1115 memcpy(pCfg, &Cfg, sizeof(PDMAUDIOBACKENDCFG));
1116 }
1117
1118 if (cbCtx.pszDefaultSink)
1119 {
1120 RTStrFree(cbCtx.pszDefaultSink);
1121 cbCtx.pszDefaultSink = NULL;
1122 }
1123
1124 if (cbCtx.pszDefaultSource)
1125 {
1126 RTStrFree(cbCtx.pszDefaultSource);
1127 cbCtx.pszDefaultSource = NULL;
1128 }
1129 }
1130 else if (fLog)
1131 LogRel(("PulseAudio: Error enumerating PulseAudio server properties\n"));
1132
1133 LogFlowFuncLeaveRC(rc);
1134 return rc;
1135}
1136
1137static int paDestroyStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1138{
1139 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1140 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1141
1142 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1143 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
1144
1145 LogFlowFuncEnter();
1146
1147 if (pStrm->pPAStream)
1148 {
1149 pa_threaded_mainloop_lock(pThis->pMainLoop);
1150
1151 pa_stream_disconnect(pStrm->pPAStream);
1152 pa_stream_unref(pStrm->pPAStream);
1153
1154 pStrm->pPAStream = NULL;
1155
1156 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1157 }
1158
1159 return VINF_SUCCESS;
1160}
1161
1162static int paDestroyStreamOut(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1163{
1164 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1165 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1166
1167 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1168 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
1169
1170 LogFlowFuncEnter();
1171
1172 if (pStrm->pPAStream)
1173 {
1174 pa_threaded_mainloop_lock(pThis->pMainLoop);
1175
1176 /* Make sure to cancel a pending draining operation, if any. */
1177 if (pStrm->pDrainOp)
1178 {
1179 pa_operation_cancel(pStrm->pDrainOp);
1180 pStrm->pDrainOp = NULL;
1181 }
1182
1183 pa_stream_disconnect(pStrm->pPAStream);
1184 pa_stream_unref(pStrm->pPAStream);
1185
1186 pStrm->pPAStream = NULL;
1187
1188 pa_threaded_mainloop_unlock(pThis->pMainLoop);
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 int paControlStreamOut(PPDMIHOSTAUDIO pInterface,
1202 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
1203{
1204 AssertPtrReturn(pInterface , VERR_INVALID_POINTER);
1205 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1206
1207 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1208 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
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->pPAStream, 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->pPAStream, paStreamCbSuccess, pStrm));
1247 if (RT_SUCCESS(rc))
1248 pStrm->pDrainOp = pa_stream_drain(pStrm->pPAStream, 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 int paControlStreamIn(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream,
1265 PDMAUDIOSTREAMCMD enmStreamCmd)
1266{
1267 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1268 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1269
1270 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1271 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
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->pPAStream, 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->pPAStream);
1295 pStrm->pu8PeekBuf = NULL;
1296 }
1297
1298 rc = paWaitFor(pThis, pa_stream_cork(pStrm->pPAStream, 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(void) drvHostPulseAudioShutdown(PPDMIHOSTAUDIO pInterface)
1313{
1314 AssertPtrReturnVoid(pInterface);
1315
1316 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1317
1318 LogFlowFuncEnter();
1319
1320 if (pThis->pMainLoop)
1321 pa_threaded_mainloop_stop(pThis->pMainLoop);
1322
1323 if (pThis->pContext)
1324 {
1325 pa_context_disconnect(pThis->pContext);
1326 pa_context_unref(pThis->pContext);
1327 pThis->pContext = NULL;
1328 }
1329
1330 if (pThis->pMainLoop)
1331 {
1332 pa_threaded_mainloop_free(pThis->pMainLoop);
1333 pThis->pMainLoop = NULL;
1334 }
1335
1336 LogFlowFuncLeave();
1337}
1338
1339static DECLCALLBACK(int) drvHostPulseAudioGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pCfg)
1340{
1341 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1342 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1343
1344 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1345
1346 return paEnumerate(pThis, pCfg, PULSEAUDIOENUMCBFLAGS_LOG /* fEnum */);
1347}
1348
1349static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvHostPulseAudioGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
1350{
1351 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
1352
1353 return PDMAUDIOBACKENDSTS_RUNNING;
1354}
1355
1356static DECLCALLBACK(int) drvHostPulseAudioStreamCreate(PPDMIHOSTAUDIO pInterface,
1357 PPDMAUDIOSTREAM pStream, PPDMAUDIOSTREAMCFG pCfg, uint32_t *pcSamples)
1358{
1359 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1360 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1361 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1362
1363 int rc;
1364 if (pCfg->enmDir == PDMAUDIODIR_IN)
1365 rc = paCreateStreamIn(pInterface, pStream, pCfg, pcSamples);
1366 else if (pStream->enmDir == PDMAUDIODIR_OUT)
1367 rc = paCreateStreamOut(pInterface, pStream, pCfg, pcSamples);
1368 else
1369 AssertFailedReturn(VERR_NOT_IMPLEMENTED);
1370
1371 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
1372 return rc;
1373}
1374
1375static DECLCALLBACK(int) drvHostPulseAudioStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1376{
1377 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1378 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1379
1380 LogFlowFunc(("%s\n", pStream->szName));
1381
1382 int rc;
1383 if (pStream->enmDir == PDMAUDIODIR_IN)
1384 rc = paDestroyStreamIn(pInterface, pStream);
1385 else if (pStream->enmDir == PDMAUDIODIR_OUT)
1386 rc = paDestroyStreamOut(pInterface, pStream);
1387 else
1388 AssertFailedReturn(VERR_NOT_IMPLEMENTED);
1389
1390 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
1391 return rc;
1392}
1393
1394static DECLCALLBACK(int) drvHostPulseAudioStreamControl(PPDMIHOSTAUDIO pInterface,
1395 PPDMAUDIOSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
1396{
1397 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1398 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1399
1400 Assert(pStream->enmCtx == PDMAUDIOSTREAMCTX_HOST);
1401
1402 int rc;
1403 if (pStream->enmDir == PDMAUDIODIR_IN)
1404 rc = paControlStreamIn(pInterface, pStream, enmStreamCmd);
1405 else if (pStream->enmDir == PDMAUDIODIR_OUT)
1406 rc = paControlStreamOut(pInterface, pStream, enmStreamCmd);
1407 else
1408 AssertFailedReturn(VERR_NOT_IMPLEMENTED);
1409
1410 LogFlowFunc(("%s: rc=%Rrc\n", pStream->szName, rc));
1411 return rc;
1412}
1413
1414static DECLCALLBACK(PDMAUDIOSTRMSTS) drvHostPulseAudioStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1415{
1416 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1417 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1418
1419 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1420 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
1421
1422 PDMAUDIOSTRMSTS strmSts = PDMAUDIOSTRMSTS_FLAG_INITIALIZED
1423 | PDMAUDIOSTRMSTS_FLAG_ENABLED;
1424
1425 pa_threaded_mainloop_lock(pThis->pMainLoop);
1426
1427 pa_context_state_t ctxState = pa_context_get_state(pThis->pContext);
1428
1429 if ( pa_context_get_state(pThis->pContext) == PA_CONTEXT_READY
1430 && pa_stream_get_state(pStrm->pPAStream) == PA_STREAM_READY)
1431 {
1432 if (pStream->enmDir == PDMAUDIODIR_IN)
1433 {
1434 strmSts |= PDMAUDIOSTRMSTS_FLAG_DATA_READABLE;
1435 }
1436 else
1437 {
1438 size_t cbSize = pa_stream_writable_size(pStrm->pPAStream);
1439 LogFlowFunc(("cbSize=%zu\n", cbSize));
1440
1441 if (cbSize >= pStrm->BufAttr.minreq)
1442 strmSts |= PDMAUDIOSTRMSTS_FLAG_DATA_WRITABLE;
1443 }
1444 }
1445
1446 pa_threaded_mainloop_unlock(pThis->pMainLoop);
1447
1448 return strmSts;
1449}
1450
1451static DECLCALLBACK(int) drvHostPulseAudioStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOSTREAM pStream)
1452{
1453 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
1454 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
1455
1456 PDRVHOSTPULSEAUDIO pThis = PDMIHOSTAUDIO_2_DRVHOSTPULSEAUDIO(pInterface);
1457 PPULSEAUDIOSTREAM pStrm = (PPULSEAUDIOSTREAM)pStream;
1458
1459 LogFlowFuncEnter();
1460
1461 /* Nothing to do here for PulseAudio. */
1462 return VINF_SUCCESS;
1463}
1464
1465/**
1466 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1467 */
1468static DECLCALLBACK(void *) drvHostPulseAudioQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1469{
1470 AssertPtrReturn(pInterface, NULL);
1471 AssertPtrReturn(pszIID, NULL);
1472
1473 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1474 PDRVHOSTPULSEAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPULSEAUDIO);
1475 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1476 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
1477
1478 return NULL;
1479}
1480
1481/**
1482 * Constructs a PulseAudio Audio driver instance.
1483 *
1484 * @copydoc FNPDMDRVCONSTRUCT
1485 */
1486static DECLCALLBACK(int) drvHostPulseAudioConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1487{
1488 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
1489
1490 PDRVHOSTPULSEAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPULSEAUDIO);
1491 LogRel(("Audio: Initializing PulseAudio driver\n"));
1492
1493 pThis->pDrvIns = pDrvIns;
1494 /* IBase */
1495 pDrvIns->IBase.pfnQueryInterface = drvHostPulseAudioQueryInterface;
1496 /* IHostAudio */
1497 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvHostPulseAudio);
1498
1499 return VINF_SUCCESS;
1500}
1501
1502/**
1503 * Destructs a PulseAudio Audio driver instance.
1504 *
1505 * @copydoc FNPDMDRVCONSTRUCT
1506 */
1507static DECLCALLBACK(void) drvHostPulseAudioDestruct(PPDMDRVINS pDrvIns)
1508{
1509 PDRVHOSTPULSEAUDIO pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPULSEAUDIO);
1510 LogFlowFuncEnter();
1511}
1512
1513/**
1514 * Char driver registration record.
1515 */
1516const PDMDRVREG g_DrvHostPulseAudio =
1517{
1518 /* u32Version */
1519 PDM_DRVREG_VERSION,
1520 /* szName */
1521 "PulseAudio",
1522 /* szRCMod */
1523 "",
1524 /* szR0Mod */
1525 "",
1526 /* pszDescription */
1527 "Pulse Audio host driver",
1528 /* fFlags */
1529 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1530 /* fClass. */
1531 PDM_DRVREG_CLASS_AUDIO,
1532 /* cMaxInstances */
1533 ~0U,
1534 /* cbInstance */
1535 sizeof(DRVHOSTPULSEAUDIO),
1536 /* pfnConstruct */
1537 drvHostPulseAudioConstruct,
1538 /* pfnDestruct */
1539 drvHostPulseAudioDestruct,
1540 /* pfnRelocate */
1541 NULL,
1542 /* pfnIOCtl */
1543 NULL,
1544 /* pfnPowerOn */
1545 NULL,
1546 /* pfnReset */
1547 NULL,
1548 /* pfnSuspend */
1549 NULL,
1550 /* pfnResume */
1551 NULL,
1552 /* pfnAttach */
1553 NULL,
1554 /* pfnDetach */
1555 NULL,
1556 /* pfnPowerOff */
1557 NULL,
1558 /* pfnSoftReset */
1559 NULL,
1560 /* u32EndVersion */
1561 PDM_DRVREG_VERSION
1562};
1563
1564static struct audio_option pulse_options[] =
1565{
1566 {"DAC_MS", AUD_OPT_INT, &s_pulseCfg.buffer_msecs_out,
1567 "DAC period size in milliseconds", NULL, 0},
1568 {"ADC_MS", AUD_OPT_INT, &s_pulseCfg.buffer_msecs_in,
1569 "ADC period size in milliseconds", NULL, 0}
1570};
1571
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