VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedOpenGL/crserverlib/server_muralfbo.c@ 45579

Last change on this file since 45579 was 45348, checked in by vboxsync, 12 years ago

crOpenGL: improved GPU data acwuisition mechanism for VRDP (disabled still)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 32.9 KB
Line 
1/* $Id: server_muralfbo.c 45348 2013-04-04 19:46:29Z vboxsync $ */
2
3/** @file
4 * VBox crOpenGL: Window to FBO redirect support.
5 */
6
7/*
8 * Copyright (C) 2010-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
19#include "server.h"
20#include "cr_string.h"
21#include "cr_mem.h"
22#include "cr_vreg.h"
23#include "render/renderspu.h"
24
25static int crServerGetPointScreen(GLint x, GLint y)
26{
27 int i;
28
29 for (i=0; i<cr_server.screenCount; ++i)
30 {
31 if ((x>=cr_server.screen[i].x && x<cr_server.screen[i].x+(int)cr_server.screen[i].w)
32 && (y>=cr_server.screen[i].y && y<cr_server.screen[i].y+(int)cr_server.screen[i].h))
33 {
34 return i;
35 }
36 }
37
38 return -1;
39}
40
41static GLboolean crServerMuralCoverScreen(CRMuralInfo *mural, int sId)
42{
43 return mural->gX < cr_server.screen[sId].x
44 && mural->gX+(int)mural->width > cr_server.screen[sId].x+(int)cr_server.screen[sId].w
45 && mural->gY < cr_server.screen[sId].y
46 && mural->gY+(int)mural->height > cr_server.screen[sId].y+(int)cr_server.screen[sId].h;
47}
48
49/* Called when a new CRMuralInfo is created
50 * or when OutputRedirect status is changed.
51 */
52void crServerSetupOutputRedirect(CRMuralInfo *mural)
53{
54 /* Unset the previous redirect. */
55 if (mural->pvOutputRedirectInstance)
56 {
57 cr_server.outputRedirect.CROREnd(mural->pvOutputRedirectInstance);
58 mural->pvOutputRedirectInstance = NULL;
59 }
60
61 /* Setup a new redirect. */
62 if (cr_server.bUseOutputRedirect)
63 {
64 /* Query supported formats. */
65 uint32_t cbFormats = 4096;
66 char *pachFormats = (char *)crAlloc(cbFormats);
67
68 if (pachFormats)
69 {
70 int rc = cr_server.outputRedirect.CRORContextProperty(cr_server.outputRedirect.pvContext,
71 0 /* H3DOR_PROP_FORMATS */, // @todo from a header
72 pachFormats, cbFormats, &cbFormats);
73 if (RT_SUCCESS(rc))
74 {
75 if (RTStrStr(pachFormats, "H3DOR_FMT_RGBA_TOPDOWN"))
76 {
77 cr_server.outputRedirect.CRORBegin(cr_server.outputRedirect.pvContext,
78 &mural->pvOutputRedirectInstance,
79 "H3DOR_FMT_RGBA_TOPDOWN"); // @todo from a header
80 }
81 }
82
83 crFree(pachFormats);
84 }
85
86 /* If this is not NULL then there was a supported format. */
87 if (mural->pvOutputRedirectInstance)
88 {
89 uint32_t cRects;
90 const RTRECT *pRects;
91
92 int rc = CrVrScrCompositorEntryRegionsGet(&mural->Compositor, &mural->CEntry, &cRects, NULL, &pRects);
93 if (!RT_SUCCESS(rc))
94 {
95 crWarning("CrVrScrCompositorEntryRegionsGet failed, rc %d", rc);
96 return;
97 }
98
99 cr_server.outputRedirect.CRORGeometry(mural->pvOutputRedirectInstance,
100 mural->hX, mural->hY,
101 mural->width, mural->height);
102 // @todo the code assumes that RTRECT == four of GLInts
103 cr_server.outputRedirect.CRORVisibleRegion(mural->pvOutputRedirectInstance,
104 mural->cVisibleRects, (RTRECT *)mural->pVisibleRects);
105 }
106 }
107}
108
109void crServerCheckMuralGeometry(CRMuralInfo *mural)
110{
111 int tlS, brS, trS, blS;
112 int overlappingScreenCount = 0, primaryS = -1 , i;
113 uint64_t winID = 0;
114 GLuint fPresentMode;
115
116 if (!mural->CreateInfo.externalID)
117 return;
118
119 CRASSERT(mural->spuWindow);
120 CRASSERT(mural->spuWindow != CR_RENDER_DEFAULT_WINDOW_ID);
121
122 if (!mural->width || !mural->height)
123 {
124 crServerRedirMuralFBO(mural, CR_SERVER_REDIR_F_NONE);
125 crServerDeleteMuralFBO(mural);
126 return;
127 }
128
129 crServerVBoxCompositionDisableEnter(mural);
130
131 tlS = crServerGetPointScreen(mural->gX, mural->gY);
132 brS = crServerGetPointScreen(mural->gX+mural->width-1, mural->gY+mural->height-1);
133
134 if ((tlS==brS && tlS>=0) || cr_server.screenCount <= 1)
135 {
136 if (cr_server.screenCount <= 1)
137 {
138 if (tlS != brS)
139 {
140 if (tlS >= 0)
141 brS = tlS;
142 else
143 tlS = brS;
144 }
145
146 primaryS = 0;
147 }
148
149 Assert(brS == tlS);
150
151 if (tlS>=0 && cr_server.screen[tlS].winID)
152 {
153 overlappingScreenCount = 1;
154 }
155 }
156 else
157 {
158 bool fFoundWindIdScreen = false;
159 trS = crServerGetPointScreen(mural->gX+mural->width-1, mural->gY);
160 blS = crServerGetPointScreen(mural->gX, mural->gY+mural->height-1);
161
162 primaryS = -1; overlappingScreenCount = 0;
163 for (i=0; i<cr_server.screenCount; ++i)
164 {
165 if ((i==tlS) || (i==brS) || (i==trS) || (i==blS)
166 || crServerMuralCoverScreen(mural, i))
167 {
168 if ((!fFoundWindIdScreen && cr_server.screen[i].winID) || primaryS<0)
169 primaryS = i;
170
171 if (cr_server.screen[i].winID)
172 {
173 overlappingScreenCount++;
174 fFoundWindIdScreen = true;
175 }
176 }
177 }
178
179 if (primaryS<0)
180 {
181 primaryS = 0;
182 }
183 }
184
185 winID = overlappingScreenCount ? cr_server.screen[primaryS].winID : 0;
186
187 if (!winID == !mural->fHasParentWindow
188 || (winID && primaryS!=mural->screenId))
189 {
190 mural->fHasParentWindow = !!winID;
191
192 renderspuSetWindowId(winID);
193 renderspuReparentWindow(mural->spuWindow);
194 renderspuSetWindowId(cr_server.screen[0].winID);
195 }
196
197 mural->screenId = primaryS;
198
199 mural->hX = mural->gX-cr_server.screen[primaryS].x;
200 mural->hY = mural->gY-cr_server.screen[primaryS].y;
201
202 fPresentMode = cr_server.fPresentMode;
203 if (!overlappingScreenCount)
204 fPresentMode &= ~CR_SERVER_REDIR_F_DISPLAY;
205 else if (overlappingScreenCount > 1)
206 fPresentMode = (fPresentMode | CR_SERVER_REDIR_F_FBO_RAM_VMFB | cr_server.fVramPresentModeDefault) & ~CR_SERVER_REDIR_F_DISPLAY;
207
208 fPresentMode = crServerRedirModeAdjust(fPresentMode);
209
210 if (!(fPresentMode & CR_SERVER_REDIR_F_FBO))
211 {
212 crServerRedirMuralFBO(mural, fPresentMode);
213 crServerDeleteMuralFBO(mural);
214 }
215 else
216 {
217 if (mural->fPresentMode & CR_SERVER_REDIR_F_FBO)
218 {
219 if (mural->width!=mural->fboWidth
220 || mural->height!=mural->fboHeight)
221 {
222 crServerRedirMuralFBO(mural, fPresentMode & CR_SERVER_REDIR_F_DISPLAY);
223 crServerDeleteMuralFBO(mural);
224 }
225 }
226
227 crServerRedirMuralFBO(mural, fPresentMode);
228 }
229
230 if (mural->fPresentMode & CR_SERVER_REDIR_F_DISPLAY)
231 {
232 CRScreenViewportInfo *pVieport = &cr_server.screenVieport[mural->screenId];
233
234 cr_server.head_spu->dispatch_table.WindowPosition(mural->spuWindow, mural->hX - pVieport->x, mural->hY - pVieport->y);
235 }
236
237 if (mural->pvOutputRedirectInstance)
238 {
239 cr_server.outputRedirect.CRORGeometry(mural->pvOutputRedirectInstance,
240 mural->hX, mural->hY,
241 mural->width, mural->height);
242 }
243
244 crServerVBoxCompositionDisableLeave(mural, GL_FALSE);
245}
246
247GLboolean crServerSupportRedirMuralFBO(void)
248{
249 static GLboolean fInited = GL_FALSE;
250 static GLboolean fSupported = GL_FALSE;
251 if (!fInited)
252 {
253 const GLubyte* pExt = cr_server.head_spu->dispatch_table.GetString(GL_REAL_EXTENSIONS);
254
255 fSupported = ( NULL!=crStrstr((const char*)pExt, "GL_ARB_framebuffer_object")
256 || NULL!=crStrstr((const char*)pExt, "GL_EXT_framebuffer_object"))
257 && NULL!=crStrstr((const char*)pExt, "GL_ARB_texture_non_power_of_two");
258 fInited = GL_TRUE;
259 }
260 return fSupported;
261}
262
263static void crServerPresentMuralVRAM(CRMuralInfo *mural, char *pixels);
264
265#define CR_SERVER_MURAL_FROM_RPW_ENTRY(_pEntry) ((CRMuralInfo*)(((uint8_t*)(_pEntry)) - RT_OFFSETOF(CRMuralInfo, RpwEntry)))
266
267static DECLCALLBACK(void) crServerMuralRpwDataCB(const struct CR_SERVER_RPW_ENTRY* pEntry, void *pvEntryTexData)
268{
269 CRMuralInfo *pMural = CR_SERVER_MURAL_FROM_RPW_ENTRY(pEntry);
270
271 Assert(&pMural->RpwEntry == pEntry);
272
273 crServerPresentMuralVRAM(pMural, pvEntryTexData);
274}
275
276static void crServerCreateMuralFBO(CRMuralInfo *mural);
277
278static bool crServerEnableMuralRpw(CRMuralInfo *mural, GLboolean fEnable)
279{
280 if (!mural->CreateInfo.externalID)
281 {
282 crWarning("trying to change Rpw setting for internal mural %d", mural->spuWindow);
283 return !fEnable;
284 }
285
286 if (fEnable)
287 {
288 if (!(mural->fPresentMode & CR_SERVER_REDIR_F_FBO_RPW))
289 {
290 int rc;
291 if (!crServerRpwIsInitialized(&cr_server.RpwWorker))
292 {
293 rc = crServerRpwInit(&cr_server.RpwWorker);
294 if (!RT_SUCCESS(rc))
295 {
296 crWarning("crServerRpwInit failed rc %d", rc);
297 return false;
298 }
299 }
300
301 CRASSERT(!mural->RpwEntry.Size.cx);
302 CRASSERT(!mural->RpwEntry.Size.cy);
303
304 if (!crServerRpwEntryIsInitialized(&mural->RpwEntry))
305 {
306 rc = crServerRpwEntryInit(&cr_server.RpwWorker, &mural->RpwEntry, mural->width, mural->height, crServerMuralRpwDataCB);
307 if (!RT_SUCCESS(rc))
308 {
309 crWarning("crServerRpwEntryInit failed rc %d", rc);
310 return false;
311 }
312 }
313 else
314 {
315 rc = crServerRpwEntryResize(&cr_server.RpwWorker, &mural->RpwEntry, mural->width, mural->height);
316 if (!RT_SUCCESS(rc))
317 {
318 crWarning("crServerRpwEntryResize failed rc %d", rc);
319 return false;
320 }
321 }
322
323 mural->fPresentMode |= CR_SERVER_REDIR_F_FBO_RPW;
324 }
325 }
326 else
327 {
328 if ((mural->fPresentMode & CR_SERVER_REDIR_F_FBO_RPW))
329 {
330// crServerRpwEntryCleanup(&cr_server.RpwWorker, &mural->RpwEntry);
331 mural->fPresentMode &= ~CR_SERVER_REDIR_F_FBO_RPW;
332 }
333 }
334
335 return true;
336}
337
338static void crServerEnableDisplayMuralFBO(CRMuralInfo *mural, GLboolean fEnable)
339{
340 if (!mural->CreateInfo.externalID)
341 {
342 crWarning("trying to change display setting for internal mural %d", mural->spuWindow);
343 return;
344 }
345
346 if (fEnable)
347 {
348 if (!(mural->fPresentMode & CR_SERVER_REDIR_F_DISPLAY))
349 {
350 if (mural->bVisible)
351 cr_server.head_spu->dispatch_table.WindowShow(mural->spuWindow, GL_TRUE);
352 mural->fPresentMode |= CR_SERVER_REDIR_F_DISPLAY;
353 }
354 }
355 else
356 {
357 if ((mural->fPresentMode & CR_SERVER_REDIR_F_DISPLAY))
358 {
359 if (mural->bVisible)
360 cr_server.head_spu->dispatch_table.WindowShow(mural->spuWindow, GL_FALSE);
361 mural->fPresentMode &= ~CR_SERVER_REDIR_F_DISPLAY;
362 }
363 }
364}
365
366void crServerRedirMuralFBO(CRMuralInfo *mural, GLuint redir)
367{
368 if (!mural->CreateInfo.externalID)
369 {
370 crWarning("trying to change redir setting for internal mural %d", mural->spuWindow);
371 return;
372 }
373
374 if (mural->fPresentMode == redir)
375 {
376// if (redir)
377// crWarning("crServerRedirMuralFBO called with the same redir status %d", redir);
378 return;
379 }
380
381 crServerVBoxCompositionDisableEnter(mural);
382
383 if (redir & CR_SERVER_REDIR_F_FBO)
384 {
385 if (!crServerSupportRedirMuralFBO())
386 {
387 crWarning("FBO not supported, can't redirect window output");
388 goto end;
389 }
390
391 if (mural->aidFBOs[0]==0)
392 {
393 crServerCreateMuralFBO(mural);
394 }
395
396 if (cr_server.curClient && cr_server.curClient->currentMural == mural)
397 {
398 if (!crStateGetCurrent()->framebufferobject.drawFB)
399 {
400 cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_DRAW_FRAMEBUFFER, mural->aidFBOs[mural->iCurDrawBuffer]);
401 }
402 if (!crStateGetCurrent()->framebufferobject.readFB)
403 {
404 cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_READ_FRAMEBUFFER, mural->aidFBOs[mural->iCurReadBuffer]);
405 }
406
407 crStateGetCurrent()->buffer.width = 0;
408 crStateGetCurrent()->buffer.height = 0;
409 }
410 }
411 else
412 {
413 if (cr_server.curClient && cr_server.curClient->currentMural == mural)
414 {
415 if (!crStateGetCurrent()->framebufferobject.drawFB)
416 {
417 cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_DRAW_FRAMEBUFFER, 0);
418 }
419 if (!crStateGetCurrent()->framebufferobject.readFB)
420 {
421 cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_READ_FRAMEBUFFER, 0);
422 }
423
424 crStateGetCurrent()->buffer.width = mural->width;
425 crStateGetCurrent()->buffer.height = mural->height;
426 }
427 }
428
429 crServerEnableMuralRpw(mural, !!(redir & CR_SERVER_REDIR_F_FBO_RPW));
430
431 crServerEnableDisplayMuralFBO(mural, !!(redir & CR_SERVER_REDIR_F_DISPLAY));
432
433 mural->fPresentMode = redir;
434
435end:
436 crServerVBoxCompositionDisableLeave(mural, GL_FALSE);
437}
438
439static void crServerCreateMuralFBO(CRMuralInfo *mural)
440{
441 CRContext *ctx = crStateGetCurrent();
442 GLuint uid, i;
443 GLenum status;
444 SPUDispatchTable *gl = &cr_server.head_spu->dispatch_table;
445 CRContextInfo *pMuralContextInfo;
446
447 CRASSERT(mural->aidFBOs[0]==0);
448 CRASSERT(mural->aidFBOs[1]==0);
449 CRASSERT(mural->width == mural->CEntry.Tex.width);
450 CRASSERT(mural->height == mural->CEntry.Tex.height);
451
452 pMuralContextInfo = cr_server.currentCtxInfo;
453 if (!pMuralContextInfo)
454 {
455 /* happens on saved state load */
456 CRASSERT(cr_server.MainContextInfo.SpuContext);
457 pMuralContextInfo = &cr_server.MainContextInfo;
458 cr_server.head_spu->dispatch_table.MakeCurrent(mural->spuWindow, 0, cr_server.MainContextInfo.SpuContext);
459 }
460
461 if (pMuralContextInfo->CreateInfo.visualBits != mural->CreateInfo.visualBits)
462 {
463 crWarning("mural visual bits do not match with current context visual bits!");
464 }
465
466 mural->cBuffers = 2;
467 mural->iBbBuffer = 0;
468 /*Color texture*/
469
470 if (crStateIsBufferBound(GL_PIXEL_UNPACK_BUFFER_ARB))
471 {
472 gl->BindBufferARB(GL_PIXEL_UNPACK_BUFFER_ARB, 0);
473 }
474
475 for (i = 0; i < mural->cBuffers; ++i)
476 {
477 gl->GenTextures(1, &mural->aidColorTexs[i]);
478 gl->BindTexture(GL_TEXTURE_2D, mural->aidColorTexs[i]);
479 gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
480 gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
481 gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP);
482 gl->TexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP);
483 gl->TexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, mural->width, mural->height,
484 0, GL_BGRA, GL_UNSIGNED_BYTE, NULL);
485 }
486
487 /*Depth&Stencil*/
488 gl->GenRenderbuffersEXT(1, &mural->idDepthStencilRB);
489 gl->BindRenderbufferEXT(GL_RENDERBUFFER_EXT, mural->idDepthStencilRB);
490 gl->RenderbufferStorageEXT(GL_RENDERBUFFER_EXT, GL_DEPTH24_STENCIL8_EXT,
491 mural->width, mural->height);
492
493 /*FBO*/
494 for (i = 0; i < mural->cBuffers; ++i)
495 {
496 gl->GenFramebuffersEXT(1, &mural->aidFBOs[i]);
497 gl->BindFramebufferEXT(GL_FRAMEBUFFER_EXT, mural->aidFBOs[i]);
498
499 gl->FramebufferTexture2DEXT(GL_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT,
500 GL_TEXTURE_2D, mural->aidColorTexs[i], 0);
501 gl->FramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT,
502 GL_RENDERBUFFER_EXT, mural->idDepthStencilRB);
503 gl->FramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_STENCIL_ATTACHMENT_EXT,
504 GL_RENDERBUFFER_EXT, mural->idDepthStencilRB);
505
506 status = gl->CheckFramebufferStatusEXT(GL_FRAMEBUFFER_EXT);
507 if (status!=GL_FRAMEBUFFER_COMPLETE_EXT)
508 {
509 crWarning("FBO status(0x%x) isn't complete", status);
510 }
511 }
512
513 mural->fboWidth = mural->width;
514 mural->fboHeight = mural->height;
515
516 mural->iCurDrawBuffer = crServerMuralFBOIdxFromBufferName(mural, ctx->buffer.drawBuffer);
517 mural->iCurReadBuffer = crServerMuralFBOIdxFromBufferName(mural, ctx->buffer.readBuffer);
518
519 /*PBO*/
520 if (cr_server.bUsePBOForReadback)
521 {
522 gl->GenBuffersARB(1, &mural->idPBO);
523 gl->BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, mural->idPBO);
524 gl->BufferDataARB(GL_PIXEL_PACK_BUFFER_ARB, mural->width*mural->height*4, 0, GL_STREAM_READ_ARB);
525 gl->BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, ctx->bufferobject.packBuffer->hwid);
526
527 if (!mural->idPBO)
528 {
529 crWarning("PBO create failed");
530 }
531 }
532
533 /*Restore gl state*/
534 uid = ctx->texture.unit[ctx->texture.curTextureUnit].currentTexture2D->hwid;
535 gl->BindTexture(GL_TEXTURE_2D, uid);
536
537 uid = ctx->framebufferobject.renderbuffer ? ctx->framebufferobject.renderbuffer->hwid:0;
538 gl->BindRenderbufferEXT(GL_RENDERBUFFER_EXT, uid);
539
540 uid = ctx->framebufferobject.drawFB ? ctx->framebufferobject.drawFB->hwid:0;
541 gl->BindFramebufferEXT(GL_DRAW_FRAMEBUFFER, uid);
542
543 uid = ctx->framebufferobject.readFB ? ctx->framebufferobject.readFB->hwid:0;
544 gl->BindFramebufferEXT(GL_READ_FRAMEBUFFER, uid);
545
546 if (crStateIsBufferBound(GL_PIXEL_UNPACK_BUFFER_ARB))
547 {
548 gl->BindBufferARB(GL_PIXEL_UNPACK_BUFFER_ARB, ctx->bufferobject.unpackBuffer->hwid);
549 }
550
551 if (crStateIsBufferBound(GL_PIXEL_PACK_BUFFER_ARB))
552 {
553 gl->BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, ctx->bufferobject.packBuffer->hwid);
554 }
555 else
556 {
557 gl->BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, 0);
558 }
559
560 CRASSERT(mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)]);
561
562 CrVrScrCompositorEntryTexNameUpdate(&mural->CEntry, mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)]);
563
564 if (mural->fRootVrOn)
565 CrVrScrCompositorEntryTexNameUpdate(&mural->RootVrCEntry, mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)]);
566}
567
568void crServerDeleteMuralFBO(CRMuralInfo *mural)
569{
570 CRASSERT(!(mural->fPresentMode & CR_SERVER_REDIR_F_FBO));
571
572 if (mural->aidFBOs[0]!=0)
573 {
574 GLuint i;
575 for (i = 0; i < mural->cBuffers; ++i)
576 {
577 cr_server.head_spu->dispatch_table.DeleteTextures(1, &mural->aidColorTexs[i]);
578 mural->aidColorTexs[i] = 0;
579 }
580
581 cr_server.head_spu->dispatch_table.DeleteRenderbuffersEXT(1, &mural->idDepthStencilRB);
582 mural->idDepthStencilRB = 0;
583
584 for (i = 0; i < mural->cBuffers; ++i)
585 {
586 cr_server.head_spu->dispatch_table.DeleteFramebuffersEXT(1, &mural->aidFBOs[i]);
587 mural->aidFBOs[i] = 0;
588 }
589 }
590
591 if (mural->idPBO!=0)
592 {
593 CRASSERT(cr_server.bUsePBOForReadback);
594 cr_server.head_spu->dispatch_table.DeleteBuffersARB(1, &mural->idPBO);
595 mural->idPBO = 0;
596 }
597
598 mural->cBuffers = 0;
599
600 if (crServerRpwEntryIsInitialized(&mural->RpwEntry))
601 crServerRpwEntryCleanup(&cr_server.RpwWorker, &mural->RpwEntry);
602}
603
604#define MIN(a, b) ((a) < (b) ? (a) : (b))
605#define MAX(a, b) ((a) > (b) ? (a) : (b))
606
607static GLboolean crServerIntersectRect(CRrecti *a, CRrecti *b, CRrecti *rect)
608{
609 CRASSERT(a && b && rect);
610
611 rect->x1 = MAX(a->x1, b->x1);
612 rect->x2 = MIN(a->x2, b->x2);
613 rect->y1 = MAX(a->y1, b->y1);
614 rect->y2 = MIN(a->y2, b->y2);
615
616 return (rect->x2>rect->x1) && (rect->y2>rect->y1);
617}
618
619static GLboolean crServerIntersectScreen(CRMuralInfo *mural, int sId, CRrecti *rect)
620{
621 rect->x1 = MAX(mural->gX, cr_server.screen[sId].x);
622 rect->x2 = MIN(mural->gX+(int)mural->fboWidth, cr_server.screen[sId].x+(int)cr_server.screen[sId].w);
623 rect->y1 = MAX(mural->gY, cr_server.screen[sId].y);
624 rect->y2 = MIN(mural->gY+(int)mural->fboHeight, cr_server.screen[sId].y+(int)cr_server.screen[sId].h);
625
626 return (rect->x2>rect->x1) && (rect->y2>rect->y1);
627}
628
629static void crServerCopySubImage(char *pDst, char* pSrc, CRrecti *pRect, int srcWidth, int srcHeight)
630{
631 int i;
632 int dstrowsize = 4*(pRect->x2-pRect->x1);
633 int srcrowsize = 4*srcWidth;
634 int height = pRect->y2-pRect->y1;
635
636 pSrc += 4*pRect->x1 + srcrowsize*(srcHeight-1-pRect->y1);
637
638 for (i=0; i<height; ++i)
639 {
640 crMemcpy(pDst, pSrc, dstrowsize);
641
642 pSrc -= srcrowsize;
643 pDst += dstrowsize;
644 }
645}
646
647static void crServerTransformRect(CRrecti *pDst, CRrecti *pSrc, int dx, int dy)
648{
649 pDst->x1 = pSrc->x1+dx;
650 pDst->x2 = pSrc->x2+dx;
651 pDst->y1 = pSrc->y1+dy;
652 pDst->y2 = pSrc->y2+dy;
653}
654
655static void crServerVBoxCompositionPresentPerform(CRMuralInfo *mural)
656{
657 CRMuralInfo *currentMural = cr_server.currentMural;
658 CRContextInfo *curCtxInfo = cr_server.currentCtxInfo;
659 GLuint idDrawFBO, idReadFBO;
660 CRContext *curCtx = curCtxInfo ? curCtxInfo->pContext : NULL;
661
662 CRASSERT(curCtx == crStateGetCurrent());
663
664 Assert(mural->fPresentMode & CR_SERVER_REDIR_F_FBO);
665 Assert(mural->fPresentMode & CR_SERVER_REDIR_F_DISPLAY);
666
667 mural->fDataPresented = GL_TRUE;
668
669 if (currentMural)
670 {
671 idDrawFBO = currentMural->aidFBOs[currentMural->iCurDrawBuffer];
672 idReadFBO = currentMural->aidFBOs[currentMural->iCurReadBuffer];
673 }
674 else
675 {
676 idDrawFBO = 0;
677 idReadFBO = 0;
678 }
679
680 crStateSwitchPrepare(NULL, curCtx, idDrawFBO, idReadFBO);
681
682 if (!mural->fRootVrOn)
683 cr_server.head_spu->dispatch_table.VBoxPresentComposition(mural->spuWindow, &mural->Compositor, &mural->CEntry);
684 else
685 cr_server.head_spu->dispatch_table.VBoxPresentComposition(mural->spuWindow, &mural->RootVrCompositor, &mural->RootVrCEntry);
686
687 crStateSwitchPostprocess(curCtx, NULL, idDrawFBO, idReadFBO);
688}
689
690void crServerVBoxCompositionPresent(CRMuralInfo *mural)
691{
692 if (!crServerVBoxCompositionPresentNeeded(mural))
693 return;
694 crServerVBoxCompositionPresentPerform(mural);
695}
696
697static void crServerVBoxCompositionReenable(CRMuralInfo *mural, GLboolean fForcePresent)
698{
699 if ((mural->fPresentMode & (CR_SERVER_REDIR_F_FBO | CR_SERVER_REDIR_F_DISPLAY)) != (CR_SERVER_REDIR_F_FBO | CR_SERVER_REDIR_F_DISPLAY)
700 || !mural->fDataPresented
701 || (!fForcePresent
702 && !crServerVBoxCompositionPresentNeeded(mural)))
703 return;
704
705 crServerVBoxCompositionPresentPerform(mural);
706}
707
708static void crServerVBoxCompositionDisable(CRMuralInfo *mural)
709{
710 if ((mural->fPresentMode & (CR_SERVER_REDIR_F_FBO | CR_SERVER_REDIR_F_DISPLAY)) != (CR_SERVER_REDIR_F_FBO | CR_SERVER_REDIR_F_DISPLAY)
711 || !mural->fDataPresented)
712 return;
713 cr_server.head_spu->dispatch_table.VBoxPresentComposition(mural->spuWindow, NULL, NULL);
714}
715
716void crServerVBoxCompositionDisableEnter(CRMuralInfo *mural)
717{
718 ++mural->cDisabled;
719 Assert(mural->cDisabled);
720 if (mural->cDisabled == 1)
721 {
722 crServerVBoxCompositionDisable(mural);
723 }
724}
725
726void crServerVBoxCompositionDisableLeave(CRMuralInfo *mural, GLboolean fForcePresentOnEnabled)
727{
728 mural->fForcePresentState = fForcePresentOnEnabled;
729 --mural->cDisabled;
730 Assert(mural->cDisabled < UINT32_MAX/2);
731 if (!mural->cDisabled)
732 {
733 crServerVBoxCompositionReenable(mural, mural->fForcePresentState);
734 mural->fForcePresentState = GL_FALSE;
735 }
736}
737
738static void crServerVBoxCompositionSetEnableStateGlobalCB(unsigned long key, void *data1, void *data2)
739{
740 CRMuralInfo *mural = (CRMuralInfo *)data1;
741
742 if (data2)
743 crServerVBoxCompositionDisableLeave(mural, GL_FALSE);
744 else
745 crServerVBoxCompositionDisableEnter(mural);
746}
747
748DECLEXPORT(void) crServerVBoxCompositionSetEnableStateGlobal(GLboolean fEnable)
749{
750 crHashtableWalk(cr_server.muralTable, crServerVBoxCompositionSetEnableStateGlobalCB, (void*)fEnable);
751
752 crHashtableWalk(cr_server.dummyMuralTable, crServerVBoxCompositionSetEnableStateGlobalCB, (void*)fEnable);
753}
754
755static void crServerPresentMuralVRAM(CRMuralInfo *mural, char *pixels)
756{
757 char *tmppixels;
758 CRrecti rect, rectwr, sectr;
759 int i, j;
760
761 if (mural->fPresentMode & CR_SERVER_REDIR_F_FBO_RAM_VMFB)
762 {
763 for (i=0; i<cr_server.screenCount; ++i)
764 {
765 if (crServerIntersectScreen(mural, i, &rect))
766 {
767 /* rect in window relative coords */
768 crServerTransformRect(&rectwr, &rect, -mural->gX, -mural->gY);
769
770 if (!mural->pVisibleRects)
771 {
772 /*we don't get any rects info for guest compiz windows, so we treat windows as visible unless explicitly received 0 visible rects*/
773 if (!mural->bReceivedRects)
774 {
775 tmppixels = crAlloc(4*(rect.x2-rect.x1)*(rect.y2-rect.y1));
776 if (!tmppixels)
777 {
778 crWarning("Out of memory in crServerPresentFBO");
779 crFree(pixels);
780 return;
781 }
782
783 crServerCopySubImage(tmppixels, pixels, &rectwr, mural->fboWidth, mural->fboHeight);
784 /*Note: pfnPresentFBO would free tmppixels*/
785 cr_server.pfnPresentFBO(tmppixels, i, rect.x1-cr_server.screen[i].x, rect.y1-cr_server.screen[i].y, rect.x2-rect.x1, rect.y2-rect.y1);
786 }
787 }
788 else
789 {
790 for (j=0; j<mural->cVisibleRects; ++j)
791 {
792 if (crServerIntersectRect(&rectwr, (CRrecti*) &mural->pVisibleRects[4*j], &sectr))
793 {
794 tmppixels = crAlloc(4*(sectr.x2-sectr.x1)*(sectr.y2-sectr.y1));
795 if (!tmppixels)
796 {
797 crWarning("Out of memory in crServerPresentFBO");
798 crFree(pixels);
799 return;
800 }
801
802 crServerCopySubImage(tmppixels, pixels, &sectr, mural->fboWidth, mural->fboHeight);
803 /*Note: pfnPresentFBO would free tmppixels*/
804 cr_server.pfnPresentFBO(tmppixels, i,
805 sectr.x1+mural->gX-cr_server.screen[i].x,
806 sectr.y1+mural->gY-cr_server.screen[i].y,
807 sectr.x2-sectr.x1, sectr.y2-sectr.y1);
808 }
809 }
810 }
811 }
812 }
813 }
814
815 if (mural->pvOutputRedirectInstance)
816 {
817 /* @todo find out why presentfbo is not called but crorframe is called. */
818 cr_server.outputRedirect.CRORFrame(mural->pvOutputRedirectInstance,
819 pixels,
820 4 * mural->fboWidth * mural->fboHeight);
821 }
822}
823
824void crServerPresentFBO(CRMuralInfo *mural)
825{
826 char *pixels=NULL;
827 GLuint idPBO;
828 CRContext *ctx = crStateGetCurrent();
829 VBOXVR_TEXTURE Tex;
830
831 CRASSERT(mural->fPresentMode & CR_SERVER_REDIR_F_FBO);
832 CRASSERT(cr_server.pfnPresentFBO || (mural->fPresentMode & CR_SERVER_REDIR_F_DISPLAY));
833
834 if (!crServerVBoxCompositionPresentNeeded(mural))
835 return;
836
837 if (mural->fPresentMode & CR_SERVER_REDIR_F_DISPLAY)
838 {
839 crServerVBoxCompositionPresentPerform(mural);
840 }
841
842 mural->fDataPresented = GL_TRUE;
843
844 if (!(mural->fPresentMode & CR_SERVER_REDIR_FGROUP_REQUIRE_FBO_RAM))
845 return;
846
847 Tex.width = mural->width;
848 Tex.height = mural->height;
849 Tex.target = GL_TEXTURE_2D;
850 Tex.hwid = mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)];
851 CRASSERT(Tex.hwid);
852
853 if (mural->fPresentMode & CR_SERVER_REDIR_F_FBO_RPW)
854 {
855 /* 1. blit to RPW entry draw texture */
856 CRMuralInfo *pCurrentMural = cr_server.currentMural;
857 CRContextInfo *pCurCtxInfo = cr_server.currentCtxInfo;
858 PCR_BLITTER pBlitter = crServerVBoxBlitterGet();
859 CRMuralInfo *pBlitterMural;
860 CR_SERVER_CTX_SWITCH CtxSwitch;
861 RTRECT Rect;
862 VBOXVR_TEXTURE DstTex;
863 CR_BLITTER_WINDOW BlitterBltInfo, CurrentBltInfo;
864 CR_BLITTER_CONTEXT CtxBltInfo;
865 int rc;
866
867 Rect.xLeft = 0;
868 Rect.yTop = 0;
869 Rect.xRight = Tex.width;
870 Rect.yBottom = Tex.height;
871
872 if (pCurrentMural && pCurrentMural->CreateInfo.visualBits == CrBltGetVisBits(pBlitter))
873 {
874 pBlitterMural = pCurrentMural;
875 }
876 else
877 {
878 pBlitterMural = crServerGetDummyMural(pCurrentMural->CreateInfo.visualBits);
879 if (!pBlitterMural)
880 {
881 crWarning("crServerGetDummyMural failed for blitter mural");
882 return;
883 }
884 }
885
886 crServerRpwEntryDrawSettingsToTex(&mural->RpwEntry, &DstTex);
887
888 cr_serverCtxSwitchPrepare(&CtxSwitch, NULL);
889
890 crServerVBoxBlitterWinInit(&CurrentBltInfo, pCurrentMural);
891 crServerVBoxBlitterWinInit(&BlitterBltInfo, pBlitterMural);
892 crServerVBoxBlitterCtxInit(&CtxBltInfo, pCurCtxInfo);
893
894 CrBltMuralSetCurrent(pBlitter, &BlitterBltInfo);
895
896 rc = CrBltEnter(pBlitter, &CtxBltInfo, &CurrentBltInfo);
897 if (RT_SUCCESS(rc))
898 {
899 CrBltBlitTexTex(pBlitter, &Tex, &Rect, &DstTex, &Rect, 1, 0);
900 CrBltLeave(pBlitter);
901 }
902 else
903 {
904 crWarning("CrBltEnter failed rc %d", rc);
905 }
906
907 cr_serverCtxSwitchPostprocess(&CtxSwitch);
908
909 if (RT_SUCCESS(rc))
910 {
911 /* 2. submit RPW entry */
912 rc = crServerRpwEntrySubmit(&cr_server.RpwWorker, &mural->RpwEntry);
913 if (!RT_SUCCESS(rc))
914 {
915 crWarning("crServerRpwEntrySubmit failed rc %d", rc);
916 }
917 }
918
919 return;
920 }
921
922 if (cr_server.bUsePBOForReadback && !mural->idPBO)
923 {
924 crWarning("Mural doesn't have PBO even though bUsePBOForReadback is set!");
925 }
926
927 idPBO = cr_server.bUsePBOForReadback ? mural->idPBO : 0;
928 if (idPBO)
929 {
930 CRASSERT(mural->fboWidth == mural->width);
931 CRASSERT(mural->fboHeight == mural->height);
932 }
933
934 pixels = CrHlpGetTexImage(ctx, &Tex, idPBO, GL_BGRA);
935 if (!pixels)
936 {
937 crWarning("CrHlpGetTexImage failed in crServerPresentFBO");
938 return;
939 }
940
941 crServerPresentMuralVRAM(mural, pixels);
942
943 CrHlpFreeTexImage(ctx, idPBO, pixels);
944}
945
946GLboolean crServerIsRedirectedToFBO()
947{
948#ifdef DEBUG_misha
949 Assert(cr_server.curClient);
950 if (cr_server.curClient)
951 {
952 Assert(cr_server.curClient->currentMural == cr_server.currentMural);
953 Assert(cr_server.curClient->currentCtxInfo == cr_server.currentCtxInfo);
954 }
955#endif
956 return cr_server.curClient
957 && cr_server.curClient->currentMural
958 && (cr_server.curClient->currentMural->fPresentMode & CR_SERVER_REDIR_F_FBO);
959}
960
961GLuint crServerMuralFBOIdxFromBufferName(CRMuralInfo *mural, GLenum buffer)
962{
963 if (buffer == GL_FRONT
964 || buffer == GL_FRONT_LEFT
965 || buffer == GL_FRONT_RIGHT)
966 return CR_SERVER_FBO_FB_IDX(mural);
967 return CR_SERVER_FBO_BB_IDX(mural);
968}
969
970void crServerMuralFBOSwapBuffers(CRMuralInfo *mural)
971{
972 CRContext *ctx = crStateGetCurrent();
973 GLuint iOldCurDrawBuffer = mural->iCurDrawBuffer;
974 GLuint iOldCurReadBuffer = mural->iCurReadBuffer;
975 mural->iBbBuffer = ((mural->iBbBuffer + 1) % (mural->cBuffers));
976 mural->iCurDrawBuffer = ((mural->iCurDrawBuffer + 1) % (mural->cBuffers));
977 mural->iCurReadBuffer = ((mural->iCurReadBuffer + 1) % (mural->cBuffers));
978 Assert(iOldCurDrawBuffer != mural->iCurDrawBuffer || mural->cBuffers == 1);
979 Assert(iOldCurReadBuffer != mural->iCurReadBuffer || mural->cBuffers == 1);
980 if (!ctx->framebufferobject.drawFB && iOldCurDrawBuffer != mural->iCurDrawBuffer)
981 {
982 cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_DRAW_FRAMEBUFFER, mural->aidFBOs[mural->iCurDrawBuffer]);
983 }
984 if (!ctx->framebufferobject.readFB && iOldCurReadBuffer != mural->iCurReadBuffer)
985 {
986 cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_READ_FRAMEBUFFER, mural->aidFBOs[mural->iCurReadBuffer]);
987 }
988 Assert(mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)]);
989 CrVrScrCompositorEntryTexNameUpdate(&mural->CEntry, mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)]);
990 if (mural->fRootVrOn)
991 CrVrScrCompositorEntryTexNameUpdate(&mural->RootVrCEntry, mural->aidColorTexs[CR_SERVER_FBO_FB_IDX(mural)]);
992}
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