VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxHeadless/VideoCapture/FFmpegFB.h@ 27105

Last change on this file since 27105 was 21520, checked in by vboxsync, 16 years ago

COM: missed pieces of scriptability support

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.4 KB
Line 
1/** @file
2 *
3 * VBox Remote Desktop Protocol.
4 * FFmpeg framebuffer interface.
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#ifndef _H_FFMPEGFB
24#define _H_FFMPEGFB
25
26#include <VBox/com/VirtualBox.h>
27
28#include <iprt/uuid.h>
29
30#include <VBox/com/com.h>
31#include <VBox/com/string.h>
32
33#include <iprt/initterm.h>
34#include <iprt/critsect.h>
35
36#ifdef DEBUG
37# define VBOX_DEBUG_FF DEBUG
38# include <avcodec.h>
39# include <avformat.h>
40# undef DEBUG
41# define DEBUG VBOX_DEBUG_FF
42#else /* DEBUG not defined */
43# include <avcodec.h>
44# include <avformat.h>
45#endif /* DEBUG not defined */
46
47class FFmpegFB : VBOX_SCRIPTABLE_IMPL(IFramebuffer)
48{
49public:
50 FFmpegFB(ULONG width, ULONG height, ULONG bitrate, com::Bstr filename);
51 virtual ~FFmpegFB();
52
53#ifndef VBOX_WITH_XPCOM
54 STDMETHOD_(ULONG, AddRef)()
55 {
56 return ::InterlockedIncrement (&refcnt);
57 }
58 STDMETHOD_(ULONG, Release)()
59 {
60 long cnt = ::InterlockedDecrement (&refcnt);
61 if (cnt == 0)
62 delete this;
63 return cnt;
64 }
65#endif
66 VBOX_SCRIPTABLE_DISPATCH_IMPL(IFramebuffer)
67
68 NS_DECL_ISUPPORTS
69
70 // public methods only for internal purposes
71 HRESULT init ();
72
73 STDMETHOD(COMGETTER(Width))(ULONG *width);
74 STDMETHOD(COMGETTER(Height))(ULONG *height);
75 STDMETHOD(Lock)();
76 STDMETHOD(Unlock)();
77 STDMETHOD(COMGETTER(Address))(BYTE **address);
78 STDMETHOD(COMGETTER(BitsPerPixel))(ULONG *bitsPerPixel);
79 STDMETHOD(COMGETTER(BytesPerLine))(ULONG *bytesPerLine);
80 STDMETHOD(COMGETTER(PixelFormat)) (ULONG *pixelFormat);
81 STDMETHOD(COMGETTER(UsesGuestVRAM)) (BOOL *usesGuestVRAM);
82 STDMETHOD(COMGETTER(HeightReduction)) (ULONG *heightReduction);
83 STDMETHOD(COMGETTER(Overlay)) (IFramebufferOverlay **aOverlay);
84 STDMETHOD(COMGETTER(WinId)) (ULONG64 *winId);
85
86 STDMETHOD(NotifyUpdate)(ULONG x, ULONG y, ULONG w, ULONG h);
87 STDMETHOD(RequestResize)(ULONG aScreenId, ULONG pixelFormat, BYTE *vram,
88 ULONG bitsPerPixel, ULONG bytesPerLine,
89 ULONG w, ULONG h, BOOL *finished);
90 STDMETHOD(VideoModeSupported)(ULONG width, ULONG height, ULONG bpp, BOOL *supported);
91 STDMETHOD(GetVisibleRegion)(BYTE *rectangles, ULONG count, ULONG *countCopied);
92 STDMETHOD(SetVisibleRegion)(BYTE *rectangles, ULONG count);
93
94 STDMETHOD(ProcessVHWACommand)(BYTE *pCommand);
95
96private:
97 /** true if url_fopen actually succeeded */
98 bool mfUrlOpen;
99 /** Guest framebuffer width */
100 ULONG mGuestWidth;
101 /** Guest framebuffer height */
102 ULONG mGuestHeight;
103 /** Bit rate used for encoding */
104 ULONG mBitRate;
105 /** Guest framebuffer pixel format */
106 ULONG mPixelFormat;
107 /** Guest framebuffer color depth */
108 ULONG mBitsPerPixel;
109 /** Name of the file we will write to */
110 com::Bstr mFileName;
111 /** Guest framebuffer line length */
112 ULONG mBytesPerLine;
113 /** MPEG frame framebuffer width */
114 ULONG mFrameWidth;
115 /** MPEG frame framebuffer height */
116 ULONG mFrameHeight;
117 /** The size of one YUV frame */
118 ULONG mYUVFrameSize;
119 /** If we can't use the video RAM directly, we allocate our own
120 * buffer */
121 uint8_t *mRGBBuffer;
122 /** The address of the buffer - can be either mRGBBuffer or the
123 * guests VRAM (HC address) if we can handle that directly */
124 uint8_t *mBufferAddress;
125 /** An intermediary RGB buffer with the same dimensions */
126 uint8_t *mTempRGBBuffer;
127 /** Frame buffer translated into YUV420 for the mpeg codec */
128 uint8_t *mYUVBuffer;
129 /** Temporary buffer into which the codec writes frames to be
130 * written into the file */
131 uint8_t *mOutBuf;
132 RTCRITSECT mCritSect;
133 /** File where we store the mpeg stream */
134 RTFILE mFile;
135 /** time at which the last "real" frame was created */
136 int64_t mLastTime;
137 /** Pointer to ffmpeg's format information context */
138 AVFormatContext *mpFormatContext;
139 /** ffmpeg context containing information about the stream */
140 AVStream *mpStream;
141 /** Information for ffmpeg describing the current frame */
142 AVFrame *mFrame;
143 /** ffmpeg pixel format of guest framebuffer */
144 int mFFMPEGPixelFormat;
145 /** Since we are building without exception support, we use this
146 to signal allocation failure in the constructor */
147 bool mOutOfMemory;
148 /** A hack: ffmpeg mpeg2 only writes a frame if something has
149 changed. So we flip the low luminance bit of the first
150 pixel every frame. */
151 bool mToggle;
152
153 HRESULT setup_library();
154 HRESULT setup_output_format();
155 HRESULT list_formats();
156 HRESULT open_codec();
157 HRESULT open_output_file();
158 void copy_to_intermediate_buffer(ULONG x, ULONG y, ULONG w, ULONG h);
159 HRESULT do_rgb_to_yuv_conversion();
160 HRESULT do_encoding_and_write();
161 HRESULT write_png();
162#ifndef VBOX_WITH_XPCOM
163 long refcnt;
164#endif
165};
166
167/**
168 * Iterator class for running through an BGRA32 image buffer and converting
169 * it to RGB.
170 */
171class FFmpegBGRA32Iter
172{
173private:
174 enum { PIX_SIZE = 4 };
175public:
176 FFmpegBGRA32Iter(unsigned aWidth, unsigned aHeight, uint8_t *aBuffer)
177 {
178 mPos = 0;
179 mSize = aWidth * aHeight * PIX_SIZE;
180 mBuffer = aBuffer;
181 }
182 /**
183 * Convert the next pixel to RGB.
184 * @returns true on success, false if we have reached the end of the buffer
185 * @param aRed where to store the red value
186 * @param aGreen where to store the green value
187 * @param aBlue where to store the blue value
188 */
189 bool getRGB(unsigned *aRed, unsigned *aGreen, unsigned *aBlue)
190 {
191 bool rc = false;
192 if (mPos + PIX_SIZE <= mSize)
193 {
194 *aRed = mBuffer[mPos + 2];
195 *aGreen = mBuffer[mPos + 1];
196 *aBlue = mBuffer[mPos];
197 mPos += PIX_SIZE;
198 rc = true;
199 }
200 return rc;
201 }
202
203 /**
204 * Skip forward by a certain number of pixels
205 * @param aPixels how many pixels to skip
206 */
207 void skip(unsigned aPixels)
208 {
209 mPos += PIX_SIZE * aPixels;
210 }
211private:
212 /** Size of the picture buffer */
213 unsigned mSize;
214 /** Current position in the picture buffer */
215 unsigned mPos;
216 /** Address of the picture buffer */
217 uint8_t *mBuffer;
218};
219
220/**
221 * Iterator class for running through an BGR24 image buffer and converting
222 * it to RGB.
223 */
224class FFmpegBGR24Iter
225{
226private:
227 enum { PIX_SIZE = 3 };
228public:
229 FFmpegBGR24Iter(unsigned aWidth, unsigned aHeight, uint8_t *aBuffer)
230 {
231 mPos = 0;
232 mSize = aWidth * aHeight * PIX_SIZE;
233 mBuffer = aBuffer;
234 }
235 /**
236 * Convert the next pixel to RGB.
237 * @returns true on success, false if we have reached the end of the buffer
238 * @param aRed where to store the red value
239 * @param aGreen where to store the green value
240 * @param aBlue where to store the blue value
241 */
242 bool getRGB(unsigned *aRed, unsigned *aGreen, unsigned *aBlue)
243 {
244 bool rc = false;
245 if (mPos + PIX_SIZE <= mSize)
246 {
247 *aRed = mBuffer[mPos + 2];
248 *aGreen = mBuffer[mPos + 1];
249 *aBlue = mBuffer[mPos];
250 mPos += PIX_SIZE;
251 rc = true;
252 }
253 return rc;
254 }
255
256 /**
257 * Skip forward by a certain number of pixels
258 * @param aPixels how many pixels to skip
259 */
260 void skip(unsigned aPixels)
261 {
262 mPos += PIX_SIZE * aPixels;
263 }
264private:
265 /** Size of the picture buffer */
266 unsigned mSize;
267 /** Current position in the picture buffer */
268 unsigned mPos;
269 /** Address of the picture buffer */
270 uint8_t *mBuffer;
271};
272
273/**
274 * Iterator class for running through an BGR565 image buffer and converting
275 * it to RGB.
276 */
277class FFmpegBGR565Iter
278{
279private:
280 enum { PIX_SIZE = 2 };
281public:
282 FFmpegBGR565Iter(unsigned aWidth, unsigned aHeight, uint8_t *aBuffer)
283 {
284 mPos = 0;
285 mSize = aWidth * aHeight * PIX_SIZE;
286 mBuffer = aBuffer;
287 }
288 /**
289 * Convert the next pixel to RGB.
290 * @returns true on success, false if we have reached the end of the buffer
291 * @param aRed where to store the red value
292 * @param aGreen where to store the green value
293 * @param aBlue where to store the blue value
294 */
295 bool getRGB(unsigned *aRed, unsigned *aGreen, unsigned *aBlue)
296 {
297 bool rc = false;
298 if (mPos + PIX_SIZE <= mSize)
299 {
300 unsigned uFull = (((unsigned) mBuffer[mPos + 1]) << 8)
301 | ((unsigned) mBuffer[mPos]);
302 *aRed = (uFull >> 8) & ~7;
303 *aGreen = (uFull >> 3) & ~3 & 0xff;
304 *aBlue = (uFull << 3) & ~7 & 0xff;
305 mPos += PIX_SIZE;
306 rc = true;
307 }
308 return rc;
309 }
310
311 /**
312 * Skip forward by a certain number of pixels
313 * @param aPixels how many pixels to skip
314 */
315 void skip(unsigned aPixels)
316 {
317 mPos += PIX_SIZE * aPixels;
318 }
319private:
320 /** Size of the picture buffer */
321 unsigned mSize;
322 /** Current position in the picture buffer */
323 unsigned mPos;
324 /** Address of the picture buffer */
325 uint8_t *mBuffer;
326};
327
328
329/**
330 * Convert an image to YUV420p format
331 * @returns true on success, false on failure
332 * @param aWidth width of image
333 * @param aHeight height of image
334 * @param aDestBuf an allocated memory buffer large enough to hold the
335 * destination image (i.e. width * height * 12bits)
336 * @param aSrcBuf the source image as an array of bytes
337 */
338template <class T>
339inline bool FFmpegWriteYUV420p(unsigned aWidth, unsigned aHeight, uint8_t *aDestBuf,
340 uint8_t *aSrcBuf)
341{
342 AssertReturn(0 == (aWidth & 1), false);
343 AssertReturn(0 == (aHeight & 1), false);
344 bool rc = true;
345 T iter1(aWidth, aHeight, aSrcBuf);
346 T iter2 = iter1;
347 iter2.skip(aWidth);
348 unsigned cPixels = aWidth * aHeight;
349 unsigned offY = 0;
350 unsigned offU = cPixels;
351 unsigned offV = cPixels + cPixels / 4;
352 for (unsigned i = 0; (i < aHeight / 2) && rc; ++i)
353 {
354 for (unsigned j = 0; (j < aWidth / 2) && rc; ++j)
355 {
356 unsigned red, green, blue, u, v;
357 rc = iter1.getRGB(&red, &green, &blue);
358 if (rc)
359 {
360 aDestBuf[offY] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
361 u = (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
362 v = (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
363 rc = iter1.getRGB(&red, &green, &blue);
364 }
365 if (rc)
366 {
367 aDestBuf[offY + 1] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
368 u += (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
369 v += (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
370 rc = iter2.getRGB(&red, &green, &blue);
371 }
372 if (rc)
373 {
374 aDestBuf[offY + aWidth] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
375 u += (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
376 v += (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
377 rc = iter2.getRGB(&red, &green, &blue);
378 }
379 if (rc)
380 {
381 aDestBuf[offY + aWidth + 1] = ((66 * red + 129 * green + 25 * blue + 128) >> 8) + 16;
382 u += (((-38 * red - 74 * green + 112 * blue + 128) >> 8) + 128) / 4;
383 v += (((112 * red - 94 * green - 18 * blue + 128) >> 8) + 128) / 4;
384 aDestBuf[offU] = u;
385 aDestBuf[offV] = v;
386 offY += 2;
387 ++offU;
388 ++offV;
389 }
390 }
391 if (rc)
392 {
393 iter1.skip(aWidth);
394 iter2.skip(aWidth);
395 offY += aWidth;
396 }
397 }
398 return rc;
399}
400
401
402/**
403 * Convert an image to RGB24 format
404 * @returns true on success, false on failure
405 * @param aWidth width of image
406 * @param aHeight height of image
407 * @param aDestBuf an allocated memory buffer large enough to hold the
408 * destination image (i.e. width * height * 12bits)
409 * @param aSrcBuf the source image as an array of bytes
410 */
411template <class T>
412inline bool FFmpegWriteRGB24(unsigned aWidth, unsigned aHeight, uint8_t *aDestBuf,
413 uint8_t *aSrcBuf)
414{
415 enum { PIX_SIZE = 3 };
416 bool rc = true;
417 AssertReturn(0 == (aWidth & 1), false);
418 AssertReturn(0 == (aHeight & 1), false);
419 T iter(aWidth, aHeight, aSrcBuf);
420 unsigned cPixels = aWidth * aHeight;
421 for (unsigned i = 0; (i < cPixels) && rc; ++i)
422 {
423 unsigned red, green, blue;
424 rc = iter.getRGB(&red, &green, &blue);
425 if (rc)
426 {
427 aDestBuf[i * PIX_SIZE] = red;
428 aDestBuf[i * PIX_SIZE + 1] = green;
429 aDestBuf[i * PIX_SIZE + 2] = blue;
430 }
431 }
432 return rc;
433}
434
435#endif /* !_H_FFMPEGFB */
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