VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedOpenGL/crserverlib/server_presenter.cpp@ 45577

Last change on this file since 45577 was 45132, checked in by vboxsync, 12 years ago

crOpenGL: seamles mode support impl; bugfizes & cleanup

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.8 KB
Line 
1/* $Id: server_presenter.cpp 45132 2013-03-21 16:11:28Z vboxsync $ */
2
3/** @file
4 * Presenter API
5 */
6
7/*
8 * Copyright (C) 2012-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_spu.h"
19#include "chromium.h"
20#include "cr_error.h"
21#include "cr_net.h"
22#include "cr_rand.h"
23#include "server_dispatch.h"
24#include "server.h"
25#include "cr_mem.h"
26#include "cr_string.h"
27#include <cr_vreg.h>
28
29#include <iprt/cdefs.h>
30#include <iprt/types.h>
31#include <iprt/asm.h>
32#include <iprt/mem.h>
33
34
35/* DISPLAY */
36
37//static DECLCALLBACK(int) crDpCbRegionsChanged(struct CR_PRESENTER *pPresenter)
38//{
39// uint32_t cRegions;
40// const RTRECT *paRegions;
41// int rc = CrPtGetRegions(pPresenter, &cRegions, &paRegions);
42// if (!RT_SUCCESS(rc))
43// {
44// crWarning("CrPtGetRegions failed, rc %d", rc);
45// return rc;
46// }
47//
48// PCR_DISPLAY pDisplay = CR_DISPLAY_FROM_PRESENTER(pPresenter);
49//
50// cr_server.head_spu->dispatch_table.WindowVisibleRegion(pDisplay->Mural.spuWindow, cRegions, (GLint*)paRegions);
51//
52// if (pDisplay->Mural.pvOutputRedirectInstance)
53// {
54// /* @todo the code assumes that RTRECT == four GLInts. */
55// cr_server.outputRedirect.CRORVisibleRegion(pDisplay->Mural.pvOutputRedirectInstance,
56// cRegions, paRegions);
57// }
58//
59// return VINF_SUCCESS;
60//}
61
62int CrDpInit(PCR_DISPLAY pDisplay)
63{
64 int rc = CrVrScrCompositorInit(&pDisplay->Compositor);
65 if (RT_SUCCESS(rc))
66 {
67 const GLint visBits = CR_RGB_BIT | CR_DOUBLE_BIT;
68 if (crServerMuralInit(&pDisplay->Mural, "", visBits, -1) >= 0)
69 {
70 return VINF_SUCCESS;
71// crServerMuralTerm(&pDisplay->Mural);
72 }
73 else
74 {
75 crWarning("crServerMuralInit failed!");
76 rc = VERR_GENERAL_FAILURE;
77 }
78 CrVrScrCompositorTerm(&pDisplay->Compositor);
79 }
80 else
81 {
82 crWarning("CrVrScrCompositorInit failed, rc %d", rc);
83 }
84 CRASSERT(RT_FAILURE(rc));
85 return rc;
86}
87
88void CrDpTerm(PCR_DISPLAY pDisplay)
89{
90 CrVrScrCompositorTerm(&pDisplay->Compositor);
91 crServerMuralTerm(&pDisplay->Mural);
92}
93
94void CrDpResize(PCR_DISPLAY pDisplay, uint32_t width, uint32_t height,
95 uint32_t stretchedWidth, uint32_t stretchedHeight)
96{
97 float StretchX, StretchY;
98 StretchX = ((float)stretchedWidth)/width;
99 StretchY = ((float)stretchedHeight)/height;
100 crServerMuralSize(&pDisplay->Mural, stretchedWidth, stretchedHeight);
101 CrVrScrCompositorLock(&pDisplay->Compositor);
102 CrVrScrCompositorSetStretching(&pDisplay->Compositor, StretchX, StretchY);
103 CrVrScrCompositorUnlock(&pDisplay->Compositor);
104}
105
106int CrDpEntryRegionsSet(PCR_DISPLAY pDisplay, PCR_DISPLAY_ENTRY pEntry, const RTPOINT *pPos, uint32_t cRegions, const RTRECT *paRegions)
107{
108 CrVrScrCompositorLock(&pDisplay->Compositor);
109 int rc = CrVrScrCompositorEntryRegionsSet(&pDisplay->Compositor, &pEntry->CEntry, pPos, cRegions, paRegions, NULL);
110 CrVrScrCompositorUnlock(&pDisplay->Compositor);
111 return rc;
112}
113
114int CrDpEntryRegionsAdd(PCR_DISPLAY pDisplay, PCR_DISPLAY_ENTRY pEntry, const RTPOINT *pPos, uint32_t cRegions, const RTRECT *paRegions)
115{
116 CrVrScrCompositorLock(&pDisplay->Compositor);
117 int rc = CrVrScrCompositorEntryRegionsAdd(&pDisplay->Compositor, &pEntry->CEntry, pPos, cRegions, paRegions, NULL);
118 CrVrScrCompositorUnlock(&pDisplay->Compositor);
119 return rc;
120}
121
122int CrDpPresentEntry(PCR_DISPLAY pDisplay, PCR_DISPLAY_ENTRY pEntry)
123{
124 GLuint idDrawFBO = 0, idReadFBO = 0;
125 CRMuralInfo *pMural = cr_server.currentMural;
126 CRContext *pCtx = cr_server.currentCtxInfo ? cr_server.currentCtxInfo->pContext : cr_server.MainContextInfo.pContext;
127
128 if (pMural)
129 {
130 idDrawFBO = pMural->aidFBOs[pMural->iCurDrawBuffer];
131 idReadFBO = pMural->aidFBOs[pMural->iCurReadBuffer];
132 }
133
134 crStateSwitchPrepare(NULL, pCtx, idDrawFBO, idReadFBO);
135
136 cr_server.head_spu->dispatch_table.VBoxPresentComposition(pDisplay->Mural.spuWindow, &pDisplay->Compositor, &pEntry->CEntry);
137
138 crStateSwitchPostprocess(pCtx, NULL, idDrawFBO, idReadFBO);
139
140 return VINF_SUCCESS;
141}
142
143void CrDpEntryInit(PCR_DISPLAY_ENTRY pEntry, const VBOXVR_TEXTURE *pTextureData)
144{
145 CrVrScrCompositorEntryInit(&pEntry->CEntry, pTextureData);
146}
147
148void CrDpEntryCleanup(PCR_DISPLAY pDisplay, PCR_DISPLAY_ENTRY pEntry)
149{
150 CrVrScrCompositorLock(&pDisplay->Compositor);
151 CrVrScrCompositorEntryRemove(&pDisplay->Compositor, &pEntry->CEntry);
152 CrVrScrCompositorUnlock(&pDisplay->Compositor);
153}
154
155int CrDemInit(PCR_DISPLAY_ENTRY_MAP pMap)
156{
157 pMap->pTextureMap = crAllocHashtable();
158 if (pMap->pTextureMap)
159 return VINF_SUCCESS;
160
161 crWarning("crAllocHashtable failed!");
162 return VERR_NO_MEMORY;
163}
164
165void CrDemTerm(PCR_DISPLAY_ENTRY_MAP pMap)
166{
167 crFreeHashtable(pMap->pTextureMap, crFree);
168}
169
170PCR_DISPLAY_ENTRY CrDemEntryGetCreate(PCR_DISPLAY_ENTRY_MAP pMap, GLuint idTexture, CRContextInfo *pCtxInfo)
171{
172 PCR_DISPLAY_ENTRY pEntry = (PCR_DISPLAY_ENTRY)crHashtableSearch(pMap->pTextureMap, idTexture);
173 if (pEntry)
174 return pEntry;
175
176 CRContext *pContext = pCtxInfo->pContext;
177 if (!pContext)
178 {
179 crWarning("pContext is null!");
180 return NULL;
181 }
182
183 CRTextureObj *pTobj = (CRTextureObj*)crHashtableSearch(pContext->shared->textureTable, idTexture);
184 if (!pTobj)
185 {
186 crWarning("pTobj is null!");
187 return NULL;
188 }
189
190 GLuint hwId = crStateGetTextureObjHWID(pTobj);
191 if (!hwId)
192 {
193 crWarning("hwId is null!");
194 return NULL;
195 }
196
197 VBOXVR_TEXTURE TextureData;
198 TextureData.width = pTobj->level[0]->width;
199 TextureData.height = pTobj->level[0]->height;
200 TextureData.target = pTobj->target;
201 TextureData.hwid = hwId;
202
203 pEntry = (PCR_DISPLAY_ENTRY)crAlloc(sizeof (*pEntry));
204 if (!pEntry)
205 {
206 crWarning("crAlloc failed allocating CR_DISPLAY_ENTRY");
207 return NULL;
208 }
209
210 CrDpEntryInit(pEntry, &TextureData);
211
212 crHashtableAdd(pMap->pTextureMap, idTexture, pEntry);
213 return pEntry;
214
215}
216
217void CrDemEntryDestroy(PCR_DISPLAY_ENTRY_MAP pMap, GLuint idTexture)
218{
219#ifdef DEBUG
220 {
221 PCR_DISPLAY_ENTRY pEntry = (PCR_DISPLAY_ENTRY)crHashtableSearch(pMap->pTextureMap, idTexture);
222 if (!pEntry)
223 {
224 crWarning("request to delete inexistent entry");
225 return;
226 }
227
228 Assert(!CrDpEntryIsUsed(pEntry));
229 }
230#endif
231 crHashtableDelete(pMap->pTextureMap, idTexture, crFree);
232}
233
234#define CR_PRESENT_SCREEN_MASK 0xffff
235#define CR_PRESENT_FLAGS_OFFSET 16
236
237#define CR_PRESENT_GET_SCREEN(_cfg) ((_cfg) & CR_PRESENT_SCREEN_MASK)
238#define CR_PRESENT_GET_FLAGS(_cfg) ((_cfg) >> CR_PRESENT_FLAGS_OFFSET)
239
240PCR_DISPLAY crServerDisplayGetInitialized(uint32_t idScreen)
241{
242 if (ASMBitTest(cr_server.DisplaysInitMap, idScreen))
243 return &cr_server.aDispplays[idScreen];
244 return NULL;
245}
246
247static PCR_DISPLAY crServerDisplayGet(uint32_t idScreen)
248{
249 if (idScreen >= CR_MAX_GUEST_MONITORS)
250 {
251 crWarning("invalid idScreen %d", idScreen);
252 return NULL;
253 }
254
255 if (ASMBitTest(cr_server.DisplaysInitMap, idScreen))
256 return &cr_server.aDispplays[idScreen];
257
258 /* the display (screen id == 0) can be initialized while doing crServerCheckInitDisplayBlitter,
259 * so re-check the bit map */
260 if (ASMBitTest(cr_server.DisplaysInitMap, idScreen))
261 return &cr_server.aDispplays[idScreen];
262
263 int rc = CrDpInit(&cr_server.aDispplays[idScreen]);
264 if (RT_SUCCESS(rc))
265 {
266 CrDpResize(&cr_server.aDispplays[idScreen],
267 cr_server.screen[idScreen].w, cr_server.screen[idScreen].h,
268 cr_server.screen[idScreen].w, cr_server.screen[idScreen].h);
269 ASMBitSet(cr_server.DisplaysInitMap, idScreen);
270 return &cr_server.aDispplays[idScreen];
271 }
272 else
273 {
274 crWarning("CrDpInit failed for screen %d", idScreen);
275 }
276
277 return NULL;
278}
279
280void CrHlpFreeTexImage(CRContext *pCurCtx, GLuint idPBO, void *pvData)
281{
282 if (idPBO)
283 {
284 cr_server.head_spu->dispatch_table.UnmapBufferARB(GL_PIXEL_PACK_BUFFER_ARB);
285 if (pCurCtx)
286 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, pCurCtx->bufferobject.packBuffer->hwid);
287 else
288 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, 0);
289 }
290 else
291 {
292 crFree(pvData);
293 if (pCurCtx && crStateIsBufferBoundForCtx(pCurCtx, GL_PIXEL_PACK_BUFFER_ARB))
294 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, pCurCtx->bufferobject.packBuffer->hwid);
295 }
296}
297
298void CrHlpPutTexImage(CRContext *pCurCtx, PVBOXVR_TEXTURE pTexture, GLenum enmFormat, void *pvData)
299{
300 CRASSERT(pTexture->hwid);
301 cr_server.head_spu->dispatch_table.BindTexture(pTexture->target, pTexture->hwid);
302
303 if (!pCurCtx || crStateIsBufferBoundForCtx(pCurCtx, GL_PIXEL_UNPACK_BUFFER_ARB))
304 {
305 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_UNPACK_BUFFER_ARB, 0);
306 }
307
308 /*read the texture, note pixels are NULL for PBO case as it's offset in the buffer*/
309 cr_server.head_spu->dispatch_table.TexSubImage2D(GL_TEXTURE_2D, 0 /* level*/, 0 /*xoffset*/, 0 /*yoffset*/, pTexture->width, pTexture->height, enmFormat, GL_UNSIGNED_BYTE, pvData);
310
311 /*restore gl state*/
312 if (pCurCtx)
313 {
314 CRTextureObj *pTObj;
315 CRTextureLevel *pTImg;
316 crStateGetTextureObjectAndImage(pCurCtx, pTexture->target, 0, &pTObj, &pTImg);
317
318 GLuint uid = pTObj->hwid;
319 CRASSERT(uid);
320 cr_server.head_spu->dispatch_table.BindTexture(pTexture->target, uid);
321 }
322 else
323 {
324 cr_server.head_spu->dispatch_table.BindTexture(pTexture->target, 0);
325 }
326
327 if (pCurCtx && crStateIsBufferBoundForCtx(pCurCtx, GL_PIXEL_UNPACK_BUFFER_ARB))
328 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_UNPACK_BUFFER_ARB, pCurCtx->bufferobject.unpackBuffer->hwid);
329}
330
331void* CrHlpGetTexImage(CRContext *pCurCtx, PVBOXVR_TEXTURE pTexture, GLuint idPBO, GLenum enmFormat)
332{
333 void *pvData = NULL;
334 cr_server.head_spu->dispatch_table.BindTexture(pTexture->target, pTexture->hwid);
335
336 if (idPBO)
337 {
338 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, idPBO);
339 }
340 else
341 {
342 if (!pCurCtx || crStateIsBufferBoundForCtx(pCurCtx, GL_PIXEL_PACK_BUFFER_ARB))
343 {
344 cr_server.head_spu->dispatch_table.BindBufferARB(GL_PIXEL_PACK_BUFFER_ARB, 0);
345 }
346
347 pvData = crAlloc(4*pTexture->width*pTexture->height);
348 if (!pvData)
349 {
350 crWarning("Out of memory in CrHlpGetTexImage");
351 return NULL;
352 }
353 }
354
355 /*read the texture, note pixels are NULL for PBO case as it's offset in the buffer*/
356 cr_server.head_spu->dispatch_table.GetTexImage(GL_TEXTURE_2D, 0, enmFormat, GL_UNSIGNED_BYTE, pvData);
357
358 /*restore gl state*/
359 if (pCurCtx)
360 {
361 CRTextureObj *pTObj;
362 CRTextureLevel *pTImg;
363 crStateGetTextureObjectAndImage(pCurCtx, pTexture->target, 0, &pTObj, &pTImg);
364
365 GLuint uid = pTObj->hwid;
366 CRASSERT(uid);
367 cr_server.head_spu->dispatch_table.BindTexture(pTexture->target, uid);
368 }
369 else
370 {
371 cr_server.head_spu->dispatch_table.BindTexture(pTexture->target, 0);
372 }
373
374 if (idPBO)
375 {
376 pvData = cr_server.head_spu->dispatch_table.MapBufferARB(GL_PIXEL_PACK_BUFFER_ARB, GL_READ_ONLY);
377 if (!pvData)
378 {
379 crWarning("Failed to MapBuffer in CrHlpGetTexImage");
380 return NULL;
381 }
382 }
383
384 CRASSERT(pvData);
385 return pvData;
386}
387
388void SERVER_DISPATCH_APIENTRY
389crServerDispatchVBoxTexPresent(GLuint texture, GLuint cfg, GLint xPos, GLint yPos, GLint cRects, const GLint *pRects)
390{
391 uint32_t idScreen = CR_PRESENT_GET_SCREEN(cfg);
392 PCR_DISPLAY pDisplay = crServerDisplayGet(idScreen);
393 if (!pDisplay)
394 {
395 crWarning("crServerDisplayGet Failed");
396 return;
397 }
398
399 PCR_DISPLAY_ENTRY pEntry = CrDemEntryGetCreate(&cr_server.PresentTexturepMap, texture, cr_server.currentCtxInfo);
400 if (!pEntry)
401 {
402 crWarning("CrDemEntryGetCreate Failed");
403 return;
404 }
405
406 RTPOINT Point = {xPos, yPos};
407 int rc = CrDpEntryRegionsAdd(pDisplay, pEntry, &Point, (uint32_t)cRects, (const RTRECT*)pRects);
408 if (!RT_SUCCESS(rc))
409 {
410 crWarning("CrDpEntrySetRegions Failed rc %d", rc);
411 return;
412 }
413
414 rc = CrDpPresentEntry(pDisplay, pEntry);
415 if (!RT_SUCCESS(rc))
416 {
417 crWarning("CrDpEntrySetRegions Failed rc %d", rc);
418 return;
419 }
420}
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