VirtualBox

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

Last change on this file since 57538 was 57451, checked in by vboxsync, 9 years ago

Audio: Try to fix chopped off samples when playing short sound files.

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