VirtualBox

source: vbox/trunk/src/VBox/GuestHost/OpenGL/util/compositor.cpp@ 53100

Last change on this file since 53100 was 50178, checked in by vboxsync, 11 years ago

crOpenGL: seamless and resize bugfixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.6 KB
Line 
1/* $Id: compositor.cpp 50178 2014-01-23 12:04:44Z vboxsync $ */
2
3/** @file
4 * Compositor impl
5 */
6
7/*
8 * Copyright (C) 2013 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18#include <cr_compositor.h>
19
20#define VBOXVR_SCR_COMPOSITOR_RECTS_UNDEFINED UINT32_MAX
21
22
23static int crVrScrCompositorRectsAssignBuffer(PVBOXVR_SCR_COMPOSITOR pCompositor, uint32_t cRects)
24{
25 Assert(cRects);
26
27 if (pCompositor->cRectsBuffer >= cRects)
28 {
29 pCompositor->cRects = cRects;
30 return VINF_SUCCESS;
31 }
32
33 if (pCompositor->cRectsBuffer)
34 {
35 Assert(pCompositor->paSrcRects);
36 RTMemFree(pCompositor->paSrcRects);
37 pCompositor->paSrcRects = NULL;
38 Assert(pCompositor->paDstRects);
39 RTMemFree(pCompositor->paDstRects);
40 pCompositor->paDstRects = NULL;
41 Assert(pCompositor->paDstUnstretchedRects);
42 RTMemFree(pCompositor->paDstUnstretchedRects);
43 pCompositor->paDstUnstretchedRects = NULL;
44 }
45 else
46 {
47 Assert(!pCompositor->paSrcRects);
48 Assert(!pCompositor->paDstRects);
49 Assert(!pCompositor->paDstUnstretchedRects);
50 }
51
52 pCompositor->paSrcRects = (PRTRECT)RTMemAlloc(sizeof (*pCompositor->paSrcRects) * cRects);
53 if (pCompositor->paSrcRects)
54 {
55 pCompositor->paDstRects = (PRTRECT)RTMemAlloc(sizeof (*pCompositor->paDstRects) * cRects);
56 if (pCompositor->paDstRects)
57 {
58 pCompositor->paDstUnstretchedRects = (PRTRECT)RTMemAlloc(sizeof (*pCompositor->paDstUnstretchedRects) * cRects);
59 if (pCompositor->paDstUnstretchedRects)
60 {
61 pCompositor->cRects = cRects;
62 pCompositor->cRectsBuffer = cRects;
63 return VINF_SUCCESS;
64 }
65
66 RTMemFree(pCompositor->paDstRects);
67 pCompositor->paDstRects = NULL;
68 }
69 else
70 {
71 WARN(("RTMemAlloc failed!"));
72 }
73 RTMemFree(pCompositor->paSrcRects);
74 pCompositor->paSrcRects = NULL;
75 }
76 else
77 {
78 WARN(("RTMemAlloc failed!"));
79 }
80
81 pCompositor->cRects = VBOXVR_SCR_COMPOSITOR_RECTS_UNDEFINED;
82 pCompositor->cRectsBuffer = 0;
83
84 return VERR_NO_MEMORY;
85}
86
87static void crVrScrCompositorRectsInvalidate(PVBOXVR_SCR_COMPOSITOR pCompositor)
88{
89 pCompositor->cRects = VBOXVR_SCR_COMPOSITOR_RECTS_UNDEFINED;
90}
91
92static DECLCALLBACK(bool) crVrScrCompositorRectsCounterCb(PVBOXVR_COMPOSITOR pCompositor, PVBOXVR_COMPOSITOR_ENTRY pEntry, void *pvVisitor)
93{
94 uint32_t* pCounter = (uint32_t*)pvVisitor;
95 Assert(VBoxVrListRectsCount(&pEntry->Vr));
96 *pCounter += VBoxVrListRectsCount(&pEntry->Vr);
97 return true;
98}
99
100typedef struct VBOXVR_SCR_COMPOSITOR_RECTS_ASSIGNER
101{
102 PRTRECT paSrcRects;
103 PRTRECT paDstRects;
104 PRTRECT paDstUnstretchedRects;
105 uint32_t cRects;
106} VBOXVR_SCR_COMPOSITOR_RECTS_ASSIGNER, *PVBOXVR_SCR_COMPOSITOR_RECTS_ASSIGNER;
107
108static DECLCALLBACK(bool) crVrScrCompositorRectsAssignerCb(PVBOXVR_COMPOSITOR pCCompositor, PVBOXVR_COMPOSITOR_ENTRY pCEntry, void *pvVisitor)
109{
110 PVBOXVR_SCR_COMPOSITOR_RECTS_ASSIGNER pData = (PVBOXVR_SCR_COMPOSITOR_RECTS_ASSIGNER)pvVisitor;
111 PVBOXVR_SCR_COMPOSITOR pCompositor = VBOXVR_SCR_COMPOSITOR_FROM_COMPOSITOR(pCCompositor);
112 PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry = VBOXVR_SCR_COMPOSITOR_ENTRY_FROM_ENTRY(pCEntry);
113 pEntry->paSrcRects = pData->paSrcRects;
114 pEntry->paDstRects = pData->paDstRects;
115 pEntry->paDstUnstretchedRects = pData->paDstUnstretchedRects;
116 uint32_t cRects = VBoxVrListRectsCount(&pCEntry->Vr);
117 Assert(cRects);
118 Assert(cRects <= pData->cRects);
119 int rc = VBoxVrListRectsGet(&pCEntry->Vr, cRects, pEntry->paDstUnstretchedRects);
120 AssertRC(rc);
121
122 if (!pEntry->Rect.xLeft && !pEntry->Rect.yTop)
123 {
124 memcpy(pEntry->paSrcRects, pEntry->paDstUnstretchedRects, cRects * sizeof (*pEntry->paSrcRects));
125 }
126 else
127 {
128 for (uint32_t i = 0; i < cRects; ++i)
129 {
130 pEntry->paSrcRects[i].xLeft = (int32_t)((pEntry->paDstUnstretchedRects[i].xLeft - pEntry->Rect.xLeft));
131 pEntry->paSrcRects[i].yTop = (int32_t)((pEntry->paDstUnstretchedRects[i].yTop - pEntry->Rect.yTop));
132 pEntry->paSrcRects[i].xRight = (int32_t)((pEntry->paDstUnstretchedRects[i].xRight - pEntry->Rect.xLeft));
133 pEntry->paSrcRects[i].yBottom = (int32_t)((pEntry->paDstUnstretchedRects[i].yBottom - pEntry->Rect.yTop));
134 }
135 }
136
137#ifndef IN_RING0
138 if (pCompositor->StretchX != 1. || pCompositor->StretchY != 1.)
139 {
140 for (uint32_t i = 0; i < cRects; ++i)
141 {
142 if (pCompositor->StretchX != 1.)
143 {
144 pEntry->paDstRects[i].xLeft = (int32_t)(pEntry->paDstUnstretchedRects[i].xLeft * pCompositor->StretchX);
145 pEntry->paDstRects[i].xRight = (int32_t)(pEntry->paDstUnstretchedRects[i].xRight * pCompositor->StretchX);
146 }
147 if (pCompositor->StretchY != 1.)
148 {
149 pEntry->paDstRects[i].yTop = (int32_t)(pEntry->paDstUnstretchedRects[i].yTop * pCompositor->StretchY);
150 pEntry->paDstRects[i].yBottom = (int32_t)(pEntry->paDstUnstretchedRects[i].yBottom * pCompositor->StretchY);
151 }
152 }
153 }
154 else
155#endif
156 {
157 memcpy(pEntry->paDstRects, pEntry->paDstUnstretchedRects, cRects * sizeof (*pEntry->paDstUnstretchedRects));
158 }
159
160#if 0//ndef IN_RING0
161 bool canZeroX = (pCompositor->StretchX < 1.);
162 bool canZeroY = (pCompositor->StretchY < 1.);
163 if (canZeroX && canZeroY)
164 {
165 /* filter out zero rectangles*/
166 uint32_t iOrig, iNew;
167 for (iOrig = 0, iNew = 0; iOrig < cRects; ++iOrig)
168 {
169 PRTRECT pOrigRect = &pEntry->paDstRects[iOrig];
170 if (pOrigRect->xLeft != pOrigRect->xRight
171 && pOrigRect->yTop != pOrigRect->yBottom)
172 continue;
173
174 if (iNew != iOrig)
175 {
176 PRTRECT pNewRect = &pEntry->paSrcRects[iNew];
177 *pNewRect = *pOrigRect;
178 }
179
180 ++iNew;
181 }
182
183 Assert(iNew <= iOrig);
184
185 uint32_t cDiff = iOrig - iNew;
186
187 if (cDiff)
188 {
189 pCompositor->cRects -= cDiff;
190 cRects -= cDiff;
191 }
192 }
193#endif
194
195 pEntry->cRects = cRects;
196 pData->paDstRects += cRects;
197 pData->paSrcRects += cRects;
198 pData->paDstUnstretchedRects += cRects;
199 pData->cRects -= cRects;
200 return true;
201}
202
203static int crVrScrCompositorRectsCheckInit(const VBOXVR_SCR_COMPOSITOR *pcCompositor)
204{
205 VBOXVR_SCR_COMPOSITOR *pCompositor = const_cast<VBOXVR_SCR_COMPOSITOR*>(pcCompositor);
206
207 if (pCompositor->cRects != VBOXVR_SCR_COMPOSITOR_RECTS_UNDEFINED)
208 return VINF_SUCCESS;
209
210 uint32_t cRects = 0;
211 VBoxVrCompositorVisit(&pCompositor->Compositor, crVrScrCompositorRectsCounterCb, &cRects);
212
213 if (!cRects)
214 {
215 pCompositor->cRects = 0;
216 return VINF_SUCCESS;
217 }
218
219 int rc = crVrScrCompositorRectsAssignBuffer(pCompositor, cRects);
220 if (!RT_SUCCESS(rc))
221 return rc;
222
223 VBOXVR_SCR_COMPOSITOR_RECTS_ASSIGNER AssignerData;
224 AssignerData.paSrcRects = pCompositor->paSrcRects;
225 AssignerData.paDstRects = pCompositor->paDstRects;
226 AssignerData.paDstUnstretchedRects = pCompositor->paDstUnstretchedRects;
227 AssignerData.cRects = pCompositor->cRects;
228 VBoxVrCompositorVisit(&pCompositor->Compositor, crVrScrCompositorRectsAssignerCb, &AssignerData);
229 Assert(!AssignerData.cRects);
230 return VINF_SUCCESS;
231}
232
233
234static int crVrScrCompositorEntryRegionsAdd(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, uint32_t cRegions, const RTRECT *paRegions, VBOXVR_SCR_COMPOSITOR_ENTRY **ppReplacedScrEntry, uint32_t *pfChangedFlags)
235{
236 uint32_t fChangedFlags = 0;
237 PVBOXVR_COMPOSITOR_ENTRY pReplacedEntry;
238 int rc = VBoxVrCompositorEntryRegionsAdd(&pCompositor->Compositor, pEntry ? &pEntry->Ce : NULL, cRegions, paRegions, &pReplacedEntry, &fChangedFlags);
239 if (!RT_SUCCESS(rc))
240 {
241 WARN(("VBoxVrCompositorEntryRegionsAdd failed, rc %d", rc));
242 return rc;
243 }
244
245 VBOXVR_SCR_COMPOSITOR_ENTRY *pReplacedScrEntry = VBOXVR_SCR_COMPOSITOR_ENTRY_FROM_ENTRY(pReplacedEntry);
246
247 if (fChangedFlags & VBOXVR_COMPOSITOR_CF_REGIONS_CHANGED)
248 {
249 crVrScrCompositorRectsInvalidate(pCompositor);
250 }
251 else if (fChangedFlags & VBOXVR_COMPOSITOR_CF_ENTRY_REPLACED)
252 {
253 Assert(pReplacedScrEntry);
254 }
255
256 if (fChangedFlags & VBOXVR_COMPOSITOR_CF_OTHER_ENTRIES_REGIONS_CHANGED)
257 {
258 CrVrScrCompositorEntrySetAllChanged(pCompositor, true);
259 }
260 else if ((fChangedFlags & VBOXVR_COMPOSITOR_CF_ENTRY_REGIONS_CHANGED) && pEntry)
261 {
262 CrVrScrCompositorEntrySetChanged(pEntry, true);
263 }
264
265 if (pfChangedFlags)
266 *pfChangedFlags = fChangedFlags;
267
268 if (ppReplacedScrEntry)
269 *ppReplacedScrEntry = pReplacedScrEntry;
270
271 return VINF_SUCCESS;
272}
273
274static int crVrScrCompositorEntryRegionsSet(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, uint32_t cRegions, const RTRECT *paRegions, bool *pfChanged)
275{
276 bool fChanged;
277 int rc = VBoxVrCompositorEntryRegionsSet(&pCompositor->Compositor, &pEntry->Ce, cRegions, paRegions, &fChanged);
278 if (!RT_SUCCESS(rc))
279 {
280 WARN(("VBoxVrCompositorEntryRegionsSet failed, rc %d", rc));
281 return rc;
282 }
283
284 if (fChanged)
285 {
286 CrVrScrCompositorEntrySetAllChanged(pCompositor, true);
287 if (!CrVrScrCompositorEntryIsInList(pEntry))
288 {
289 pEntry->cRects = 0;
290 pEntry->paSrcRects = NULL;
291 pEntry->paDstRects = NULL;
292 pEntry->paDstUnstretchedRects = NULL;
293 }
294 crVrScrCompositorRectsInvalidate(pCompositor);
295 }
296
297
298 if (pfChanged)
299 *pfChanged = fChanged;
300 return VINF_SUCCESS;
301}
302
303static int crVrScrCompositorEntryPositionSet(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, const RTPOINT *pPos, bool *pfChanged)
304{
305 if (pfChanged)
306 *pfChanged = false;
307 if (pEntry && (pEntry->Rect.xLeft != pPos->x || pEntry->Rect.yTop != pPos->y))
308 {
309 if (VBoxVrCompositorEntryIsInList(&pEntry->Ce))
310 {
311 int rc = VBoxVrCompositorEntryRegionsTranslate(&pCompositor->Compositor, &pEntry->Ce, pPos->x - pEntry->Rect.xLeft, pPos->y - pEntry->Rect.yTop, pfChanged);
312 if (!RT_SUCCESS(rc))
313 {
314 WARN(("VBoxVrCompositorEntryRegionsTranslate failed rc %d", rc));
315 return rc;
316 }
317
318 crVrScrCompositorRectsInvalidate(pCompositor);
319 }
320
321 VBoxRectMove(&pEntry->Rect, pPos->x, pPos->y);
322 CrVrScrCompositorEntrySetChanged(pEntry, true);
323
324 if (pfChanged)
325 *pfChanged = true;
326 }
327 return VINF_SUCCESS;
328}
329
330static int crVrScrCompositorEntryEnsureRegionsBounds(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, bool *pfChanged)
331{
332 RTRECT Rect;
333 Rect.xLeft = RT_MAX(pCompositor->Rect.xLeft, pEntry->Rect.xLeft);
334 Rect.yTop = RT_MAX(pCompositor->Rect.yTop, pEntry->Rect.yTop);
335 Rect.xRight = RT_MIN(pCompositor->Rect.xRight, pEntry->Rect.xRight);
336 Rect.yBottom = RT_MIN(pCompositor->Rect.yBottom, pEntry->Rect.yBottom);
337 bool fChanged = false;
338
339 if (pfChanged)
340 *pfChanged = false;
341
342 int rc = CrVrScrCompositorEntryRegionsIntersect(pCompositor, pEntry, 1, &Rect, &fChanged);
343 if (!RT_SUCCESS(rc))
344 WARN(("CrVrScrCompositorEntryRegionsIntersect failed, rc %d", rc));
345
346 if (pfChanged)
347 *pfChanged = fChanged;
348 return rc;
349}
350
351VBOXVREGDECL(int) CrVrScrCompositorEntryRegionsAdd(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, const RTPOINT *pPos, uint32_t cRegions, const RTRECT *paRegions, bool fPosRelated, VBOXVR_SCR_COMPOSITOR_ENTRY **ppReplacedScrEntry, uint32_t *pfChangeFlags)
352{
353 int rc;
354 uint32_t fChangeFlags = 0;
355 bool fPosChanged = false;
356 RTRECT *paTranslatedRects = NULL;
357 if (pPos)
358 {
359 rc = crVrScrCompositorEntryPositionSet(pCompositor, pEntry, pPos, &fPosChanged);
360 if (!RT_SUCCESS(rc))
361 {
362 WARN(("RegionsAdd: crVrScrCompositorEntryPositionSet failed rc %d", rc));
363 return rc;
364 }
365 }
366
367 if (fPosRelated)
368 {
369 if (!pEntry)
370 {
371 WARN(("Entry is expected to be specified for pos-related regions"));
372 return VERR_INVALID_PARAMETER;
373 }
374
375 if (cRegions && (pEntry->Rect.xLeft || pEntry->Rect.yTop))
376 {
377 paTranslatedRects = (RTRECT*)RTMemAlloc(sizeof (RTRECT) * cRegions);
378 if (!paTranslatedRects)
379 {
380 WARN(("RTMemAlloc failed"));
381 return VERR_NO_MEMORY;
382 }
383 memcpy (paTranslatedRects, paRegions, sizeof (RTRECT) * cRegions);
384 for (uint32_t i = 0; i < cRegions; ++i)
385 {
386 VBoxRectTranslate(&paTranslatedRects[i], pEntry->Rect.xLeft, pEntry->Rect.yTop);
387 paRegions = paTranslatedRects;
388 }
389 }
390 }
391
392 rc = crVrScrCompositorEntryRegionsAdd(pCompositor, pEntry, cRegions, paRegions, ppReplacedScrEntry, &fChangeFlags);
393 if (!RT_SUCCESS(rc))
394 {
395 WARN(("crVrScrCompositorEntryRegionsAdd failed, rc %d", rc));
396 goto done;
397 }
398
399 if ((fPosChanged || (fChangeFlags & VBOXVR_COMPOSITOR_CF_ENTRY_REGIONS_CHANGED)) && pEntry)
400 {
401 bool fAdjusted = false;
402 rc = crVrScrCompositorEntryEnsureRegionsBounds(pCompositor, pEntry, &fAdjusted);
403 if (!RT_SUCCESS(rc))
404 {
405 WARN(("crVrScrCompositorEntryEnsureRegionsBounds failed, rc %d", rc));
406 goto done;
407 }
408
409 if (fAdjusted)
410 {
411 if (CrVrScrCompositorEntryIsUsed(pEntry))
412 {
413 fChangeFlags &= ~VBOXVR_COMPOSITOR_CF_ENTRY_REPLACED;
414 fChangeFlags |= VBOXVR_COMPOSITOR_CF_REGIONS_CHANGED | VBOXVR_COMPOSITOR_CF_ENTRY_REGIONS_CHANGED;
415 }
416 else
417 {
418 fChangeFlags = 0;
419 }
420 }
421 }
422
423 if (fChangeFlags & VBOXVR_COMPOSITOR_CF_ENTRY_REPLACED)
424 fPosChanged = false;
425 else if (ppReplacedScrEntry)
426 *ppReplacedScrEntry = NULL;
427
428 if (pfChangeFlags)
429 {
430 if (fPosChanged)
431 {
432 /* means entry was in list and was moved, so regions changed */
433 *pfChangeFlags = VBOXVR_COMPOSITOR_CF_REGIONS_CHANGED | VBOXVR_COMPOSITOR_CF_ENTRY_REGIONS_CHANGED | VBOXVR_COMPOSITOR_CF_OTHER_ENTRIES_REGIONS_CHANGED;
434 }
435 else
436 *pfChangeFlags = fChangeFlags;
437 }
438
439done:
440
441 if (paTranslatedRects)
442 RTMemFree(paTranslatedRects);
443
444 return rc;
445}
446
447VBOXVREGDECL(int) CrVrScrCompositorEntryRectSet(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, const RTRECT *pRect)
448{
449 if (!memcmp(&pEntry->Rect, pRect, sizeof (*pRect)))
450 {
451 return VINF_SUCCESS;
452 }
453 RTPOINT Point = {pRect->xLeft, pRect->yTop};
454 bool fChanged = false;
455 int rc = crVrScrCompositorEntryPositionSet(pCompositor, pEntry, &Point, &fChanged);
456 if (!RT_SUCCESS(rc))
457 {
458 WARN(("crVrScrCompositorEntryPositionSet failed %d", rc));
459 return rc;
460 }
461
462 pEntry->Rect = *pRect;
463
464 if (!CrVrScrCompositorEntryIsUsed(pEntry))
465 return VINF_SUCCESS;
466
467 rc = crVrScrCompositorEntryEnsureRegionsBounds(pCompositor, pEntry, NULL);
468 if (!RT_SUCCESS(rc))
469 {
470 WARN(("crVrScrCompositorEntryEnsureRegionsBounds failed, rc %d", rc));
471 return rc;
472 }
473
474 return VINF_SUCCESS;
475}
476
477VBOXVREGDECL(int) CrVrScrCompositorEntryTexAssign(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, CR_TEXDATA *pTex)
478{
479 if (pEntry->pTex == pTex)
480 return VINF_SUCCESS;
481
482 if (pEntry->pTex)
483 CrTdRelease(pEntry->pTex);
484 if (pTex)
485 CrTdAddRef(pTex);
486 pEntry->pTex = pTex;
487 return VINF_SUCCESS;
488}
489
490VBOXVREGDECL(int) CrVrScrCompositorEntryRegionsSet(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, const RTPOINT *pPos, uint32_t cRegions, const RTRECT *paRegions, bool fPosRelated, bool *pfChanged)
491{
492 /* @todo: the fChanged sate calculation is really rough now, this is enough for now though */
493 bool fChanged = false, fPosChanged = false;
494 bool fWasInList = CrVrScrCompositorEntryIsInList(pEntry);
495 RTRECT *paTranslatedRects = NULL;
496 int rc = CrVrScrCompositorEntryRemove(pCompositor, pEntry);
497 if (!RT_SUCCESS(rc))
498 {
499 WARN(("RegionsSet: CrVrScrCompositorEntryRemove failed rc %d", rc));
500 return rc;
501 }
502
503 if (pPos)
504 {
505 rc = crVrScrCompositorEntryPositionSet(pCompositor, pEntry, pPos, &fPosChanged);
506 if (!RT_SUCCESS(rc))
507 {
508 WARN(("RegionsSet: crVrScrCompositorEntryPositionSet failed rc %d", rc));
509 return rc;
510 }
511 }
512
513 if (fPosRelated)
514 {
515 if (!pEntry)
516 {
517 WARN(("Entry is expected to be specified for pos-related regions"));
518 return VERR_INVALID_PARAMETER;
519 }
520
521 if (cRegions && (pEntry->Rect.xLeft || pEntry->Rect.yTop))
522 {
523 paTranslatedRects = (RTRECT*)RTMemAlloc(sizeof (RTRECT) * cRegions);
524 if (!paTranslatedRects)
525 {
526 WARN(("RTMemAlloc failed"));
527 return VERR_NO_MEMORY;
528 }
529 memcpy (paTranslatedRects, paRegions, sizeof (RTRECT) * cRegions);
530 for (uint32_t i = 0; i < cRegions; ++i)
531 {
532 VBoxRectTranslate(&paTranslatedRects[i], pEntry->Rect.xLeft, pEntry->Rect.yTop);
533 paRegions = paTranslatedRects;
534 }
535 }
536 }
537
538 rc = crVrScrCompositorEntryRegionsSet(pCompositor, pEntry, cRegions, paRegions, &fChanged);
539 if (!RT_SUCCESS(rc))
540 {
541 WARN(("crVrScrCompositorEntryRegionsSet failed, rc %d", rc));
542 return rc;
543 }
544
545 if (fChanged && CrVrScrCompositorEntryIsUsed(pEntry))
546 {
547 rc = crVrScrCompositorEntryEnsureRegionsBounds(pCompositor, pEntry, NULL);
548 if (!RT_SUCCESS(rc))
549 {
550 WARN(("crVrScrCompositorEntryEnsureRegionsBounds failed, rc %d", rc));
551 return rc;
552 }
553 }
554
555 if (pfChanged)
556 *pfChanged = fPosChanged || fChanged || fWasInList;
557
558 return VINF_SUCCESS;
559}
560
561VBOXVREGDECL(int) CrVrScrCompositorEntryListIntersect(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, const VBOXVR_LIST *pList2, bool *pfChanged)
562{
563 bool fChanged = false;
564 int rc = VBoxVrCompositorEntryListIntersect(&pCompositor->Compositor, &pEntry->Ce, pList2, &fChanged);
565 if (!RT_SUCCESS(rc))
566 {
567 WARN(("RegionsIntersect: VBoxVrCompositorEntryRegionsIntersect failed rc %d", rc));
568 return rc;
569 }
570
571 if (fChanged)
572 {
573 CrVrScrCompositorEntrySetChanged(pEntry, true);
574 crVrScrCompositorRectsInvalidate(pCompositor);
575 }
576
577 if (pfChanged)
578 *pfChanged = fChanged;
579
580 return VINF_SUCCESS;
581}
582
583VBOXVREGDECL(int) CrVrScrCompositorEntryRegionsIntersect(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, uint32_t cRegions, const RTRECT *paRegions, bool *pfChanged)
584{
585 bool fChanged = false;
586 int rc = VBoxVrCompositorEntryRegionsIntersect(&pCompositor->Compositor, &pEntry->Ce, cRegions, paRegions, &fChanged);
587 if (!RT_SUCCESS(rc))
588 {
589 WARN(("RegionsIntersect: VBoxVrCompositorEntryRegionsIntersect failed rc %d", rc));
590 return rc;
591 }
592
593 if (fChanged)
594 crVrScrCompositorRectsInvalidate(pCompositor);
595
596 if (pfChanged)
597 *pfChanged = fChanged;
598
599 return VINF_SUCCESS;
600}
601
602VBOXVREGDECL(int) CrVrScrCompositorEntryListIntersectAll(PVBOXVR_SCR_COMPOSITOR pCompositor, const VBOXVR_LIST *pList2, bool *pfChanged)
603{
604 VBOXVR_SCR_COMPOSITOR_ITERATOR Iter;
605 CrVrScrCompositorIterInit(pCompositor, &Iter);
606 PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry;
607 int rc = VINF_SUCCESS;
608 bool fChanged = false;
609
610 while ((pEntry = CrVrScrCompositorIterNext(&Iter)) != NULL)
611 {
612 bool fTmpChanged = false;
613 int tmpRc = CrVrScrCompositorEntryListIntersect(pCompositor, pEntry, pList2, &fTmpChanged);
614 if (RT_SUCCESS(tmpRc))
615 {
616 fChanged |= fTmpChanged;
617 }
618 else
619 {
620 WARN(("CrVrScrCompositorEntryRegionsIntersect failed, rc %d", tmpRc));
621 rc = tmpRc;
622 }
623 }
624
625 if (pfChanged)
626 *pfChanged = fChanged;
627
628 return rc;
629}
630
631VBOXVREGDECL(int) CrVrScrCompositorEntryRegionsIntersectAll(PVBOXVR_SCR_COMPOSITOR pCompositor, uint32_t cRegions, const RTRECT *paRegions, bool *pfChanged)
632{
633 VBOXVR_SCR_COMPOSITOR_ITERATOR Iter;
634 CrVrScrCompositorIterInit(pCompositor, &Iter);
635 PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry;
636 int rc = VINF_SUCCESS;
637 bool fChanged = false;
638
639 while ((pEntry = CrVrScrCompositorIterNext(&Iter)) != NULL)
640 {
641 bool fTmpChanged = false;
642 int tmpRc = CrVrScrCompositorEntryRegionsIntersect(pCompositor, pEntry, cRegions, paRegions, &fTmpChanged);
643 if (RT_SUCCESS(tmpRc))
644 {
645 fChanged |= fTmpChanged;
646 }
647 else
648 {
649 WARN(("CrVrScrCompositorEntryRegionsIntersect failed, rc %d", tmpRc));
650 rc = tmpRc;
651 }
652 }
653
654 if (pfChanged)
655 *pfChanged = fChanged;
656
657 return rc;
658}
659
660VBOXVREGDECL(int) CrVrScrCompositorEntryPosSet(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, const RTPOINT *pPos)
661{
662 int rc = crVrScrCompositorEntryPositionSet(pCompositor, pEntry, pPos, NULL);
663 if (!RT_SUCCESS(rc))
664 {
665 WARN(("RegionsSet: crVrScrCompositorEntryPositionSet failed rc %d", rc));
666 return rc;
667 }
668
669 rc = crVrScrCompositorEntryEnsureRegionsBounds(pCompositor, pEntry, NULL);
670 if (!RT_SUCCESS(rc))
671 {
672 WARN(("RegionsSet: crVrScrCompositorEntryEnsureRegionsBounds failed rc %d", rc));
673 return rc;
674 }
675
676 return VINF_SUCCESS;
677}
678
679/* regions are valid until the next CrVrScrCompositor call */
680VBOXVREGDECL(int) CrVrScrCompositorEntryRegionsGet(const VBOXVR_SCR_COMPOSITOR *pCompositor, const VBOXVR_SCR_COMPOSITOR_ENTRY *pEntry, uint32_t *pcRegions, const RTRECT **ppaSrcRegions, const RTRECT **ppaDstRegions, const RTRECT **ppaDstUnstretchedRects)
681{
682 if (CrVrScrCompositorEntryIsUsed(pEntry))
683 {
684 int rc = crVrScrCompositorRectsCheckInit(pCompositor);
685 if (!RT_SUCCESS(rc))
686 {
687 WARN(("crVrScrCompositorRectsCheckInit failed, rc %d", rc));
688 return rc;
689 }
690 }
691
692 Assert(pCompositor->cRects != VBOXVR_SCR_COMPOSITOR_RECTS_UNDEFINED);
693
694 *pcRegions = pEntry->cRects;
695 if (ppaSrcRegions)
696 *ppaSrcRegions = pEntry->paSrcRects;
697 if (ppaDstRegions)
698 *ppaDstRegions = pEntry->paDstRects;
699 if (ppaDstUnstretchedRects)
700 *ppaDstUnstretchedRects = pEntry->paDstUnstretchedRects;
701
702 return VINF_SUCCESS;
703}
704
705VBOXVREGDECL(uint32_t) CrVrScrCompositorEntryFlagsCombinedGet(const VBOXVR_SCR_COMPOSITOR *pCompositor, const VBOXVR_SCR_COMPOSITOR_ENTRY *pEntry)
706{
707 return CRBLT_FOP_COMBINE(pCompositor->fFlags, pEntry->fFlags);
708}
709
710VBOXVREGDECL(void) CrVrScrCompositorEntryFlagsSet(PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, uint32_t fFlags)
711{
712 if (pEntry->fFlags == fFlags)
713 return;
714
715 pEntry->fFlags = fFlags;
716 CrVrScrCompositorEntrySetChanged(pEntry, true);
717}
718
719static void crVrScrCompositorEntryDataCleanup(PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry)
720{
721 pEntry->cRects = 0;
722 pEntry->paSrcRects = NULL;
723 pEntry->paDstRects = NULL;
724 pEntry->paDstUnstretchedRects = NULL;
725}
726
727static void crVrScrCompositorEntryDataCopy(PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, PVBOXVR_SCR_COMPOSITOR_ENTRY pToEntry)
728{
729 pToEntry->cRects = pEntry->cRects;
730 pToEntry->paSrcRects = pEntry->paSrcRects;
731 pToEntry->paDstRects = pEntry->paDstRects;
732 pToEntry->paDstUnstretchedRects = pEntry->paDstUnstretchedRects;
733 crVrScrCompositorEntryDataCleanup(pEntry);
734}
735
736VBOXVREGDECL(int) CrVrScrCompositorEntryRemove(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry)
737{
738 if (!VBoxVrCompositorEntryRemove(&pCompositor->Compositor, &pEntry->Ce))
739 return VINF_SUCCESS;
740
741 CrVrScrCompositorEntrySetChanged(pEntry, true);
742 crVrScrCompositorEntryDataCleanup(pEntry);
743
744 crVrScrCompositorRectsInvalidate(pCompositor);
745 return VINF_SUCCESS;
746}
747
748VBOXVREGDECL(bool) CrVrScrCompositorEntryReplace(PVBOXVR_SCR_COMPOSITOR pCompositor, PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry, PVBOXVR_SCR_COMPOSITOR_ENTRY pNewEntry)
749{
750 Assert(!CrVrScrCompositorEntryIsUsed(pNewEntry));
751
752 if (!VBoxVrCompositorEntryReplace(&pCompositor->Compositor, &pEntry->Ce, &pNewEntry->Ce))
753 return false;
754
755 CrVrScrCompositorEntrySetChanged(pEntry, true);
756 crVrScrCompositorEntryDataCopy(pEntry, pNewEntry);
757 CrVrScrCompositorEntrySetChanged(pNewEntry, true);
758
759 return true;
760}
761
762static DECLCALLBACK(void) crVrScrCompositorEntryReleasedCB(const struct VBOXVR_COMPOSITOR *pCompositor, PVBOXVR_COMPOSITOR_ENTRY pEntry, PVBOXVR_COMPOSITOR_ENTRY pReplacingEntry)
763{
764 PVBOXVR_SCR_COMPOSITOR_ENTRY pCEntry = VBOXVR_SCR_COMPOSITOR_ENTRY_FROM_ENTRY(pEntry);
765
766 CrVrScrCompositorEntrySetChanged(pCEntry, true);
767
768 Assert(!CrVrScrCompositorEntryIsInList(pCEntry));
769
770 if (pReplacingEntry)
771 {
772 PVBOXVR_SCR_COMPOSITOR_ENTRY pCReplacingEntry = VBOXVR_SCR_COMPOSITOR_ENTRY_FROM_ENTRY(pReplacingEntry);
773 Assert(CrVrScrCompositorEntryIsInList(pCReplacingEntry));
774 pCReplacingEntry->cRects = pCEntry->cRects;
775 pCReplacingEntry->paSrcRects = pCEntry->paSrcRects;
776 pCReplacingEntry->paDstRects = pCEntry->paDstRects;
777 pCReplacingEntry->paDstUnstretchedRects = pCEntry->paDstUnstretchedRects;
778 }
779
780 if (pCEntry->pfnEntryReleased)
781 {
782 PVBOXVR_SCR_COMPOSITOR_ENTRY pCReplacingEntry = pReplacingEntry ? VBOXVR_SCR_COMPOSITOR_ENTRY_FROM_ENTRY(pReplacingEntry) : NULL;
783 PVBOXVR_SCR_COMPOSITOR pCConpositor = VBOXVR_SCR_COMPOSITOR_FROM_COMPOSITOR(pCompositor);
784 pCEntry->pfnEntryReleased(pCConpositor, pCEntry, pCReplacingEntry);
785 }
786}
787
788VBOXVREGDECL(int) CrVrScrCompositorRectSet(PVBOXVR_SCR_COMPOSITOR pCompositor, const RTRECT *pRect, bool *pfChanged)
789{
790 if (!memcmp(&pCompositor->Rect, pRect, sizeof (pCompositor->Rect)))
791 {
792 if (pfChanged)
793 *pfChanged = false;
794 return VINF_SUCCESS;
795 }
796
797 pCompositor->Rect = *pRect;
798
799 VBOXVR_SCR_COMPOSITOR_ITERATOR Iter;
800 CrVrScrCompositorIterInit(pCompositor, &Iter);
801 PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry;
802 while ((pEntry = CrVrScrCompositorIterNext(&Iter)) != NULL)
803 {
804 int rc = crVrScrCompositorEntryEnsureRegionsBounds(pCompositor, pEntry, NULL);
805 if (!RT_SUCCESS(rc))
806 {
807 WARN(("crVrScrCompositorEntryEnsureRegionsBounds failed, rc %d", rc));
808 return rc;
809 }
810 }
811
812 return VINF_SUCCESS;
813}
814
815VBOXVREGDECL(void) CrVrScrCompositorInit(PVBOXVR_SCR_COMPOSITOR pCompositor, const RTRECT *pRect)
816{
817 memset(pCompositor, 0, sizeof (*pCompositor));
818 VBoxVrCompositorInit(&pCompositor->Compositor, crVrScrCompositorEntryReleasedCB);
819 pCompositor->fFlags = CRBLT_F_LINEAR | CRBLT_F_INVERT_YCOORDS;
820 if (pRect)
821 pCompositor->Rect = *pRect;
822#ifndef IN_RING0
823 pCompositor->StretchX = 1.0;
824 pCompositor->StretchY = 1.0;
825#endif
826}
827
828VBOXVREGDECL(void) CrVrScrCompositorRegionsClear(PVBOXVR_SCR_COMPOSITOR pCompositor, bool *pfChanged)
829{
830 /* set changed flag first, while entries are in the list and we have them */
831 CrVrScrCompositorEntrySetAllChanged(pCompositor, true);
832 VBoxVrCompositorRegionsClear(&pCompositor->Compositor, pfChanged);
833 crVrScrCompositorRectsInvalidate(pCompositor);
834}
835
836VBOXVREGDECL(void) CrVrScrCompositorClear(PVBOXVR_SCR_COMPOSITOR pCompositor)
837{
838 CrVrScrCompositorRegionsClear(pCompositor, NULL);
839 if (pCompositor->paDstRects)
840 {
841 RTMemFree(pCompositor->paDstRects);
842 pCompositor->paDstRects = NULL;
843 }
844 if (pCompositor->paSrcRects)
845 {
846 RTMemFree(pCompositor->paSrcRects);
847 pCompositor->paSrcRects = NULL;
848 }
849 if (pCompositor->paDstUnstretchedRects)
850 {
851 RTMemFree(pCompositor->paDstUnstretchedRects);
852 pCompositor->paDstUnstretchedRects = NULL;
853 }
854
855 pCompositor->cRects = 0;
856 pCompositor->cRectsBuffer = 0;
857}
858
859VBOXVREGDECL(void) CrVrScrCompositorEntrySetAllChanged(PVBOXVR_SCR_COMPOSITOR pCompositor, bool fChanged)
860{
861 VBOXVR_SCR_COMPOSITOR_ITERATOR CIter;
862 PVBOXVR_SCR_COMPOSITOR_ENTRY pCurEntry;
863 CrVrScrCompositorIterInit(pCompositor, &CIter);
864
865 while ((pCurEntry = CrVrScrCompositorIterNext(&CIter)) != NULL)
866 {
867 CrVrScrCompositorEntrySetChanged(pCurEntry, fChanged);
868 }
869}
870
871#ifndef IN_RING0
872VBOXVREGDECL(void) CrVrScrCompositorSetStretching(PVBOXVR_SCR_COMPOSITOR pCompositor, float StretchX, float StretchY)
873{
874 if (pCompositor->StretchX == StretchX && pCompositor->StretchY == StretchY)
875 return;
876
877 pCompositor->StretchX = StretchX;
878 pCompositor->StretchY = StretchY;
879 crVrScrCompositorRectsInvalidate(pCompositor);
880 CrVrScrCompositorEntrySetAllChanged(pCompositor, true);
881}
882#endif
883
884/* regions are valid until the next CrVrScrCompositor call */
885VBOXVREGDECL(int) CrVrScrCompositorRegionsGet(const VBOXVR_SCR_COMPOSITOR *pCompositor, uint32_t *pcRegions, const RTRECT **ppaSrcRegions, const RTRECT **ppaDstRegions, const RTRECT **ppaDstUnstretchedRects)
886{
887 int rc = crVrScrCompositorRectsCheckInit(pCompositor);
888 if (!RT_SUCCESS(rc))
889 {
890 WARN(("crVrScrCompositorRectsCheckInit failed, rc %d", rc));
891 return rc;
892 }
893
894 Assert(pCompositor->cRects != VBOXVR_SCR_COMPOSITOR_RECTS_UNDEFINED);
895
896 *pcRegions = pCompositor->cRects;
897 if (ppaSrcRegions)
898 *ppaSrcRegions = pCompositor->paSrcRects;
899 if (ppaDstRegions)
900 *ppaDstRegions = pCompositor->paDstRects;
901 if (ppaDstUnstretchedRects)
902 *ppaDstUnstretchedRects = pCompositor->paDstUnstretchedRects;
903
904 return VINF_SUCCESS;
905}
906
907typedef struct VBOXVR_SCR_COMPOSITOR_VISITOR_CB
908{
909 PFNVBOXVRSCRCOMPOSITOR_VISITOR pfnVisitor;
910 void *pvVisitor;
911} VBOXVR_SCR_COMPOSITOR_VISITOR_CB, *PVBOXVR_SCR_COMPOSITOR_VISITOR_CB;
912
913static DECLCALLBACK(bool) crVrScrCompositorVisitCb(PVBOXVR_COMPOSITOR pCCompositor, PVBOXVR_COMPOSITOR_ENTRY pCEntry, void *pvVisitor)
914{
915 PVBOXVR_SCR_COMPOSITOR_VISITOR_CB pData = (PVBOXVR_SCR_COMPOSITOR_VISITOR_CB)pvVisitor;
916 PVBOXVR_SCR_COMPOSITOR pCompositor = VBOXVR_SCR_COMPOSITOR_FROM_COMPOSITOR(pCCompositor);
917 PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry = VBOXVR_SCR_COMPOSITOR_ENTRY_FROM_ENTRY(pCEntry);
918 return pData->pfnVisitor(pCompositor, pEntry, pData->pvVisitor);
919}
920
921VBOXVREGDECL(void) CrVrScrCompositorVisit(PVBOXVR_SCR_COMPOSITOR pCompositor, PFNVBOXVRSCRCOMPOSITOR_VISITOR pfnVisitor, void *pvVisitor)
922{
923 VBOXVR_SCR_COMPOSITOR_VISITOR_CB Data;
924 Data.pfnVisitor = pfnVisitor;
925 Data.pvVisitor = pvVisitor;
926 VBoxVrCompositorVisit(&pCompositor->Compositor, crVrScrCompositorVisitCb, &Data);
927}
928
929VBOXVREGDECL(int) CrVrScrCompositorClone(const VBOXVR_SCR_COMPOSITOR *pCompositor, PVBOXVR_SCR_COMPOSITOR pDstCompositor, PFNVBOXVR_SCR_COMPOSITOR_ENTRY_FOR pfnEntryFor, void* pvEntryFor)
930{
931 /* for simplicity just copy from one to another */
932 CrVrScrCompositorInit(pDstCompositor, CrVrScrCompositorRectGet(pCompositor));
933 VBOXVR_SCR_COMPOSITOR_CONST_ITERATOR CIter;
934 const VBOXVR_SCR_COMPOSITOR_ENTRY *pEntry;
935 CrVrScrCompositorConstIterInit(pCompositor, &CIter);
936 int rc = VINF_SUCCESS;
937 uint32_t cRects;
938 const RTRECT *pRects;
939
940 while ((pEntry = CrVrScrCompositorConstIterNext(&CIter)) != NULL)
941 {
942 /* get source rects, that will be non-stretched and entry pos - pased */
943 rc = CrVrScrCompositorEntryRegionsGet(pCompositor, pEntry, &cRects, NULL, NULL, &pRects);
944 if (!RT_SUCCESS(rc))
945 {
946 WARN(("CrVrScrCompositorEntryRegionsGet failed, rc %d", rc));
947 return rc;
948 }
949
950 PVBOXVR_SCR_COMPOSITOR_ENTRY pDstEntry = pfnEntryFor(pEntry, pvEntryFor);
951 if (!pDstEntry)
952 {
953 WARN(("pfnEntryFor failed"));
954 return VERR_INVALID_STATE;
955 }
956
957 rc = CrVrScrCompositorEntryRegionsSet(pDstCompositor, pDstEntry, NULL, cRects, pRects, false, NULL);
958 if (!RT_SUCCESS(rc))
959 {
960 WARN(("CrVrScrCompositorEntryRegionsSet failed, rc %d", rc));
961 return rc;
962 }
963 }
964
965 return rc;
966}
967
968VBOXVREGDECL(int) CrVrScrCompositorIntersectList(PVBOXVR_SCR_COMPOSITOR pCompositor, const VBOXVR_LIST *pVr, bool *pfChanged)
969{
970 VBOXVR_SCR_COMPOSITOR_ITERATOR CIter;
971 PVBOXVR_SCR_COMPOSITOR_ENTRY pEntry;
972 CrVrScrCompositorIterInit(pCompositor, &CIter);
973 int rc = VINF_SUCCESS;
974 bool fChanged = false;
975
976 while ((pEntry = CrVrScrCompositorIterNext(&CIter)) != NULL)
977 {
978 bool fCurChanged = false;
979
980 rc = CrVrScrCompositorEntryListIntersect(pCompositor, pEntry, pVr, &fCurChanged);
981 if (!RT_SUCCESS(rc))
982 {
983 WARN(("CrVrScrCompositorEntryRegionsSet failed, rc %d", rc));
984 break;
985 }
986
987 fChanged |= fCurChanged;
988 }
989
990 if (pfChanged)
991 *pfChanged = fChanged;
992
993 return rc;
994}
995
996VBOXVREGDECL(int) CrVrScrCompositorIntersectedList(const VBOXVR_SCR_COMPOSITOR *pCompositor, const VBOXVR_LIST *pVr, PVBOXVR_SCR_COMPOSITOR pDstCompositor, PFNVBOXVR_SCR_COMPOSITOR_ENTRY_FOR pfnEntryFor, void* pvEntryFor, bool *pfChanged)
997{
998 int rc = CrVrScrCompositorClone(pCompositor, pDstCompositor, pfnEntryFor, pvEntryFor);
999 if (!RT_SUCCESS(rc))
1000 {
1001 WARN(("CrVrScrCompositorClone failed, rc %d", rc));
1002 return rc;
1003 }
1004
1005 rc = CrVrScrCompositorIntersectList(pDstCompositor, pVr, pfChanged);
1006 if (!RT_SUCCESS(rc))
1007 {
1008 WARN(("CrVrScrCompositorIntersectList failed, rc %d", rc));
1009 CrVrScrCompositorClear(pDstCompositor);
1010 return rc;
1011 }
1012
1013 return VINF_SUCCESS;
1014}
1015
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