VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/VideoRec.cpp@ 46107

Last change on this file since 46107 was 45962, checked in by vboxsync, 12 years ago

Main/VPX: more fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
  • Property svn:mergeinfo set to (toggle deleted branches)
    /branches/VBox-3.0/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp58652,​70973
    /branches/VBox-3.2/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp66309,​66318
    /branches/VBox-4.0/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp70873
    /branches/VBox-4.1/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp74233
    /branches/dsen/gui/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp79076-79078,​79089,​79109-79110,​79112-79113,​79127-79130,​79134,​79141,​79151,​79155,​79157-79159,​79193,​79197
    /branches/dsen/gui2/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp79224,​79228,​79233,​79235,​79258,​79262-79263,​79273,​79341,​79345,​79354,​79357,​79387-79388,​79559-79569,​79572-79573,​79578,​79581-79582,​79590-79591,​79598-79599,​79602-79603,​79605-79606,​79632,​79635,​79637,​79644
    /branches/dsen/gui3/src/VBox/Frontends/VBoxHeadless/VideoCapture/EncodeAndWrite.cpp79645-79692
File size: 25.6 KB
Line 
1/* $Id: VideoRec.cpp 45962 2013-05-08 20:39:55Z vboxsync $ */
2/** @file
3 * Encodes the screen content in VPX format.
4 */
5
6/*
7 * Copyright (C) 2012-2013 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#define LOG_GROUP LOG_GROUP_MAIN
19#include <VBox/log.h>
20#include <iprt/asm.h>
21#include <iprt/assert.h>
22#include <iprt/semaphore.h>
23#include <iprt/thread.h>
24
25#include <VBox/com/VirtualBox.h>
26#include <VBox/com/com.h>
27#include <VBox/com/string.h>
28
29#include "EbmlWriter.h"
30#include "VideoRec.h"
31
32#define VPX_CODEC_DISABLE_COMPAT 1
33#include <vpx/vp8cx.h>
34#include <vpx/vpx_image.h>
35
36/** Default VPX codec to use */
37#define DEFAULTCODEC (vpx_codec_vp8_cx())
38
39static int videoRecEncodeAndWrite(PVIDEORECSTREAM pStrm);
40static int videoRecRGBToYUV(PVIDEORECSTREAM pStrm);
41
42/* encoding */
43enum
44{
45 VIDREC_UNINITIALIZED = 0,
46 /* initialized */
47 VIDREC_INITIALIZED = 1,
48 /* signal that we are terminating */
49 VIDREC_TERMINATING = 2,
50 /* confirmation that the worker thread terminated */
51 VIDREC_TERMINATED = 3
52};
53
54typedef struct VIDEORECSTREAM
55{
56 /* container context */
57 EbmlGlobal Ebml;
58 /* VPX codec context */
59 vpx_codec_ctx_t VpxCodec;
60 /* VPX configuration */
61 vpx_codec_enc_cfg_t VpxConfig;
62 /* X resolution */
63 uint32_t uTargetWidth;
64 /* Y resolution */
65 uint32_t uTargetHeight;
66 /* X resolution of the last encoded picture */
67 uint32_t uLastSourceWidth;
68 /* Y resolution of the last encoded picture */
69 uint32_t uLastSourceHeight;
70 /* current frame number */
71 uint32_t cFrame;
72 /* RGB buffer containing the most recent frame of the framebuffer */
73 uint8_t *pu8RgbBuf;
74 /* YUV buffer the encode function fetches the frame from */
75 uint8_t *pu8YuvBuf;
76 /* VPX image context */
77 vpx_image_t VpxRawImage;
78 /* true if video recording is enabled */
79 bool fEnabled;
80 /* true if the RGB buffer is filled */
81 bool fRgbFilled;
82 /* pixel format of the current frame */
83 uint32_t u32PixelFormat;
84 /* minimal delay between two frames */
85 uint32_t uDelay;
86 /* time stamp of the last frame we encoded */
87 uint64_t u64LastTimeStamp;
88 /* time stamp of the current frame */
89 uint64_t u64TimeStamp;
90} VIDEORECSTREAM;
91
92typedef struct VIDEORECCONTEXT
93{
94 /* semaphore */
95 RTSEMEVENT WaitEvent;
96 /* true if video recording is enabled */
97 bool fEnabled;
98 /* worker thread */
99 RTTHREAD Thread;
100 /* see VIDREC_xxx */
101 uint32_t uState;
102 /* number of stream contexts */
103 uint32_t cScreens;
104 /* video recording stream contexts */
105 VIDEORECSTREAM Strm[1];
106} VIDEORECCONTEXT;
107
108
109/**
110 * Iterator class for running through a BGRA32 image buffer and converting
111 * it to RGB.
112 */
113class ColorConvBGRA32Iter
114{
115private:
116 enum { PIX_SIZE = 4 };
117public:
118 ColorConvBGRA32Iter(unsigned aWidth, unsigned aHeight, uint8_t *aBuf)
119 {
120 LogFlow(("width = %d height=%d aBuf=%lx\n", aWidth, aHeight, aBuf));
121 mPos = 0;
122 mSize = aWidth * aHeight * PIX_SIZE;
123 mBuf = aBuf;
124 }
125 /**
126 * Convert the next pixel to RGB.
127 * @returns true on success, false if we have reached the end of the buffer
128 * @param aRed where to store the red value
129 * @param aGreen where to store the green value
130 * @param aBlue where to store the blue value
131 */
132 bool getRGB(unsigned *aRed, unsigned *aGreen, unsigned *aBlue)
133 {
134 bool rc = false;
135 if (mPos + PIX_SIZE <= mSize)
136 {
137 *aRed = mBuf[mPos + 2];
138 *aGreen = mBuf[mPos + 1];
139 *aBlue = mBuf[mPos ];
140 mPos += PIX_SIZE;
141 rc = true;
142 }
143 return rc;
144 }
145
146 /**
147 * Skip forward by a certain number of pixels
148 * @param aPixels how many pixels to skip
149 */
150 void skip(unsigned aPixels)
151 {
152 mPos += PIX_SIZE * aPixels;
153 }
154private:
155 /** Size of the picture buffer */
156 unsigned mSize;
157 /** Current position in the picture buffer */
158 unsigned mPos;
159 /** Address of the picture buffer */
160 uint8_t *mBuf;
161};
162
163/**
164 * Iterator class for running through an BGR24 image buffer and converting
165 * it to RGB.
166 */
167class ColorConvBGR24Iter
168{
169private:
170 enum { PIX_SIZE = 3 };
171public:
172 ColorConvBGR24Iter(unsigned aWidth, unsigned aHeight, uint8_t *aBuf)
173 {
174 mPos = 0;
175 mSize = aWidth * aHeight * PIX_SIZE;
176 mBuf = aBuf;
177 }
178 /**
179 * Convert the next pixel to RGB.
180 * @returns true on success, false if we have reached the end of the buffer
181 * @param aRed where to store the red value
182 * @param aGreen where to store the green value
183 * @param aBlue where to store the blue value
184 */
185 bool getRGB(unsigned *aRed, unsigned *aGreen, unsigned *aBlue)
186 {
187 bool rc = false;
188 if (mPos + PIX_SIZE <= mSize)
189 {
190 *aRed = mBuf[mPos + 2];
191 *aGreen = mBuf[mPos + 1];
192 *aBlue = mBuf[mPos ];
193 mPos += PIX_SIZE;
194 rc = true;
195 }
196 return rc;
197 }
198
199 /**
200 * Skip forward by a certain number of pixels
201 * @param aPixels how many pixels to skip
202 */
203 void skip(unsigned aPixels)
204 {
205 mPos += PIX_SIZE * aPixels;
206 }
207private:
208 /** Size of the picture buffer */
209 unsigned mSize;
210 /** Current position in the picture buffer */
211 unsigned mPos;
212 /** Address of the picture buffer */
213 uint8_t *mBuf;
214};
215
216/**
217 * Iterator class for running through an BGR565 image buffer and converting
218 * it to RGB.
219 */
220class ColorConvBGR565Iter
221{
222private:
223 enum { PIX_SIZE = 2 };
224public:
225 ColorConvBGR565Iter(unsigned aWidth, unsigned aHeight, uint8_t *aBuf)
226 {
227 mPos = 0;
228 mSize = aWidth * aHeight * PIX_SIZE;
229 mBuf = aBuf;
230 }
231 /**
232 * Convert the next pixel to RGB.
233 * @returns true on success, false if we have reached the end of the buffer
234 * @param aRed where to store the red value
235 * @param aGreen where to store the green value
236 * @param aBlue where to store the blue value
237 */
238 bool getRGB(unsigned *aRed, unsigned *aGreen, unsigned *aBlue)
239 {
240 bool rc = false;
241 if (mPos + PIX_SIZE <= mSize)
242 {
243 unsigned uFull = (((unsigned) mBuf[mPos + 1]) << 8)
244 | ((unsigned) mBuf[mPos]);
245 *aRed = (uFull >> 8) & ~7;
246 *aGreen = (uFull >> 3) & ~3 & 0xff;
247 *aBlue = (uFull << 3) & ~7 & 0xff;
248 mPos += PIX_SIZE;
249 rc = true;
250 }
251 return rc;
252 }
253
254 /**
255 * Skip forward by a certain number of pixels
256 * @param aPixels how many pixels to skip
257 */
258 void skip(unsigned aPixels)
259 {
260 mPos += PIX_SIZE * aPixels;
261 }
262private:
263 /** Size of the picture buffer */
264 unsigned mSize;
265 /** Current position in the picture buffer */
266 unsigned mPos;
267 /** Address of the picture buffer */
268 uint8_t *mBuf;
269};
270
271/**
272 * Convert an image to YUV420p format
273 * @returns true on success, false on failure
274 * @param aWidth width of image
275 * @param aHeight height of image
276 * @param aDestBuf an allocated memory buffer large enough to hold the
277 * destination image (i.e. width * height * 12bits)
278 * @param aSrcBuf the source image as an array of bytes
279 */
280template <class T>
281inline bool colorConvWriteYUV420p(unsigned aWidth, unsigned aHeight,
282 uint8_t *aDestBuf, uint8_t *aSrcBuf)
283{
284 AssertReturn(0 == (aWidth & 1), false);
285 AssertReturn(0 == (aHeight & 1), false);
286 bool rc = true;
287 T iter1(aWidth, aHeight, aSrcBuf);
288 T iter2 = iter1;
289 iter2.skip(aWidth);
290 unsigned cPixels = aWidth * aHeight;
291 unsigned offY = 0;
292 unsigned offU = cPixels;
293 unsigned offV = cPixels + cPixels / 4;
294 for (unsigned i = 0; (i < aHeight / 2) && rc; ++i)
295 {
296 for (unsigned j = 0; (j < aWidth / 2) && rc; ++j)
297 {
298 unsigned red, green, blue, u, v;
299 rc = iter1.getRGB(&red, &green, &blue);
300 if (rc)
301 {
302 aDestBuf[offY] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
303 u = (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
304 v = (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
305 rc = iter1.getRGB(&red, &green, &blue);
306 }
307 if (rc)
308 {
309 aDestBuf[offY + 1] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
310 u += (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
311 v += (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
312 rc = iter2.getRGB(&red, &green, &blue);
313 }
314 if (rc)
315 {
316 aDestBuf[offY + aWidth] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
317 u += (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
318 v += (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
319 rc = iter2.getRGB(&red, &green, &blue);
320 }
321 if (rc)
322 {
323 aDestBuf[offY + aWidth + 1] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
324 u += (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
325 v += (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
326 aDestBuf[offU] = u;
327 aDestBuf[offV] = v;
328 offY += 2;
329 ++offU;
330 ++offV;
331 }
332 }
333 if (rc)
334 {
335 iter1.skip(aWidth);
336 iter2.skip(aWidth);
337 offY += aWidth;
338 }
339 }
340 return rc;
341}
342
343/**
344 * Convert an image to RGB24 format
345 * @returns true on success, false on failure
346 * @param aWidth width of image
347 * @param aHeight height of image
348 * @param aDestBuf an allocated memory buffer large enough to hold the
349 * destination image (i.e. width * height * 12bits)
350 * @param aSrcBuf the source image as an array of bytes
351 */
352template <class T>
353inline bool colorConvWriteRGB24(unsigned aWidth, unsigned aHeight,
354 uint8_t *aDestBuf, uint8_t *aSrcBuf)
355{
356 enum { PIX_SIZE = 3 };
357 bool rc = true;
358 AssertReturn(0 == (aWidth & 1), false);
359 AssertReturn(0 == (aHeight & 1), false);
360 T iter(aWidth, aHeight, aSrcBuf);
361 unsigned cPixels = aWidth * aHeight;
362 for (unsigned i = 0; i < cPixels && rc; ++i)
363 {
364 unsigned red, green, blue;
365 rc = iter.getRGB(&red, &green, &blue);
366 if (rc)
367 {
368 aDestBuf[i * PIX_SIZE ] = red;
369 aDestBuf[i * PIX_SIZE + 1] = green;
370 aDestBuf[i * PIX_SIZE + 2] = blue;
371 }
372 }
373 return rc;
374}
375
376/**
377 * Worker thread for all streams.
378 *
379 * RGB/YUV conversion and encoding.
380 */
381static DECLCALLBACK(int) videoRecThread(RTTHREAD Thread, void *pvUser)
382{
383 PVIDEORECCONTEXT pCtx = (PVIDEORECCONTEXT)pvUser;
384 for (;;)
385 {
386 int rc = RTSemEventWait(pCtx->WaitEvent, RT_INDEFINITE_WAIT);
387 AssertRCBreak(rc);
388
389 if (ASMAtomicReadU32(&pCtx->uState) == VIDREC_TERMINATING)
390 break;
391 for (unsigned uScreen = 0; uScreen < pCtx->cScreens; uScreen++)
392 {
393 PVIDEORECSTREAM pStrm = &pCtx->Strm[uScreen];
394 if (ASMAtomicReadBool(&pStrm->fRgbFilled))
395 {
396 rc = videoRecRGBToYUV(pStrm);
397 ASMAtomicWriteBool(&pStrm->fRgbFilled, false);
398 if (RT_SUCCESS(rc))
399 rc = videoRecEncodeAndWrite(pStrm);
400 if (RT_FAILURE(rc))
401 {
402 static unsigned cErrors = 100;
403 if (cErrors > 0)
404 {
405 LogRel(("Error %Rrc encoding / writing video frame\n", rc));
406 cErrors--;
407 }
408 }
409 }
410 }
411 }
412
413 ASMAtomicWriteU32(&pCtx->uState, VIDREC_TERMINATED);
414 return VINF_SUCCESS;
415}
416
417/**
418 * VideoRec utility function to create video recording context.
419 *
420 * @returns IPRT status code.
421 * @param ppCtx Video recording context
422 * @param cScreens Number of screens.
423 */
424int VideoRecContextCreate(PVIDEORECCONTEXT *ppCtx, uint32_t cScreens)
425{
426 PVIDEORECCONTEXT pCtx = (PVIDEORECCONTEXT)RTMemAllocZ(RT_OFFSETOF(VIDEORECCONTEXT, Strm[cScreens]));
427 *ppCtx = pCtx;
428 AssertPtrReturn(pCtx, VERR_NO_MEMORY);
429
430 pCtx->cScreens = cScreens;
431 for (unsigned uScreen = 0; uScreen < cScreens; uScreen++)
432 pCtx->Strm[uScreen].Ebml.last_pts_ms = -1;
433
434 int rc = RTSemEventCreate(&pCtx->WaitEvent);
435 AssertRCReturn(rc, rc);
436
437 rc = RTThreadCreate(&pCtx->Thread, videoRecThread, (void*)pCtx, 0,
438 RTTHREADTYPE_MAIN_WORKER, RTTHREADFLAGS_WAITABLE, "VideoRec");
439 AssertRCReturn(rc, rc);
440
441 ASMAtomicWriteU32(&pCtx->uState, VIDREC_INITIALIZED);
442 return VINF_SUCCESS;
443}
444
445/**
446 * VideoRec utility function to initialize video recording context.
447 *
448 * @returns IPRT status code.
449 * @param pCtx Pointer to video recording context to initialize Framebuffer width.
450 * @param uScreeen Screen number.
451 * @param strFile File to save the recorded data
452 * @param uTargetWidth Width of the target image in the video recoriding file (movie)
453 * @param uTargetHeight Height of the target image in video recording file.
454 */
455int VideoRecStrmInit(PVIDEORECCONTEXT pCtx, uint32_t uScreen, const char *pszFile,
456 uint32_t uWidth, uint32_t uHeight, uint32_t uRate, uint32_t uFps)
457{
458 AssertPtrReturn(pCtx, VERR_INVALID_PARAMETER);
459 AssertReturn(uScreen < pCtx->cScreens, VERR_INVALID_PARAMETER);
460
461 PVIDEORECSTREAM pStrm = &pCtx->Strm[uScreen];
462 pStrm->uTargetWidth = uWidth;
463 pStrm->uTargetHeight = uHeight;
464 pStrm->pu8RgbBuf = (uint8_t *)RTMemAllocZ(uWidth * uHeight * 4);
465 AssertReturn(pStrm->pu8RgbBuf, VERR_NO_MEMORY);
466
467 int rc = RTFileOpen(&pStrm->Ebml.file, pszFile,
468 RTFILE_O_CREATE_REPLACE | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
469 if (RT_FAILURE(rc))
470 {
471 LogFlow(("Failed to open the video capture output File (%Rrc)\n", rc));
472 return rc;
473 }
474
475 vpx_codec_err_t rcv = vpx_codec_enc_config_default(DEFAULTCODEC, &pStrm->VpxConfig, 0);
476 if (rcv != VPX_CODEC_OK)
477 {
478 LogFlow(("Failed to configure codec\n", vpx_codec_err_to_string(rcv)));
479 return VERR_INVALID_PARAMETER;
480 }
481
482 /* target bitrate in kilobits per second */
483 pStrm->VpxConfig.rc_target_bitrate = uRate;
484 /* frame width */
485 pStrm->VpxConfig.g_w = uWidth;
486 /* frame height */
487 pStrm->VpxConfig.g_h = uHeight;
488 /* 1ms per frame */
489 pStrm->VpxConfig.g_timebase.num = 1;
490 pStrm->VpxConfig.g_timebase.den = 1000;
491 /* disable multithreading */
492 pStrm->VpxConfig.g_threads = 0;
493 pStrm->uDelay = 1000 / uFps;
494
495 struct vpx_rational arg_framerate = { 30, 1 };
496 rc = Ebml_WriteWebMFileHeader(&pStrm->Ebml, &pStrm->VpxConfig, &arg_framerate);
497 AssertRCReturn(rc, rc);
498
499 /* Initialize codec */
500 rcv = vpx_codec_enc_init(&pStrm->VpxCodec, DEFAULTCODEC, &pStrm->VpxConfig, 0);
501 if (rcv != VPX_CODEC_OK)
502 {
503 LogFlow(("Failed to initialize VP8 encoder %s", vpx_codec_err_to_string(rcv)));
504 return VERR_INVALID_PARAMETER;
505 }
506
507 if (!vpx_img_alloc(&pStrm->VpxRawImage, VPX_IMG_FMT_I420, uWidth, uHeight, 1))
508 {
509 LogFlow(("Failed to allocate image %dx%d", uWidth, uHeight));
510 return VERR_NO_MEMORY;
511 }
512 pStrm->pu8YuvBuf = pStrm->VpxRawImage.planes[0];
513
514 pCtx->fEnabled = true;
515 pStrm->fEnabled = true;
516 return VINF_SUCCESS;
517}
518
519/**
520 * VideoRec utility function to close the video recording context.
521 *
522 * @param pCtx Pointer to video recording context.
523 */
524void VideoRecContextClose(PVIDEORECCONTEXT pCtx)
525{
526 if (!pCtx)
527 return;
528
529 if (ASMAtomicReadU32(&pCtx->uState) != VIDREC_INITIALIZED)
530 return;
531
532 ASMAtomicWriteU32(&pCtx->uState, VIDREC_TERMINATING);
533 RTSemEventSignal(pCtx->WaitEvent);
534 RTThreadWait(pCtx->Thread, 10000, NULL);
535 RTSemEventDestroy(pCtx->WaitEvent);
536
537 for (unsigned uScreen = 0; uScreen < pCtx->cScreens; uScreen++)
538 {
539 PVIDEORECSTREAM pStrm = &pCtx->Strm[uScreen];
540 if (pStrm->fEnabled)
541 {
542 if (pStrm->Ebml.file != NIL_RTFILE)
543 {
544 int rc = Ebml_WriteWebMFileFooter(&pStrm->Ebml, 0);
545 AssertRC(rc);
546 RTFileClose(pStrm->Ebml.file);
547 pStrm->Ebml.file = NIL_RTFILE;
548 }
549 if (pStrm->Ebml.cue_list)
550 {
551 RTMemFree(pStrm->Ebml.cue_list);
552 pStrm->Ebml.cue_list = NULL;
553 }
554 vpx_img_free(&pStrm->VpxRawImage);
555 vpx_codec_err_t rcv = vpx_codec_destroy(&pStrm->VpxCodec);
556 Assert(rcv == VPX_CODEC_OK);
557 RTMemFree(pStrm->pu8RgbBuf);
558 pStrm->pu8RgbBuf = NULL;
559 }
560 }
561
562 ASMAtomicWriteU32(&pCtx->uState, VIDREC_UNINITIALIZED);
563}
564
565/**
566 * VideoRec utility function to check if recording is enabled.
567 *
568 * @returns true if recording is enabled
569 * @param pCtx Pointer to video recording context.
570 */
571bool VideoRecIsEnabled(PVIDEORECCONTEXT pCtx)
572{
573 if (!pCtx)
574 return false;
575
576 return pCtx->fEnabled;
577}
578
579/**
580 * VideoRec utility function to encode the source image and write the encoded
581 * image to target file.
582 *
583 * @returns IPRT status code.
584 * @param pCtx Pointer to video recording context.
585 * @param uSourceWidth Width of the source image.
586 * @param uSourceHeight Height of the source image.
587 */
588static int videoRecEncodeAndWrite(PVIDEORECSTREAM pStrm)
589{
590 /* presentation time stamp */
591 vpx_codec_pts_t pts = pStrm->u64TimeStamp;
592 vpx_codec_err_t rcv = vpx_codec_encode(&pStrm->VpxCodec,
593 &pStrm->VpxRawImage,
594 pts /* time stamp */,
595 10 /* how long to show this frame */,
596 0 /* flags */,
597 VPX_DL_REALTIME /* deadline */);
598 if (rcv != VPX_CODEC_OK)
599 {
600 LogFlow(("Failed to encode:%s\n", vpx_codec_err_to_string(rcv)));
601 return VERR_GENERAL_FAILURE;
602 }
603
604 vpx_codec_iter_t iter = NULL;
605 int rc = VERR_NO_DATA;
606 for (;;)
607 {
608 const vpx_codec_cx_pkt_t *pkt = vpx_codec_get_cx_data(&pStrm->VpxCodec, &iter);
609 if (!pkt)
610 break;
611 switch (pkt->kind)
612 {
613 case VPX_CODEC_CX_FRAME_PKT:
614 rc = Ebml_WriteWebMBlock(&pStrm->Ebml, &pStrm->VpxConfig, pkt);
615 break;
616 default:
617 LogFlow(("Unexpected CODEC Packet.\n"));
618 break;
619 }
620 }
621
622 pStrm->cFrame++;
623 return rc;
624}
625
626/**
627 * VideoRec utility function to convert RGB to YUV.
628 *
629 * @returns IPRT status code.
630 * @param pCtx Pointer to video recording context.
631 */
632static int videoRecRGBToYUV(PVIDEORECSTREAM pStrm)
633{
634 switch (pStrm->u32PixelFormat)
635 {
636 case VPX_IMG_FMT_RGB32:
637 LogFlow(("32 bit\n"));
638 if (!colorConvWriteYUV420p<ColorConvBGRA32Iter>(pStrm->uTargetWidth,
639 pStrm->uTargetHeight,
640 pStrm->pu8YuvBuf,
641 pStrm->pu8RgbBuf))
642 return VERR_GENERAL_FAILURE;
643 break;
644 case VPX_IMG_FMT_RGB24:
645 LogFlow(("24 bit\n"));
646 if (!colorConvWriteYUV420p<ColorConvBGR24Iter>(pStrm->uTargetWidth,
647 pStrm->uTargetHeight,
648 pStrm->pu8YuvBuf,
649 pStrm->pu8RgbBuf))
650 return VERR_GENERAL_FAILURE;
651 break;
652 case VPX_IMG_FMT_RGB565:
653 LogFlow(("565 bit\n"));
654 if (!colorConvWriteYUV420p<ColorConvBGR565Iter>(pStrm->uTargetWidth,
655 pStrm->uTargetHeight,
656 pStrm->pu8YuvBuf,
657 pStrm->pu8RgbBuf))
658 return VERR_GENERAL_FAILURE;
659 break;
660 default:
661 return VERR_GENERAL_FAILURE;
662 }
663 return VINF_SUCCESS;
664}
665
666/**
667 * VideoRec utility function to copy source image (FrameBuf) to
668 * intermediate RGB buffer.
669 *
670 * @returns IPRT status code.
671 * @param pCtx Pointer to the video recording context.
672 * @param uScreen Screen number.
673 * @param x Starting x coordinate of the source buffer (Framebuffer).
674 * @param y Starting y coordinate of the source buffer (Framebuffer).
675 * @param uPixelFormat Pixel Format.
676 * @param uBitsPerPixel Bits Per Pixel
677 * @param uBytesPerLine Bytes per source scanlineName.
678 * @param uSourceWidth Width of the source image (framebuffer).
679 * @param uSourceHeight Height of the source image (framebuffer).
680 * @param pu8BufAddr Pointer to source image(framebuffer).
681 * @param u64TimeStamp Time stamp (milliseconds).
682 */
683int VideoRecCopyToIntBuf(PVIDEORECCONTEXT pCtx, uint32_t uScreen, uint32_t x, uint32_t y,
684 uint32_t uPixelFormat, uint32_t uBitsPerPixel, uint32_t uBytesPerLine,
685 uint32_t uSourceWidth, uint32_t uSourceHeight, uint8_t *pu8BufAddr,
686 uint64_t u64TimeStamp)
687{
688 AssertPtrReturn(pu8BufAddr, VERR_INVALID_PARAMETER);
689 AssertReturn(uSourceWidth, VERR_INVALID_PARAMETER);
690 AssertReturn(uSourceHeight, VERR_INVALID_PARAMETER);
691 AssertReturn(uScreen < pCtx->cScreens, VERR_INVALID_PARAMETER);
692 AssertReturn(pCtx->uState == VIDREC_INITIALIZED, VERR_INVALID_STATE);
693
694 PVIDEORECSTREAM pStrm = &pCtx->Strm[uScreen];
695
696 if (u64TimeStamp < pStrm->u64LastTimeStamp + pStrm->uDelay)
697 return VINF_TRY_AGAIN; /* respect maximum frames per second */
698
699 if (ASMAtomicReadBool(&pStrm->fRgbFilled))
700 return VERR_TRY_AGAIN; /* previous frame not yet encoded */
701
702 pStrm->u64LastTimeStamp = u64TimeStamp;
703
704 int xDiff = ((int)pStrm->uTargetWidth - (int)uSourceWidth) / 2;
705 uint32_t w = uSourceWidth;
706 if ((int)w + xDiff + (int)x <= 0) /* nothing visible */
707 return VERR_INVALID_PARAMETER;
708
709 uint32_t destX;
710 if ((int)x < -xDiff)
711 {
712 w += xDiff + x;
713 x = -xDiff;
714 destX = 0;
715 }
716 else
717 destX = x + xDiff;
718
719 uint32_t h = uSourceHeight;
720 int yDiff = ((int)pStrm->uTargetHeight - (int)uSourceHeight) / 2;
721 if ((int)h + yDiff + (int)y <= 0) /* nothing visible */
722 return VERR_INVALID_PARAMETER;
723
724 uint32_t destY;
725 if ((int)y < -yDiff)
726 {
727 h += yDiff + (int)y;
728 y = -yDiff;
729 destY = 0;
730 }
731 else
732 destY = y + yDiff;
733
734 if ( destX > pStrm->uTargetWidth
735 || destY > pStrm->uTargetHeight)
736 return VERR_INVALID_PARAMETER; /* nothing visible */
737
738 if (destX + w > pStrm->uTargetWidth)
739 w = pStrm->uTargetWidth - destX;
740
741 if (destY + h > pStrm->uTargetHeight)
742 h = pStrm->uTargetHeight - destY;
743
744 /* Calculate bytes per pixel */
745 uint32_t bpp = 1;
746 if (uPixelFormat == FramebufferPixelFormat_FOURCC_RGB)
747 {
748 switch (uBitsPerPixel)
749 {
750 case 32:
751 pStrm->u32PixelFormat = VPX_IMG_FMT_RGB32;
752 bpp = 4;
753 break;
754 case 24:
755 pStrm->u32PixelFormat = VPX_IMG_FMT_RGB24;
756 bpp = 3;
757 break;
758 case 16:
759 pStrm->u32PixelFormat = VPX_IMG_FMT_RGB565;
760 bpp = 2;
761 break;
762 default:
763 AssertMsgFailed(("Unknown color depth! mBitsPerPixel=%d\n", uBitsPerPixel));
764 break;
765 }
766 }
767 else
768 AssertMsgFailed(("Unknown pixel format! mPixelFormat=%d\n", uPixelFormat));
769
770 /* One of the dimensions of the current frame is smaller than before so
771 * clear the entire buffer to prevent artifacts from the previous frame */
772 if ( uSourceWidth < pStrm->uLastSourceWidth
773 || uSourceHeight < pStrm->uLastSourceHeight)
774 memset(pStrm->pu8RgbBuf, 0, pStrm->uTargetWidth * pStrm->uTargetHeight * 4);
775
776 pStrm->uLastSourceWidth = uSourceWidth;
777 pStrm->uLastSourceHeight = uSourceHeight;
778
779 /* Calculate start offset in source and destination buffers */
780 uint32_t offSrc = y * uBytesPerLine + x * bpp;
781 uint32_t offDst = (destY * pStrm->uTargetWidth + destX) * bpp;
782 /* do the copy */
783 for (unsigned int i = 0; i < h; i++)
784 {
785 /* Overflow check */
786 Assert(offSrc + w * bpp <= uSourceHeight * uBytesPerLine);
787 Assert(offDst + w * bpp <= pStrm->uTargetHeight * pStrm->uTargetWidth * bpp);
788 memcpy(pStrm->pu8RgbBuf + offDst, pu8BufAddr + offSrc, w * bpp);
789 offSrc += uBytesPerLine;
790 offDst += pStrm->uTargetWidth * bpp;
791 }
792
793 pStrm->u64TimeStamp = u64TimeStamp;
794
795 ASMAtomicWriteBool(&pStrm->fRgbFilled, true);
796 RTSemEventSignal(pCtx->WaitEvent);
797
798 return VINF_SUCCESS;
799}
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