VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/DrvAudioVideoRec.cpp@ 65934

Last change on this file since 65934 was 65699, checked in by vboxsync, 8 years ago

Audio: Bugfixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 30.6 KB
Line 
1/* $Id: DrvAudioVideoRec.cpp 65699 2017-02-09 13:25:02Z vboxsync $ */
2/** @file
3 * Video recording audio backend for Main.
4 */
5
6/*
7 * Copyright (C) 2016-2017 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 * This driver is part of Main and is responsible for providing audio
20 * data to Main's video capturing feature.
21 *
22 * The driver itself implements a PDM host audio backend, which in turn
23 * provides the driver with the required audio data and audio events.
24 *
25 * For now there is support for the following destinations (called "sinks"):
26 *
27 * - Direct writing of .webm files to the host.
28 * - Communicating with Main via the Console object to send the encoded audio data to.
29 * The Console object in turn then will route the data to the Display / video capturing interface then.
30 */
31
32/*********************************************************************************************************************************
33* Header Files *
34*********************************************************************************************************************************/
35#define LOG_GROUP LOG_GROUP_DRV_HOST_AUDIO
36#include <VBox/log.h>
37#include "DrvAudioVideoRec.h"
38#include "ConsoleImpl.h"
39
40#include "../../Devices/Audio/DrvAudio.h"
41#include "EbmlWriter.h"
42
43#include <iprt/mem.h>
44#include <iprt/cdefs.h>
45
46#include <VBox/vmm/pdmaudioifs.h>
47#include <VBox/vmm/pdmdrv.h>
48#include <VBox/vmm/cfgm.h>
49#include <VBox/err.h>
50
51#ifdef VBOX_WITH_LIBOPUS
52# include <opus.h>
53#endif
54
55/*********************************************************************************************************************************
56* Structures and Typedefs *
57*********************************************************************************************************************************/
58
59/**
60 * Enumeration for specifying the recording container type.
61 */
62typedef enum AVRECCONTAINERTYPE
63{
64 /** Unknown / invalid container type. */
65 AVRECCONTAINERTYPE_UNKNOWN = 0,
66 /** Recorded data goes to Main / Console. */
67 AVRECCONTAINERTYPE_MAIN_CONSOLE = 1,
68 /** Recorded data will be written a .webm file. */
69 AVRECCONTAINERTYPE_WEBM = 2
70} AVRECCONTAINERTYPE;
71
72/**
73 * Structure for keeping generic container parameters.
74 */
75typedef struct AVRECCONTAINERPARMS
76{
77 /** The container's type. */
78 AVRECCONTAINERTYPE enmType;
79
80} AVRECCONTAINERPARMS, *PAVRECCONTAINERPARMS;
81
82/**
83 * Structure for keeping container-specific data.
84 */
85typedef struct AVRECCONTAINER
86{
87 /** Generic container parameters. */
88 AVRECCONTAINERPARMS Parms;
89
90 union
91 {
92 struct
93 {
94 /** Pointer to Console. */
95 Console *pConsole;
96 } Main;
97
98 struct
99 {
100 /** Pointer to WebM container to write recorded audio data to.
101 * See the AVRECMODE enumeration for more information. */
102 WebMWriter *pWebM;
103 /** Assigned track number from WebM container. */
104 uint8_t uTrack;
105 } WebM;
106 };
107} AVRECCONTAINER, *PAVRECCONTAINER;
108
109/**
110 * Structure for keeping generic codec parameters.
111 */
112typedef struct AVRECCODECPARMS
113{
114 /** The encoding rate to use. */
115 uint32_t uHz;
116 /** Duration of the frame in samples (per channel).
117 *
118 * For Opus, valid frame size are:
119 * ms Frame size
120 * 2.5 120
121 * 5 240
122 * 10 480
123 * 20 (Default) 960
124 * 40 1920
125 * 60 2880
126 */
127 /** Number of audio channels to encode.
128 * Currently we only supported stereo (2) channels. */
129 uint8_t cChannels;
130 /** The codec's bitrate. 0 if not used / cannot be specified. */
131 uint32_t uBitrate;
132
133} AVRECCODECPARMS, *PAVRECCODECPARMS;
134
135/**
136 * Structure for keeping codec-specific data.
137 */
138typedef struct AVRECCODEC
139{
140 /** Generic codec parameters. */
141 AVRECCODECPARMS Parms;
142 union
143 {
144#ifdef VBOX_WITH_LIBOPUS
145 struct
146 {
147 /** Encoder we're going to use. */
148 OpusEncoder *pEnc;
149 /** Number of samples per frame. */
150 uint32_t csFrame;
151 /** The maximum frame size (in samples) we can handle. */
152 uint32_t csFrameMax;
153 } Opus;
154#endif /* VBOX_WITH_LIBOPUS */
155 };
156
157#ifdef VBOX_WITH_STATISTICS /** @todo Make these real STAM values. */
158 struct
159 {
160 /** Number of frames encoded. */
161 uint64_t cEncFrames;
162 /** Total time (in ms) of already encoded audio data. */
163 uint64_t msEncTotal;
164 } STAM;
165#endif /* VBOX_WITH_STATISTICS */
166
167} AVRECCODEC, *PAVRECCODEC;
168
169typedef struct AVRECSINK
170{
171 /** @todo Add types for container / codec as soon as we implement more stuff. */
172
173 /** Container data to use for data processing. */
174 AVRECCONTAINER Con;
175 /** Codec data this stream uses for encoding. */
176 AVRECCODEC Codec;
177} AVRECSINK, *PAVRECSINK;
178
179/**
180 * Audio video recording (output) stream.
181 */
182typedef struct AVRECSTREAM
183{
184 /** The stream's acquired configuration. */
185 PPDMAUDIOSTREAMCFG pCfg;
186 /** (Audio) frame buffer. */
187 PRTCIRCBUF pCircBuf;
188 /** Pointer to sink to use for writing. */
189 PAVRECSINK pSink;
190} AVRECSTREAM, *PAVRECSTREAM;
191
192/**
193 * Video recording audio driver instance data.
194 */
195typedef struct DRVAUDIOVIDEOREC
196{
197 /** Pointer to audio video recording object. */
198 AudioVideoRec *pAudioVideoRec;
199 /** Pointer to the driver instance structure. */
200 PPDMDRVINS pDrvIns;
201 /** Pointer to host audio interface. */
202 PDMIHOSTAUDIO IHostAudio;
203 /** Pointer to the console object. */
204 ComObjPtr<Console> pConsole;
205 /** Pointer to the DrvAudio port interface that is above us. */
206 PPDMIAUDIOCONNECTOR pDrvAudio;
207 /** The driver's sink for writing output to. */
208 AVRECSINK Sink;
209} DRVAUDIOVIDEOREC, *PDRVAUDIOVIDEOREC;
210
211/** Makes DRVAUDIOVIDEOREC out of PDMIHOSTAUDIO. */
212#define PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface) \
213 ( (PDRVAUDIOVIDEOREC)((uintptr_t)pInterface - RT_OFFSETOF(DRVAUDIOVIDEOREC, IHostAudio)) )
214
215/**
216 * Initializes a recording sink.
217 *
218 * @returns IPRT status code.
219 * @param pThis Driver instance.
220 * @param pSink Sink to initialize.
221 * @param pConParms Container parameters to set.
222 * @param pCodecParms Codec parameters to set.
223 */
224static int avRecSinkInit(PDRVAUDIOVIDEOREC pThis, PAVRECSINK pSink, PAVRECCONTAINERPARMS pConParms, PAVRECCODECPARMS pCodecParms)
225{
226 uint32_t uHz = pCodecParms->uHz;
227
228 /* Opus only supports certain input sample rates in an efficient manner.
229 * So make sure that we use those by resampling the data to the requested rate. */
230 if (uHz > 24000) uHz = 48000;
231 else if (uHz > 16000) uHz = 24000;
232 else if (uHz > 12000) uHz = 16000;
233 else if (uHz > 8000 ) uHz = 12000;
234 else uHz = 8000;
235
236 OpusEncoder *pEnc = NULL;
237
238 int orc;
239 pEnc = opus_encoder_create(pCodecParms->uHz, pCodecParms->cChannels, OPUS_APPLICATION_AUDIO, &orc);
240 if (orc != OPUS_OK)
241 {
242 LogRel(("VideoRec: Audio codec failed to initialize: %s\n", opus_strerror(orc)));
243 return VERR_AUDIO_BACKEND_INIT_FAILED;
244 }
245
246 AssertPtr(pEnc);
247
248 opus_encoder_ctl(pEnc, OPUS_SET_BITRATE(pCodecParms->uBitrate));
249 if (orc != OPUS_OK)
250 {
251 LogRel(("VideoRec: Audio codec failed to set bitrate (%RU32): %s\n", pCodecParms->uBitrate, opus_strerror(orc)));
252 return VERR_AUDIO_BACKEND_INIT_FAILED;
253 }
254
255 LogRel(("VideoRec: Recording audio in %RU16Hz, %RU8 channels, %RU32 bpS\n",
256 pCodecParms->uHz, pCodecParms->cChannels, pCodecParms->uBitrate / 1000));
257
258 int rc;
259
260 try
261 {
262 switch (pConParms->enmType)
263 {
264 case AVRECCONTAINERTYPE_MAIN_CONSOLE:
265 {
266 pSink->Con.Main.pConsole = pThis->pConsole;
267
268 rc = VINF_SUCCESS;
269 break;
270 }
271
272 case AVRECCONTAINERTYPE_WEBM:
273 {
274 rc = VINF_SUCCESS;
275 break;
276 }
277
278 default:
279 rc = VERR_NOT_SUPPORTED;
280 break;
281 }
282 }
283 catch (std::bad_alloc)
284 {
285 rc = VERR_NO_MEMORY;
286 }
287
288 if (RT_SUCCESS(rc))
289 {
290 pSink->Con.Parms.enmType = pConParms->enmType;
291
292 pSink->Codec.Parms.uHz = uHz;
293 pSink->Codec.Parms.cChannels = pCodecParms->cChannels;
294 pSink->Codec.Parms.uBitrate = pCodecParms->uBitrate;
295
296 pSink->Codec.Opus.pEnc = pEnc;
297 pSink->Codec.Opus.csFrame = uHz / 50;
298
299#ifdef VBOX_WITH_STATISTICS
300 pSink->Codec.STAM.cEncFrames = 0;
301 pSink->Codec.STAM.msEncTotal = 0;
302#endif
303
304 /* Calculate the maximum frame size. */
305 pSink->Codec.Opus.csFrameMax = 48000 /* Maximum sample rate Opus can handle */
306 * pCodecParms->cChannels; /* Number of channels */
307 }
308
309 return rc;
310}
311
312
313/**
314 * Shuts down (closes) a recording sink,
315 *
316 * @returns IPRT status code.
317 * @param pSink Recording sink to shut down.
318 */
319static void avRecSinkShutdown(PAVRECSINK pSink)
320{
321 AssertPtrReturnVoid(pSink);
322
323#ifdef VBOX_WITH_LIBOPUS
324 if (pSink->Codec.Opus.pEnc)
325 {
326 opus_encoder_destroy(pSink->Codec.Opus.pEnc);
327 pSink->Codec.Opus.pEnc = NULL;
328 }
329#endif
330 switch (pSink->Con.Parms.enmType)
331 {
332 case AVRECCONTAINERTYPE_WEBM:
333 {
334 if (pSink->Con.WebM.pWebM)
335 {
336 int rc2 = pSink->Con.WebM.pWebM->Close();
337 AssertRC(rc2);
338
339 delete pSink->Con.WebM.pWebM;
340 pSink->Con.WebM.pWebM = NULL;
341 }
342 break;
343 }
344
345 case AVRECCONTAINERTYPE_MAIN_CONSOLE:
346 default:
347 break;
348 }
349}
350
351
352/**
353 * Creates an audio output stream and associates it with the specified recording sink.
354 *
355 * @returns IPRT status code.
356 * @param pThis Driver instance.
357 * @param pStreamAV Audio output stream to create.
358 * @param pSink Recording sink to associate audio output stream to.
359 * @param pCfgReq Requested configuration by the audio backend.
360 * @param pCfgAcq Acquired configuration by the audio output stream.
361 */
362static int avRecCreateStreamOut(PDRVAUDIOVIDEOREC pThis, PAVRECSTREAM pStreamAV,
363 PAVRECSINK pSink, PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
364{
365 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
366 AssertPtrReturn(pStreamAV, VERR_INVALID_POINTER);
367 AssertPtrReturn(pSink, VERR_INVALID_POINTER);
368 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
369 AssertPtrReturn(pCfgAcq, VERR_INVALID_POINTER);
370
371 if (pCfgReq->DestSource.Dest != PDMAUDIOPLAYBACKDEST_FRONT)
372 {
373 AssertFailed();
374
375 if (pCfgAcq)
376 pCfgAcq->cSampleBufferHint = 0;
377
378 LogRel2(("VideoRec: Support for surround audio not implemented yet\n"));
379 return VERR_NOT_SUPPORTED;
380 }
381
382 int rc = VINF_SUCCESS;
383
384#ifdef VBOX_WITH_LIBOPUS
385 /* If we only record audio, create our own WebM writer instance here. */
386 if (pSink->Con.Parms.enmType == AVRECCONTAINERTYPE_WEBM)
387 {
388 pSink->Con.WebM.pWebM = new WebMWriter();
389 rc = pSink->Con.WebM.pWebM->Create("/tmp/acap.webm", RTFILE_O_CREATE_REPLACE | RTFILE_O_WRITE | RTFILE_O_DENY_WRITE, /** @todo Fix path! */
390 WebMWriter::AudioCodec_Opus, WebMWriter::VideoCodec_None);
391 if (RT_SUCCESS(rc))
392 rc = pSink->Con.WebM.pWebM->AddAudioTrack(pSink->Codec.Parms.uHz, pCfgReq->Props.cChannels, pCfgReq->Props.cBits,
393 &pSink->Con.WebM.uTrack);
394 }
395
396 if (RT_FAILURE(rc))
397 return rc;
398
399 rc = RTCircBufCreate(&pStreamAV->pCircBuf, (pSink->Codec.Opus.csFrame * pSink->Codec.Parms.cChannels) * sizeof(uint16_t));
400 if (RT_SUCCESS(rc))
401 {
402 pStreamAV->pSink = pSink; /* Assign sink to stream. */
403
404 if (pCfgAcq)
405 {
406 /* Make sure to let the driver backend know that we need the audio data in
407 * a specific sampling rate Opus is optimized for. */
408 pCfgAcq->Props.uHz = pSink->Codec.Parms.uHz;
409 pCfgAcq->Props.cShift = PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(pCfgAcq->Props.cBits, pCfgAcq->Props.cChannels);
410 pCfgAcq->cSampleBufferHint = _4K; /** @todo Make this configurable. */
411 }
412 }
413#else
414 RT_NOREF(pThis, pSink, pStreamAV, pCfgReq, pCfgAcq);
415 rc = VERR_NOT_SUPPORTED;
416#endif /* VBOX_WITH_LIBOPUS */
417
418 LogFlowFuncLeaveRC(rc);
419 return rc;
420}
421
422
423/**
424 * Destroys (closes) an audio output stream.
425 *
426 * @returns IPRT status code.
427 * @param pThis Driver instance.
428 * @param pStreamAV Audio output stream to destroy.
429 */
430static int avRecDestroyStreamOut(PDRVAUDIOVIDEOREC pThis, PAVRECSTREAM pStreamAV)
431{
432 RT_NOREF(pThis);
433
434 if (pStreamAV->pCircBuf)
435 {
436 RTCircBufDestroy(pStreamAV->pCircBuf);
437 pStreamAV->pCircBuf = NULL;
438 }
439
440 return VINF_SUCCESS;
441}
442
443
444/**
445 * Controls an audio output stream
446 *
447 * @returns IPRT status code.
448 * @param pThis Driver instance.
449 * @param pStreamAV Audio output stream to control.
450 * @param enmStreamCmd Stream command to issue.
451 */
452static int avRecControlStreamOut(PDRVAUDIOVIDEOREC pThis,
453 PAVRECSTREAM pStreamAV, PDMAUDIOSTREAMCMD enmStreamCmd)
454{
455 RT_NOREF(pThis, pStreamAV);
456
457 switch (enmStreamCmd)
458 {
459 case PDMAUDIOSTREAMCMD_ENABLE:
460 case PDMAUDIOSTREAMCMD_DISABLE:
461 case PDMAUDIOSTREAMCMD_RESUME:
462 case PDMAUDIOSTREAMCMD_PAUSE:
463 break;
464
465 default:
466 AssertMsgFailed(("Invalid command %ld\n", enmStreamCmd));
467 break;
468 }
469
470 return VINF_SUCCESS;
471}
472
473
474/**
475 * @interface_method_impl{PDMIHOSTAUDIO,pfnInit}
476 */
477static DECLCALLBACK(int) drvAudioVideoRecInit(PPDMIHOSTAUDIO pInterface)
478{
479 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
480
481 LogFlowFuncEnter();
482
483 PDRVAUDIOVIDEOREC pThis = PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface);
484
485 AVRECCONTAINERPARMS ContainerParms;
486 ContainerParms.enmType = AVRECCONTAINERTYPE_MAIN_CONSOLE; /** @todo Make this configurable. */
487
488 AVRECCODECPARMS CodecParms;
489 CodecParms.uHz = 48000; /** @todo Make this configurable. */
490 CodecParms.cChannels = 2; /** @todo Make this configurable. */
491 CodecParms.uBitrate = 196000; /** @todo Make this configurable. */
492
493 int rc = avRecSinkInit(pThis, &pThis->Sink, &ContainerParms, &CodecParms);
494 if (RT_FAILURE(rc))
495 {
496 LogRel(("VideoRec: Audio recording driver failed to initialize, rc=%Rrc\n", rc));
497 }
498 else
499 LogRel2(("VideoRec: Audio recording driver initialized\n"));
500
501 return rc;
502}
503
504
505/**
506 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCapture}
507 */
508static DECLCALLBACK(int) drvAudioVideoRecStreamCapture(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
509 void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead)
510{
511 RT_NOREF(pInterface, pStream, pvBuf, cbBuf);
512
513 if (pcbRead)
514 *pcbRead = 0;
515
516 return VINF_SUCCESS;
517}
518
519
520/**
521 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamPlay}
522 */
523static DECLCALLBACK(int) drvAudioVideoRecStreamPlay(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
524 const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten)
525{
526 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
527 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
528 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
529 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
530 /* pcbWritten is optional. */
531
532 PDRVAUDIOVIDEOREC pThis = PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface);
533 RT_NOREF(pThis);
534 PAVRECSTREAM pStreamAV = (PAVRECSTREAM)pStream;
535
536 int rc = VINF_SUCCESS;
537
538 uint32_t cbWrittenTotal = 0;
539
540 /*
541 * Call the encoder with the data.
542 */
543#ifdef VBOX_WITH_LIBOPUS
544 PAVRECSINK pSink = pStreamAV->pSink;
545 AssertPtr(pSink);
546 PRTCIRCBUF pCircBuf = pStreamAV->pCircBuf;
547 AssertPtr(pCircBuf);
548
549 void *pvCircBuf;
550 size_t cbCircBuf;
551
552 uint32_t cbToWrite = cbBuf;
553
554 /*
555 * Fetch as much as we can into our internal ring buffer.
556 */
557 while ( cbToWrite
558 && RTCircBufFree(pCircBuf))
559 {
560 RTCircBufAcquireWriteBlock(pCircBuf, cbToWrite, &pvCircBuf, &cbCircBuf);
561
562 if (cbCircBuf)
563 {
564 memcpy(pvCircBuf, (uint8_t *)pvBuf + cbWrittenTotal, cbCircBuf),
565 cbWrittenTotal += cbCircBuf;
566 Assert(cbToWrite >= cbCircBuf);
567 cbToWrite -= cbCircBuf;
568 }
569
570 RTCircBufReleaseWriteBlock(pCircBuf, cbCircBuf);
571
572 if ( RT_FAILURE(rc)
573 || !cbCircBuf)
574 {
575 break;
576 }
577 }
578
579 /*
580 * Process our internal ring buffer and encode the data.
581 */
582
583 uint8_t abSrc[_64K]; /** @todo Fix! */
584 size_t cbSrc;
585
586 const uint32_t csFrame = pSink->Codec.Opus.csFrame;
587 const uint32_t cbFrame = PDMAUDIOSTREAMCFG_S2B(pStreamAV->pCfg, csFrame);
588
589 while (RTCircBufUsed(pCircBuf) >= cbFrame)
590 {
591 cbSrc = 0;
592
593 while (cbSrc < cbFrame)
594 {
595 RTCircBufAcquireReadBlock(pCircBuf, cbFrame - cbSrc, &pvCircBuf, &cbCircBuf);
596
597 if (cbCircBuf)
598 {
599 memcpy(&abSrc[cbSrc], pvCircBuf, cbCircBuf);
600
601 cbSrc += cbCircBuf;
602 Assert(cbSrc <= sizeof(abSrc));
603 }
604
605 RTCircBufReleaseReadBlock(pCircBuf, cbCircBuf);
606
607 if (!cbCircBuf)
608 break;
609 }
610
611# ifdef DEBUG_andy
612 RTFILE fh;
613 RTFileOpen(&fh, "/tmp/acap.pcm", RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
614 RTFileWrite(fh, abSrc, cbSrc, NULL);
615 RTFileClose(fh);
616# endif
617
618 Assert(cbSrc == cbFrame);
619
620 /*
621 * Opus always encodes PER FRAME, that is, exactly 2.5, 5, 10, 20, 40 or 60 ms of audio data.
622 *
623 * A packet can have up to 120ms worth of audio data.
624 * Anything > 120ms of data will result in a "corrupted package" error message by
625 * by decoding application.
626 */
627 uint8_t abDst[_64K]; /** @todo Fix! */
628 size_t cbDst = sizeof(abDst);
629
630 /* Call the encoder to encode one frame per iteration. */
631 opus_int32 cbWritten = opus_encode(pSink->Codec.Opus.pEnc,
632 (opus_int16 *)abSrc, csFrame, abDst, cbDst);
633 if (cbWritten > 0)
634 {
635# ifdef VBOX_WITH_STATISTICS
636 /* Get overall frames encoded. */
637 uint32_t cEncFrames = opus_packet_get_nb_frames(abDst, cbDst);
638 uint32_t cEncSamplesPerFrame = opus_packet_get_samples_per_frame(abDst, pSink->Codec.Parms.uHz);
639 uint32_t csEnc = cEncFrames * cEncSamplesPerFrame;
640
641 pSink->Codec.STAM.cEncFrames += cEncFrames;
642 pSink->Codec.STAM.msEncTotal += 20 /* Default 20 ms */ * cEncFrames;
643
644 LogFunc(("%RU64ms [%RU64 frames]: cbSrc=%zu, cbDst=%zu, cEncFrames=%RU32, cEncSamplesPerFrame=%RU32, csEnc=%RU32\n",
645 pSink->Codec.STAM.msEncTotal, pSink->Codec.STAM.cEncFrames,
646 cbSrc, cbDst, cEncFrames, cEncSamplesPerFrame, csEnc));
647# endif
648 Assert((uint32_t)cbWritten <= cbDst);
649 cbDst = RT_MIN((uint32_t)cbWritten, cbDst); /* Update cbDst to actual bytes encoded (written). */
650
651 switch (pSink->Con.Parms.enmType)
652 {
653 case AVRECCONTAINERTYPE_MAIN_CONSOLE:
654 {
655 HRESULT hr = pSink->Con.Main.pConsole->i_audioVideoRecSendAudio(abDst, cbDst, RTTimeMilliTS() /* Now */);
656 Assert(hr == S_OK);
657
658 break;
659 }
660
661 case AVRECCONTAINERTYPE_WEBM:
662 {
663 WebMWriter::BlockData_Opus blockData = { abDst, cbDst, RTTimeMilliTS() };
664 rc = pSink->Con.WebM.pWebM->WriteBlock(pSink->Con.WebM.uTrack, &blockData, sizeof(blockData));
665 AssertRC(rc);
666
667 break;
668 }
669
670 default:
671 AssertFailedStmt(rc = VERR_NOT_IMPLEMENTED);
672 break;
673 }
674 }
675 else if (cbWritten < 0)
676 {
677 AssertMsgFailed(("Encoding failed: %s\n", opus_strerror(cbWritten)));
678 rc = VERR_INVALID_PARAMETER;
679 }
680
681 if (RT_FAILURE(rc))
682 break;
683 }
684#else
685 rc = VERR_NOT_SUPPORTED;
686#endif /* VBOX_WITH_LIBOPUS */
687
688 /*
689 * Always report back all samples acquired, regardless of whether the
690 * encoder actually did process those.
691 */
692 if (pcbWritten)
693 *pcbWritten = cbWrittenTotal;
694
695 LogFlowFunc(("csReadTotal=%RU32, rc=%Rrc\n", cbWrittenTotal, rc));
696 return rc;
697}
698
699
700/**
701 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetConfig}
702 */
703static DECLCALLBACK(int) drvAudioVideoRecGetConfig(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDCFG pBackendCfg)
704{
705 RT_NOREF(pInterface);
706 AssertPtrReturn(pBackendCfg, VERR_INVALID_POINTER);
707
708 pBackendCfg->cbStreamOut = sizeof(AVRECSTREAM);
709 pBackendCfg->cbStreamIn = 0;
710 pBackendCfg->cMaxStreamsIn = 0;
711 pBackendCfg->cMaxStreamsOut = UINT32_MAX;
712
713 return VINF_SUCCESS;
714}
715
716
717/**
718 * @interface_method_impl{PDMIHOSTAUDIO,pfnShutdown}
719 */
720static DECLCALLBACK(void) drvAudioVideoRecShutdown(PPDMIHOSTAUDIO pInterface)
721{
722 LogFlowFuncEnter();
723
724 PDRVAUDIOVIDEOREC pThis = PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface);
725
726 avRecSinkShutdown(&pThis->Sink);
727}
728
729
730/**
731 * @interface_method_impl{PDMIHOSTAUDIO,pfnGetStatus}
732 */
733static DECLCALLBACK(PDMAUDIOBACKENDSTS) drvAudioVideoRecGetStatus(PPDMIHOSTAUDIO pInterface, PDMAUDIODIR enmDir)
734{
735 RT_NOREF(enmDir);
736 AssertPtrReturn(pInterface, PDMAUDIOBACKENDSTS_UNKNOWN);
737
738 return PDMAUDIOBACKENDSTS_RUNNING;
739}
740
741
742/**
743 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamCreate}
744 */
745static DECLCALLBACK(int) drvAudioVideoRecStreamCreate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream,
746 PPDMAUDIOSTREAMCFG pCfgReq, PPDMAUDIOSTREAMCFG pCfgAcq)
747{
748 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
749 AssertPtrReturn(pCfgReq, VERR_INVALID_POINTER);
750 AssertPtrReturn(pCfgAcq, VERR_INVALID_POINTER);
751
752 if (pCfgReq->enmDir == PDMAUDIODIR_IN)
753 return VERR_NOT_SUPPORTED;
754
755 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
756
757 PDRVAUDIOVIDEOREC pThis = PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface);
758 PAVRECSTREAM pStreamAV = (PAVRECSTREAM)pStream;
759
760 /* For now we only have one sink, namely the driver's one.
761 * Later each stream could have its own one, to e.g. router different stream to different sinks .*/
762 PAVRECSINK pSink = &pThis->Sink;
763
764 int rc = avRecCreateStreamOut(pThis, pStreamAV, pSink, pCfgReq, pCfgAcq);
765 if (RT_SUCCESS(rc))
766 {
767 pStreamAV->pCfg = DrvAudioHlpStreamCfgDup(pCfgAcq);
768 if (!pStreamAV->pCfg)
769 rc = VERR_NO_MEMORY;
770 }
771
772 return rc;
773}
774
775
776/**
777 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamDestroy}
778 */
779static DECLCALLBACK(int) drvAudioVideoRecStreamDestroy(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
780{
781 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
782 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
783
784 PDRVAUDIOVIDEOREC pThis = PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface);
785 PAVRECSTREAM pStreamAV = (PAVRECSTREAM)pStream;
786
787 if (!pStreamAV->pCfg) /* Not (yet) configured? Skip. */
788 return VINF_SUCCESS;
789
790 int rc = VINF_SUCCESS;
791
792 if (pStreamAV->pCfg->enmDir == PDMAUDIODIR_OUT)
793 rc = avRecDestroyStreamOut(pThis, pStreamAV);
794
795 if (RT_SUCCESS(rc))
796 {
797 DrvAudioHlpStreamCfgFree(pStreamAV->pCfg);
798 pStreamAV->pCfg = NULL;
799 }
800
801 return rc;
802}
803
804
805/**
806 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamControl}
807 */
808static DECLCALLBACK(int) drvAudioVideoRecStreamControl(PPDMIHOSTAUDIO pInterface,
809 PPDMAUDIOBACKENDSTREAM pStream, PDMAUDIOSTREAMCMD enmStreamCmd)
810{
811 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
812 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
813
814 PDRVAUDIOVIDEOREC pThis = PDMIHOSTAUDIO_2_DRVAUDIOVIDEOREC(pInterface);
815 PAVRECSTREAM pStreamAV = (PAVRECSTREAM)pStream;
816
817 if (!pStreamAV->pCfg) /* Not (yet) configured? Skip. */
818 return VINF_SUCCESS;
819
820 if (pStreamAV->pCfg->enmDir == PDMAUDIODIR_OUT)
821 return avRecControlStreamOut(pThis, pStreamAV, enmStreamCmd);
822
823 return VINF_SUCCESS;
824}
825
826
827/**
828 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetReadable}
829 */
830static DECLCALLBACK(uint32_t) drvAudioVideoRecStreamGetReadable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
831{
832 RT_NOREF(pInterface, pStream);
833
834 return 0; /* Video capturing does not provide any input. */
835}
836
837
838/**
839 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetWritable}
840 */
841static DECLCALLBACK(uint32_t) drvAudioVideoRecStreamGetWritable(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
842{
843 RT_NOREF(pInterface, pStream);
844
845 return UINT32_MAX;
846}
847
848
849/**
850 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamGetStatus}
851 */
852static DECLCALLBACK(PDMAUDIOSTRMSTS) drvAudioVideoRecStreamGetStatus(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
853{
854 RT_NOREF(pInterface, pStream);
855
856 return (PDMAUDIOSTRMSTS_FLAG_INITIALIZED | PDMAUDIOSTRMSTS_FLAG_ENABLED);
857}
858
859
860/**
861 * @interface_method_impl{PDMIHOSTAUDIO,pfnStreamIterate}
862 */
863static DECLCALLBACK(int) drvAudioVideoRecStreamIterate(PPDMIHOSTAUDIO pInterface, PPDMAUDIOBACKENDSTREAM pStream)
864{
865 AssertPtrReturn(pInterface, VERR_INVALID_POINTER);
866 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
867
868 LogFlowFuncEnter();
869
870 /* Nothing to do here for video recording. */
871 return VINF_SUCCESS;
872}
873
874
875/**
876 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
877 */
878static DECLCALLBACK(void *) drvAudioVideoRecQueryInterface(PPDMIBASE pInterface, const char *pszIID)
879{
880 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
881 PDRVAUDIOVIDEOREC pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIOVIDEOREC);
882
883 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
884 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTAUDIO, &pThis->IHostAudio);
885 return NULL;
886}
887
888
889AudioVideoRec::AudioVideoRec(Console *pConsole)
890 : mpDrv(NULL)
891 , mpConsole(pConsole)
892{
893}
894
895
896AudioVideoRec::~AudioVideoRec(void)
897{
898 if (mpDrv)
899 {
900 mpDrv->pAudioVideoRec = NULL;
901 mpDrv = NULL;
902 }
903}
904
905
906/**
907 * Construct a audio video recording driver instance.
908 *
909 * @copydoc FNPDMDRVCONSTRUCT
910 */
911/* static */
912DECLCALLBACK(int) AudioVideoRec::drvConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
913{
914 RT_NOREF(fFlags);
915
916 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
917 PDRVAUDIOVIDEOREC pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIOVIDEOREC);
918
919 AssertPtrReturn(pDrvIns, VERR_INVALID_POINTER);
920 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
921
922 LogRel(("Audio: Initializing video recording audio driver\n"));
923 LogFlowFunc(("fFlags=0x%x\n", fFlags));
924
925 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
926 ("Configuration error: Not possible to attach anything to this driver!\n"),
927 VERR_PDM_DRVINS_NO_ATTACH);
928
929 /*
930 * Init the static parts.
931 */
932 pThis->pDrvIns = pDrvIns;
933 /* IBase */
934 pDrvIns->IBase.pfnQueryInterface = drvAudioVideoRecQueryInterface;
935 /* IHostAudio */
936 PDMAUDIO_IHOSTAUDIO_CALLBACKS(drvAudioVideoRec);
937
938 /*
939 * Get the Console object pointer.
940 */
941 void *pvUser;
942 int rc = CFGMR3QueryPtr(pCfg, "ObjectConsole", &pvUser); /** @todo r=andy Get rid of this hack and use IHostAudio::SetCallback. */
943 AssertMsgRCReturn(rc, ("Confguration error: No/bad \"ObjectConsole\" value, rc=%Rrc\n", rc), rc);
944
945 /* CFGM tree saves the pointer to Console in the Object node of AudioVideoRec. */
946 pThis->pConsole = (Console *)pvUser;
947
948 /*
949 * Get the pointer to the audio driver instance.
950 */
951 rc = CFGMR3QueryPtr(pCfg, "Object", &pvUser); /** @todo r=andy Get rid of this hack and use IHostAudio::SetCallback. */
952 AssertMsgRCReturn(rc, ("Confguration error: No/bad \"Object\" value, rc=%Rrc\n", rc), rc);
953
954 pThis->pAudioVideoRec = (AudioVideoRec *)pvUser;
955 pThis->pAudioVideoRec->mpDrv = pThis;
956
957 /*
958 * Get the interface for the above driver (DrvAudio) to make mixer/conversion calls.
959 * Described in CFGM tree.
960 */
961 pThis->pDrvAudio = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIAUDIOCONNECTOR);
962 AssertMsgReturn(pThis->pDrvAudio, ("Configuration error: No upper interface specified!\n"), VERR_PDM_MISSING_INTERFACE_ABOVE);
963
964 return VINF_SUCCESS;
965}
966
967
968/**
969 * @interface_method_impl{PDMDRVREG,pfnDestruct}
970 */
971/* static */
972DECLCALLBACK(void) AudioVideoRec::drvDestruct(PPDMDRVINS pDrvIns)
973{
974 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
975 PDRVAUDIOVIDEOREC pThis = PDMINS_2_DATA(pDrvIns, PDRVAUDIOVIDEOREC);
976 LogFlowFuncEnter();
977
978 /*
979 * If the AudioVideoRec object is still alive, we must clear it's reference to
980 * us since we'll be invalid when we return from this method.
981 */
982 if (pThis->pAudioVideoRec)
983 {
984 pThis->pAudioVideoRec->mpDrv = NULL;
985 pThis->pAudioVideoRec = NULL;
986 }
987}
988
989
990/**
991 * Video recording audio driver registration record.
992 */
993const PDMDRVREG AudioVideoRec::DrvReg =
994{
995 PDM_DRVREG_VERSION,
996 /* szName */
997 "AudioVideoRec",
998 /* szRCMod */
999 "",
1000 /* szR0Mod */
1001 "",
1002 /* pszDescription */
1003 "Audio driver for video recording",
1004 /* fFlags */
1005 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1006 /* fClass. */
1007 PDM_DRVREG_CLASS_AUDIO,
1008 /* cMaxInstances */
1009 ~0U,
1010 /* cbInstance */
1011 sizeof(DRVAUDIOVIDEOREC),
1012 /* pfnConstruct */
1013 AudioVideoRec::drvConstruct,
1014 /* pfnDestruct */
1015 AudioVideoRec::drvDestruct,
1016 /* pfnRelocate */
1017 NULL,
1018 /* pfnIOCtl */
1019 NULL,
1020 /* pfnPowerOn */
1021 NULL,
1022 /* pfnReset */
1023 NULL,
1024 /* pfnSuspend */
1025 NULL,
1026 /* pfnResume */
1027 NULL,
1028 /* pfnAttach */
1029 NULL,
1030 /* pfnDetach */
1031 NULL,
1032 /* pfnPowerOff */
1033 NULL,
1034 /* pfnSoftReset */
1035 NULL,
1036 /* u32EndVersion */
1037 PDM_DRVREG_VERSION
1038};
1039
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