VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/DisplayImpl.cpp@ 62233

Last change on this file since 62233 was 62139, checked in by vboxsync, 9 years ago

bugref:8389: Main: fixed video recording with VBoxheadless

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 152.1 KB
Line 
1/* $Id: DisplayImpl.cpp 62139 2016-07-08 13:32:43Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include "DisplayImpl.h"
19#include "DisplayUtils.h"
20#include "ConsoleImpl.h"
21#include "ConsoleVRDPServer.h"
22#include "GuestImpl.h"
23#include "VMMDev.h"
24
25#include "AutoCaller.h"
26#include "Logging.h"
27
28/* generated header */
29#include "VBoxEvents.h"
30
31#include <iprt/semaphore.h>
32#include <iprt/thread.h>
33#include <iprt/asm.h>
34#include <iprt/time.h>
35#include <iprt/cpp/utils.h>
36#include <iprt/alloca.h>
37
38#include <VBox/vmm/pdmdrv.h>
39#if defined(DEBUG) || defined(VBOX_STRICT) /* for VM_ASSERT_EMT(). */
40# include <VBox/vmm/vm.h>
41#endif
42
43#ifdef VBOX_WITH_VIDEOHWACCEL
44# include <VBox/VBoxVideo.h>
45#endif
46
47#if defined(VBOX_WITH_CROGL) || defined(VBOX_WITH_CRHGSMI)
48# include <VBox/HostServices/VBoxCrOpenGLSvc.h>
49#endif
50
51#include <VBox/com/array.h>
52
53#ifdef VBOX_WITH_VPX
54# include <iprt/path.h>
55# include "VideoRec.h"
56#endif
57
58#ifdef VBOX_WITH_CROGL
59typedef enum
60{
61 CRVREC_STATE_IDLE,
62 CRVREC_STATE_SUBMITTED
63} CRVREC_STATE;
64#endif
65
66/**
67 * Display driver instance data.
68 *
69 * @implements PDMIDISPLAYCONNECTOR
70 */
71typedef struct DRVMAINDISPLAY
72{
73 /** Pointer to the display object. */
74 Display *pDisplay;
75 /** Pointer to the driver instance structure. */
76 PPDMDRVINS pDrvIns;
77 /** Pointer to the keyboard port interface of the driver/device above us. */
78 PPDMIDISPLAYPORT pUpPort;
79 /** Our display connector interface. */
80 PDMIDISPLAYCONNECTOR IConnector;
81#if defined(VBOX_WITH_VIDEOHWACCEL) || defined(VBOX_WITH_CRHGSMI)
82 /** VBVA callbacks */
83 PPDMIDISPLAYVBVACALLBACKS pVBVACallbacks;
84#endif
85} DRVMAINDISPLAY, *PDRVMAINDISPLAY;
86
87/** Converts PDMIDISPLAYCONNECTOR pointer to a DRVMAINDISPLAY pointer. */
88#define PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface) RT_FROM_MEMBER(pInterface, DRVMAINDISPLAY, IConnector)
89
90// constructor / destructor
91/////////////////////////////////////////////////////////////////////////////
92
93Display::Display()
94 : mParent(NULL), mfIsCr3DEnabled(false)
95{
96}
97
98Display::~Display()
99{
100}
101
102
103HRESULT Display::FinalConstruct()
104{
105 int rc = videoAccelConstruct(&mVideoAccelLegacy);
106 AssertRC(rc);
107
108 mfVideoAccelVRDP = false;
109 mfu32SupportedOrders = 0;
110 mcVideoAccelVRDPRefs = 0;
111
112 mfSeamlessEnabled = false;
113 mpRectVisibleRegion = NULL;
114 mcRectVisibleRegion = 0;
115
116#ifdef VBOX_WITH_CROGL
117 mfCrOglDataHidden = false;
118#endif
119
120 mpDrv = NULL;
121 mpVMMDev = NULL;
122 mfVMMDevInited = false;
123
124 rc = RTCritSectInit(&mVideoAccelLock);
125 AssertRC(rc);
126
127#ifdef VBOX_WITH_HGSMI
128 mu32UpdateVBVAFlags = 0;
129 mfVMMDevSupportsGraphics = false;
130 mfGuestVBVACapabilities = 0;
131 mfHostCursorCapabilities = 0;
132#endif
133#ifdef VBOX_WITH_VPX
134 rc = RTCritSectInit(&mVideoCaptureLock);
135 AssertRC(rc);
136
137 mpVideoRecCtx = NULL;
138 for (unsigned i = 0; i < RT_ELEMENTS(maVideoRecEnabled); i++)
139 maVideoRecEnabled[i] = true;
140#endif
141
142#ifdef VBOX_WITH_CRHGSMI
143 mhCrOglSvc = NULL;
144 rc = RTCritSectRwInit(&mCrOglLock);
145 AssertRC(rc);
146#endif
147#ifdef VBOX_WITH_CROGL
148 RT_ZERO(mCrOglCallbacks);
149 RT_ZERO(mCrOglScreenshotData);
150 mfCrOglVideoRecState = CRVREC_STATE_IDLE;
151 mCrOglScreenshotData.u32Screen = CRSCREEN_ALL;
152 mCrOglScreenshotData.pvContext = this;
153 mCrOglScreenshotData.pfnScreenshotBegin = i_displayCrVRecScreenshotBegin;
154 mCrOglScreenshotData.pfnScreenshotPerform = i_displayCrVRecScreenshotPerform;
155 mCrOglScreenshotData.pfnScreenshotEnd = i_displayCrVRecScreenshotEnd;
156#endif
157
158 return BaseFinalConstruct();
159}
160
161void Display::FinalRelease()
162{
163 uninit();
164
165#ifdef VBOX_WITH_VPX
166 if (RTCritSectIsInitialized(&mVideoCaptureLock))
167 {
168 RTCritSectDelete(&mVideoCaptureLock);
169 RT_ZERO(mVideoCaptureLock);
170 }
171#endif
172
173 videoAccelDestroy(&mVideoAccelLegacy);
174 i_saveVisibleRegion(0, NULL);
175
176 if (RTCritSectIsInitialized(&mVideoAccelLock))
177 {
178 RTCritSectDelete(&mVideoAccelLock);
179 RT_ZERO(mVideoAccelLock);
180 }
181
182#ifdef VBOX_WITH_CRHGSMI
183 if (RTCritSectRwIsInitialized(&mCrOglLock))
184 {
185 RTCritSectRwDelete(&mCrOglLock);
186 RT_ZERO(mCrOglLock);
187 }
188#endif
189 BaseFinalRelease();
190}
191
192// public initializer/uninitializer for internal purposes only
193/////////////////////////////////////////////////////////////////////////////
194
195#define kMaxSizeThumbnail 64
196
197/**
198 * Save thumbnail and screenshot of the guest screen.
199 */
200static int displayMakeThumbnail(uint8_t *pbData, uint32_t cx, uint32_t cy,
201 uint8_t **ppu8Thumbnail, uint32_t *pcbThumbnail, uint32_t *pcxThumbnail, uint32_t *pcyThumbnail)
202{
203 int rc = VINF_SUCCESS;
204
205 uint8_t *pu8Thumbnail = NULL;
206 uint32_t cbThumbnail = 0;
207 uint32_t cxThumbnail = 0;
208 uint32_t cyThumbnail = 0;
209
210 if (cx > cy)
211 {
212 cxThumbnail = kMaxSizeThumbnail;
213 cyThumbnail = (kMaxSizeThumbnail * cy) / cx;
214 }
215 else
216 {
217 cyThumbnail = kMaxSizeThumbnail;
218 cxThumbnail = (kMaxSizeThumbnail * cx) / cy;
219 }
220
221 LogRelFlowFunc(("%dx%d -> %dx%d\n", cx, cy, cxThumbnail, cyThumbnail));
222
223 cbThumbnail = cxThumbnail * 4 * cyThumbnail;
224 pu8Thumbnail = (uint8_t *)RTMemAlloc(cbThumbnail);
225
226 if (pu8Thumbnail)
227 {
228 uint8_t *dst = pu8Thumbnail;
229 uint8_t *src = pbData;
230 int dstW = cxThumbnail;
231 int dstH = cyThumbnail;
232 int srcW = cx;
233 int srcH = cy;
234 int iDeltaLine = cx * 4;
235
236 BitmapScale32(dst,
237 dstW, dstH,
238 src,
239 iDeltaLine,
240 srcW, srcH);
241
242 *ppu8Thumbnail = pu8Thumbnail;
243 *pcbThumbnail = cbThumbnail;
244 *pcxThumbnail = cxThumbnail;
245 *pcyThumbnail = cyThumbnail;
246 }
247 else
248 {
249 rc = VERR_NO_MEMORY;
250 }
251
252 return rc;
253}
254
255#ifdef VBOX_WITH_CROGL
256typedef struct
257{
258 CRVBOXHGCMTAKESCREENSHOT Base;
259
260 /* 32bpp small RGB image. */
261 uint8_t *pu8Thumbnail;
262 uint32_t cbThumbnail;
263 uint32_t cxThumbnail;
264 uint32_t cyThumbnail;
265
266 /* PNG screenshot. */
267 uint8_t *pu8PNG;
268 uint32_t cbPNG;
269 uint32_t cxPNG;
270 uint32_t cyPNG;
271} VBOX_DISPLAY_SAVESCREENSHOT_DATA;
272
273static DECLCALLBACK(void) displaySaveScreenshotReport(void *pvCtx, uint32_t uScreen,
274 uint32_t x, uint32_t y, uint32_t uBitsPerPixel,
275 uint32_t uBytesPerLine, uint32_t uGuestWidth, uint32_t uGuestHeight,
276 uint8_t *pu8BufferAddress, uint64_t u64TimeStamp)
277{
278 VBOX_DISPLAY_SAVESCREENSHOT_DATA *pData = (VBOX_DISPLAY_SAVESCREENSHOT_DATA*)pvCtx;
279 displayMakeThumbnail(pu8BufferAddress, uGuestWidth, uGuestHeight, &pData->pu8Thumbnail,
280 &pData->cbThumbnail, &pData->cxThumbnail, &pData->cyThumbnail);
281 int rc = DisplayMakePNG(pu8BufferAddress, uGuestWidth, uGuestHeight, &pData->pu8PNG,
282 &pData->cbPNG, &pData->cxPNG, &pData->cyPNG, 1);
283 if (RT_FAILURE(rc))
284 {
285 AssertMsgFailed(("DisplayMakePNG failed (rc=%Rrc)\n", rc));
286 if (pData->pu8PNG)
287 {
288 RTMemFree(pData->pu8PNG);
289 pData->pu8PNG = NULL;
290 }
291 pData->cbPNG = 0;
292 pData->cxPNG = 0;
293 pData->cyPNG = 0;
294 }
295}
296#endif
297
298DECLCALLBACK(void) Display::i_displaySSMSaveScreenshot(PSSMHANDLE pSSM, void *pvUser)
299{
300 Display *that = static_cast<Display*>(pvUser);
301
302 /* 32bpp small RGB image. */
303 uint8_t *pu8Thumbnail = NULL;
304 uint32_t cbThumbnail = 0;
305 uint32_t cxThumbnail = 0;
306 uint32_t cyThumbnail = 0;
307
308 /* PNG screenshot. */
309 uint8_t *pu8PNG = NULL;
310 uint32_t cbPNG = 0;
311 uint32_t cxPNG = 0;
312 uint32_t cyPNG = 0;
313
314 Console::SafeVMPtr ptrVM(that->mParent);
315 if (ptrVM.isOk())
316 {
317#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
318 BOOL f3DSnapshot = FALSE;
319 if ( that->mfIsCr3DEnabled
320 && that->mCrOglCallbacks.pfnHasData
321 && that->mCrOglCallbacks.pfnHasData())
322 {
323 VMMDev *pVMMDev = that->mParent->i_getVMMDev();
324 if (pVMMDev)
325 {
326 VBOX_DISPLAY_SAVESCREENSHOT_DATA *pScreenshot;
327 pScreenshot = (VBOX_DISPLAY_SAVESCREENSHOT_DATA*)RTMemAllocZ(sizeof(*pScreenshot));
328 if (pScreenshot)
329 {
330 /* screen id or CRSCREEN_ALL to specify all enabled */
331 pScreenshot->Base.u32Screen = 0;
332 pScreenshot->Base.u32Width = 0;
333 pScreenshot->Base.u32Height = 0;
334 pScreenshot->Base.u32Pitch = 0;
335 pScreenshot->Base.pvBuffer = NULL;
336 pScreenshot->Base.pvContext = pScreenshot;
337 pScreenshot->Base.pfnScreenshotBegin = NULL;
338 pScreenshot->Base.pfnScreenshotPerform = displaySaveScreenshotReport;
339 pScreenshot->Base.pfnScreenshotEnd = NULL;
340
341 VBOXCRCMDCTL_HGCM data;
342 data.Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
343 data.Hdr.u32Function = SHCRGL_HOST_FN_TAKE_SCREENSHOT;
344
345 data.aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
346 data.aParms[0].u.pointer.addr = &pScreenshot->Base;
347 data.aParms[0].u.pointer.size = sizeof(pScreenshot->Base);
348
349 int rc = that->i_crCtlSubmitSync(&data.Hdr, sizeof(data));
350 if (RT_SUCCESS(rc))
351 {
352 if (pScreenshot->pu8PNG)
353 {
354 pu8Thumbnail = pScreenshot->pu8Thumbnail;
355 cbThumbnail = pScreenshot->cbThumbnail;
356 cxThumbnail = pScreenshot->cxThumbnail;
357 cyThumbnail = pScreenshot->cyThumbnail;
358
359 /* PNG screenshot. */
360 pu8PNG = pScreenshot->pu8PNG;
361 cbPNG = pScreenshot->cbPNG;
362 cxPNG = pScreenshot->cxPNG;
363 cyPNG = pScreenshot->cyPNG;
364 f3DSnapshot = TRUE;
365 }
366 else
367 AssertMsgFailed(("no png\n"));
368 }
369 else
370 AssertMsgFailed(("SHCRGL_HOST_FN_TAKE_SCREENSHOT failed (rc=%Rrc)\n", rc));
371
372
373 RTMemFree(pScreenshot);
374 }
375 }
376 }
377
378 if (!f3DSnapshot)
379#endif
380 {
381 /* Query RGB bitmap. */
382 /* SSM code is executed on EMT(0), therefore no need to use VMR3ReqCallWait. */
383 uint8_t *pbData = NULL;
384 size_t cbData = 0;
385 uint32_t cx = 0;
386 uint32_t cy = 0;
387 bool fFreeMem = false;
388 int rc = Display::i_displayTakeScreenshotEMT(that, VBOX_VIDEO_PRIMARY_SCREEN, &pbData, &cbData, &cx, &cy, &fFreeMem);
389
390 /*
391 * It is possible that success is returned but everything is 0 or NULL.
392 * (no display attached if a VM is running with VBoxHeadless on OSE for example)
393 */
394 if (RT_SUCCESS(rc) && pbData)
395 {
396 Assert(cx && cy);
397
398 /* Prepare a small thumbnail and a PNG screenshot. */
399 displayMakeThumbnail(pbData, cx, cy, &pu8Thumbnail, &cbThumbnail, &cxThumbnail, &cyThumbnail);
400 rc = DisplayMakePNG(pbData, cx, cy, &pu8PNG, &cbPNG, &cxPNG, &cyPNG, 1);
401 if (RT_FAILURE(rc))
402 {
403 if (pu8PNG)
404 {
405 RTMemFree(pu8PNG);
406 pu8PNG = NULL;
407 }
408 cbPNG = 0;
409 cxPNG = 0;
410 cyPNG = 0;
411 }
412
413 if (fFreeMem)
414 RTMemFree(pbData);
415 else
416 that->mpDrv->pUpPort->pfnFreeScreenshot(that->mpDrv->pUpPort, pbData);
417 }
418 }
419 }
420 else
421 {
422 LogFunc(("Failed to get VM pointer 0x%x\n", ptrVM.rc()));
423 }
424
425 /* Regardless of rc, save what is available:
426 * Data format:
427 * uint32_t cBlocks;
428 * [blocks]
429 *
430 * Each block is:
431 * uint32_t cbBlock; if 0 - no 'block data'.
432 * uint32_t typeOfBlock; 0 - 32bpp RGB bitmap, 1 - PNG, ignored if 'cbBlock' is 0.
433 * [block data]
434 *
435 * Block data for bitmap and PNG:
436 * uint32_t cx;
437 * uint32_t cy;
438 * [image data]
439 */
440 SSMR3PutU32(pSSM, 2); /* Write thumbnail and PNG screenshot. */
441
442 /* First block. */
443 SSMR3PutU32(pSSM, cbThumbnail + 2 * sizeof(uint32_t));
444 SSMR3PutU32(pSSM, 0); /* Block type: thumbnail. */
445
446 if (cbThumbnail)
447 {
448 SSMR3PutU32(pSSM, cxThumbnail);
449 SSMR3PutU32(pSSM, cyThumbnail);
450 SSMR3PutMem(pSSM, pu8Thumbnail, cbThumbnail);
451 }
452
453 /* Second block. */
454 SSMR3PutU32(pSSM, cbPNG + 2 * sizeof(uint32_t));
455 SSMR3PutU32(pSSM, 1); /* Block type: png. */
456
457 if (cbPNG)
458 {
459 SSMR3PutU32(pSSM, cxPNG);
460 SSMR3PutU32(pSSM, cyPNG);
461 SSMR3PutMem(pSSM, pu8PNG, cbPNG);
462 }
463
464 RTMemFree(pu8PNG);
465 RTMemFree(pu8Thumbnail);
466}
467
468DECLCALLBACK(int)
469Display::i_displaySSMLoadScreenshot(PSSMHANDLE pSSM, void *pvUser, uint32_t uVersion, uint32_t uPass)
470{
471 Display *that = static_cast<Display*>(pvUser);
472
473 if (uVersion != sSSMDisplayScreenshotVer)
474 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
475 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
476
477 /* Skip data. */
478 uint32_t cBlocks;
479 int rc = SSMR3GetU32(pSSM, &cBlocks);
480 AssertRCReturn(rc, rc);
481
482 for (uint32_t i = 0; i < cBlocks; i++)
483 {
484 uint32_t cbBlock;
485 rc = SSMR3GetU32(pSSM, &cbBlock);
486 AssertRCBreak(rc);
487
488 uint32_t typeOfBlock;
489 rc = SSMR3GetU32(pSSM, &typeOfBlock);
490 AssertRCBreak(rc);
491
492 LogRelFlowFunc(("[%d] type %d, size %d bytes\n", i, typeOfBlock, cbBlock));
493
494 /* Note: displaySSMSaveScreenshot writes size of a block = 8 and
495 * do not write any data if the image size was 0.
496 * @todo Fix and increase saved state version.
497 */
498 if (cbBlock > 2 * sizeof(uint32_t))
499 {
500 rc = SSMR3Skip(pSSM, cbBlock);
501 AssertRCBreak(rc);
502 }
503 }
504
505 return rc;
506}
507
508/**
509 * Save/Load some important guest state
510 */
511DECLCALLBACK(void)
512Display::i_displaySSMSave(PSSMHANDLE pSSM, void *pvUser)
513{
514 Display *that = static_cast<Display*>(pvUser);
515
516 SSMR3PutU32(pSSM, that->mcMonitors);
517 for (unsigned i = 0; i < that->mcMonitors; i++)
518 {
519 SSMR3PutU32(pSSM, that->maFramebuffers[i].u32Offset);
520 SSMR3PutU32(pSSM, that->maFramebuffers[i].u32MaxFramebufferSize);
521 SSMR3PutU32(pSSM, that->maFramebuffers[i].u32InformationSize);
522 SSMR3PutU32(pSSM, that->maFramebuffers[i].w);
523 SSMR3PutU32(pSSM, that->maFramebuffers[i].h);
524 SSMR3PutS32(pSSM, that->maFramebuffers[i].xOrigin);
525 SSMR3PutS32(pSSM, that->maFramebuffers[i].yOrigin);
526 SSMR3PutU32(pSSM, that->maFramebuffers[i].flags);
527 }
528 SSMR3PutS32(pSSM, that->xInputMappingOrigin);
529 SSMR3PutS32(pSSM, that->yInputMappingOrigin);
530 SSMR3PutU32(pSSM, that->cxInputMapping);
531 SSMR3PutU32(pSSM, that->cyInputMapping);
532 SSMR3PutU32(pSSM, that->mfGuestVBVACapabilities);
533 SSMR3PutU32(pSSM, that->mfHostCursorCapabilities);
534}
535
536DECLCALLBACK(int)
537Display::i_displaySSMLoad(PSSMHANDLE pSSM, void *pvUser, uint32_t uVersion, uint32_t uPass)
538{
539 Display *that = static_cast<Display*>(pvUser);
540
541 if ( uVersion != sSSMDisplayVer
542 && uVersion != sSSMDisplayVer2
543 && uVersion != sSSMDisplayVer3
544 && uVersion != sSSMDisplayVer4
545 && uVersion != sSSMDisplayVer5)
546 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
547 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
548
549 uint32_t cMonitors;
550 int rc = SSMR3GetU32(pSSM, &cMonitors);
551 if (cMonitors != that->mcMonitors)
552 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("Number of monitors changed (%d->%d)!"), cMonitors, that->mcMonitors);
553
554 for (uint32_t i = 0; i < cMonitors; i++)
555 {
556 SSMR3GetU32(pSSM, &that->maFramebuffers[i].u32Offset);
557 SSMR3GetU32(pSSM, &that->maFramebuffers[i].u32MaxFramebufferSize);
558 SSMR3GetU32(pSSM, &that->maFramebuffers[i].u32InformationSize);
559 if ( uVersion == sSSMDisplayVer2
560 || uVersion == sSSMDisplayVer3
561 || uVersion == sSSMDisplayVer4
562 || uVersion == sSSMDisplayVer5)
563 {
564 uint32_t w;
565 uint32_t h;
566 SSMR3GetU32(pSSM, &w);
567 SSMR3GetU32(pSSM, &h);
568 that->maFramebuffers[i].w = w;
569 that->maFramebuffers[i].h = h;
570 }
571 if ( uVersion == sSSMDisplayVer3
572 || uVersion == sSSMDisplayVer4
573 || uVersion == sSSMDisplayVer5)
574 {
575 int32_t xOrigin;
576 int32_t yOrigin;
577 uint32_t flags;
578 SSMR3GetS32(pSSM, &xOrigin);
579 SSMR3GetS32(pSSM, &yOrigin);
580 SSMR3GetU32(pSSM, &flags);
581 that->maFramebuffers[i].xOrigin = xOrigin;
582 that->maFramebuffers[i].yOrigin = yOrigin;
583 that->maFramebuffers[i].flags = (uint16_t)flags;
584 that->maFramebuffers[i].fDisabled = (that->maFramebuffers[i].flags & VBVA_SCREEN_F_DISABLED) != 0;
585 }
586 }
587 if ( uVersion == sSSMDisplayVer4
588 || uVersion == sSSMDisplayVer5)
589 {
590 SSMR3GetS32(pSSM, &that->xInputMappingOrigin);
591 SSMR3GetS32(pSSM, &that->yInputMappingOrigin);
592 SSMR3GetU32(pSSM, &that->cxInputMapping);
593 SSMR3GetU32(pSSM, &that->cyInputMapping);
594 }
595 if (uVersion == sSSMDisplayVer5)
596 {
597 SSMR3GetU32(pSSM, &that->mfGuestVBVACapabilities);
598 SSMR3GetU32(pSSM, &that->mfHostCursorCapabilities);
599 }
600
601 return VINF_SUCCESS;
602}
603
604/**
605 * Initializes the display object.
606 *
607 * @returns COM result indicator
608 * @param parent handle of our parent object
609 * @param qemuConsoleData address of common console data structure
610 */
611HRESULT Display::init(Console *aParent)
612{
613 ComAssertRet(aParent, E_INVALIDARG);
614 /* Enclose the state transition NotReady->InInit->Ready */
615 AutoInitSpan autoInitSpan(this);
616 AssertReturn(autoInitSpan.isOk(), E_FAIL);
617
618 unconst(mParent) = aParent;
619
620 mfSourceBitmapEnabled = true;
621 fVGAResizing = false;
622
623 ULONG ul;
624 mParent->i_machine()->COMGETTER(MonitorCount)(&ul);
625 mcMonitors = ul;
626 xInputMappingOrigin = 0;
627 yInputMappingOrigin = 0;
628 cxInputMapping = 0;
629 cyInputMapping = 0;
630
631 for (ul = 0; ul < mcMonitors; ul++)
632 {
633 maFramebuffers[ul].u32Offset = 0;
634 maFramebuffers[ul].u32MaxFramebufferSize = 0;
635 maFramebuffers[ul].u32InformationSize = 0;
636
637 maFramebuffers[ul].pFramebuffer = NULL;
638 /* All secondary monitors are disabled at startup. */
639 maFramebuffers[ul].fDisabled = ul > 0;
640
641 maFramebuffers[ul].u32Caps = 0;
642
643 maFramebuffers[ul].updateImage.pu8Address = NULL;
644 maFramebuffers[ul].updateImage.cbLine = 0;
645
646 maFramebuffers[ul].xOrigin = 0;
647 maFramebuffers[ul].yOrigin = 0;
648
649 maFramebuffers[ul].w = 0;
650 maFramebuffers[ul].h = 0;
651
652 maFramebuffers[ul].flags = maFramebuffers[ul].fDisabled? VBVA_SCREEN_F_DISABLED: 0;
653
654 maFramebuffers[ul].u16BitsPerPixel = 0;
655 maFramebuffers[ul].pu8FramebufferVRAM = NULL;
656 maFramebuffers[ul].u32LineSize = 0;
657
658 maFramebuffers[ul].pHostEvents = NULL;
659
660 maFramebuffers[ul].fDefaultFormat = false;
661
662#ifdef VBOX_WITH_HGSMI
663 maFramebuffers[ul].fVBVAEnabled = false;
664 maFramebuffers[ul].fVBVAForceResize = false;
665 maFramebuffers[ul].fRenderThreadMode = false;
666 maFramebuffers[ul].pVBVAHostFlags = NULL;
667#endif /* VBOX_WITH_HGSMI */
668#ifdef VBOX_WITH_CROGL
669 RT_ZERO(maFramebuffers[ul].pendingViewportInfo);
670#endif
671 }
672
673 {
674 // register listener for state change events
675 ComPtr<IEventSource> es;
676 mParent->COMGETTER(EventSource)(es.asOutParam());
677 com::SafeArray<VBoxEventType_T> eventTypes;
678 eventTypes.push_back(VBoxEventType_OnStateChanged);
679 es->RegisterListener(this, ComSafeArrayAsInParam(eventTypes), true);
680 }
681
682 /* Cache the 3D settings. */
683 BOOL fIs3DEnabled = FALSE;
684 mParent->i_machine()->COMGETTER(Accelerate3DEnabled)(&fIs3DEnabled);
685 GraphicsControllerType_T enmGpuType = (GraphicsControllerType_T)GraphicsControllerType_VBoxVGA;
686 mParent->i_machine()->COMGETTER(GraphicsControllerType)(&enmGpuType);
687 mfIsCr3DEnabled = fIs3DEnabled && enmGpuType == GraphicsControllerType_VBoxVGA;
688
689 /* Confirm a successful initialization */
690 autoInitSpan.setSucceeded();
691
692 return S_OK;
693}
694
695/**
696 * Uninitializes the instance and sets the ready flag to FALSE.
697 * Called either from FinalRelease() or by the parent when it gets destroyed.
698 */
699void Display::uninit()
700{
701 LogRelFlowFunc(("this=%p\n", this));
702
703 /* Enclose the state transition Ready->InUninit->NotReady */
704 AutoUninitSpan autoUninitSpan(this);
705 if (autoUninitSpan.uninitDone())
706 return;
707
708 unsigned uScreenId;
709 for (uScreenId = 0; uScreenId < mcMonitors; uScreenId++)
710 {
711 maFramebuffers[uScreenId].pSourceBitmap.setNull();
712 maFramebuffers[uScreenId].updateImage.pSourceBitmap.setNull();
713 maFramebuffers[uScreenId].updateImage.pu8Address = NULL;
714 maFramebuffers[uScreenId].updateImage.cbLine = 0;
715 maFramebuffers[uScreenId].pFramebuffer.setNull();
716#ifdef VBOX_WITH_VPX
717 maFramebuffers[uScreenId].videoCapture.pSourceBitmap.setNull();
718#endif
719 }
720
721 if (mParent)
722 {
723 ComPtr<IEventSource> es;
724 mParent->COMGETTER(EventSource)(es.asOutParam());
725 es->UnregisterListener(this);
726 }
727
728 unconst(mParent) = NULL;
729
730 if (mpDrv)
731 mpDrv->pDisplay = NULL;
732
733 mpDrv = NULL;
734 mpVMMDev = NULL;
735 mfVMMDevInited = true;
736}
737
738/**
739 * Register the SSM methods. Called by the power up thread to be able to
740 * pass pVM
741 */
742int Display::i_registerSSM(PUVM pUVM)
743{
744 /* Version 2 adds width and height of the framebuffer; version 3 adds
745 * the framebuffer offset in the virtual desktop and the framebuffer flags;
746 * version 4 adds guest to host input event mapping and version 5 adds
747 * guest VBVA and host cursor capabilities.
748 */
749 int rc = SSMR3RegisterExternal(pUVM, "DisplayData", 0, sSSMDisplayVer5,
750 mcMonitors * sizeof(uint32_t) * 8 + sizeof(uint32_t),
751 NULL, NULL, NULL,
752 NULL, i_displaySSMSave, NULL,
753 NULL, i_displaySSMLoad, NULL, this);
754 AssertRCReturn(rc, rc);
755
756 /*
757 * Register loaders for old saved states where iInstance was
758 * 3 * sizeof(uint32_t *) due to a code mistake.
759 */
760 rc = SSMR3RegisterExternal(pUVM, "DisplayData", 12 /*uInstance*/, sSSMDisplayVer, 0 /*cbGuess*/,
761 NULL, NULL, NULL,
762 NULL, NULL, NULL,
763 NULL, i_displaySSMLoad, NULL, this);
764 AssertRCReturn(rc, rc);
765
766 rc = SSMR3RegisterExternal(pUVM, "DisplayData", 24 /*uInstance*/, sSSMDisplayVer, 0 /*cbGuess*/,
767 NULL, NULL, NULL,
768 NULL, NULL, NULL,
769 NULL, i_displaySSMLoad, NULL, this);
770 AssertRCReturn(rc, rc);
771
772 /* uInstance is an arbitrary value greater than 1024. Such a value will ensure a quick seek in saved state file. */
773 rc = SSMR3RegisterExternal(pUVM, "DisplayScreenshot", 1100 /*uInstance*/, sSSMDisplayScreenshotVer, 0 /*cbGuess*/,
774 NULL, NULL, NULL,
775 NULL, i_displaySSMSaveScreenshot, NULL,
776 NULL, i_displaySSMLoadScreenshot, NULL, this);
777
778 AssertRCReturn(rc, rc);
779
780 return VINF_SUCCESS;
781}
782
783DECLCALLBACK(void) Display::i_displayCrCmdFree(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd, int rc, void *pvCompletion)
784{
785 Assert(pvCompletion);
786 RTMemFree(pvCompletion);
787}
788
789#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
790int Display::i_crOglWindowsShow(bool fShow)
791{
792 if (!mfCrOglDataHidden == !!fShow)
793 return VINF_SUCCESS;
794
795 if (!mhCrOglSvc)
796 {
797 /* No 3D or the VMSVGA3d kind. */
798 Assert(!mfIsCr3DEnabled);
799 return VERR_INVALID_STATE;
800 }
801
802 VMMDev *pVMMDev = mParent->i_getVMMDev();
803 if (!pVMMDev)
804 {
805 AssertMsgFailed(("no vmmdev\n"));
806 return VERR_INVALID_STATE;
807 }
808
809 VBOXCRCMDCTL_HGCM *pData = (VBOXCRCMDCTL_HGCM*)RTMemAlloc(sizeof(VBOXCRCMDCTL_HGCM));
810 if (!pData)
811 {
812 AssertMsgFailed(("RTMemAlloc failed\n"));
813 return VERR_NO_MEMORY;
814 }
815
816 pData->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
817 pData->Hdr.u32Function = SHCRGL_HOST_FN_WINDOWS_SHOW;
818
819 pData->aParms[0].type = VBOX_HGCM_SVC_PARM_32BIT;
820 pData->aParms[0].u.uint32 = (uint32_t)fShow;
821
822 int rc = i_crCtlSubmit(&pData->Hdr, sizeof(*pData), i_displayCrCmdFree, pData);
823 if (RT_SUCCESS(rc))
824 mfCrOglDataHidden = !fShow;
825 else
826 {
827 AssertMsgFailed(("crCtlSubmit failed (rc=%Rrc)\n", rc));
828 RTMemFree(pData);
829 }
830
831 return rc;
832}
833#endif
834
835
836// public methods only for internal purposes
837/////////////////////////////////////////////////////////////////////////////
838
839int Display::i_notifyCroglResize(const PVBVAINFOVIEW pView, const PVBVAINFOSCREEN pScreen, void *pvVRAM)
840{
841#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
842 if (maFramebuffers[pScreen->u32ViewIndex].fRenderThreadMode)
843 return VINF_SUCCESS; /* nop it */
844
845 if (mfIsCr3DEnabled)
846 {
847 int rc = VERR_INVALID_STATE;
848 if (mhCrOglSvc)
849 {
850 VMMDev *pVMMDev = mParent->i_getVMMDev();
851 if (pVMMDev)
852 {
853 VBOXCRCMDCTL_HGCM *pCtl;
854 pCtl = (VBOXCRCMDCTL_HGCM*)RTMemAlloc(sizeof(CRVBOXHGCMDEVRESIZE) + sizeof(VBOXCRCMDCTL_HGCM));
855 if (pCtl)
856 {
857 CRVBOXHGCMDEVRESIZE *pData = (CRVBOXHGCMDEVRESIZE*)(pCtl+1);
858 pData->Screen = *pScreen;
859 pData->pvVRAM = pvVRAM;
860
861 pCtl->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
862 pCtl->Hdr.u32Function = SHCRGL_HOST_FN_DEV_RESIZE;
863 pCtl->aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
864 pCtl->aParms[0].u.pointer.addr = pData;
865 pCtl->aParms[0].u.pointer.size = sizeof(*pData);
866
867 rc = i_crCtlSubmit(&pCtl->Hdr, sizeof(*pCtl), i_displayCrCmdFree, pCtl);
868 if (RT_FAILURE(rc))
869 {
870 AssertMsgFailed(("crCtlSubmit failed (rc=%Rrc)\n", rc));
871 RTMemFree(pCtl);
872 }
873 }
874 else
875 rc = VERR_NO_MEMORY;
876 }
877 }
878
879 return rc;
880 }
881#endif /* #if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL) */
882 return VINF_SUCCESS;
883}
884
885/**
886 * Handles display resize event.
887 *
888 * @param w New display width
889 * @param h New display height
890 *
891 * @thread EMT
892 */
893int Display::i_handleDisplayResize(unsigned uScreenId, uint32_t bpp, void *pvVRAM,
894 uint32_t cbLine, uint32_t w, uint32_t h, uint16_t flags)
895{
896 LogRel(("Display::handleDisplayResize: uScreenId=%d pvVRAM=%p w=%d h=%d bpp=%d cbLine=0x%X flags=0x%X\n", uScreenId,
897 pvVRAM, w, h, bpp, cbLine, flags));
898
899 if (uScreenId >= mcMonitors)
900 return VINF_SUCCESS;
901
902 DISPLAYFBINFO *pFBInfo = &maFramebuffers[uScreenId];
903
904 /* Reset the update mode. */
905 pFBInfo->updateImage.pSourceBitmap.setNull();
906 pFBInfo->updateImage.pu8Address = NULL;
907 pFBInfo->updateImage.cbLine = 0;
908
909 if (uScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
910 {
911 pFBInfo->w = w;
912 pFBInfo->h = h;
913
914 pFBInfo->u16BitsPerPixel = (uint16_t)bpp;
915 pFBInfo->pu8FramebufferVRAM = (uint8_t *)pvVRAM;
916 pFBInfo->u32LineSize = cbLine;
917 pFBInfo->flags = flags;
918 }
919
920 /* Guest screen image will be invalid during resize, make sure that it is not updated. */
921 if (uScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
922 {
923 mpDrv->pUpPort->pfnSetRenderVRAM(mpDrv->pUpPort, false);
924
925 mpDrv->IConnector.pbData = NULL;
926 mpDrv->IConnector.cbScanline = 0;
927 mpDrv->IConnector.cBits = 32; /* DevVGA does not work with cBits == 0. */
928 mpDrv->IConnector.cx = 0;
929 mpDrv->IConnector.cy = 0;
930 }
931
932 maFramebuffers[uScreenId].pSourceBitmap.setNull();
933
934 if (!maFramebuffers[uScreenId].pFramebuffer.isNull())
935 {
936 HRESULT hr = maFramebuffers[uScreenId].pFramebuffer->NotifyChange(uScreenId, 0, 0, w, h); /* @todo origin */
937 LogFunc(("NotifyChange hr %08X\n", hr));
938 NOREF(hr);
939 }
940
941 bool fUpdateImage = RT_BOOL(pFBInfo->u32Caps & FramebufferCapabilities_UpdateImage);
942 if (fUpdateImage && !pFBInfo->pFramebuffer.isNull())
943 {
944 ComPtr<IDisplaySourceBitmap> pSourceBitmap;
945 HRESULT hr = QuerySourceBitmap(uScreenId, pSourceBitmap.asOutParam());
946 if (SUCCEEDED(hr))
947 {
948 BYTE *pAddress = NULL;
949 ULONG ulWidth = 0;
950 ULONG ulHeight = 0;
951 ULONG ulBitsPerPixel = 0;
952 ULONG ulBytesPerLine = 0;
953 BitmapFormat_T bitmapFormat = BitmapFormat_Opaque;
954
955 hr = pSourceBitmap->QueryBitmapInfo(&pAddress,
956 &ulWidth,
957 &ulHeight,
958 &ulBitsPerPixel,
959 &ulBytesPerLine,
960 &bitmapFormat);
961 if (SUCCEEDED(hr))
962 {
963 pFBInfo->updateImage.pSourceBitmap = pSourceBitmap;
964 pFBInfo->updateImage.pu8Address = pAddress;
965 pFBInfo->updateImage.cbLine = ulBytesPerLine;
966 }
967 }
968 }
969
970 /* Inform the VRDP server about the change of display parameters. */
971 LogRelFlowFunc(("Calling VRDP\n"));
972 mParent->i_consoleVRDPServer()->SendResize();
973
974 /* And re-send the seamless rectangles if necessary. */
975 if (mfSeamlessEnabled)
976 i_handleSetVisibleRegion(mcRectVisibleRegion, mpRectVisibleRegion);
977
978#ifdef VBOX_WITH_VPX
979 videoCaptureScreenChanged(uScreenId);
980#endif
981
982 LogRelFlowFunc(("[%d]: default format %d\n", uScreenId, pFBInfo->fDefaultFormat));
983
984 return VINF_SUCCESS;
985}
986
987static void i_checkCoordBounds(int *px, int *py, int *pw, int *ph, int cx, int cy)
988{
989 /* Correct negative x and y coordinates. */
990 if (*px < 0)
991 {
992 *px += *pw; /* Compute xRight which is also the new width. */
993
994 *pw = (*px < 0)? 0: *px;
995
996 *px = 0;
997 }
998
999 if (*py < 0)
1000 {
1001 *py += *ph; /* Compute xBottom, which is also the new height. */
1002
1003 *ph = (*py < 0)? 0: *py;
1004
1005 *py = 0;
1006 }
1007
1008 /* Also check if coords are greater than the display resolution. */
1009 if (*px + *pw > cx)
1010 {
1011 *pw = cx > *px? cx - *px: 0;
1012 }
1013
1014 if (*py + *ph > cy)
1015 {
1016 *ph = cy > *py? cy - *py: 0;
1017 }
1018}
1019
1020void Display::i_handleDisplayUpdate(unsigned uScreenId, int x, int y, int w, int h)
1021{
1022 /*
1023 * Always runs under either VBVA lock or, for HGSMI, DevVGA lock.
1024 * Safe to use VBVA vars and take the framebuffer lock.
1025 */
1026
1027#ifdef DEBUG_sunlover
1028 LogFlowFunc(("[%d] %d,%d %dx%d\n",
1029 uScreenId, x, y, w, h));
1030#endif /* DEBUG_sunlover */
1031
1032 /* No updates for a disabled guest screen. */
1033 if (maFramebuffers[uScreenId].fDisabled)
1034 return;
1035
1036 /* No updates for a blank guest screen. */
1037 /** @note Disabled for now, as the GUI does not update the picture when we
1038 * first blank. */
1039 /* if (maFramebuffers[uScreenId].flags & VBVA_SCREEN_F_BLANK)
1040 return; */
1041
1042 i_checkCoordBounds(&x, &y, &w, &h, maFramebuffers[uScreenId].w,
1043 maFramebuffers[uScreenId].h);
1044
1045 IFramebuffer *pFramebuffer = maFramebuffers[uScreenId].pFramebuffer;
1046 if (pFramebuffer != NULL)
1047 {
1048 if (w != 0 && h != 0)
1049 {
1050 bool fUpdateImage = RT_BOOL(maFramebuffers[uScreenId].u32Caps & FramebufferCapabilities_UpdateImage);
1051 if (RT_LIKELY(!fUpdateImage))
1052 {
1053 pFramebuffer->NotifyUpdate(x, y, w, h);
1054 }
1055 else
1056 {
1057 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1058
1059 DISPLAYFBINFO *pFBInfo = &maFramebuffers[uScreenId];
1060
1061 if (!pFBInfo->updateImage.pSourceBitmap.isNull())
1062 {
1063 Assert(pFBInfo->updateImage.pu8Address);
1064
1065 size_t cbData = w * h * 4;
1066 com::SafeArray<BYTE> image(cbData);
1067
1068 uint8_t *pu8Dst = image.raw();
1069 const uint8_t *pu8Src = pFBInfo->updateImage.pu8Address + pFBInfo->updateImage.cbLine * y + x * 4;
1070
1071 int i;
1072 for (i = y; i < y + h; ++i)
1073 {
1074 memcpy(pu8Dst, pu8Src, w * 4);
1075 pu8Dst += w * 4;
1076 pu8Src += pFBInfo->updateImage.cbLine;
1077 }
1078
1079 pFramebuffer->NotifyUpdateImage(x, y, w, h, ComSafeArrayAsInParam(image));
1080 }
1081 }
1082 }
1083 }
1084
1085#ifndef VBOX_WITH_HGSMI
1086 if (!mVideoAccelLegacy.fVideoAccelEnabled)
1087 {
1088#else
1089 if (!mVideoAccelLegacy.fVideoAccelEnabled && !maFramebuffers[uScreenId].fVBVAEnabled)
1090 {
1091#endif /* VBOX_WITH_HGSMI */
1092 /* When VBVA is enabled, the VRDP server is informed
1093 * either in VideoAccelFlush or displayVBVAUpdateProcess.
1094 * Inform the server here only if VBVA is disabled.
1095 */
1096 mParent->i_consoleVRDPServer()->SendUpdateBitmap(uScreenId, x, y, w, h);
1097 }
1098}
1099
1100void Display::i_updateGuestGraphicsFacility(void)
1101{
1102 Guest* pGuest = mParent->i_getGuest();
1103 AssertPtrReturnVoid(pGuest);
1104 /* The following is from GuestImpl.cpp. */
1105 /** @todo A nit: The timestamp is wrong on saved state restore. Would be better
1106 * to move the graphics and seamless capability -> facility translation to
1107 * VMMDev so this could be saved. */
1108 RTTIMESPEC TimeSpecTS;
1109 RTTimeNow(&TimeSpecTS);
1110
1111 if ( mfVMMDevSupportsGraphics
1112 || (mfGuestVBVACapabilities & VBVACAPS_VIDEO_MODE_HINTS) != 0)
1113 pGuest->i_setAdditionsStatus(VBoxGuestFacilityType_Graphics,
1114 VBoxGuestFacilityStatus_Active,
1115 0 /*fFlags*/, &TimeSpecTS);
1116 else
1117 pGuest->i_setAdditionsStatus(VBoxGuestFacilityType_Graphics,
1118 VBoxGuestFacilityStatus_Inactive,
1119 0 /*fFlags*/, &TimeSpecTS);
1120}
1121
1122void Display::i_handleUpdateVMMDevSupportsGraphics(bool fSupportsGraphics)
1123{
1124 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1125 if (mfVMMDevSupportsGraphics == fSupportsGraphics)
1126 return;
1127 mfVMMDevSupportsGraphics = fSupportsGraphics;
1128 i_updateGuestGraphicsFacility();
1129 /* The VMMDev interface notifies the console. */
1130}
1131
1132void Display::i_handleUpdateGuestVBVACapabilities(uint32_t fNewCapabilities)
1133{
1134 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1135 bool fNotify = (fNewCapabilities & VBVACAPS_VIDEO_MODE_HINTS) != (mfGuestVBVACapabilities & VBVACAPS_VIDEO_MODE_HINTS);
1136
1137 mfGuestVBVACapabilities = fNewCapabilities;
1138 if (!fNotify)
1139 return;
1140 i_updateGuestGraphicsFacility();
1141 /* Tell the console about it */
1142 mParent->i_onAdditionsStateChange();
1143}
1144
1145void Display::i_handleUpdateVBVAInputMapping(int32_t xOrigin, int32_t yOrigin, uint32_t cx, uint32_t cy)
1146{
1147 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1148
1149 xInputMappingOrigin = xOrigin;
1150 yInputMappingOrigin = yOrigin;
1151 cxInputMapping = cx;
1152 cyInputMapping = cy;
1153
1154 /* Re-send the seamless rectangles if necessary. */
1155 if (mfSeamlessEnabled)
1156 i_handleSetVisibleRegion(mcRectVisibleRegion, mpRectVisibleRegion);
1157}
1158
1159/**
1160 * Returns the upper left and lower right corners of the virtual framebuffer.
1161 * The lower right is "exclusive" (i.e. first pixel beyond the framebuffer),
1162 * and the origin is (0, 0), not (1, 1) like the GUI returns.
1163 */
1164void Display::i_getFramebufferDimensions(int32_t *px1, int32_t *py1,
1165 int32_t *px2, int32_t *py2)
1166{
1167 int32_t x1 = 0, y1 = 0, x2 = 0, y2 = 0;
1168 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1169
1170 AssertPtrReturnVoid(px1);
1171 AssertPtrReturnVoid(py1);
1172 AssertPtrReturnVoid(px2);
1173 AssertPtrReturnVoid(py2);
1174 LogRelFlowFunc(("\n"));
1175
1176 if (!mpDrv)
1177 return;
1178 /* If VBVA is not in use then this flag will not be set and this
1179 * will still work as it should. */
1180 if (!maFramebuffers[0].fDisabled)
1181 {
1182 x1 = (int32_t)maFramebuffers[0].xOrigin;
1183 y1 = (int32_t)maFramebuffers[0].yOrigin;
1184 x2 = (int32_t)maFramebuffers[0].w + (int32_t)maFramebuffers[0].xOrigin;
1185 y2 = (int32_t)maFramebuffers[0].h + (int32_t)maFramebuffers[0].yOrigin;
1186 }
1187 if (cxInputMapping && cyInputMapping)
1188 {
1189 x1 = xInputMappingOrigin;
1190 y1 = yInputMappingOrigin;
1191 x2 = xInputMappingOrigin + cxInputMapping;
1192 y2 = yInputMappingOrigin + cyInputMapping;
1193 }
1194 else
1195 for (unsigned i = 1; i < mcMonitors; ++i)
1196 {
1197 if (!maFramebuffers[i].fDisabled)
1198 {
1199 x1 = RT_MIN(x1, maFramebuffers[i].xOrigin);
1200 y1 = RT_MIN(y1, maFramebuffers[i].yOrigin);
1201 x2 = RT_MAX(x2, maFramebuffers[i].xOrigin + (int32_t)maFramebuffers[i].w);
1202 y2 = RT_MAX(y2, maFramebuffers[i].yOrigin + (int32_t)maFramebuffers[i].h);
1203 }
1204 }
1205 *px1 = x1;
1206 *py1 = y1;
1207 *px2 = x2;
1208 *py2 = y2;
1209}
1210
1211HRESULT Display::i_reportHostCursorCapabilities(uint32_t fCapabilitiesAdded, uint32_t fCapabilitiesRemoved)
1212{
1213 /* Do we need this to access mParent? I presume that the safe VM pointer
1214 * ensures that mpDrv will remain valid. */
1215 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1216 uint32_t fHostCursorCapabilities = (mfHostCursorCapabilities | fCapabilitiesAdded)
1217 & ~fCapabilitiesRemoved;
1218
1219 Console::SafeVMPtr ptrVM(mParent);
1220 if (!ptrVM.isOk())
1221 return ptrVM.rc();
1222 if (mfHostCursorCapabilities == fHostCursorCapabilities)
1223 return S_OK;
1224 CHECK_CONSOLE_DRV(mpDrv);
1225 alock.release(); /* Release before calling up for lock order reasons. */
1226 mpDrv->pUpPort->pfnReportHostCursorCapabilities(mpDrv->pUpPort, fCapabilitiesAdded, fCapabilitiesRemoved);
1227 mfHostCursorCapabilities = fHostCursorCapabilities;
1228 return S_OK;
1229}
1230
1231HRESULT Display::i_reportHostCursorPosition(int32_t x, int32_t y)
1232{
1233 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1234 uint32_t xAdj = (uint32_t)RT_MAX(x - xInputMappingOrigin, 0);
1235 uint32_t yAdj = (uint32_t)RT_MAX(y - yInputMappingOrigin, 0);
1236 xAdj = RT_MIN(xAdj, cxInputMapping);
1237 yAdj = RT_MIN(yAdj, cyInputMapping);
1238
1239 Console::SafeVMPtr ptrVM(mParent);
1240 if (!ptrVM.isOk())
1241 return ptrVM.rc();
1242 CHECK_CONSOLE_DRV(mpDrv);
1243 alock.release(); /* Release before calling up for lock order reasons. */
1244 mpDrv->pUpPort->pfnReportHostCursorPosition(mpDrv->pUpPort, xAdj, yAdj);
1245 return S_OK;
1246}
1247
1248static bool displayIntersectRect(RTRECT *prectResult,
1249 const RTRECT *prect1,
1250 const RTRECT *prect2)
1251{
1252 /* Initialize result to an empty record. */
1253 memset(prectResult, 0, sizeof(RTRECT));
1254
1255 int xLeftResult = RT_MAX(prect1->xLeft, prect2->xLeft);
1256 int xRightResult = RT_MIN(prect1->xRight, prect2->xRight);
1257
1258 if (xLeftResult < xRightResult)
1259 {
1260 /* There is intersection by X. */
1261
1262 int yTopResult = RT_MAX(prect1->yTop, prect2->yTop);
1263 int yBottomResult = RT_MIN(prect1->yBottom, prect2->yBottom);
1264
1265 if (yTopResult < yBottomResult)
1266 {
1267 /* There is intersection by Y. */
1268
1269 prectResult->xLeft = xLeftResult;
1270 prectResult->yTop = yTopResult;
1271 prectResult->xRight = xRightResult;
1272 prectResult->yBottom = yBottomResult;
1273
1274 return true;
1275 }
1276 }
1277
1278 return false;
1279}
1280
1281int Display::i_saveVisibleRegion(uint32_t cRect, PRTRECT pRect)
1282{
1283 RTRECT *pRectVisibleRegion = NULL;
1284
1285 if (pRect == mpRectVisibleRegion)
1286 return VINF_SUCCESS;
1287 if (cRect != 0)
1288 {
1289 pRectVisibleRegion = (RTRECT *)RTMemAlloc(cRect * sizeof(RTRECT));
1290 if (!pRectVisibleRegion)
1291 {
1292 return VERR_NO_MEMORY;
1293 }
1294 memcpy(pRectVisibleRegion, pRect, cRect * sizeof(RTRECT));
1295 }
1296 if (mpRectVisibleRegion)
1297 RTMemFree(mpRectVisibleRegion);
1298 mcRectVisibleRegion = cRect;
1299 mpRectVisibleRegion = pRectVisibleRegion;
1300 return VINF_SUCCESS;
1301}
1302
1303int Display::i_handleSetVisibleRegion(uint32_t cRect, PRTRECT pRect)
1304{
1305 RTRECT *pVisibleRegion = (RTRECT *)RTMemTmpAlloc( RT_MAX(cRect, 1)
1306 * sizeof(RTRECT));
1307 LogRel2(("%s: cRect=%u\n", __PRETTY_FUNCTION__, cRect));
1308 if (!pVisibleRegion)
1309 {
1310 return VERR_NO_TMP_MEMORY;
1311 }
1312 int rc = i_saveVisibleRegion(cRect, pRect);
1313 if (RT_FAILURE(rc))
1314 {
1315 RTMemTmpFree(pVisibleRegion);
1316 return rc;
1317 }
1318
1319 unsigned uScreenId;
1320 for (uScreenId = 0; uScreenId < mcMonitors; uScreenId++)
1321 {
1322 DISPLAYFBINFO *pFBInfo = &maFramebuffers[uScreenId];
1323
1324 if ( !pFBInfo->pFramebuffer.isNull()
1325 && RT_BOOL(pFBInfo->u32Caps & FramebufferCapabilities_VisibleRegion))
1326 {
1327 /* Prepare a new array of rectangles which intersect with the framebuffer.
1328 */
1329 RTRECT rectFramebuffer;
1330 rectFramebuffer.xLeft = pFBInfo->xOrigin - xInputMappingOrigin;
1331 rectFramebuffer.yTop = pFBInfo->yOrigin - yInputMappingOrigin;
1332 rectFramebuffer.xRight = rectFramebuffer.xLeft + pFBInfo->w;
1333 rectFramebuffer.yBottom = rectFramebuffer.yTop + pFBInfo->h;
1334
1335 uint32_t cRectVisibleRegion = 0;
1336
1337 uint32_t i;
1338 for (i = 0; i < cRect; i++)
1339 {
1340 if (displayIntersectRect(&pVisibleRegion[cRectVisibleRegion], &pRect[i], &rectFramebuffer))
1341 {
1342 pVisibleRegion[cRectVisibleRegion].xLeft -= rectFramebuffer.xLeft;
1343 pVisibleRegion[cRectVisibleRegion].yTop -= rectFramebuffer.yTop;
1344 pVisibleRegion[cRectVisibleRegion].xRight -= rectFramebuffer.xLeft;
1345 pVisibleRegion[cRectVisibleRegion].yBottom -= rectFramebuffer.yTop;
1346
1347 cRectVisibleRegion++;
1348 }
1349 }
1350 pFBInfo->pFramebuffer->SetVisibleRegion((BYTE *)pVisibleRegion, cRectVisibleRegion);
1351 }
1352 }
1353
1354#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
1355 VMMDev *vmmDev = mParent->i_getVMMDev();
1356 if (mfIsCr3DEnabled && vmmDev)
1357 {
1358 if (mhCrOglSvc)
1359 {
1360 VBOXCRCMDCTL_HGCM *pCtl;
1361 pCtl = (VBOXCRCMDCTL_HGCM*)RTMemAlloc(RT_MAX(cRect, 1) * sizeof(RTRECT) + sizeof(VBOXCRCMDCTL_HGCM));
1362 if (pCtl)
1363 {
1364 RTRECT *pRectsCopy = (RTRECT*)(pCtl+1);
1365 memcpy(pRectsCopy, pRect, cRect * sizeof(RTRECT));
1366
1367 pCtl->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
1368 pCtl->Hdr.u32Function = SHCRGL_HOST_FN_SET_VISIBLE_REGION;
1369
1370 pCtl->aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
1371 pCtl->aParms[0].u.pointer.addr = pRectsCopy;
1372 pCtl->aParms[0].u.pointer.size = cRect * sizeof(RTRECT);
1373
1374 rc = i_crCtlSubmit(&pCtl->Hdr, sizeof(*pCtl), i_displayCrCmdFree, pCtl);
1375 if (!RT_SUCCESS(rc))
1376 {
1377 AssertMsgFailed(("crCtlSubmit failed (rc=%Rrc)\n", rc));
1378 RTMemFree(pCtl);
1379 }
1380 }
1381 else
1382 AssertMsgFailed(("failed to allocate rects memory\n"));
1383 }
1384 else
1385 AssertMsgFailed(("mhCrOglSvc is NULL\n"));
1386 }
1387#endif
1388
1389 RTMemTmpFree(pVisibleRegion);
1390
1391 return VINF_SUCCESS;
1392}
1393
1394int Display::i_handleQueryVisibleRegion(uint32_t *pcRect, PRTRECT pRect)
1395{
1396 // @todo Currently not used by the guest and is not implemented in framebuffers. Remove?
1397 return VERR_NOT_SUPPORTED;
1398}
1399
1400#ifdef VBOX_WITH_HGSMI
1401static void vbvaSetMemoryFlagsHGSMI(unsigned uScreenId,
1402 uint32_t fu32SupportedOrders,
1403 bool fVideoAccelVRDP,
1404 DISPLAYFBINFO *pFBInfo)
1405{
1406 LogRelFlowFunc(("HGSMI[%d]: %p\n", uScreenId, pFBInfo->pVBVAHostFlags));
1407
1408 if (pFBInfo->pVBVAHostFlags)
1409 {
1410 uint32_t fu32HostEvents = VBOX_VIDEO_INFO_HOST_EVENTS_F_VRDP_RESET;
1411
1412 if (pFBInfo->fVBVAEnabled)
1413 {
1414 fu32HostEvents |= VBVA_F_MODE_ENABLED;
1415
1416 if (fVideoAccelVRDP)
1417 {
1418 fu32HostEvents |= VBVA_F_MODE_VRDP;
1419 }
1420 }
1421
1422 ASMAtomicWriteU32(&pFBInfo->pVBVAHostFlags->u32HostEvents, fu32HostEvents);
1423 ASMAtomicWriteU32(&pFBInfo->pVBVAHostFlags->u32SupportedOrders, fu32SupportedOrders);
1424
1425 LogRelFlowFunc((" fu32HostEvents = 0x%08X, fu32SupportedOrders = 0x%08X\n", fu32HostEvents, fu32SupportedOrders));
1426 }
1427}
1428
1429static void vbvaSetMemoryFlagsAllHGSMI(uint32_t fu32SupportedOrders,
1430 bool fVideoAccelVRDP,
1431 DISPLAYFBINFO *paFBInfos,
1432 unsigned cFBInfos)
1433{
1434 unsigned uScreenId;
1435
1436 for (uScreenId = 0; uScreenId < cFBInfos; uScreenId++)
1437 {
1438 vbvaSetMemoryFlagsHGSMI(uScreenId, fu32SupportedOrders, fVideoAccelVRDP, &paFBInfos[uScreenId]);
1439 }
1440}
1441#endif /* VBOX_WITH_HGSMI */
1442
1443int Display::VideoAccelEnableVMMDev(bool fEnable, VBVAMEMORY *pVbvaMemory)
1444{
1445 LogFlowFunc(("%d %p\n", fEnable, pVbvaMemory));
1446 int rc = videoAccelEnterVMMDev(&mVideoAccelLegacy);
1447 if (RT_SUCCESS(rc))
1448 {
1449 rc = i_VideoAccelEnable(fEnable, pVbvaMemory, mpDrv->pUpPort);
1450 videoAccelLeaveVMMDev(&mVideoAccelLegacy);
1451 }
1452 LogFlowFunc(("leave %Rrc\n", rc));
1453 return rc;
1454}
1455
1456int Display::VideoAccelEnableVGA(bool fEnable, VBVAMEMORY *pVbvaMemory)
1457{
1458 LogFlowFunc(("%d %p\n", fEnable, pVbvaMemory));
1459 int rc = videoAccelEnterVGA(&mVideoAccelLegacy);
1460 if (RT_SUCCESS(rc))
1461 {
1462 rc = i_VideoAccelEnable(fEnable, pVbvaMemory, mpDrv->pUpPort);
1463 videoAccelLeaveVGA(&mVideoAccelLegacy);
1464 }
1465 LogFlowFunc(("leave %Rrc\n", rc));
1466 return rc;
1467}
1468
1469void Display::VideoAccelFlushVMMDev(void)
1470{
1471 LogFlowFunc(("enter\n"));
1472 int rc = videoAccelEnterVMMDev(&mVideoAccelLegacy);
1473 if (RT_SUCCESS(rc))
1474 {
1475 i_VideoAccelFlush(mpDrv->pUpPort);
1476 videoAccelLeaveVMMDev(&mVideoAccelLegacy);
1477 }
1478 LogFlowFunc(("leave\n"));
1479}
1480
1481/* Called always by one VRDP server thread. Can be thread-unsafe.
1482 */
1483void Display::i_VideoAccelVRDP(bool fEnable)
1484{
1485 LogRelFlowFunc(("fEnable = %d\n", fEnable));
1486
1487 VIDEOACCEL *pVideoAccel = &mVideoAccelLegacy;
1488
1489 int c = fEnable?
1490 ASMAtomicIncS32(&mcVideoAccelVRDPRefs):
1491 ASMAtomicDecS32(&mcVideoAccelVRDPRefs);
1492
1493 Assert (c >= 0);
1494
1495 /* This can run concurrently with Display videoaccel state change. */
1496 RTCritSectEnter(&mVideoAccelLock);
1497
1498 if (c == 0)
1499 {
1500 /* The last client has disconnected, and the accel can be
1501 * disabled.
1502 */
1503 Assert(fEnable == false);
1504
1505 mfVideoAccelVRDP = false;
1506 mfu32SupportedOrders = 0;
1507
1508 i_vbvaSetMemoryFlags(pVideoAccel->pVbvaMemory, pVideoAccel->fVideoAccelEnabled, mfVideoAccelVRDP, mfu32SupportedOrders,
1509 maFramebuffers, mcMonitors);
1510#ifdef VBOX_WITH_HGSMI
1511 /* Here is VRDP-IN thread. Process the request in vbvaUpdateBegin under DevVGA lock on an EMT. */
1512 ASMAtomicIncU32(&mu32UpdateVBVAFlags);
1513#endif /* VBOX_WITH_HGSMI */
1514
1515 LogRel(("VBVA: VRDP acceleration has been disabled.\n"));
1516 }
1517 else if ( c == 1
1518 && !mfVideoAccelVRDP)
1519 {
1520 /* The first client has connected. Enable the accel.
1521 */
1522 Assert(fEnable == true);
1523
1524 mfVideoAccelVRDP = true;
1525 /* Supporting all orders. */
1526 mfu32SupportedOrders = ~0;
1527
1528 i_vbvaSetMemoryFlags(pVideoAccel->pVbvaMemory, pVideoAccel->fVideoAccelEnabled, mfVideoAccelVRDP, mfu32SupportedOrders,
1529 maFramebuffers, mcMonitors);
1530#ifdef VBOX_WITH_HGSMI
1531 /* Here is VRDP-IN thread. Process the request in vbvaUpdateBegin under DevVGA lock on an EMT. */
1532 ASMAtomicIncU32(&mu32UpdateVBVAFlags);
1533#endif /* VBOX_WITH_HGSMI */
1534
1535 LogRel(("VBVA: VRDP acceleration has been requested.\n"));
1536 }
1537 else
1538 {
1539 /* A client is connected or disconnected but there is no change in the
1540 * accel state. It remains enabled.
1541 */
1542 Assert(mfVideoAccelVRDP == true);
1543 }
1544
1545 RTCritSectLeave(&mVideoAccelLock);
1546}
1547
1548void Display::i_notifyPowerDown(void)
1549{
1550 LogRelFlowFunc(("\n"));
1551
1552 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1553
1554 /* Source bitmaps are not available anymore. */
1555 mfSourceBitmapEnabled = false;
1556
1557 alock.release();
1558
1559 /* Resize all displays to tell framebuffers to forget current source bitmap. */
1560 unsigned uScreenId = mcMonitors;
1561 while (uScreenId > 0)
1562 {
1563 --uScreenId;
1564
1565 DISPLAYFBINFO *pFBInfo = &maFramebuffers[uScreenId];
1566 if (!pFBInfo->fDisabled)
1567 {
1568 i_handleDisplayResize(uScreenId, 32,
1569 pFBInfo->pu8FramebufferVRAM,
1570 pFBInfo->u32LineSize,
1571 pFBInfo->w,
1572 pFBInfo->h,
1573 pFBInfo->flags);
1574 }
1575 }
1576}
1577
1578// Wrapped IDisplay methods
1579/////////////////////////////////////////////////////////////////////////////
1580HRESULT Display::getScreenResolution(ULONG aScreenId, ULONG *aWidth, ULONG *aHeight, ULONG *aBitsPerPixel,
1581 LONG *aXOrigin, LONG *aYOrigin, GuestMonitorStatus_T *aGuestMonitorStatus)
1582{
1583 LogRelFlowFunc(("aScreenId=%RU32\n", aScreenId));
1584
1585 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1586
1587 if (aScreenId >= mcMonitors)
1588 return E_INVALIDARG;
1589
1590 DISPLAYFBINFO *pFBInfo = &maFramebuffers[aScreenId];
1591
1592 GuestMonitorStatus_T guestMonitorStatus = GuestMonitorStatus_Enabled;
1593 if (pFBInfo->flags & VBVA_SCREEN_F_DISABLED)
1594 guestMonitorStatus = GuestMonitorStatus_Disabled;
1595
1596 if (aWidth)
1597 *aWidth = pFBInfo->w;
1598 if (aHeight)
1599 *aHeight = pFBInfo->h;
1600 if (aBitsPerPixel)
1601 *aBitsPerPixel = pFBInfo->u16BitsPerPixel;
1602 if (aXOrigin)
1603 *aXOrigin = pFBInfo->xOrigin;
1604 if (aYOrigin)
1605 *aYOrigin = pFBInfo->yOrigin;
1606 if (aGuestMonitorStatus)
1607 *aGuestMonitorStatus = guestMonitorStatus;
1608
1609 return S_OK;
1610}
1611
1612
1613HRESULT Display::attachFramebuffer(ULONG aScreenId, const ComPtr<IFramebuffer> &aFramebuffer, com::Guid &aId)
1614{
1615 LogRelFlowFunc(("aScreenId = %d\n", aScreenId));
1616
1617 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1618
1619 if (aScreenId >= mcMonitors)
1620 return setError(E_INVALIDARG, tr("AttachFramebuffer: Invalid screen %d (total %d)"),
1621 aScreenId, mcMonitors);
1622
1623 DISPLAYFBINFO *pFBInfo = &maFramebuffers[aScreenId];
1624 if (!pFBInfo->pFramebuffer.isNull())
1625 return setError(E_FAIL, tr("AttachFramebuffer: Framebuffer already attached to %d"),
1626 aScreenId);
1627
1628 pFBInfo->pFramebuffer = aFramebuffer;
1629 pFBInfo->framebufferId.create();
1630 aId = pFBInfo->framebufferId;
1631
1632 SafeArray<FramebufferCapabilities_T> caps;
1633 pFBInfo->pFramebuffer->COMGETTER(Capabilities)(ComSafeArrayAsOutParam(caps));
1634 pFBInfo->u32Caps = 0;
1635 size_t i;
1636 for (i = 0; i < caps.size(); ++i)
1637 pFBInfo->u32Caps |= caps[i];
1638
1639 alock.release();
1640
1641 /* The driver might not have been constructed yet */
1642 if (mpDrv)
1643 {
1644 /* Setup the new framebuffer. */
1645 i_handleDisplayResize(aScreenId, pFBInfo->u16BitsPerPixel,
1646 pFBInfo->pu8FramebufferVRAM,
1647 pFBInfo->u32LineSize,
1648 pFBInfo->w,
1649 pFBInfo->h,
1650 pFBInfo->flags);
1651 }
1652
1653 Console::SafeVMPtrQuiet ptrVM(mParent);
1654 if (ptrVM.isOk())
1655 {
1656#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
1657 if (mfIsCr3DEnabled)
1658 {
1659 VBOXCRCMDCTL_HGCM data;
1660 RT_ZERO(data);
1661 data.Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
1662 data.Hdr.u32Function = SHCRGL_HOST_FN_SCREEN_CHANGED;
1663
1664 data.aParms[0].type = VBOX_HGCM_SVC_PARM_32BIT;
1665 data.aParms[0].u.uint32 = aScreenId;
1666
1667 int vrc = i_crCtlSubmitSync(&data.Hdr, sizeof(data));
1668 AssertRC(vrc);
1669 }
1670#endif /* defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL) */
1671
1672 VMR3ReqCallNoWaitU(ptrVM.rawUVM(), VMCPUID_ANY, (PFNRT)Display::i_InvalidateAndUpdateEMT,
1673 3, this, aScreenId, false);
1674 }
1675
1676 LogRelFlowFunc(("Attached to %d %RTuuid\n", aScreenId, aId.raw()));
1677 return S_OK;
1678}
1679
1680HRESULT Display::detachFramebuffer(ULONG aScreenId, const com::Guid &aId)
1681{
1682 LogRelFlowFunc(("aScreenId = %d %RTuuid\n", aScreenId, aId.raw()));
1683
1684 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1685
1686 if (aScreenId >= mcMonitors)
1687 return setError(E_INVALIDARG, tr("DetachFramebuffer: Invalid screen %d (total %d)"),
1688 aScreenId, mcMonitors);
1689
1690 DISPLAYFBINFO *pFBInfo = &maFramebuffers[aScreenId];
1691
1692 if (pFBInfo->framebufferId != aId)
1693 {
1694 LogRelFlowFunc(("Invalid framebuffer aScreenId = %d, attached %p\n", aScreenId, pFBInfo->framebufferId.raw()));
1695 return setError(E_FAIL, tr("DetachFramebuffer: Invalid framebuffer object"));
1696 }
1697
1698 pFBInfo->pFramebuffer.setNull();
1699 pFBInfo->framebufferId.clear();
1700
1701 alock.release();
1702
1703#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
1704 Console::SafeVMPtrQuiet ptrVM(mParent);
1705 if (ptrVM.isOk())
1706 {
1707 if (mfIsCr3DEnabled)
1708 {
1709 VBOXCRCMDCTL_HGCM data;
1710 RT_ZERO(data);
1711 data.Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
1712 data.Hdr.u32Function = SHCRGL_HOST_FN_SCREEN_CHANGED;
1713
1714 data.aParms[0].type = VBOX_HGCM_SVC_PARM_32BIT;
1715 data.aParms[0].u.uint32 = aScreenId;
1716
1717 int vrc = i_crCtlSubmitSync(&data.Hdr, sizeof(data));
1718 AssertRC(vrc);
1719 }
1720 }
1721#endif /* defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL) */
1722
1723 return S_OK;
1724}
1725
1726HRESULT Display::queryFramebuffer(ULONG aScreenId, ComPtr<IFramebuffer> &aFramebuffer)
1727{
1728 LogRelFlowFunc(("aScreenId = %d\n", aScreenId));
1729
1730 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1731
1732 if (aScreenId >= mcMonitors)
1733 return setError(E_INVALIDARG, tr("QueryFramebuffer: Invalid screen %d (total %d)"),
1734 aScreenId, mcMonitors);
1735
1736 DISPLAYFBINFO *pFBInfo = &maFramebuffers[aScreenId];
1737
1738 pFBInfo->pFramebuffer.queryInterfaceTo(aFramebuffer.asOutParam());
1739
1740 return S_OK;
1741}
1742
1743HRESULT Display::setVideoModeHint(ULONG aDisplay, BOOL aEnabled,
1744 BOOL aChangeOrigin, LONG aOriginX, LONG aOriginY,
1745 ULONG aWidth, ULONG aHeight, ULONG aBitsPerPixel)
1746{
1747 if (aWidth == 0 || aHeight == 0 || aBitsPerPixel == 0)
1748 {
1749 /* Some of parameters must not change. Query current mode. */
1750 ULONG ulWidth = 0;
1751 ULONG ulHeight = 0;
1752 ULONG ulBitsPerPixel = 0;
1753 HRESULT hr = getScreenResolution(aDisplay, &ulWidth, &ulHeight, &ulBitsPerPixel, NULL, NULL, NULL);
1754 if (FAILED(hr))
1755 return hr;
1756
1757 /* Assign current values to not changing parameters. */
1758 if (aWidth == 0)
1759 aWidth = ulWidth;
1760 if (aHeight == 0)
1761 aHeight = ulHeight;
1762 if (aBitsPerPixel == 0)
1763 aBitsPerPixel = ulBitsPerPixel;
1764 }
1765
1766 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1767
1768 if (aDisplay >= mcMonitors)
1769 return E_INVALIDARG;
1770
1771 CHECK_CONSOLE_DRV(mpDrv);
1772
1773 /*
1774 * It is up to the guest to decide whether the hint is
1775 * valid. Therefore don't do any VRAM sanity checks here.
1776 */
1777
1778 /* Have to release the lock because the pfnRequestDisplayChange
1779 * will call EMT. */
1780 alock.release();
1781
1782 /* We always send the hint to the graphics card in case the guest enables
1783 * support later. For now we notify exactly when support is enabled. */
1784 mpDrv->pUpPort->pfnSendModeHint(mpDrv->pUpPort, aWidth, aHeight,
1785 aBitsPerPixel, aDisplay,
1786 aChangeOrigin ? aOriginX : ~0,
1787 aChangeOrigin ? aOriginY : ~0,
1788 RT_BOOL(aEnabled),
1789 mfGuestVBVACapabilities
1790 & VBVACAPS_VIDEO_MODE_HINTS);
1791 if ( mfGuestVBVACapabilities & VBVACAPS_VIDEO_MODE_HINTS
1792 && !(mfGuestVBVACapabilities & VBVACAPS_IRQ))
1793 {
1794 mParent->i_sendACPIMonitorHotPlugEvent();
1795 }
1796
1797 /* We currently never suppress the VMMDev hint if the guest has requested
1798 * it. Specifically the video graphics driver may not be responsible for
1799 * screen positioning in the guest virtual desktop, and the component
1800 * responsible may want to get the hint from VMMDev. */
1801 VMMDev *pVMMDev = mParent->i_getVMMDev();
1802 if (pVMMDev)
1803 {
1804 PPDMIVMMDEVPORT pVMMDevPort = pVMMDev->getVMMDevPort();
1805 if (pVMMDevPort)
1806 pVMMDevPort->pfnRequestDisplayChange(pVMMDevPort, aWidth, aHeight, aBitsPerPixel,
1807 aDisplay, aOriginX, aOriginY,
1808 RT_BOOL(aEnabled), RT_BOOL(aChangeOrigin));
1809 }
1810 return S_OK;
1811}
1812
1813HRESULT Display::setSeamlessMode(BOOL enabled)
1814{
1815 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1816
1817 /* Have to release the lock because the pfnRequestSeamlessChange will call EMT. */
1818 alock.release();
1819
1820 VMMDev *pVMMDev = mParent->i_getVMMDev();
1821 if (pVMMDev)
1822 {
1823 PPDMIVMMDEVPORT pVMMDevPort = pVMMDev->getVMMDevPort();
1824 if (pVMMDevPort)
1825 pVMMDevPort->pfnRequestSeamlessChange(pVMMDevPort, !!enabled);
1826 }
1827 mfSeamlessEnabled = RT_BOOL(enabled);
1828
1829#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
1830 if (!enabled)
1831 {
1832 VMMDev *vmmDev = mParent->i_getVMMDev();
1833 if (mfIsCr3DEnabled && vmmDev)
1834 {
1835 VBOXCRCMDCTL_HGCM *pData = (VBOXCRCMDCTL_HGCM*)RTMemAlloc(sizeof(VBOXCRCMDCTL_HGCM));
1836 if (!pData)
1837 {
1838 AssertMsgFailed(("RTMemAlloc failed\n"));
1839 return VERR_NO_MEMORY;
1840 }
1841
1842 pData->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
1843 pData->Hdr.u32Function = SHCRGL_HOST_FN_SET_VISIBLE_REGION;
1844
1845 pData->aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
1846 pData->aParms[0].u.pointer.addr = NULL;
1847 pData->aParms[0].u.pointer.size = 0; /* <- means null rects, NULL pRects address and 0 rects means "disable" */
1848
1849 int rc = i_crCtlSubmit(&pData->Hdr, sizeof(*pData), i_displayCrCmdFree, pData);
1850 if (!RT_SUCCESS(rc))
1851 {
1852 AssertMsgFailed(("crCtlSubmit failed (rc=%Rrc)\n", rc));
1853 RTMemFree(pData);
1854 }
1855 }
1856 }
1857#endif
1858 return S_OK;
1859}
1860
1861#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
1862BOOL Display::i_displayCheckTakeScreenshotCrOgl(Display *pDisplay, ULONG aScreenId, uint8_t *pbData,
1863 uint32_t u32Width, uint32_t u32Height)
1864{
1865 if ( pDisplay->mfIsCr3DEnabled
1866 && pDisplay->mCrOglCallbacks.pfnHasData
1867 && pDisplay->mCrOglCallbacks.pfnHasData())
1868 {
1869 VMMDev *pVMMDev = pDisplay->mParent->i_getVMMDev();
1870 if (pVMMDev)
1871 {
1872 CRVBOXHGCMTAKESCREENSHOT *pScreenshot = (CRVBOXHGCMTAKESCREENSHOT *)RTMemAlloc(sizeof(*pScreenshot));
1873 if (pScreenshot)
1874 {
1875 /* screen id or CRSCREEN_ALL to specify all enabled */
1876 pScreenshot->u32Screen = aScreenId;
1877 pScreenshot->u32Width = u32Width;
1878 pScreenshot->u32Height = u32Height;
1879 pScreenshot->u32Pitch = u32Width * 4;
1880 pScreenshot->pvBuffer = pbData;
1881 pScreenshot->pvContext = NULL;
1882 pScreenshot->pfnScreenshotBegin = NULL;
1883 pScreenshot->pfnScreenshotPerform = NULL;
1884 pScreenshot->pfnScreenshotEnd = NULL;
1885
1886 VBOXCRCMDCTL_HGCM data;
1887 data.Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
1888 data.Hdr.u32Function = SHCRGL_HOST_FN_TAKE_SCREENSHOT;
1889
1890 data.aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
1891 data.aParms[0].u.pointer.addr = pScreenshot;
1892 data.aParms[0].u.pointer.size = sizeof(*pScreenshot);
1893
1894 int rc = pDisplay->i_crCtlSubmitSync(&data.Hdr, sizeof(data));
1895
1896 RTMemFree(pScreenshot);
1897
1898 if (RT_SUCCESS(rc))
1899 return TRUE;
1900 AssertMsgFailed(("failed to get screenshot data from crOgl (rc=%Rrc)\n", rc));
1901 /* fall back to the non-3d mechanism */
1902 }
1903 }
1904 }
1905 return FALSE;
1906}
1907#endif
1908
1909/* static */
1910int Display::i_displayTakeScreenshotEMT(Display *pDisplay, ULONG aScreenId, uint8_t **ppbData, size_t *pcbData,
1911 uint32_t *pcx, uint32_t *pcy, bool *pfMemFree)
1912{
1913 int rc;
1914 if ( aScreenId == VBOX_VIDEO_PRIMARY_SCREEN
1915 && pDisplay->maFramebuffers[aScreenId].fVBVAEnabled == false) /* A non-VBVA mode. */
1916 {
1917 rc = pDisplay->mpDrv->pUpPort->pfnTakeScreenshot(pDisplay->mpDrv->pUpPort, ppbData, pcbData, pcx, pcy);
1918 *pfMemFree = false;
1919 }
1920 else if (aScreenId < pDisplay->mcMonitors)
1921 {
1922 DISPLAYFBINFO *pFBInfo = &pDisplay->maFramebuffers[aScreenId];
1923
1924 uint32_t width = pFBInfo->w;
1925 uint32_t height = pFBInfo->h;
1926
1927 /* Allocate 32 bit per pixel bitmap. */
1928 size_t cbRequired = width * 4 * height;
1929
1930 if (cbRequired)
1931 {
1932 uint8_t *pbDst = (uint8_t *)RTMemAlloc(cbRequired);
1933 if (pbDst != NULL)
1934 {
1935 if (pFBInfo->flags & VBVA_SCREEN_F_ACTIVE)
1936 {
1937 /* Copy guest VRAM to the allocated 32bpp buffer. */
1938 const uint8_t *pu8Src = pFBInfo->pu8FramebufferVRAM;
1939 int32_t xSrc = 0;
1940 int32_t ySrc = 0;
1941 uint32_t u32SrcWidth = width;
1942 uint32_t u32SrcHeight = height;
1943 uint32_t u32SrcLineSize = pFBInfo->u32LineSize;
1944 uint32_t u32SrcBitsPerPixel = pFBInfo->u16BitsPerPixel;
1945
1946 int32_t xDst = 0;
1947 int32_t yDst = 0;
1948 uint32_t u32DstWidth = u32SrcWidth;
1949 uint32_t u32DstHeight = u32SrcHeight;
1950 uint32_t u32DstLineSize = u32DstWidth * 4;
1951 uint32_t u32DstBitsPerPixel = 32;
1952
1953 rc = pDisplay->mpDrv->pUpPort->pfnCopyRect(pDisplay->mpDrv->pUpPort,
1954 width, height,
1955 pu8Src,
1956 xSrc, ySrc,
1957 u32SrcWidth, u32SrcHeight,
1958 u32SrcLineSize, u32SrcBitsPerPixel,
1959 pbDst,
1960 xDst, yDst,
1961 u32DstWidth, u32DstHeight,
1962 u32DstLineSize, u32DstBitsPerPixel);
1963 }
1964 else
1965 {
1966 memset(pbDst, 0, cbRequired);
1967 rc = VINF_SUCCESS;
1968 }
1969 if (RT_SUCCESS(rc))
1970 {
1971 *ppbData = pbDst;
1972 *pcbData = cbRequired;
1973 *pcx = width;
1974 *pcy = height;
1975 *pfMemFree = true;
1976 }
1977 else
1978 {
1979 RTMemFree(pbDst);
1980
1981 /* CopyRect can fail if VBVA was paused in VGA device, retry using the generic method. */
1982 if ( rc == VERR_INVALID_STATE
1983 && aScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
1984 {
1985 rc = pDisplay->mpDrv->pUpPort->pfnTakeScreenshot(pDisplay->mpDrv->pUpPort, ppbData, pcbData, pcx, pcy);
1986 *pfMemFree = false;
1987 }
1988 }
1989 }
1990 else
1991 rc = VERR_NO_MEMORY;
1992 }
1993 else
1994 {
1995 /* No image. */
1996 *ppbData = NULL;
1997 *pcbData = 0;
1998 *pcx = 0;
1999 *pcy = 0;
2000 *pfMemFree = true;
2001 rc = VINF_SUCCESS;
2002 }
2003 }
2004 else
2005 rc = VERR_INVALID_PARAMETER;
2006 return rc;
2007}
2008
2009static int i_displayTakeScreenshot(PUVM pUVM, Display *pDisplay, struct DRVMAINDISPLAY *pDrv, ULONG aScreenId,
2010 BYTE *address, ULONG width, ULONG height)
2011{
2012#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
2013 /*
2014 * CrOgl screenshot hook/hack.
2015 */
2016 if (Display::i_displayCheckTakeScreenshotCrOgl(pDisplay, aScreenId, (uint8_t *)address, width, height))
2017 return VINF_SUCCESS;
2018#endif
2019
2020 uint8_t *pbData = NULL;
2021 size_t cbData = 0;
2022 uint32_t cx = 0;
2023 uint32_t cy = 0;
2024 bool fFreeMem = false;
2025 int vrc = VINF_SUCCESS;
2026
2027 int cRetries = 5;
2028 while (cRetries-- > 0)
2029 {
2030 /* Note! Not sure if the priority call is such a good idea here, but
2031 it would be nice to have an accurate screenshot for the bug
2032 report if the VM deadlocks. */
2033 vrc = VMR3ReqPriorityCallWaitU(pUVM, VMCPUID_ANY, (PFNRT)Display::i_displayTakeScreenshotEMT, 7,
2034 pDisplay, aScreenId, &pbData, &cbData, &cx, &cy, &fFreeMem);
2035 if (vrc != VERR_TRY_AGAIN)
2036 {
2037 break;
2038 }
2039
2040 RTThreadSleep(10);
2041 }
2042
2043 if (RT_SUCCESS(vrc) && pbData)
2044 {
2045 if (cx == width && cy == height)
2046 {
2047 /* No scaling required. */
2048 memcpy(address, pbData, cbData);
2049 }
2050 else
2051 {
2052 /* Scale. */
2053 LogRelFlowFunc(("SCALE: %dx%d -> %dx%d\n", cx, cy, width, height));
2054
2055 uint8_t *dst = address;
2056 uint8_t *src = pbData;
2057 int dstW = width;
2058 int dstH = height;
2059 int srcW = cx;
2060 int srcH = cy;
2061 int iDeltaLine = cx * 4;
2062
2063 BitmapScale32(dst,
2064 dstW, dstH,
2065 src,
2066 iDeltaLine,
2067 srcW, srcH);
2068 }
2069
2070 if (fFreeMem)
2071 RTMemFree(pbData);
2072 else
2073 {
2074 /* This can be called from any thread. */
2075 pDrv->pUpPort->pfnFreeScreenshot(pDrv->pUpPort, pbData);
2076 }
2077 }
2078
2079 return vrc;
2080}
2081
2082HRESULT Display::takeScreenShotWorker(ULONG aScreenId,
2083 BYTE *aAddress,
2084 ULONG aWidth,
2085 ULONG aHeight,
2086 BitmapFormat_T aBitmapFormat,
2087 ULONG *pcbOut)
2088{
2089 HRESULT rc = S_OK;
2090
2091 /* Do not allow too small and too large screenshots. This also filters out negative
2092 * values passed as either 'aWidth' or 'aHeight'.
2093 */
2094 CheckComArgExpr(aWidth, aWidth != 0 && aWidth <= 32767);
2095 CheckComArgExpr(aHeight, aHeight != 0 && aHeight <= 32767);
2096
2097 if ( aBitmapFormat != BitmapFormat_BGR0
2098 && aBitmapFormat != BitmapFormat_BGRA
2099 && aBitmapFormat != BitmapFormat_RGBA
2100 && aBitmapFormat != BitmapFormat_PNG)
2101 {
2102 return setError(E_NOTIMPL,
2103 tr("Unsupported screenshot format 0x%08X"), aBitmapFormat);
2104 }
2105
2106 Console::SafeVMPtr ptrVM(mParent);
2107 if (!ptrVM.isOk())
2108 return ptrVM.rc();
2109
2110 int vrc = i_displayTakeScreenshot(ptrVM.rawUVM(), this, mpDrv, aScreenId, aAddress, aWidth, aHeight);
2111
2112 if (RT_SUCCESS(vrc))
2113 {
2114 const size_t cbData = aWidth * 4 * aHeight;
2115
2116 /* Most of uncompressed formats. */
2117 *pcbOut = (ULONG)cbData;
2118
2119 if (aBitmapFormat == BitmapFormat_BGR0)
2120 {
2121 /* Do nothing. */
2122 }
2123 else if (aBitmapFormat == BitmapFormat_BGRA)
2124 {
2125 uint32_t *pu32 = (uint32_t *)aAddress;
2126 size_t cPixels = aWidth * aHeight;
2127 while (cPixels--)
2128 {
2129 *pu32++ |= UINT32_C(0xFF000000);
2130 }
2131 }
2132 else if (aBitmapFormat == BitmapFormat_RGBA)
2133 {
2134 uint8_t *pu8 = aAddress;
2135 size_t cPixels = aWidth * aHeight;
2136 while (cPixels--)
2137 {
2138 uint8_t u8 = pu8[0];
2139 pu8[0] = pu8[2];
2140 pu8[2] = u8;
2141 pu8[3] = 0xFF;
2142
2143 pu8 += 4;
2144 }
2145 }
2146 else if (aBitmapFormat == BitmapFormat_PNG)
2147 {
2148 uint8_t *pu8PNG = NULL;
2149 uint32_t cbPNG = 0;
2150 uint32_t cxPNG = 0;
2151 uint32_t cyPNG = 0;
2152
2153 vrc = DisplayMakePNG(aAddress, aWidth, aHeight, &pu8PNG, &cbPNG, &cxPNG, &cyPNG, 0);
2154 if (RT_SUCCESS(vrc))
2155 {
2156 if (cbPNG <= cbData)
2157 {
2158 memcpy(aAddress, pu8PNG, cbPNG);
2159 *pcbOut = cbPNG;
2160 }
2161 else
2162 {
2163 rc = setError(E_FAIL,
2164 tr("PNG is larger than 32bpp bitmap"));
2165 }
2166 }
2167 else
2168 {
2169 rc = setError(VBOX_E_IPRT_ERROR,
2170 tr("Could not convert screenshot to PNG (%Rrc)"), vrc);
2171 }
2172 RTMemFree(pu8PNG);
2173 }
2174 }
2175 else if (vrc == VERR_TRY_AGAIN)
2176 rc = setError(E_UNEXPECTED,
2177 tr("Screenshot is not available at this time"));
2178 else if (RT_FAILURE(vrc))
2179 rc = setError(VBOX_E_IPRT_ERROR,
2180 tr("Could not take a screenshot (%Rrc)"), vrc);
2181
2182 return rc;
2183}
2184
2185HRESULT Display::takeScreenShot(ULONG aScreenId,
2186 BYTE *aAddress,
2187 ULONG aWidth,
2188 ULONG aHeight,
2189 BitmapFormat_T aBitmapFormat)
2190{
2191 HRESULT rc = S_OK;
2192
2193 LogRelFlowFunc(("[%d] address=%p, width=%d, height=%d, format 0x%08X\n",
2194 aScreenId, aAddress, aWidth, aHeight, aBitmapFormat));
2195
2196 ULONG cbOut = 0;
2197 rc = takeScreenShotWorker(aScreenId, aAddress, aWidth, aHeight, aBitmapFormat, &cbOut);
2198 NOREF(cbOut);
2199
2200 LogRelFlowFunc(("%Rhrc\n", rc));
2201 return rc;
2202}
2203
2204HRESULT Display::takeScreenShotToArray(ULONG aScreenId,
2205 ULONG aWidth,
2206 ULONG aHeight,
2207 BitmapFormat_T aBitmapFormat,
2208 std::vector<BYTE> &aScreenData)
2209{
2210 HRESULT rc = S_OK;
2211
2212 LogRelFlowFunc(("[%d] width=%d, height=%d, format 0x%08X\n",
2213 aScreenId, aWidth, aHeight, aBitmapFormat));
2214
2215 /* Do not allow too small and too large screenshots. This also filters out negative
2216 * values passed as either 'aWidth' or 'aHeight'.
2217 */
2218 CheckComArgExpr(aWidth, aWidth != 0 && aWidth <= 32767);
2219 CheckComArgExpr(aHeight, aHeight != 0 && aHeight <= 32767);
2220
2221 const size_t cbData = aWidth * 4 * aHeight;
2222 aScreenData.resize(cbData);
2223
2224 ULONG cbOut = 0;
2225 rc = takeScreenShotWorker(aScreenId, &aScreenData.front(), aWidth, aHeight, aBitmapFormat, &cbOut);
2226 if (FAILED(rc))
2227 cbOut = 0;
2228
2229 aScreenData.resize(cbOut);
2230
2231 LogRelFlowFunc(("%Rhrc\n", rc));
2232 return rc;
2233}
2234
2235
2236int Display::i_VideoCaptureEnableScreens(ComSafeArrayIn(BOOL, aScreens))
2237{
2238#ifdef VBOX_WITH_VPX
2239 com::SafeArray<BOOL> Screens(ComSafeArrayInArg(aScreens));
2240 for (unsigned i = 0; i < Screens.size(); i++)
2241 {
2242 bool fChanged = maVideoRecEnabled[i] != RT_BOOL(Screens[i]);
2243
2244 maVideoRecEnabled[i] = RT_BOOL(Screens[i]);
2245
2246 if (fChanged && i < mcMonitors)
2247 videoCaptureScreenChanged(i);
2248
2249 }
2250 return VINF_SUCCESS;
2251#else
2252 return VERR_NOT_IMPLEMENTED;
2253#endif
2254}
2255
2256/**
2257 * Start video capturing. Does nothing if capturing is already active.
2258 */
2259int Display::i_VideoCaptureStart()
2260{
2261#ifdef VBOX_WITH_VPX
2262 if (VideoRecIsEnabled(mpVideoRecCtx))
2263 return VINF_SUCCESS;
2264
2265 int rc = VideoRecContextCreate(&mpVideoRecCtx, mcMonitors);
2266 if (RT_FAILURE(rc))
2267 {
2268 LogFlow(("Failed to create video recording context (%Rrc)!\n", rc));
2269 return rc;
2270 }
2271 ComPtr<IMachine> pMachine = mParent->i_machine();
2272 com::SafeArray<BOOL> screens;
2273 HRESULT hrc = pMachine->COMGETTER(VideoCaptureScreens)(ComSafeArrayAsOutParam(screens));
2274 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2275 for (unsigned i = 0; i < RT_ELEMENTS(maVideoRecEnabled); i++)
2276 maVideoRecEnabled[i] = i < screens.size() && screens[i];
2277 ULONG ulWidth;
2278 hrc = pMachine->COMGETTER(VideoCaptureWidth)(&ulWidth);
2279 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2280 ULONG ulHeight;
2281 hrc = pMachine->COMGETTER(VideoCaptureHeight)(&ulHeight);
2282 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2283 ULONG ulRate;
2284 hrc = pMachine->COMGETTER(VideoCaptureRate)(&ulRate);
2285 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2286 ULONG ulFPS;
2287 hrc = pMachine->COMGETTER(VideoCaptureFPS)(&ulFPS);
2288 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2289 BSTR strFile;
2290 hrc = pMachine->COMGETTER(VideoCaptureFile)(&strFile);
2291 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2292 ULONG ulMaxTime;
2293 hrc = pMachine->COMGETTER(VideoCaptureMaxTime)(&ulMaxTime);
2294 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2295 ULONG ulMaxSize;
2296 hrc = pMachine->COMGETTER(VideoCaptureMaxFileSize)(&ulMaxSize);
2297 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2298 BSTR strOptions;
2299 hrc = pMachine->COMGETTER(VideoCaptureOptions)(&strOptions);
2300 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
2301
2302 RTTIMESPEC ts;
2303 RTTimeNow(&ts);
2304 RTTIME time;
2305 RTTimeExplode(&time, &ts);
2306 for (unsigned uScreen = 0; uScreen < mcMonitors; uScreen++)
2307 {
2308 char *pszAbsPath = RTPathAbsDup(com::Utf8Str(strFile).c_str());
2309 char *pszSuff = RTPathSuffix(pszAbsPath);
2310 if (pszSuff)
2311 pszSuff = RTStrDup(pszSuff);
2312 RTPathStripSuffix(pszAbsPath);
2313 if (!pszAbsPath)
2314 rc = VERR_INVALID_PARAMETER;
2315 if (!pszSuff)
2316 pszSuff = RTStrDup(".webm");
2317 char *pszName = NULL;
2318 if (RT_SUCCESS(rc))
2319 {
2320 if (mcMonitors > 1)
2321 rc = RTStrAPrintf(&pszName, "%s-%u%s", pszAbsPath, uScreen+1, pszSuff);
2322 else
2323 rc = RTStrAPrintf(&pszName, "%s%s", pszAbsPath, pszSuff);
2324 }
2325 if (RT_SUCCESS(rc))
2326 {
2327 rc = VideoRecStrmInit(mpVideoRecCtx, uScreen,
2328 pszName, ulWidth, ulHeight,
2329 ulRate, ulFPS, ulMaxTime,
2330 ulMaxSize, com::Utf8Str(strOptions).c_str());
2331 if (rc == VERR_ALREADY_EXISTS)
2332 {
2333 RTStrFree(pszName);
2334 pszName = NULL;
2335
2336 if (mcMonitors > 1)
2337 rc = RTStrAPrintf(&pszName, "%s-%04d-%02u-%02uT%02u-%02u-%02u-%09uZ-%u%s",
2338 pszAbsPath, time.i32Year, time.u8Month, time.u8MonthDay,
2339 time.u8Hour, time.u8Minute, time.u8Second, time.u32Nanosecond,
2340 uScreen+1, pszSuff);
2341 else
2342 rc = RTStrAPrintf(&pszName, "%s-%04d-%02u-%02uT%02u-%02u-%02u-%09uZ%s",
2343 pszAbsPath, time.i32Year, time.u8Month, time.u8MonthDay,
2344 time.u8Hour, time.u8Minute, time.u8Second, time.u32Nanosecond,
2345 pszSuff);
2346 if (RT_SUCCESS(rc))
2347 rc = VideoRecStrmInit(mpVideoRecCtx, uScreen,
2348 pszName, ulWidth, ulHeight, ulRate,
2349 ulFPS, ulMaxTime,
2350 ulMaxSize, com::Utf8Str(strOptions).c_str());
2351 }
2352 }
2353
2354 if (RT_SUCCESS(rc))
2355 {
2356 LogRel(("Display::VideoCaptureStart: WebM/VP8 video recording screen #%u with %ux%u @ %u kbps, %u fps to '%s' "
2357 "enabled\n", uScreen, ulWidth, ulHeight, ulRate, ulFPS, pszName));
2358
2359 videoCaptureScreenChanged(uScreen);
2360 }
2361 else
2362 LogRel(("Display::VideoCaptureStart: Failed to initialize video recording context #%u (%Rrc)!\n", uScreen, rc));
2363 RTStrFree(pszName);
2364 RTStrFree(pszSuff);
2365 RTStrFree(pszAbsPath);
2366 }
2367 return rc;
2368#else
2369 return VERR_NOT_IMPLEMENTED;
2370#endif
2371}
2372
2373/**
2374 * Stop video capturing. Does nothing if video capturing is not active.
2375 */
2376void Display::i_VideoCaptureStop()
2377{
2378#ifdef VBOX_WITH_VPX
2379 if (VideoRecIsEnabled(mpVideoRecCtx))
2380 LogRel(("Display::VideoCaptureStop: WebM/VP8 video recording stopped\n"));
2381 VideoRecContextClose(mpVideoRecCtx);
2382 mpVideoRecCtx = NULL;
2383
2384 unsigned uScreenId;
2385 for (uScreenId = 0; uScreenId < mcMonitors; ++uScreenId)
2386 videoCaptureScreenChanged(uScreenId);
2387#endif
2388}
2389
2390#ifdef VBOX_WITH_VPX
2391void Display::videoCaptureScreenChanged(unsigned uScreenId)
2392{
2393 ComPtr<IDisplaySourceBitmap> pSourceBitmap;
2394
2395 if (VideoRecIsEnabled(mpVideoRecCtx) && maVideoRecEnabled[uScreenId])
2396 {
2397 /* Get a new source bitmap which will be used by video capture code. */
2398 QuerySourceBitmap(uScreenId, pSourceBitmap.asOutParam());
2399 }
2400
2401 int rc = RTCritSectEnter(&mVideoCaptureLock);
2402 if (RT_SUCCESS(rc))
2403 {
2404 maFramebuffers[uScreenId].videoCapture.pSourceBitmap = pSourceBitmap;
2405 RTCritSectLeave(&mVideoCaptureLock);
2406 }
2407}
2408#endif
2409
2410int Display::i_drawToScreenEMT(Display *pDisplay, ULONG aScreenId, BYTE *address,
2411 ULONG x, ULONG y, ULONG width, ULONG height)
2412{
2413 int rc = VINF_SUCCESS;
2414
2415 DISPLAYFBINFO *pFBInfo = &pDisplay->maFramebuffers[aScreenId];
2416
2417 if (aScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
2418 {
2419 rc = pDisplay->mpDrv->pUpPort->pfnDisplayBlt(pDisplay->mpDrv->pUpPort, address, x, y, width, height);
2420 }
2421 else if (aScreenId < pDisplay->mcMonitors)
2422 {
2423 /* Copy the bitmap to the guest VRAM. */
2424 const uint8_t *pu8Src = address;
2425 int32_t xSrc = 0;
2426 int32_t ySrc = 0;
2427 uint32_t u32SrcWidth = width;
2428 uint32_t u32SrcHeight = height;
2429 uint32_t u32SrcLineSize = width * 4;
2430 uint32_t u32SrcBitsPerPixel = 32;
2431
2432 uint8_t *pu8Dst = pFBInfo->pu8FramebufferVRAM;
2433 int32_t xDst = x;
2434 int32_t yDst = y;
2435 uint32_t u32DstWidth = pFBInfo->w;
2436 uint32_t u32DstHeight = pFBInfo->h;
2437 uint32_t u32DstLineSize = pFBInfo->u32LineSize;
2438 uint32_t u32DstBitsPerPixel = pFBInfo->u16BitsPerPixel;
2439
2440 rc = pDisplay->mpDrv->pUpPort->pfnCopyRect(pDisplay->mpDrv->pUpPort,
2441 width, height,
2442 pu8Src,
2443 xSrc, ySrc,
2444 u32SrcWidth, u32SrcHeight,
2445 u32SrcLineSize, u32SrcBitsPerPixel,
2446 pu8Dst,
2447 xDst, yDst,
2448 u32DstWidth, u32DstHeight,
2449 u32DstLineSize, u32DstBitsPerPixel);
2450 if (RT_SUCCESS(rc))
2451 {
2452 if (!pFBInfo->pSourceBitmap.isNull())
2453 {
2454 /* Update the changed screen area. When source bitmap uses VRAM directly, just notify
2455 * frontend to update. And for default format, render the guest VRAM to the source bitmap.
2456 */
2457 if ( pFBInfo->fDefaultFormat
2458 && !pFBInfo->fDisabled)
2459 {
2460 BYTE *pAddress = NULL;
2461 ULONG ulWidth = 0;
2462 ULONG ulHeight = 0;
2463 ULONG ulBitsPerPixel = 0;
2464 ULONG ulBytesPerLine = 0;
2465 BitmapFormat_T bitmapFormat = BitmapFormat_Opaque;
2466
2467 HRESULT hrc = pFBInfo->pSourceBitmap->QueryBitmapInfo(&pAddress,
2468 &ulWidth,
2469 &ulHeight,
2470 &ulBitsPerPixel,
2471 &ulBytesPerLine,
2472 &bitmapFormat);
2473 if (SUCCEEDED(hrc))
2474 {
2475 pu8Src = pFBInfo->pu8FramebufferVRAM;
2476 xSrc = x;
2477 ySrc = y;
2478 u32SrcWidth = pFBInfo->w;
2479 u32SrcHeight = pFBInfo->h;
2480 u32SrcLineSize = pFBInfo->u32LineSize;
2481 u32SrcBitsPerPixel = pFBInfo->u16BitsPerPixel;
2482
2483 /* Default format is 32 bpp. */
2484 pu8Dst = pAddress;
2485 xDst = xSrc;
2486 yDst = ySrc;
2487 u32DstWidth = u32SrcWidth;
2488 u32DstHeight = u32SrcHeight;
2489 u32DstLineSize = u32DstWidth * 4;
2490 u32DstBitsPerPixel = 32;
2491
2492 pDisplay->mpDrv->pUpPort->pfnCopyRect(pDisplay->mpDrv->pUpPort,
2493 width, height,
2494 pu8Src,
2495 xSrc, ySrc,
2496 u32SrcWidth, u32SrcHeight,
2497 u32SrcLineSize, u32SrcBitsPerPixel,
2498 pu8Dst,
2499 xDst, yDst,
2500 u32DstWidth, u32DstHeight,
2501 u32DstLineSize, u32DstBitsPerPixel);
2502 }
2503 }
2504 }
2505
2506 pDisplay->i_handleDisplayUpdate(aScreenId, x, y, width, height);
2507 }
2508 }
2509 else
2510 {
2511 rc = VERR_INVALID_PARAMETER;
2512 }
2513
2514 if (RT_SUCCESS(rc))
2515 pDisplay->mParent->i_consoleVRDPServer()->SendUpdateBitmap(aScreenId, x, y, width, height);
2516
2517 return rc;
2518}
2519
2520HRESULT Display::drawToScreen(ULONG aScreenId, BYTE *aAddress, ULONG aX, ULONG aY, ULONG aWidth, ULONG aHeight)
2521{
2522 /// @todo (r=dmik) this function may take too long to complete if the VM
2523 // is doing something like saving state right now. Which, in case if it
2524 // is called on the GUI thread, will make it unresponsive. We should
2525 // check the machine state here (by enclosing the check and VMRequCall
2526 // within the Console lock to make it atomic).
2527
2528 LogRelFlowFunc(("aAddress=%p, x=%d, y=%d, width=%d, height=%d\n",
2529 (void *)aAddress, aX, aY, aWidth, aHeight));
2530
2531 CheckComArgExpr(aWidth, aWidth != 0);
2532 CheckComArgExpr(aHeight, aHeight != 0);
2533
2534 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2535
2536 CHECK_CONSOLE_DRV(mpDrv);
2537
2538 Console::SafeVMPtr ptrVM(mParent);
2539 if (!ptrVM.isOk())
2540 return ptrVM.rc();
2541
2542 /* Release lock because the call scheduled on EMT may also try to take it. */
2543 alock.release();
2544
2545 /*
2546 * Again we're lazy and make the graphics device do all the
2547 * dirty conversion work.
2548 */
2549 int rcVBox = VMR3ReqCallWaitU(ptrVM.rawUVM(), VMCPUID_ANY, (PFNRT)Display::i_drawToScreenEMT, 7,
2550 this, aScreenId, aAddress, aX, aY, aWidth, aHeight);
2551
2552 /*
2553 * If the function returns not supported, we'll have to do all the
2554 * work ourselves using the framebuffer.
2555 */
2556 HRESULT rc = S_OK;
2557 if (rcVBox == VERR_NOT_SUPPORTED || rcVBox == VERR_NOT_IMPLEMENTED)
2558 {
2559 /** @todo implement generic fallback for screen blitting. */
2560 rc = E_NOTIMPL;
2561 }
2562 else if (RT_FAILURE(rcVBox))
2563 rc = setError(VBOX_E_IPRT_ERROR,
2564 tr("Could not draw to the screen (%Rrc)"), rcVBox);
2565//@todo
2566// else
2567// {
2568// /* All ok. Redraw the screen. */
2569// handleDisplayUpdate(x, y, width, height);
2570// }
2571
2572 LogRelFlowFunc(("rc=%Rhrc\n", rc));
2573 return rc;
2574}
2575
2576int Display::i_InvalidateAndUpdateEMT(Display *pDisplay, unsigned uId, bool fUpdateAll)
2577{
2578 LogRelFlowFunc(("uId=%d, fUpdateAll %d\n", uId, fUpdateAll));
2579
2580 unsigned uScreenId;
2581 for (uScreenId = (fUpdateAll ? 0 : uId); uScreenId < pDisplay->mcMonitors; uScreenId++)
2582 {
2583 DISPLAYFBINFO *pFBInfo = &pDisplay->maFramebuffers[uScreenId];
2584
2585 if ( !pFBInfo->fVBVAEnabled
2586 && uScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
2587 {
2588 pDisplay->mpDrv->pUpPort->pfnUpdateDisplayAll(pDisplay->mpDrv->pUpPort, /* fFailOnResize = */ true);
2589 }
2590 else
2591 {
2592 if (!pFBInfo->fDisabled)
2593 {
2594 /* Render complete VRAM screen to the framebuffer.
2595 * When framebuffer uses VRAM directly, just notify it to update.
2596 */
2597 if (pFBInfo->fDefaultFormat && !pFBInfo->pSourceBitmap.isNull())
2598 {
2599 BYTE *pAddress = NULL;
2600 ULONG ulWidth = 0;
2601 ULONG ulHeight = 0;
2602 ULONG ulBitsPerPixel = 0;
2603 ULONG ulBytesPerLine = 0;
2604 BitmapFormat_T bitmapFormat = BitmapFormat_Opaque;
2605
2606 HRESULT hrc = pFBInfo->pSourceBitmap->QueryBitmapInfo(&pAddress,
2607 &ulWidth,
2608 &ulHeight,
2609 &ulBitsPerPixel,
2610 &ulBytesPerLine,
2611 &bitmapFormat);
2612 if (SUCCEEDED(hrc))
2613 {
2614 uint32_t width = pFBInfo->w;
2615 uint32_t height = pFBInfo->h;
2616
2617 const uint8_t *pu8Src = pFBInfo->pu8FramebufferVRAM;
2618 int32_t xSrc = 0;
2619 int32_t ySrc = 0;
2620 uint32_t u32SrcWidth = pFBInfo->w;
2621 uint32_t u32SrcHeight = pFBInfo->h;
2622 uint32_t u32SrcLineSize = pFBInfo->u32LineSize;
2623 uint32_t u32SrcBitsPerPixel = pFBInfo->u16BitsPerPixel;
2624
2625 /* Default format is 32 bpp. */
2626 uint8_t *pu8Dst = pAddress;
2627 int32_t xDst = xSrc;
2628 int32_t yDst = ySrc;
2629 uint32_t u32DstWidth = u32SrcWidth;
2630 uint32_t u32DstHeight = u32SrcHeight;
2631 uint32_t u32DstLineSize = u32DstWidth * 4;
2632 uint32_t u32DstBitsPerPixel = 32;
2633
2634 /* if uWidth != pFBInfo->w and uHeight != pFBInfo->h
2635 * implies resize of Framebuffer is in progress and
2636 * copyrect should not be called.
2637 */
2638 if (ulWidth == pFBInfo->w && ulHeight == pFBInfo->h)
2639 {
2640 pDisplay->mpDrv->pUpPort->pfnCopyRect(pDisplay->mpDrv->pUpPort,
2641 width, height,
2642 pu8Src,
2643 xSrc, ySrc,
2644 u32SrcWidth, u32SrcHeight,
2645 u32SrcLineSize, u32SrcBitsPerPixel,
2646 pu8Dst,
2647 xDst, yDst,
2648 u32DstWidth, u32DstHeight,
2649 u32DstLineSize, u32DstBitsPerPixel);
2650 }
2651 }
2652 }
2653
2654 pDisplay->i_handleDisplayUpdate(uScreenId, 0, 0, pFBInfo->w, pFBInfo->h);
2655 }
2656 }
2657 if (!fUpdateAll)
2658 break;
2659 }
2660 LogRelFlowFunc(("done\n"));
2661 return VINF_SUCCESS;
2662}
2663
2664/**
2665 * Does a full invalidation of the VM display and instructs the VM
2666 * to update it immediately.
2667 *
2668 * @returns COM status code
2669 */
2670
2671HRESULT Display::invalidateAndUpdate()
2672{
2673 LogRelFlowFunc(("\n"));
2674
2675 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2676
2677 CHECK_CONSOLE_DRV(mpDrv);
2678
2679 Console::SafeVMPtr ptrVM(mParent);
2680 if (!ptrVM.isOk())
2681 return ptrVM.rc();
2682
2683 HRESULT rc = S_OK;
2684
2685 LogRelFlowFunc(("Sending DPYUPDATE request\n"));
2686
2687 /* Have to release the lock when calling EMT. */
2688 alock.release();
2689
2690 int rcVBox = VMR3ReqCallNoWaitU(ptrVM.rawUVM(), VMCPUID_ANY, (PFNRT)Display::i_InvalidateAndUpdateEMT,
2691 3, this, 0, true);
2692 alock.acquire();
2693
2694 if (RT_FAILURE(rcVBox))
2695 rc = setError(VBOX_E_IPRT_ERROR,
2696 tr("Could not invalidate and update the screen (%Rrc)"), rcVBox);
2697
2698 LogRelFlowFunc(("rc=%Rhrc\n", rc));
2699 return rc;
2700}
2701
2702HRESULT Display::invalidateAndUpdateScreen(ULONG aScreenId)
2703{
2704 LogRelFlowFunc(("\n"));
2705
2706 HRESULT rc = S_OK;
2707
2708 Console::SafeVMPtr ptrVM(mParent);
2709 if (!ptrVM.isOk())
2710 return ptrVM.rc();
2711
2712 int rcVBox = VMR3ReqCallNoWaitU(ptrVM.rawUVM(), VMCPUID_ANY, (PFNRT)Display::i_InvalidateAndUpdateEMT,
2713 3, this, aScreenId, false);
2714 if (RT_FAILURE(rcVBox))
2715 rc = setError(VBOX_E_IPRT_ERROR,
2716 tr("Could not invalidate and update the screen %d (%Rrc)"), aScreenId, rcVBox);
2717
2718 LogRelFlowFunc(("rc=%Rhrc\n", rc));
2719 return rc;
2720}
2721
2722HRESULT Display::completeVHWACommand(BYTE *aCommand)
2723{
2724#ifdef VBOX_WITH_VIDEOHWACCEL
2725 mpDrv->pVBVACallbacks->pfnVHWACommandCompleteAsync(mpDrv->pVBVACallbacks, (PVBOXVHWACMD)aCommand);
2726 return S_OK;
2727#else
2728 return E_NOTIMPL;
2729#endif
2730}
2731
2732HRESULT Display::viewportChanged(ULONG aScreenId, ULONG aX, ULONG aY, ULONG aWidth, ULONG aHeight)
2733{
2734 AssertMsgReturn(aScreenId < mcMonitors, ("aScreendId=%d mcMonitors=%d\n", aScreenId, mcMonitors), E_INVALIDARG);
2735
2736#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
2737 if (mfIsCr3DEnabled)
2738 {
2739 int rc = i_crViewportNotify(aScreenId, aX, aY, aWidth, aHeight);
2740 if (RT_FAILURE(rc))
2741 {
2742 DISPLAYFBINFO *pFb = &maFramebuffers[aScreenId];
2743 pFb->pendingViewportInfo.fPending = true;
2744 pFb->pendingViewportInfo.x = aX;
2745 pFb->pendingViewportInfo.y = aY;
2746 pFb->pendingViewportInfo.width = aWidth;
2747 pFb->pendingViewportInfo.height = aHeight;
2748 }
2749 }
2750#endif /* VBOX_WITH_CROGL && VBOX_WITH_HGCM */
2751
2752 /* The driver might not have been constructed yet */
2753 if (mpDrv && mpDrv->pUpPort->pfnSetViewport)
2754 mpDrv->pUpPort->pfnSetViewport(mpDrv->pUpPort, aScreenId, aX, aY, aWidth, aHeight);
2755
2756 return S_OK;
2757}
2758
2759HRESULT Display::querySourceBitmap(ULONG aScreenId,
2760 ComPtr<IDisplaySourceBitmap> &aDisplaySourceBitmap)
2761{
2762 LogRelFlowFunc(("aScreenId = %d\n", aScreenId));
2763
2764 Console::SafeVMPtr ptrVM(mParent);
2765 if (!ptrVM.isOk())
2766 return ptrVM.rc();
2767
2768 bool fSetRenderVRAM = false;
2769 bool fInvalidate = false;
2770
2771 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2772
2773 if (aScreenId >= mcMonitors)
2774 return setError(E_INVALIDARG, tr("QuerySourceBitmap: Invalid screen %d (total %d)"),
2775 aScreenId, mcMonitors);
2776
2777 if (!mfSourceBitmapEnabled)
2778 {
2779 aDisplaySourceBitmap = NULL;
2780 return E_FAIL;
2781 }
2782
2783 DISPLAYFBINFO *pFBInfo = &maFramebuffers[aScreenId];
2784
2785 /* No source bitmap for a blank guest screen. */
2786 if (pFBInfo->flags & VBVA_SCREEN_F_BLANK)
2787 {
2788 aDisplaySourceBitmap = NULL;
2789 return E_FAIL;
2790 }
2791
2792 HRESULT hr = S_OK;
2793
2794 if (pFBInfo->pSourceBitmap.isNull())
2795 {
2796 /* Create a new object. */
2797 ComObjPtr<DisplaySourceBitmap> obj;
2798 hr = obj.createObject();
2799 if (SUCCEEDED(hr))
2800 hr = obj->init(this, aScreenId, pFBInfo);
2801
2802 if (SUCCEEDED(hr))
2803 {
2804 bool fDefaultFormat = !obj->i_usesVRAM();
2805
2806 if (aScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
2807 {
2808 /* Start buffer updates. */
2809 BYTE *pAddress = NULL;
2810 ULONG ulWidth = 0;
2811 ULONG ulHeight = 0;
2812 ULONG ulBitsPerPixel = 0;
2813 ULONG ulBytesPerLine = 0;
2814 BitmapFormat_T bitmapFormat = BitmapFormat_Opaque;
2815
2816 obj->QueryBitmapInfo(&pAddress,
2817 &ulWidth,
2818 &ulHeight,
2819 &ulBitsPerPixel,
2820 &ulBytesPerLine,
2821 &bitmapFormat);
2822
2823 mpDrv->IConnector.pbData = pAddress;
2824 mpDrv->IConnector.cbScanline = ulBytesPerLine;
2825 mpDrv->IConnector.cBits = ulBitsPerPixel;
2826 mpDrv->IConnector.cx = ulWidth;
2827 mpDrv->IConnector.cy = ulHeight;
2828
2829 fSetRenderVRAM = fDefaultFormat;
2830 }
2831
2832 /* Make sure that the bitmap contains the latest image. */
2833 fInvalidate = fDefaultFormat;
2834
2835 pFBInfo->pSourceBitmap = obj;
2836 pFBInfo->fDefaultFormat = fDefaultFormat;
2837 }
2838 }
2839
2840 if (SUCCEEDED(hr))
2841 {
2842 pFBInfo->pSourceBitmap.queryInterfaceTo(aDisplaySourceBitmap.asOutParam());
2843 }
2844
2845 /* Leave the IDisplay lock because the VGA device must not be called under it. */
2846 alock.release();
2847
2848 if (SUCCEEDED(hr))
2849 {
2850 if (fSetRenderVRAM)
2851 {
2852 mpDrv->pUpPort->pfnSetRenderVRAM(mpDrv->pUpPort, true);
2853 }
2854
2855 if (fInvalidate)
2856 VMR3ReqCallWaitU(ptrVM.rawUVM(), VMCPUID_ANY, (PFNRT)Display::i_InvalidateAndUpdateEMT,
2857 3, this, aScreenId, false);
2858 }
2859
2860 LogRelFlowFunc(("%Rhrc\n", hr));
2861 return hr;
2862}
2863
2864HRESULT Display::getGuestScreenLayout(std::vector<ComPtr<IGuestScreenInfo> > &aGuestScreenLayout)
2865{
2866 NOREF(aGuestScreenLayout);
2867 return E_NOTIMPL;
2868}
2869
2870HRESULT Display::setScreenLayout(ScreenLayoutMode_T aScreenLayoutMode,
2871 const std::vector<ComPtr<IGuestScreenInfo> > &aGuestScreenInfo)
2872{
2873 NOREF(aScreenLayoutMode);
2874 NOREF(aGuestScreenInfo);
2875 return E_NOTIMPL;
2876}
2877
2878// wrapped IEventListener method
2879HRESULT Display::handleEvent(const ComPtr<IEvent> &aEvent)
2880{
2881 VBoxEventType_T aType = VBoxEventType_Invalid;
2882
2883 aEvent->COMGETTER(Type)(&aType);
2884 switch (aType)
2885 {
2886 case VBoxEventType_OnStateChanged:
2887 {
2888 ComPtr<IStateChangedEvent> scev = aEvent;
2889 Assert(scev);
2890 MachineState_T machineState;
2891 scev->COMGETTER(State)(&machineState);
2892 if ( machineState == MachineState_Running
2893 || machineState == MachineState_Teleporting
2894 || machineState == MachineState_LiveSnapshotting
2895 || machineState == MachineState_DeletingSnapshotOnline
2896 )
2897 {
2898 LogRelFlowFunc(("Machine is running.\n"));
2899
2900#ifdef VBOX_WITH_CROGL
2901 i_crOglWindowsShow(true);
2902#endif
2903 }
2904 else
2905 {
2906#ifdef VBOX_WITH_CROGL
2907 if (machineState == MachineState_Paused)
2908 i_crOglWindowsShow(false);
2909#endif
2910 }
2911 break;
2912 }
2913 default:
2914 AssertFailed();
2915 }
2916
2917 return S_OK;
2918}
2919
2920
2921// private methods
2922/////////////////////////////////////////////////////////////////////////////
2923
2924#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
2925int Display::i_crViewportNotify(ULONG aScreenId, ULONG x, ULONG y, ULONG width, ULONG height)
2926{
2927 VMMDev *pVMMDev = mParent->i_getVMMDev();
2928 if (!pVMMDev)
2929 return VERR_INVALID_STATE;
2930
2931 size_t cbData = RT_UOFFSETOF(VBOXCRCMDCTL_HGCM, aParms[5]);
2932 VBOXCRCMDCTL_HGCM *pData = (VBOXCRCMDCTL_HGCM *)alloca(cbData);
2933
2934 pData->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
2935 pData->Hdr.u32Function = SHCRGL_HOST_FN_VIEWPORT_CHANGED;
2936
2937 pData->aParms[0].type = VBOX_HGCM_SVC_PARM_32BIT;
2938 pData->aParms[0].u.uint32 = aScreenId;
2939
2940 pData->aParms[1].type = VBOX_HGCM_SVC_PARM_32BIT;
2941 pData->aParms[1].u.uint32 = x;
2942
2943 pData->aParms[2].type = VBOX_HGCM_SVC_PARM_32BIT;
2944 pData->aParms[2].u.uint32 = y;
2945
2946 pData->aParms[3].type = VBOX_HGCM_SVC_PARM_32BIT;
2947 pData->aParms[3].u.uint32 = width;
2948
2949 pData->aParms[4].type = VBOX_HGCM_SVC_PARM_32BIT;
2950 pData->aParms[4].u.uint32 = height;
2951
2952 return i_crCtlSubmitSyncIfHasDataForScreen(aScreenId, &pData->Hdr, (uint32_t)cbData);
2953}
2954#endif
2955
2956#ifdef VBOX_WITH_CRHGSMI
2957void Display::i_setupCrHgsmiData(void)
2958{
2959 VMMDev *pVMMDev = mParent->i_getVMMDev();
2960 Assert(pVMMDev);
2961 int rc = RTCritSectRwEnterExcl(&mCrOglLock);
2962 AssertRC(rc);
2963
2964 if (pVMMDev)
2965 rc = pVMMDev->hgcmHostSvcHandleCreate("VBoxSharedCrOpenGL", &mhCrOglSvc);
2966 else
2967 rc = VERR_GENERAL_FAILURE;
2968
2969 if (RT_SUCCESS(rc))
2970 {
2971 Assert(mhCrOglSvc);
2972 /* setup command completion callback */
2973 VBOXVDMACMD_CHROMIUM_CTL_CRHGSMI_SETUP_MAINCB Completion;
2974 Completion.Hdr.enmType = VBOXVDMACMD_CHROMIUM_CTL_TYPE_CRHGSMI_SETUP_MAINCB;
2975 Completion.Hdr.cbCmd = sizeof(Completion);
2976 Completion.hCompletion = mpDrv->pVBVACallbacks;
2977 Completion.pfnCompletion = mpDrv->pVBVACallbacks->pfnCrHgsmiCommandCompleteAsync;
2978
2979 VBOXHGCMSVCPARM parm;
2980 parm.type = VBOX_HGCM_SVC_PARM_PTR;
2981 parm.u.pointer.addr = &Completion;
2982 parm.u.pointer.size = 0;
2983
2984 rc = pVMMDev->hgcmHostCall("VBoxSharedCrOpenGL", SHCRGL_HOST_FN_CRHGSMI_CTL, 1, &parm);
2985 if (RT_SUCCESS(rc))
2986 mCrOglCallbacks = Completion.MainInterface;
2987 else
2988 AssertMsgFailed(("VBOXVDMACMD_CHROMIUM_CTL_TYPE_CRHGSMI_SETUP_COMPLETION failed (rc=%Rrc)\n", rc));
2989 }
2990
2991 if (RT_FAILURE(rc))
2992 mhCrOglSvc = NULL;
2993
2994 RTCritSectRwLeaveExcl(&mCrOglLock);
2995}
2996
2997void Display::i_destructCrHgsmiData(void)
2998{
2999 int rc = RTCritSectRwEnterExcl(&mCrOglLock);
3000 AssertRC(rc);
3001 mhCrOglSvc = NULL;
3002 RTCritSectRwLeaveExcl(&mCrOglLock);
3003}
3004#endif /* VBOX_WITH_CRHGSMI */
3005
3006/**
3007 * Handle display resize event issued by the VGA device for the primary screen.
3008 *
3009 * @see PDMIDISPLAYCONNECTOR::pfnResize
3010 */
3011DECLCALLBACK(int) Display::i_displayResizeCallback(PPDMIDISPLAYCONNECTOR pInterface,
3012 uint32_t bpp, void *pvVRAM, uint32_t cbLine, uint32_t cx, uint32_t cy)
3013{
3014 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3015 Display *pThis = pDrv->pDisplay;
3016
3017 LogRelFlowFunc(("bpp %d, pvVRAM %p, cbLine %d, cx %d, cy %d\n",
3018 bpp, pvVRAM, cbLine, cx, cy));
3019
3020 bool f = ASMAtomicCmpXchgBool(&pThis->fVGAResizing, true, false);
3021 if (!f)
3022 {
3023 /* This is a result of recursive call when the source bitmap is being updated
3024 * during a VGA resize. Tell the VGA device to ignore the call.
3025 *
3026 * @todo It is a workaround, actually pfnUpdateDisplayAll must
3027 * fail on resize.
3028 */
3029 LogRel(("displayResizeCallback: already processing\n"));
3030 return VINF_VGA_RESIZE_IN_PROGRESS;
3031 }
3032
3033 int rc = pThis->i_handleDisplayResize(VBOX_VIDEO_PRIMARY_SCREEN, bpp, pvVRAM, cbLine, cx, cy, VBVA_SCREEN_F_ACTIVE);
3034
3035 /* Restore the flag. */
3036 f = ASMAtomicCmpXchgBool(&pThis->fVGAResizing, false, true);
3037 AssertRelease(f);
3038
3039 return rc;
3040}
3041
3042/**
3043 * Handle display update.
3044 *
3045 * @see PDMIDISPLAYCONNECTOR::pfnUpdateRect
3046 */
3047DECLCALLBACK(void) Display::i_displayUpdateCallback(PPDMIDISPLAYCONNECTOR pInterface,
3048 uint32_t x, uint32_t y, uint32_t cx, uint32_t cy)
3049{
3050 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3051
3052#ifdef DEBUG_sunlover
3053 LogFlowFunc(("fVideoAccelEnabled = %d, %d,%d %dx%d\n",
3054 pDrv->pDisplay->mVideoAccelLegacy.fVideoAccelEnabled, x, y, cx, cy));
3055#endif /* DEBUG_sunlover */
3056
3057 /* This call does update regardless of VBVA status.
3058 * But in VBVA mode this is called only as result of
3059 * pfnUpdateDisplayAll in the VGA device.
3060 */
3061
3062 pDrv->pDisplay->i_handleDisplayUpdate(VBOX_VIDEO_PRIMARY_SCREEN, x, y, cx, cy);
3063}
3064
3065/**
3066 * Periodic display refresh callback.
3067 *
3068 * @see PDMIDISPLAYCONNECTOR::pfnRefresh
3069 * @thread EMT
3070 */
3071/*static*/ DECLCALLBACK(void) Display::i_displayRefreshCallback(PPDMIDISPLAYCONNECTOR pInterface)
3072{
3073 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3074
3075#ifdef DEBUG_sunlover_2
3076 LogFlowFunc(("pDrv->pDisplay->mfVideoAccelEnabled = %d\n",
3077 pDrv->pDisplay->mfVideoAccelEnabled));
3078#endif /* DEBUG_sunlover_2 */
3079
3080 Display *pDisplay = pDrv->pDisplay;
3081 unsigned uScreenId;
3082
3083 int rc = pDisplay->i_videoAccelRefreshProcess(pDrv->pUpPort);
3084 if (rc != VINF_TRY_AGAIN) /* Means 'do nothing' here. */
3085 {
3086 if (rc == VWRN_INVALID_STATE)
3087 {
3088 /* No VBVA do a display update. */
3089 DISPLAYFBINFO *pFBInfo = &pDisplay->maFramebuffers[VBOX_VIDEO_PRIMARY_SCREEN];
3090 pDrv->pUpPort->pfnUpdateDisplay(pDrv->pUpPort);
3091 }
3092
3093 /* Inform the VRDP server that the current display update sequence is
3094 * completed. At this moment the framebuffer memory contains a definite
3095 * image, that is synchronized with the orders already sent to VRDP client.
3096 * The server can now process redraw requests from clients or initial
3097 * fullscreen updates for new clients.
3098 */
3099 for (uScreenId = 0; uScreenId < pDisplay->mcMonitors; uScreenId++)
3100 {
3101 DISPLAYFBINFO *pFBInfo = &pDisplay->maFramebuffers[uScreenId];
3102
3103 Assert(pDisplay->mParent && pDisplay->mParent->i_consoleVRDPServer());
3104 pDisplay->mParent->i_consoleVRDPServer()->SendUpdate(uScreenId, NULL, 0);
3105 }
3106 }
3107
3108#ifdef VBOX_WITH_VPX
3109 if (VideoRecIsEnabled(pDisplay->mpVideoRecCtx))
3110 {
3111 do {
3112# if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
3113 if (pDisplay->mfIsCr3DEnabled)
3114 {
3115 if (ASMAtomicCmpXchgU32(&pDisplay->mfCrOglVideoRecState, CRVREC_STATE_SUBMITTED, CRVREC_STATE_IDLE))
3116 {
3117 if ( pDisplay->mCrOglCallbacks.pfnHasData
3118 && pDisplay->mCrOglCallbacks.pfnHasData())
3119 {
3120 /* submit */
3121 VBOXCRCMDCTL_HGCM *pData = &pDisplay->mCrOglScreenshotCtl;
3122
3123 pData->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
3124 pData->Hdr.u32Function = SHCRGL_HOST_FN_TAKE_SCREENSHOT;
3125
3126 pData->aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
3127 pData->aParms[0].u.pointer.addr = &pDisplay->mCrOglScreenshotData;
3128 pData->aParms[0].u.pointer.size = sizeof(pDisplay->mCrOglScreenshotData);
3129 rc = pDisplay->i_crCtlSubmit(&pData->Hdr, sizeof(*pData), Display::i_displayVRecCompletion, pDisplay);
3130 if (RT_SUCCESS(rc))
3131 break;
3132 AssertMsgFailed(("crCtlSubmit failed (rc=%Rrc)\n", rc));
3133 }
3134
3135 /* no 3D data available, or error has occured,
3136 * go the straight way */
3137 ASMAtomicWriteU32(&pDisplay->mfCrOglVideoRecState, CRVREC_STATE_IDLE);
3138 }
3139 else
3140 {
3141 /* record request is still in progress, don't do anything */
3142 break;
3143 }
3144 }
3145# endif /* VBOX_WITH_HGCM && VBOX_WITH_CROGL */
3146
3147 uint64_t u64Now = RTTimeProgramMilliTS();
3148 for (uScreenId = 0; uScreenId < pDisplay->mcMonitors; uScreenId++)
3149 {
3150 if (!pDisplay->maVideoRecEnabled[uScreenId])
3151 continue;
3152
3153 if (VideoRecIsFull(pDisplay->mpVideoRecCtx, uScreenId, u64Now))
3154 {
3155 pDisplay->i_VideoCaptureStop();
3156 pDisplay->mParent->i_machine()->COMSETTER(VideoCaptureEnabled)(false);
3157 break;
3158 }
3159
3160 DISPLAYFBINFO *pFBInfo = &pDisplay->maFramebuffers[uScreenId];
3161 if (!pFBInfo->fDisabled)
3162 {
3163 ComPtr<IDisplaySourceBitmap> pSourceBitmap;
3164 int rc2 = RTCritSectEnter(&pDisplay->mVideoCaptureLock);
3165 if (RT_SUCCESS(rc2))
3166 {
3167 pSourceBitmap = pFBInfo->videoCapture.pSourceBitmap;
3168 RTCritSectLeave(&pDisplay->mVideoCaptureLock);
3169 }
3170
3171 if (!pSourceBitmap.isNull())
3172 {
3173 BYTE *pbAddress = NULL;
3174 ULONG ulWidth = 0;
3175 ULONG ulHeight = 0;
3176 ULONG ulBitsPerPixel = 0;
3177 ULONG ulBytesPerLine = 0;
3178 BitmapFormat_T bitmapFormat = BitmapFormat_Opaque;
3179 HRESULT hr = pSourceBitmap->QueryBitmapInfo(&pbAddress,
3180 &ulWidth,
3181 &ulHeight,
3182 &ulBitsPerPixel,
3183 &ulBytesPerLine,
3184 &bitmapFormat);
3185 if (SUCCEEDED(hr) && pbAddress)
3186 rc = VideoRecCopyToIntBuf(pDisplay->mpVideoRecCtx, uScreenId, 0, 0,
3187 BitmapFormat_BGR,
3188 ulBitsPerPixel, ulBytesPerLine, ulWidth, ulHeight,
3189 pbAddress, u64Now);
3190 else
3191 rc = VERR_NOT_SUPPORTED;
3192
3193 pSourceBitmap.setNull();
3194 }
3195 else
3196 rc = VERR_NOT_SUPPORTED;
3197
3198 if (rc == VINF_TRY_AGAIN)
3199 break;
3200 }
3201 }
3202 } while (0);
3203 }
3204#endif /* VBOX_WITH_VPX */
3205
3206#ifdef DEBUG_sunlover_2
3207 LogFlowFunc(("leave\n"));
3208#endif /* DEBUG_sunlover_2 */
3209}
3210
3211/**
3212 * Reset notification
3213 *
3214 * @see PDMIDISPLAYCONNECTOR::pfnReset
3215 */
3216DECLCALLBACK(void) Display::i_displayResetCallback(PPDMIDISPLAYCONNECTOR pInterface)
3217{
3218 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3219
3220 LogRelFlowFunc(("\n"));
3221
3222 /* Disable VBVA mode. */
3223 pDrv->pDisplay->VideoAccelEnableVGA(false, NULL);
3224}
3225
3226/**
3227 * LFBModeChange notification
3228 *
3229 * @see PDMIDISPLAYCONNECTOR::pfnLFBModeChange
3230 */
3231DECLCALLBACK(void) Display::i_displayLFBModeChangeCallback(PPDMIDISPLAYCONNECTOR pInterface, bool fEnabled)
3232{
3233 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3234
3235 LogRelFlowFunc(("fEnabled=%d\n", fEnabled));
3236
3237 NOREF(fEnabled);
3238
3239 /* Disable VBVA mode in any case. The guest driver reenables VBVA mode if necessary. */
3240 pDrv->pDisplay->VideoAccelEnableVGA(false, NULL);
3241}
3242
3243/**
3244 * Adapter information change notification.
3245 *
3246 * @see PDMIDISPLAYCONNECTOR::pfnProcessAdapterData
3247 */
3248DECLCALLBACK(void) Display::i_displayProcessAdapterDataCallback(PPDMIDISPLAYCONNECTOR pInterface, void *pvVRAM,
3249 uint32_t u32VRAMSize)
3250{
3251 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3252 pDrv->pDisplay->processAdapterData(pvVRAM, u32VRAMSize);
3253}
3254
3255/**
3256 * Display information change notification.
3257 *
3258 * @see PDMIDISPLAYCONNECTOR::pfnProcessDisplayData
3259 */
3260DECLCALLBACK(void) Display::i_displayProcessDisplayDataCallback(PPDMIDISPLAYCONNECTOR pInterface,
3261 void *pvVRAM, unsigned uScreenId)
3262{
3263 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3264 pDrv->pDisplay->processDisplayData(pvVRAM, uScreenId);
3265}
3266
3267#ifdef VBOX_WITH_VIDEOHWACCEL
3268
3269#ifndef S_FALSE
3270# define S_FALSE ((HRESULT)1L)
3271#endif
3272
3273int Display::i_handleVHWACommandProcess(PVBOXVHWACMD pCommand)
3274{
3275 unsigned id = (unsigned)pCommand->iDisplay;
3276 int rc = VINF_SUCCESS;
3277 if (id >= mcMonitors)
3278 return VERR_INVALID_PARAMETER;
3279
3280 ComPtr<IFramebuffer> pFramebuffer;
3281 AutoReadLock arlock(this COMMA_LOCKVAL_SRC_POS);
3282 pFramebuffer = maFramebuffers[id].pFramebuffer;
3283 bool fVHWASupported = RT_BOOL(maFramebuffers[id].u32Caps & FramebufferCapabilities_VHWA);
3284 arlock.release();
3285
3286 if (pFramebuffer == NULL || !fVHWASupported)
3287 return VERR_NOT_IMPLEMENTED; /* Implementation is not available. */
3288
3289 HRESULT hr = pFramebuffer->ProcessVHWACommand((BYTE*)pCommand);
3290 if (hr == S_FALSE)
3291 return VINF_SUCCESS;
3292 else if (SUCCEEDED(hr))
3293 return VINF_CALLBACK_RETURN;
3294 else if (hr == E_ACCESSDENIED)
3295 return VERR_INVALID_STATE; /* notify we can not handle request atm */
3296 else if (hr == E_NOTIMPL)
3297 return VERR_NOT_IMPLEMENTED;
3298 return VERR_GENERAL_FAILURE;
3299}
3300
3301DECLCALLBACK(int) Display::i_displayVHWACommandProcess(PPDMIDISPLAYCONNECTOR pInterface, PVBOXVHWACMD pCommand)
3302{
3303 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3304
3305 return pDrv->pDisplay->i_handleVHWACommandProcess(pCommand);
3306}
3307#endif
3308
3309#ifdef VBOX_WITH_CRHGSMI
3310void Display::i_handleCrHgsmiCommandCompletion(int32_t result, uint32_t u32Function, PVBOXHGCMSVCPARM pParam)
3311{
3312 mpDrv->pVBVACallbacks->pfnCrHgsmiCommandCompleteAsync(mpDrv->pVBVACallbacks,
3313 (PVBOXVDMACMD_CHROMIUM_CMD)pParam->u.pointer.addr, result);
3314}
3315
3316void Display::i_handleCrHgsmiControlCompletion(int32_t result, uint32_t u32Function, PVBOXHGCMSVCPARM pParam)
3317{
3318 PVBOXVDMACMD_CHROMIUM_CTL pCtl = (PVBOXVDMACMD_CHROMIUM_CTL)pParam->u.pointer.addr;
3319 mpDrv->pVBVACallbacks->pfnCrHgsmiControlCompleteAsync(mpDrv->pVBVACallbacks, pCtl, result);
3320}
3321
3322void Display::i_handleCrHgsmiCommandProcess(PVBOXVDMACMD_CHROMIUM_CMD pCmd, uint32_t cbCmd)
3323{
3324 int rc = VERR_NOT_SUPPORTED;
3325 VBOXHGCMSVCPARM parm;
3326 parm.type = VBOX_HGCM_SVC_PARM_PTR;
3327 parm.u.pointer.addr = pCmd;
3328 parm.u.pointer.size = cbCmd;
3329
3330 if (mhCrOglSvc)
3331 {
3332 VMMDev *pVMMDev = mParent->i_getVMMDev();
3333 if (pVMMDev)
3334 {
3335 /* no completion callback is specified with this call,
3336 * the CrOgl code will complete the CrHgsmi command once it processes it */
3337 rc = pVMMDev->hgcmHostFastCallAsync(mhCrOglSvc, SHCRGL_HOST_FN_CRHGSMI_CMD, &parm, NULL, NULL);
3338 AssertRC(rc);
3339 if (RT_SUCCESS(rc))
3340 return;
3341 }
3342 else
3343 rc = VERR_INVALID_STATE;
3344 }
3345
3346 /* we are here because something went wrong with command processing, complete it */
3347 i_handleCrHgsmiCommandCompletion(rc, SHCRGL_HOST_FN_CRHGSMI_CMD, &parm);
3348}
3349
3350void Display::i_handleCrHgsmiControlProcess(PVBOXVDMACMD_CHROMIUM_CTL pCtl, uint32_t cbCtl)
3351{
3352 int rc = VERR_NOT_SUPPORTED;
3353 VBOXHGCMSVCPARM parm;
3354 parm.type = VBOX_HGCM_SVC_PARM_PTR;
3355 parm.u.pointer.addr = pCtl;
3356 parm.u.pointer.size = cbCtl;
3357
3358 if (mhCrOglSvc)
3359 {
3360 VMMDev *pVMMDev = mParent->i_getVMMDev();
3361 if (pVMMDev)
3362 {
3363 bool fCheckPendingViewport = (pCtl->enmType == VBOXVDMACMD_CHROMIUM_CTL_TYPE_CRHGSMI_SETUP);
3364 rc = pVMMDev->hgcmHostFastCallAsync(mhCrOglSvc, SHCRGL_HOST_FN_CRHGSMI_CTL, &parm,
3365 Display::i_displayCrHgsmiControlCompletion, this);
3366 AssertRC(rc);
3367 if (RT_SUCCESS(rc))
3368 {
3369 if (fCheckPendingViewport)
3370 {
3371 ULONG ul;
3372 for (ul = 0; ul < mcMonitors; ul++)
3373 {
3374 DISPLAYFBINFO *pFb = &maFramebuffers[ul];
3375 if (!pFb->pendingViewportInfo.fPending)
3376 continue;
3377
3378 rc = i_crViewportNotify(ul, pFb->pendingViewportInfo.x, pFb->pendingViewportInfo.y,
3379 pFb->pendingViewportInfo.width, pFb->pendingViewportInfo.height);
3380 if (RT_SUCCESS(rc))
3381 pFb->pendingViewportInfo.fPending = false;
3382 else
3383 {
3384 AssertMsgFailed(("crViewportNotify failed (rc=%Rrc)\n", rc));
3385 rc = VINF_SUCCESS;
3386 }
3387 }
3388 }
3389 return;
3390 }
3391 }
3392 else
3393 rc = VERR_INVALID_STATE;
3394 }
3395
3396 /* we are here because something went wrong with command processing, complete it */
3397 i_handleCrHgsmiControlCompletion(rc, SHCRGL_HOST_FN_CRHGSMI_CTL, &parm);
3398}
3399
3400DECLCALLBACK(void) Display::i_displayCrHgsmiCommandProcess(PPDMIDISPLAYCONNECTOR pInterface, PVBOXVDMACMD_CHROMIUM_CMD pCmd,
3401 uint32_t cbCmd)
3402{
3403 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3404
3405 pDrv->pDisplay->i_handleCrHgsmiCommandProcess(pCmd, cbCmd);
3406}
3407
3408DECLCALLBACK(void) Display::i_displayCrHgsmiControlProcess(PPDMIDISPLAYCONNECTOR pInterface, PVBOXVDMACMD_CHROMIUM_CTL pCmd,
3409 uint32_t cbCmd)
3410{
3411 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3412
3413 pDrv->pDisplay->i_handleCrHgsmiControlProcess(pCmd, cbCmd);
3414}
3415
3416DECLCALLBACK(void) Display::i_displayCrHgsmiCommandCompletion(int32_t result, uint32_t u32Function, PVBOXHGCMSVCPARM pParam,
3417 void *pvContext)
3418{
3419 AssertMsgFailed(("not expected!\n"));
3420 Display *pDisplay = (Display *)pvContext;
3421 pDisplay->i_handleCrHgsmiCommandCompletion(result, u32Function, pParam);
3422}
3423
3424DECLCALLBACK(void) Display::i_displayCrHgsmiControlCompletion(int32_t result, uint32_t u32Function, PVBOXHGCMSVCPARM pParam,
3425 void *pvContext)
3426{
3427 Display *pDisplay = (Display *)pvContext;
3428 pDisplay->i_handleCrHgsmiControlCompletion(result, u32Function, pParam);
3429
3430}
3431#endif
3432
3433#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
3434DECLCALLBACK(void) Display::i_displayCrHgcmCtlSubmitCompletion(int32_t result, uint32_t u32Function, PVBOXHGCMSVCPARM pParam,
3435 void *pvContext)
3436{
3437 VBOXCRCMDCTL *pCmd = (VBOXCRCMDCTL*)pParam->u.pointer.addr;
3438 if (pCmd->u.pfnInternal)
3439 ((PFNCRCTLCOMPLETION)pCmd->u.pfnInternal)(pCmd, pParam->u.pointer.size, result, pvContext);
3440}
3441
3442int Display::i_handleCrHgcmCtlSubmit(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd,
3443 PFNCRCTLCOMPLETION pfnCompletion,
3444 void *pvCompletion)
3445{
3446 VMMDev *pVMMDev = mParent ? mParent->i_getVMMDev() : NULL;
3447 if (!pVMMDev)
3448 {
3449 AssertMsgFailed(("no vmmdev\n"));
3450 return VERR_INVALID_STATE;
3451 }
3452
3453 Assert(mhCrOglSvc);
3454 VBOXHGCMSVCPARM parm;
3455 parm.type = VBOX_HGCM_SVC_PARM_PTR;
3456 parm.u.pointer.addr = pCmd;
3457 parm.u.pointer.size = cbCmd;
3458
3459 pCmd->u.pfnInternal = (void(*)())pfnCompletion;
3460 int rc = pVMMDev->hgcmHostFastCallAsync(mhCrOglSvc, SHCRGL_HOST_FN_CTL, &parm, i_displayCrHgcmCtlSubmitCompletion,
3461 pvCompletion);
3462 if (!RT_SUCCESS(rc))
3463 AssertMsgFailed(("hgcmHostFastCallAsync failed (rc=%Rrc)\n", rc));
3464
3465 return rc;
3466}
3467
3468DECLCALLBACK(int) Display::i_displayCrHgcmCtlSubmit(PPDMIDISPLAYCONNECTOR pInterface,
3469 struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd,
3470 PFNCRCTLCOMPLETION pfnCompletion,
3471 void *pvCompletion)
3472{
3473 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3474 Display *pThis = pDrv->pDisplay;
3475 return pThis->i_handleCrHgcmCtlSubmit(pCmd, cbCmd, pfnCompletion, pvCompletion);
3476}
3477
3478int Display::i_crCtlSubmit(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd, PFNCRCTLCOMPLETION pfnCompletion, void *pvCompletion)
3479{
3480 int rc = RTCritSectRwEnterShared(&mCrOglLock);
3481 if (RT_SUCCESS(rc))
3482 {
3483 if (mhCrOglSvc)
3484 rc = mpDrv->pVBVACallbacks->pfnCrCtlSubmit(mpDrv->pVBVACallbacks, pCmd, cbCmd, pfnCompletion, pvCompletion);
3485 else
3486 rc = VERR_NOT_SUPPORTED;
3487
3488 RTCritSectRwLeaveShared(&mCrOglLock);
3489 }
3490 return rc;
3491}
3492
3493int Display::i_crCtlSubmitSync(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd)
3494{
3495 int rc = RTCritSectRwEnterShared(&mCrOglLock);
3496 if (RT_SUCCESS(rc))
3497 {
3498 if (mhCrOglSvc)
3499 rc = mpDrv->pVBVACallbacks->pfnCrCtlSubmitSync(mpDrv->pVBVACallbacks, pCmd, cbCmd);
3500 else
3501 rc = VERR_NOT_SUPPORTED;
3502
3503 RTCritSectRwLeaveShared(&mCrOglLock);
3504 }
3505 return rc;
3506}
3507
3508int Display::i_crCtlSubmitAsyncCmdCopy(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd)
3509{
3510 VBOXCRCMDCTL* pCmdCopy = (VBOXCRCMDCTL*)RTMemAlloc(cbCmd);
3511 if (!pCmdCopy)
3512 {
3513 LogRel(("RTMemAlloc failed\n"));
3514 return VERR_NO_MEMORY;
3515 }
3516
3517 memcpy(pCmdCopy, pCmd, cbCmd);
3518
3519 int rc = i_crCtlSubmit(pCmdCopy, cbCmd, i_displayCrCmdFree, pCmdCopy);
3520 if (RT_FAILURE(rc))
3521 {
3522 LogRel(("crCtlSubmit failed (rc=%Rrc)\n", rc));
3523 RTMemFree(pCmdCopy);
3524 return rc;
3525 }
3526
3527 return VINF_SUCCESS;
3528}
3529
3530int Display::i_crCtlSubmitSyncIfHasDataForScreen(uint32_t u32ScreenID, struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd)
3531{
3532 int rc = RTCritSectRwEnterShared(&mCrOglLock);
3533 AssertRCReturn(rc, rc);
3534
3535 if ( mCrOglCallbacks.pfnHasDataForScreen
3536 && mCrOglCallbacks.pfnHasDataForScreen(u32ScreenID))
3537 rc = i_crCtlSubmitSync(pCmd, cbCmd);
3538 else
3539 rc = i_crCtlSubmitAsyncCmdCopy(pCmd, cbCmd);
3540
3541 RTCritSectRwLeaveShared(&mCrOglLock);
3542
3543 return rc;
3544}
3545
3546bool Display::i_handleCrVRecScreenshotBegin(uint32_t uScreen, uint64_t u64TimeStamp)
3547{
3548# if VBOX_WITH_VPX
3549 return VideoRecIsReady(mpVideoRecCtx, uScreen, u64TimeStamp);
3550# else
3551 return false;
3552# endif
3553}
3554
3555void Display::i_handleCrVRecScreenshotEnd(uint32_t uScreen, uint64_t u64TimeStamp)
3556{
3557}
3558
3559void Display::i_handleCrVRecScreenshotPerform(uint32_t uScreen,
3560 uint32_t x, uint32_t y, uint32_t uPixelFormat,
3561 uint32_t uBitsPerPixel, uint32_t uBytesPerLine,
3562 uint32_t uGuestWidth, uint32_t uGuestHeight,
3563 uint8_t *pu8BufferAddress, uint64_t u64TimeStamp)
3564{
3565 Assert(mfCrOglVideoRecState == CRVREC_STATE_SUBMITTED);
3566# if VBOX_WITH_VPX
3567 int rc = VideoRecCopyToIntBuf(mpVideoRecCtx, uScreen, x, y,
3568 uPixelFormat,
3569 uBitsPerPixel, uBytesPerLine,
3570 uGuestWidth, uGuestHeight,
3571 pu8BufferAddress, u64TimeStamp);
3572 Assert(rc == VINF_SUCCESS /* || rc == VERR_TRY_AGAIN || rc == VINF_TRY_AGAIN*/);
3573# endif
3574}
3575
3576void Display::i_handleVRecCompletion()
3577{
3578 Assert(mfCrOglVideoRecState == CRVREC_STATE_SUBMITTED);
3579 ASMAtomicWriteU32(&mfCrOglVideoRecState, CRVREC_STATE_IDLE);
3580}
3581
3582#endif /* VBOX_WITH_HGCM && VBOX_WITH_CROGL */
3583
3584HRESULT Display::notifyScaleFactorChange(ULONG aScreenId, ULONG aScaleFactorWMultiplied, ULONG aScaleFactorHMultiplied)
3585{
3586#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
3587 HRESULT hr = E_UNEXPECTED;
3588
3589 if (aScreenId >= mcMonitors)
3590 return E_INVALIDARG;
3591
3592 /* 3D acceleration enabled in VM config. */
3593 if (mfIsCr3DEnabled)
3594 {
3595 /* VBoxSharedCrOpenGL HGCM host service is running. */
3596 if (mhCrOglSvc)
3597 {
3598 VMMDev *pVMMDev = mParent->i_getVMMDev();
3599 if (pVMMDev)
3600 {
3601 VBOXCRCMDCTL_HGCM *pCtl;
3602 pCtl = (VBOXCRCMDCTL_HGCM *)RTMemAlloc(sizeof(CRVBOXHGCMSETSCALEFACTOR) + sizeof(VBOXCRCMDCTL_HGCM));
3603 if (pCtl)
3604 {
3605 CRVBOXHGCMSETSCALEFACTOR *pData = (CRVBOXHGCMSETSCALEFACTOR *)(pCtl + 1);
3606 int rc;
3607
3608 pData->u32Screen = aScreenId;
3609 pData->u32ScaleFactorWMultiplied = aScaleFactorWMultiplied;
3610 pData->u32ScaleFactorHMultiplied = aScaleFactorHMultiplied;
3611
3612 pCtl->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
3613 pCtl->Hdr.u32Function = SHCRGL_HOST_FN_SET_SCALE_FACTOR;
3614 pCtl->aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
3615 pCtl->aParms[0].u.pointer.addr = pData;
3616 pCtl->aParms[0].u.pointer.size = sizeof(*pData);
3617
3618 rc = i_crCtlSubmitSync(&pCtl->Hdr, sizeof(*pCtl));
3619 if (RT_FAILURE(rc))
3620 AssertMsgFailed(("crCtlSubmitSync failed (rc=%Rrc)\n", rc));
3621 else
3622 hr = S_OK;
3623
3624 RTMemFree(pCtl);
3625 }
3626 else
3627 {
3628 LogRel(("Running out of memory on attempt to set OpenGL content scale factor. Ignored.\n"));
3629 hr = E_OUTOFMEMORY;
3630 }
3631 }
3632 else
3633 LogRel(("Internal error occurred on attempt to set OpenGL content scale factor. Ignored.\n"));
3634 }
3635 else
3636 LogRel(("Attempt to specify OpenGL content scale factor while corresponding HGCM host service not yet runing. Ignored.\n"));
3637 }
3638 else
3639# if 0 /** @todo Thank you so very much from anyone using VMSVGA3d! */
3640 AssertMsgFailed(("Attempt to specify OpenGL content scale factor while 3D acceleration is disabled in VM config. Ignored.\n"));
3641# else
3642 {
3643 hr = S_OK;
3644 /* Need an interface like this here (and the #ifdefs needs adjusting):
3645 PPDMIDISPLAYPORT pUpPort = mpDrv ? mpDrv->pUpPort : NULL;
3646 if (pUpPort && pUpPort->pfnSetScaleFactor)
3647 pUpPort->pfnSetScaleFactor(pUpPort, aScreeId, aScaleFactorWMultiplied, aScaleFactorHMultiplied); */
3648 }
3649# endif
3650
3651 return hr;
3652#else
3653 AssertMsgFailed(("Attempt to specify OpenGL content scale factor while corresponding functionality is disabled."));
3654 return E_UNEXPECTED;
3655#endif /* VBOX_WITH_HGCM && VBOX_WITH_CROGL */
3656}
3657
3658HRESULT Display::notifyHiDPIOutputPolicyChange(BOOL fUnscaledHiDPI)
3659{
3660#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
3661 HRESULT hr = E_UNEXPECTED;
3662
3663 /* 3D acceleration enabled in VM config. */
3664 if (mfIsCr3DEnabled)
3665 {
3666 /* VBoxSharedCrOpenGL HGCM host service is running. */
3667 if (mhCrOglSvc)
3668 {
3669 VMMDev *pVMMDev = mParent->i_getVMMDev();
3670 if (pVMMDev)
3671 {
3672 VBOXCRCMDCTL_HGCM *pCtl;
3673 pCtl = (VBOXCRCMDCTL_HGCM *)RTMemAlloc(sizeof(CRVBOXHGCMSETUNSCALEDHIDPIOUTPUT) + sizeof(VBOXCRCMDCTL_HGCM));
3674 if (pCtl)
3675 {
3676 CRVBOXHGCMSETUNSCALEDHIDPIOUTPUT *pData = (CRVBOXHGCMSETUNSCALEDHIDPIOUTPUT *)(pCtl + 1);
3677 int rc;
3678
3679 pData->fUnscaledHiDPI = RT_BOOL(fUnscaledHiDPI);
3680
3681 pCtl->Hdr.enmType = VBOXCRCMDCTL_TYPE_HGCM;
3682 pCtl->Hdr.u32Function = SHCRGL_HOST_FN_SET_UNSCALED_HIDPI;
3683 pCtl->aParms[0].type = VBOX_HGCM_SVC_PARM_PTR;
3684 pCtl->aParms[0].u.pointer.addr = pData;
3685 pCtl->aParms[0].u.pointer.size = sizeof(*pData);
3686
3687 rc = i_crCtlSubmitSync(&pCtl->Hdr, sizeof(*pCtl));
3688 if (RT_FAILURE(rc))
3689 AssertMsgFailed(("crCtlSubmitSync failed (rc=%Rrc)\n", rc));
3690 else
3691 hr = S_OK;
3692
3693 RTMemFree(pCtl);
3694 }
3695 else
3696 {
3697 LogRel(("Running out of memory on attempt to notify OpenGL about HiDPI output scaling policy change. Ignored.\n"));
3698 hr = E_OUTOFMEMORY;
3699 }
3700 }
3701 else
3702 LogRel(("Internal error occurred on attempt to notify OpenGL about HiDPI output scaling policy change. Ignored.\n"));
3703 }
3704 else
3705 LogRel(("Attempt to notify OpenGL about HiDPI output scaling policy change while corresponding HGCM host service not yet runing. Ignored.\n"));
3706 }
3707 else
3708 {
3709 hr = S_OK;
3710 /* Need an interface like this here (and the #ifdefs needs adjusting):
3711 PPDMIDISPLAYPORT pUpPort = mpDrv ? mpDrv->pUpPort : NULL;
3712 if (pUpPort && pUpPort->pfnSetScaleFactor)
3713 pUpPort->pfnSetScaleFactor(pUpPort, aScreeId, aScaleFactorWMultiplied, aScaleFactorHMultiplied); */
3714 }
3715
3716 return hr;
3717#else
3718 AssertMsgFailed(("Attempt to notify OpenGL about HiDPI output scaling policy change while corresponding functionality is disabled."));
3719 return E_UNEXPECTED;
3720#endif /* VBOX_WITH_HGCM && VBOX_WITH_CROGL */
3721}
3722
3723#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
3724DECLCALLBACK(void) Display::i_displayCrVRecScreenshotPerform(void *pvCtx, uint32_t uScreen,
3725 uint32_t x, uint32_t y,
3726 uint32_t uBitsPerPixel, uint32_t uBytesPerLine,
3727 uint32_t uGuestWidth, uint32_t uGuestHeight,
3728 uint8_t *pu8BufferAddress, uint64_t u64TimeStamp)
3729{
3730 Display *pDisplay = (Display *)pvCtx;
3731 pDisplay->i_handleCrVRecScreenshotPerform(uScreen,
3732 x, y, BitmapFormat_BGR, uBitsPerPixel,
3733 uBytesPerLine, uGuestWidth, uGuestHeight,
3734 pu8BufferAddress, u64TimeStamp);
3735}
3736
3737DECLCALLBACK(bool) Display::i_displayCrVRecScreenshotBegin(void *pvCtx, uint32_t uScreen, uint64_t u64TimeStamp)
3738{
3739 Display *pDisplay = (Display *)pvCtx;
3740 return pDisplay->i_handleCrVRecScreenshotBegin(uScreen, u64TimeStamp);
3741}
3742
3743DECLCALLBACK(void) Display::i_displayCrVRecScreenshotEnd(void *pvCtx, uint32_t uScreen, uint64_t u64TimeStamp)
3744{
3745 Display *pDisplay = (Display *)pvCtx;
3746 pDisplay->i_handleCrVRecScreenshotEnd(uScreen, u64TimeStamp);
3747}
3748
3749DECLCALLBACK(void) Display::i_displayVRecCompletion(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd, int rc, void *pvCompletion)
3750{
3751 Display *pDisplay = (Display *)pvCompletion;
3752 pDisplay->i_handleVRecCompletion();
3753}
3754
3755#endif
3756
3757
3758#ifdef VBOX_WITH_HGSMI
3759DECLCALLBACK(int) Display::i_displayVBVAEnable(PPDMIDISPLAYCONNECTOR pInterface, unsigned uScreenId, PVBVAHOSTFLAGS pHostFlags,
3760 bool fRenderThreadMode)
3761{
3762 LogRelFlowFunc(("uScreenId %d\n", uScreenId));
3763
3764 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3765 Display *pThis = pDrv->pDisplay;
3766
3767 if (pThis->maFramebuffers[uScreenId].fVBVAEnabled && pThis->maFramebuffers[uScreenId].fRenderThreadMode != fRenderThreadMode)
3768 {
3769 LogRel(("Enabling different vbva mode\n"));
3770#ifdef DEBUG_misha
3771 AssertMsgFailed(("enabling different vbva mode\n"));
3772#endif
3773 return VERR_INVALID_STATE;
3774 }
3775
3776 pThis->maFramebuffers[uScreenId].fVBVAEnabled = true;
3777 pThis->maFramebuffers[uScreenId].pVBVAHostFlags = pHostFlags;
3778 pThis->maFramebuffers[uScreenId].fRenderThreadMode = fRenderThreadMode;
3779 pThis->maFramebuffers[uScreenId].fVBVAForceResize = true;
3780
3781 vbvaSetMemoryFlagsHGSMI(uScreenId, pThis->mfu32SupportedOrders, pThis->mfVideoAccelVRDP, &pThis->maFramebuffers[uScreenId]);
3782
3783 return VINF_SUCCESS;
3784}
3785
3786DECLCALLBACK(void) Display::i_displayVBVADisable(PPDMIDISPLAYCONNECTOR pInterface, unsigned uScreenId)
3787{
3788 LogRelFlowFunc(("uScreenId %d\n", uScreenId));
3789
3790 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3791 Display *pThis = pDrv->pDisplay;
3792
3793 DISPLAYFBINFO *pFBInfo = &pThis->maFramebuffers[uScreenId];
3794
3795 bool fRenderThreadMode = pFBInfo->fRenderThreadMode;
3796
3797 if (uScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
3798 {
3799 /* Make sure that the primary screen is visible now.
3800 * The guest can't use VBVA anymore, so only only the VGA device output works.
3801 */
3802 if (pFBInfo->fDisabled)
3803 {
3804 pFBInfo->fDisabled = false;
3805 fireGuestMonitorChangedEvent(pThis->mParent->i_getEventSource(),
3806 GuestMonitorChangedEventType_Enabled,
3807 uScreenId,
3808 pFBInfo->xOrigin, pFBInfo->yOrigin,
3809 pFBInfo->w, pFBInfo->h);
3810 }
3811 }
3812
3813 pFBInfo->fVBVAEnabled = false;
3814 pFBInfo->fVBVAForceResize = false;
3815 pFBInfo->fRenderThreadMode = false;
3816
3817 vbvaSetMemoryFlagsHGSMI(uScreenId, 0, false, pFBInfo);
3818
3819 pFBInfo->pVBVAHostFlags = NULL;
3820
3821 if (!fRenderThreadMode && uScreenId == VBOX_VIDEO_PRIMARY_SCREEN)
3822 {
3823 /* Force full screen update, because VGA device must take control, do resize, etc. */
3824 pThis->mpDrv->pUpPort->pfnUpdateDisplayAll(pThis->mpDrv->pUpPort, /* fFailOnResize = */ false);
3825 }
3826}
3827
3828DECLCALLBACK(void) Display::i_displayVBVAUpdateBegin(PPDMIDISPLAYCONNECTOR pInterface, unsigned uScreenId)
3829{
3830 LogFlowFunc(("uScreenId %d\n", uScreenId));
3831
3832 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3833 Display *pThis = pDrv->pDisplay;
3834 DISPLAYFBINFO *pFBInfo = &pThis->maFramebuffers[uScreenId];
3835
3836 if (ASMAtomicReadU32(&pThis->mu32UpdateVBVAFlags) > 0)
3837 {
3838 vbvaSetMemoryFlagsAllHGSMI(pThis->mfu32SupportedOrders, pThis->mfVideoAccelVRDP, pThis->maFramebuffers,
3839 pThis->mcMonitors);
3840 ASMAtomicDecU32(&pThis->mu32UpdateVBVAFlags);
3841 }
3842}
3843
3844DECLCALLBACK(void) Display::i_displayVBVAUpdateProcess(PPDMIDISPLAYCONNECTOR pInterface, unsigned uScreenId,
3845 const PVBVACMDHDR pCmd, size_t cbCmd)
3846{
3847 LogFlowFunc(("uScreenId %d pCmd %p cbCmd %d, @%d,%d %dx%d\n", uScreenId, pCmd, cbCmd, pCmd->x, pCmd->y, pCmd->w, pCmd->h));
3848
3849 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3850 Display *pThis = pDrv->pDisplay;
3851 DISPLAYFBINFO *pFBInfo = &pThis->maFramebuffers[uScreenId];
3852
3853 if (pFBInfo->fDefaultFormat)
3854 {
3855 /* Make sure that framebuffer contains the same image as the guest VRAM. */
3856 if ( uScreenId == VBOX_VIDEO_PRIMARY_SCREEN
3857 && !pFBInfo->fDisabled)
3858 {
3859 pDrv->pUpPort->pfnUpdateDisplayRect(pDrv->pUpPort, pCmd->x, pCmd->y, pCmd->w, pCmd->h);
3860 }
3861 else if ( !pFBInfo->pSourceBitmap.isNull()
3862 && !pFBInfo->fDisabled)
3863 {
3864 /* Render VRAM content to the framebuffer. */
3865 BYTE *pAddress = NULL;
3866 ULONG ulWidth = 0;
3867 ULONG ulHeight = 0;
3868 ULONG ulBitsPerPixel = 0;
3869 ULONG ulBytesPerLine = 0;
3870 BitmapFormat_T bitmapFormat = BitmapFormat_Opaque;
3871
3872 HRESULT hrc = pFBInfo->pSourceBitmap->QueryBitmapInfo(&pAddress,
3873 &ulWidth,
3874 &ulHeight,
3875 &ulBitsPerPixel,
3876 &ulBytesPerLine,
3877 &bitmapFormat);
3878 if (SUCCEEDED(hrc))
3879 {
3880 uint32_t width = pCmd->w;
3881 uint32_t height = pCmd->h;
3882
3883 const uint8_t *pu8Src = pFBInfo->pu8FramebufferVRAM;
3884 int32_t xSrc = pCmd->x - pFBInfo->xOrigin;
3885 int32_t ySrc = pCmd->y - pFBInfo->yOrigin;
3886 uint32_t u32SrcWidth = pFBInfo->w;
3887 uint32_t u32SrcHeight = pFBInfo->h;
3888 uint32_t u32SrcLineSize = pFBInfo->u32LineSize;
3889 uint32_t u32SrcBitsPerPixel = pFBInfo->u16BitsPerPixel;
3890
3891 uint8_t *pu8Dst = pAddress;
3892 int32_t xDst = xSrc;
3893 int32_t yDst = ySrc;
3894 uint32_t u32DstWidth = u32SrcWidth;
3895 uint32_t u32DstHeight = u32SrcHeight;
3896 uint32_t u32DstLineSize = u32DstWidth * 4;
3897 uint32_t u32DstBitsPerPixel = 32;
3898
3899 pDrv->pUpPort->pfnCopyRect(pDrv->pUpPort,
3900 width, height,
3901 pu8Src,
3902 xSrc, ySrc,
3903 u32SrcWidth, u32SrcHeight,
3904 u32SrcLineSize, u32SrcBitsPerPixel,
3905 pu8Dst,
3906 xDst, yDst,
3907 u32DstWidth, u32DstHeight,
3908 u32DstLineSize, u32DstBitsPerPixel);
3909 }
3910 }
3911 }
3912
3913 VBVACMDHDR hdrSaved = *pCmd;
3914
3915 VBVACMDHDR *pHdrUnconst = (VBVACMDHDR *)pCmd;
3916
3917 pHdrUnconst->x -= (int16_t)pFBInfo->xOrigin;
3918 pHdrUnconst->y -= (int16_t)pFBInfo->yOrigin;
3919
3920 /* @todo new SendUpdate entry which can get a separate cmd header or coords. */
3921 pThis->mParent->i_consoleVRDPServer()->SendUpdate(uScreenId, pCmd, (uint32_t)cbCmd);
3922
3923 *pHdrUnconst = hdrSaved;
3924}
3925
3926DECLCALLBACK(void) Display::i_displayVBVAUpdateEnd(PPDMIDISPLAYCONNECTOR pInterface, unsigned uScreenId, int32_t x, int32_t y,
3927 uint32_t cx, uint32_t cy)
3928{
3929 LogFlowFunc(("uScreenId %d %d,%d %dx%d\n", uScreenId, x, y, cx, cy));
3930
3931 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
3932 Display *pThis = pDrv->pDisplay;
3933 DISPLAYFBINFO *pFBInfo = &pThis->maFramebuffers[uScreenId];
3934
3935 /* @todo handleFramebufferUpdate (uScreenId,
3936 * x - pThis->maFramebuffers[uScreenId].xOrigin,
3937 * y - pThis->maFramebuffers[uScreenId].yOrigin,
3938 * cx, cy);
3939 */
3940 pThis->i_handleDisplayUpdate(uScreenId, x - pFBInfo->xOrigin, y - pFBInfo->yOrigin, cx, cy);
3941}
3942
3943#ifdef DEBUG_sunlover
3944static void logVBVAResize(const PVBVAINFOVIEW pView, const PVBVAINFOSCREEN pScreen, const DISPLAYFBINFO *pFBInfo)
3945{
3946 LogRel(("displayVBVAResize: [%d] %s\n"
3947 " pView->u32ViewIndex %d\n"
3948 " pView->u32ViewOffset 0x%08X\n"
3949 " pView->u32ViewSize 0x%08X\n"
3950 " pView->u32MaxScreenSize 0x%08X\n"
3951 " pScreen->i32OriginX %d\n"
3952 " pScreen->i32OriginY %d\n"
3953 " pScreen->u32StartOffset 0x%08X\n"
3954 " pScreen->u32LineSize 0x%08X\n"
3955 " pScreen->u32Width %d\n"
3956 " pScreen->u32Height %d\n"
3957 " pScreen->u16BitsPerPixel %d\n"
3958 " pScreen->u16Flags 0x%04X\n"
3959 " pFBInfo->u32Offset 0x%08X\n"
3960 " pFBInfo->u32MaxFramebufferSize 0x%08X\n"
3961 " pFBInfo->u32InformationSize 0x%08X\n"
3962 " pFBInfo->fDisabled %d\n"
3963 " xOrigin, yOrigin, w, h: %d,%d %dx%d\n"
3964 " pFBInfo->u16BitsPerPixel %d\n"
3965 " pFBInfo->pu8FramebufferVRAM %p\n"
3966 " pFBInfo->u32LineSize 0x%08X\n"
3967 " pFBInfo->flags 0x%04X\n"
3968 " pFBInfo->pHostEvents %p\n"
3969 " pFBInfo->fDefaultFormat %d\n"
3970 " pFBInfo->fVBVAEnabled %d\n"
3971 " pFBInfo->fVBVAForceResize %d\n"
3972 " pFBInfo->pVBVAHostFlags %p\n"
3973 "",
3974 pScreen->u32ViewIndex,
3975 (pScreen->u16Flags & VBVA_SCREEN_F_DISABLED)? "DISABLED": "ENABLED",
3976 pView->u32ViewIndex,
3977 pView->u32ViewOffset,
3978 pView->u32ViewSize,
3979 pView->u32MaxScreenSize,
3980 pScreen->i32OriginX,
3981 pScreen->i32OriginY,
3982 pScreen->u32StartOffset,
3983 pScreen->u32LineSize,
3984 pScreen->u32Width,
3985 pScreen->u32Height,
3986 pScreen->u16BitsPerPixel,
3987 pScreen->u16Flags,
3988 pFBInfo->u32Offset,
3989 pFBInfo->u32MaxFramebufferSize,
3990 pFBInfo->u32InformationSize,
3991 pFBInfo->fDisabled,
3992 pFBInfo->xOrigin,
3993 pFBInfo->yOrigin,
3994 pFBInfo->w,
3995 pFBInfo->h,
3996 pFBInfo->u16BitsPerPixel,
3997 pFBInfo->pu8FramebufferVRAM,
3998 pFBInfo->u32LineSize,
3999 pFBInfo->flags,
4000 pFBInfo->pHostEvents,
4001 pFBInfo->fDefaultFormat,
4002 pFBInfo->fVBVAEnabled,
4003 pFBInfo->fVBVAForceResize,
4004 pFBInfo->pVBVAHostFlags
4005 ));
4006}
4007#endif /* DEBUG_sunlover */
4008
4009DECLCALLBACK(int) Display::i_displayVBVAResize(PPDMIDISPLAYCONNECTOR pInterface, const PVBVAINFOVIEW pView,
4010 const PVBVAINFOSCREEN pScreen, void *pvVRAM, bool fResetInputMapping)
4011{
4012 LogRelFlowFunc(("pScreen %p, pvVRAM %p\n", pScreen, pvVRAM));
4013
4014 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
4015 Display *pThis = pDrv->pDisplay;
4016
4017 DISPLAYFBINFO *pFBInfo = &pThis->maFramebuffers[pScreen->u32ViewIndex];
4018
4019 if (pScreen->u16Flags & VBVA_SCREEN_F_DISABLED)
4020 {
4021 pThis->i_notifyCroglResize(pView, pScreen, pvVRAM);
4022
4023 pFBInfo->fDisabled = true;
4024 pFBInfo->flags = pScreen->u16Flags;
4025
4026 /* Ask the framebuffer to resize using a default format. The framebuffer will be black.
4027 * So if the frontend does not support GuestMonitorChangedEventType_Disabled event,
4028 * the VM window will be black. */
4029 uint32_t u32Width = pFBInfo->w ? pFBInfo->w : 640;
4030 uint32_t u32Height = pFBInfo->h ? pFBInfo->h : 480;
4031 pThis->i_handleDisplayResize(pScreen->u32ViewIndex, 0, (uint8_t *)NULL, 0,
4032 u32Width, u32Height, pScreen->u16Flags);
4033
4034 fireGuestMonitorChangedEvent(pThis->mParent->i_getEventSource(),
4035 GuestMonitorChangedEventType_Disabled,
4036 pScreen->u32ViewIndex,
4037 0, 0, 0, 0);
4038 return VINF_SUCCESS;
4039 }
4040
4041 /* If display was disabled or there is no framebuffer, a resize will be required,
4042 * because the framebuffer was/will be changed.
4043 */
4044 bool fResize = pFBInfo->fDisabled || pFBInfo->pFramebuffer.isNull();
4045
4046 if (pFBInfo->fVBVAForceResize)
4047 {
4048 /* VBVA was just enabled. Do the resize. */
4049 fResize = true;
4050 pFBInfo->fVBVAForceResize = false;
4051 }
4052
4053 /* If the screen if blanked, then do a resize request to make sure that the framebuffer
4054 * switches to the default format.
4055 */
4056 fResize = fResize || RT_BOOL((pScreen->u16Flags ^ pFBInfo->flags) & VBVA_SCREEN_F_BLANK);
4057
4058 /* Check if this is a real resize or a notification about the screen origin.
4059 * The guest uses this VBVAResize call for both.
4060 */
4061 fResize = fResize
4062 || pFBInfo->u16BitsPerPixel != pScreen->u16BitsPerPixel
4063 || pFBInfo->pu8FramebufferVRAM != (uint8_t *)pvVRAM + pScreen->u32StartOffset
4064 || pFBInfo->u32LineSize != pScreen->u32LineSize
4065 || pFBInfo->w != pScreen->u32Width
4066 || pFBInfo->h != pScreen->u32Height;
4067
4068 bool fNewOrigin = pFBInfo->xOrigin != pScreen->i32OriginX
4069 || pFBInfo->yOrigin != pScreen->i32OriginY;
4070
4071 if (fNewOrigin || fResize)
4072 pThis->i_notifyCroglResize(pView, pScreen, pvVRAM);
4073
4074 if (pFBInfo->fDisabled)
4075 {
4076 pFBInfo->fDisabled = false;
4077 fireGuestMonitorChangedEvent(pThis->mParent->i_getEventSource(),
4078 GuestMonitorChangedEventType_Enabled,
4079 pScreen->u32ViewIndex,
4080 pScreen->i32OriginX, pScreen->i32OriginY,
4081 pScreen->u32Width, pScreen->u32Height);
4082 /* Continue to update pFBInfo. */
4083 }
4084
4085 pFBInfo->u32Offset = pView->u32ViewOffset; /* Not used in HGSMI. */
4086 pFBInfo->u32MaxFramebufferSize = pView->u32MaxScreenSize; /* Not used in HGSMI. */
4087 pFBInfo->u32InformationSize = 0; /* Not used in HGSMI. */
4088
4089 pFBInfo->xOrigin = pScreen->i32OriginX;
4090 pFBInfo->yOrigin = pScreen->i32OriginY;
4091
4092 pFBInfo->w = pScreen->u32Width;
4093 pFBInfo->h = pScreen->u32Height;
4094
4095 pFBInfo->u16BitsPerPixel = pScreen->u16BitsPerPixel;
4096 pFBInfo->pu8FramebufferVRAM = (uint8_t *)pvVRAM + pScreen->u32StartOffset;
4097 pFBInfo->u32LineSize = pScreen->u32LineSize;
4098
4099 pFBInfo->flags = pScreen->u16Flags;
4100
4101 if (fResetInputMapping)
4102 {
4103 pThis->xInputMappingOrigin = 0;
4104 pThis->yInputMappingOrigin = 0;
4105 pThis->cxInputMapping = 0;
4106 pThis->cyInputMapping = 0;
4107 }
4108
4109 if (fNewOrigin)
4110 {
4111 fireGuestMonitorChangedEvent(pThis->mParent->i_getEventSource(),
4112 GuestMonitorChangedEventType_NewOrigin,
4113 pScreen->u32ViewIndex,
4114 pScreen->i32OriginX, pScreen->i32OriginY,
4115 0, 0);
4116 }
4117
4118 if (!fResize)
4119 {
4120 /* No parameters of the framebuffer have actually changed. */
4121 if (fNewOrigin)
4122 {
4123 /* VRDP server still need this notification. */
4124 LogRelFlowFunc(("Calling VRDP\n"));
4125 pThis->mParent->i_consoleVRDPServer()->SendResize();
4126 }
4127 return VINF_SUCCESS;
4128 }
4129
4130 /* Do a regular resize. */
4131 return pThis->i_handleDisplayResize(pScreen->u32ViewIndex, pScreen->u16BitsPerPixel,
4132 (uint8_t *)pvVRAM + pScreen->u32StartOffset,
4133 pScreen->u32LineSize, pScreen->u32Width, pScreen->u32Height, pScreen->u16Flags);
4134}
4135
4136DECLCALLBACK(int) Display::i_displayVBVAMousePointerShape(PPDMIDISPLAYCONNECTOR pInterface, bool fVisible, bool fAlpha,
4137 uint32_t xHot, uint32_t yHot,
4138 uint32_t cx, uint32_t cy,
4139 const void *pvShape)
4140{
4141 LogFlowFunc(("\n"));
4142
4143 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
4144 Display *pThis = pDrv->pDisplay;
4145
4146 uint32_t cbShape = 0;
4147 if (pvShape)
4148 {
4149 cbShape = (cx + 7) / 8 * cy; /* size of the AND mask */
4150 cbShape = ((cbShape + 3) & ~3) + cx * 4 * cy; /* + gap + size of the XOR mask */
4151 }
4152
4153 /* Tell the console about it */
4154 pDrv->pDisplay->mParent->i_onMousePointerShapeChange(fVisible, fAlpha,
4155 xHot, yHot, cx, cy, (uint8_t *)pvShape, cbShape);
4156
4157 return VINF_SUCCESS;
4158}
4159
4160DECLCALLBACK(void) Display::i_displayVBVAGuestCapabilityUpdate(PPDMIDISPLAYCONNECTOR pInterface, uint32_t fCapabilities)
4161{
4162 LogFlowFunc(("\n"));
4163
4164 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
4165 Display *pThis = pDrv->pDisplay;
4166
4167 pThis->i_handleUpdateGuestVBVACapabilities(fCapabilities);
4168}
4169
4170DECLCALLBACK(void) Display::i_displayVBVAInputMappingUpdate(PPDMIDISPLAYCONNECTOR pInterface, int32_t xOrigin, int32_t yOrigin,
4171 uint32_t cx, uint32_t cy)
4172{
4173 LogFlowFunc(("\n"));
4174
4175 PDRVMAINDISPLAY pDrv = PDMIDISPLAYCONNECTOR_2_MAINDISPLAY(pInterface);
4176 Display *pThis = pDrv->pDisplay;
4177
4178 pThis->i_handleUpdateVBVAInputMapping(xOrigin, yOrigin, cx, cy);
4179}
4180
4181#endif /* VBOX_WITH_HGSMI */
4182
4183/**
4184 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
4185 */
4186DECLCALLBACK(void *) Display::i_drvQueryInterface(PPDMIBASE pInterface, const char *pszIID)
4187{
4188 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
4189 PDRVMAINDISPLAY pDrv = PDMINS_2_DATA(pDrvIns, PDRVMAINDISPLAY);
4190 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
4191 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIDISPLAYCONNECTOR, &pDrv->IConnector);
4192 return NULL;
4193}
4194
4195
4196/**
4197 * Destruct a display driver instance.
4198 *
4199 * @returns VBox status code.
4200 * @param pDrvIns The driver instance data.
4201 */
4202DECLCALLBACK(void) Display::i_drvDestruct(PPDMDRVINS pDrvIns)
4203{
4204 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
4205 PDRVMAINDISPLAY pThis = PDMINS_2_DATA(pDrvIns, PDRVMAINDISPLAY);
4206 LogRelFlowFunc(("iInstance=%d\n", pDrvIns->iInstance));
4207
4208 pThis->pUpPort->pfnSetRenderVRAM(pThis->pUpPort, false);
4209
4210 pThis->IConnector.pbData = NULL;
4211 pThis->IConnector.cbScanline = 0;
4212 pThis->IConnector.cBits = 32;
4213 pThis->IConnector.cx = 0;
4214 pThis->IConnector.cy = 0;
4215
4216 if (pThis->pDisplay)
4217 {
4218 AutoWriteLock displayLock(pThis->pDisplay COMMA_LOCKVAL_SRC_POS);
4219#ifdef VBOX_WITH_VPX
4220 pThis->pDisplay->i_VideoCaptureStop();
4221#endif
4222#ifdef VBOX_WITH_CRHGSMI
4223 pThis->pDisplay->i_destructCrHgsmiData();
4224#endif
4225 pThis->pDisplay->mpDrv = NULL;
4226 pThis->pDisplay->mpVMMDev = NULL;
4227 }
4228}
4229
4230
4231/**
4232 * Construct a display driver instance.
4233 *
4234 * @copydoc FNPDMDRVCONSTRUCT
4235 */
4236DECLCALLBACK(int) Display::i_drvConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
4237{
4238 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
4239 PDRVMAINDISPLAY pThis = PDMINS_2_DATA(pDrvIns, PDRVMAINDISPLAY);
4240 LogRelFlowFunc(("iInstance=%d\n", pDrvIns->iInstance));
4241
4242 /*
4243 * Validate configuration.
4244 */
4245 if (!CFGMR3AreValuesValid(pCfg, "Object\0"))
4246 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
4247 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
4248 ("Configuration error: Not possible to attach anything to this driver!\n"),
4249 VERR_PDM_DRVINS_NO_ATTACH);
4250
4251 /*
4252 * Init Interfaces.
4253 */
4254 pDrvIns->IBase.pfnQueryInterface = Display::i_drvQueryInterface;
4255
4256 pThis->IConnector.pfnResize = Display::i_displayResizeCallback;
4257 pThis->IConnector.pfnUpdateRect = Display::i_displayUpdateCallback;
4258 pThis->IConnector.pfnRefresh = Display::i_displayRefreshCallback;
4259 pThis->IConnector.pfnReset = Display::i_displayResetCallback;
4260 pThis->IConnector.pfnLFBModeChange = Display::i_displayLFBModeChangeCallback;
4261 pThis->IConnector.pfnProcessAdapterData = Display::i_displayProcessAdapterDataCallback;
4262 pThis->IConnector.pfnProcessDisplayData = Display::i_displayProcessDisplayDataCallback;
4263#ifdef VBOX_WITH_VIDEOHWACCEL
4264 pThis->IConnector.pfnVHWACommandProcess = Display::i_displayVHWACommandProcess;
4265#endif
4266#ifdef VBOX_WITH_CRHGSMI
4267 pThis->IConnector.pfnCrHgsmiCommandProcess = Display::i_displayCrHgsmiCommandProcess;
4268 pThis->IConnector.pfnCrHgsmiControlProcess = Display::i_displayCrHgsmiControlProcess;
4269#endif
4270#if defined(VBOX_WITH_HGCM) && defined(VBOX_WITH_CROGL)
4271 pThis->IConnector.pfnCrHgcmCtlSubmit = Display::i_displayCrHgcmCtlSubmit;
4272#endif
4273#ifdef VBOX_WITH_HGSMI
4274 pThis->IConnector.pfnVBVAEnable = Display::i_displayVBVAEnable;
4275 pThis->IConnector.pfnVBVADisable = Display::i_displayVBVADisable;
4276 pThis->IConnector.pfnVBVAUpdateBegin = Display::i_displayVBVAUpdateBegin;
4277 pThis->IConnector.pfnVBVAUpdateProcess = Display::i_displayVBVAUpdateProcess;
4278 pThis->IConnector.pfnVBVAUpdateEnd = Display::i_displayVBVAUpdateEnd;
4279 pThis->IConnector.pfnVBVAResize = Display::i_displayVBVAResize;
4280 pThis->IConnector.pfnVBVAMousePointerShape = Display::i_displayVBVAMousePointerShape;
4281 pThis->IConnector.pfnVBVAGuestCapabilityUpdate = Display::i_displayVBVAGuestCapabilityUpdate;
4282 pThis->IConnector.pfnVBVAInputMappingUpdate = Display::i_displayVBVAInputMappingUpdate;
4283#endif
4284
4285 /*
4286 * Get the IDisplayPort interface of the above driver/device.
4287 */
4288 pThis->pUpPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIDISPLAYPORT);
4289 if (!pThis->pUpPort)
4290 {
4291 AssertMsgFailed(("Configuration error: No display port interface above!\n"));
4292 return VERR_PDM_MISSING_INTERFACE_ABOVE;
4293 }
4294#if defined(VBOX_WITH_VIDEOHWACCEL) || defined(VBOX_WITH_CRHGSMI)
4295 pThis->pVBVACallbacks = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIDISPLAYVBVACALLBACKS);
4296 if (!pThis->pVBVACallbacks)
4297 {
4298 AssertMsgFailed(("Configuration error: No VBVA callback interface above!\n"));
4299 return VERR_PDM_MISSING_INTERFACE_ABOVE;
4300 }
4301#endif
4302 /*
4303 * Get the Display object pointer and update the mpDrv member.
4304 */
4305 void *pv;
4306 int rc = CFGMR3QueryPtr(pCfg, "Object", &pv);
4307 if (RT_FAILURE(rc))
4308 {
4309 AssertMsgFailed(("Configuration error: No/bad \"Object\" value! rc=%Rrc\n", rc));
4310 return rc;
4311 }
4312 Display *pDisplay = (Display *)pv; /** @todo Check this cast! */
4313 pThis->pDisplay = pDisplay;
4314 pThis->pDisplay->mpDrv = pThis;
4315
4316 /* Disable VRAM to a buffer copy initially. */
4317 pThis->pUpPort->pfnSetRenderVRAM(pThis->pUpPort, false);
4318 pThis->IConnector.cBits = 32; /* DevVGA does nothing otherwise. */
4319
4320 /*
4321 * Start periodic screen refreshes
4322 */
4323 pThis->pUpPort->pfnSetRefreshRate(pThis->pUpPort, 20);
4324
4325#ifdef VBOX_WITH_CRHGSMI
4326 pDisplay->i_setupCrHgsmiData();
4327#endif
4328
4329#ifdef VBOX_WITH_VPX
4330 ComPtr<IMachine> pMachine = pDisplay->mParent->i_machine();
4331 BOOL fEnabled = false;
4332 HRESULT hrc = pMachine->COMGETTER(VideoCaptureEnabled)(&fEnabled);
4333 AssertComRCReturn(hrc, VERR_COM_UNEXPECTED);
4334 if (fEnabled)
4335 {
4336 rc = pDisplay->i_VideoCaptureStart();
4337 fireVideoCaptureChangedEvent(pDisplay->mParent->i_getEventSource());
4338 }
4339#endif
4340
4341 return rc;
4342}
4343
4344
4345/**
4346 * Display driver registration record.
4347 */
4348const PDMDRVREG Display::DrvReg =
4349{
4350 /* u32Version */
4351 PDM_DRVREG_VERSION,
4352 /* szName */
4353 "MainDisplay",
4354 /* szRCMod */
4355 "",
4356 /* szR0Mod */
4357 "",
4358 /* pszDescription */
4359 "Main display driver (Main as in the API).",
4360 /* fFlags */
4361 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
4362 /* fClass. */
4363 PDM_DRVREG_CLASS_DISPLAY,
4364 /* cMaxInstances */
4365 ~0U,
4366 /* cbInstance */
4367 sizeof(DRVMAINDISPLAY),
4368 /* pfnConstruct */
4369 Display::i_drvConstruct,
4370 /* pfnDestruct */
4371 Display::i_drvDestruct,
4372 /* pfnRelocate */
4373 NULL,
4374 /* pfnIOCtl */
4375 NULL,
4376 /* pfnPowerOn */
4377 NULL,
4378 /* pfnReset */
4379 NULL,
4380 /* pfnSuspend */
4381 NULL,
4382 /* pfnResume */
4383 NULL,
4384 /* pfnAttach */
4385 NULL,
4386 /* pfnDetach */
4387 NULL,
4388 /* pfnPowerOff */
4389 NULL,
4390 /* pfnSoftReset */
4391 NULL,
4392 /* u32EndVersion */
4393 PDM_DRVREG_VERSION
4394};
4395
4396/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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