VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedOpenGL/crserver/crservice.cpp@ 29806

Last change on this file since 29806 was 29019, checked in by vboxsync, 15 years ago

crOpenGL: rever back to 8 texture untits, to avoid memory overwrite in some code parts

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 19.6 KB
Line 
1/* $Id: crservice.cpp 29019 2010-05-04 13:57:05Z vboxsync $ */
2
3/** @file
4 * VBox crOpenGL: Host service entry points.
5 */
6
7/*
8 * Copyright (C) 2006-2008 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#define __STDC_CONSTANT_MACROS /* needed for a definition in iprt/string.h */
20
21#ifdef RT_OS_WINDOWS
22#include <iprt/alloc.h>
23#include <iprt/string.h>
24#include <iprt/assert.h>
25#include <iprt/stream.h>
26#include <VBox/ssm.h>
27#include <VBox/hgcmsvc.h>
28#include <VBox/HostServices/VBoxCrOpenGLSvc.h>
29#include "cr_server.h"
30#define LOG_GROUP LOG_GROUP_SHARED_CROPENGL
31#include <VBox/log.h>
32
33#include <VBox/com/com.h>
34#include <VBox/com/string.h>
35#include <VBox/com/array.h>
36#include <VBox/com/Guid.h>
37#include <VBox/com/ErrorInfo.h>
38#include <VBox/com/EventQueue.h>
39#include <VBox/com/VirtualBox.h>
40#include <VBox/com/assert.h>
41
42#else
43#include <VBox/com/VirtualBox.h>
44#include <iprt/assert.h>
45#include <VBox/ssm.h>
46#include <VBox/hgcmsvc.h>
47#include <VBox/HostServices/VBoxCrOpenGLSvc.h>
48
49#include "cr_server.h"
50#define LOG_GROUP LOG_GROUP_SHARED_CROPENGL
51#include <VBox/log.h>
52#include <VBox/com/ErrorInfo.h>
53#endif /* RT_OS_WINDOWS */
54
55#include <VBox/com/errorprint.h>
56
57PVBOXHGCMSVCHELPERS g_pHelpers;
58static IConsole* g_pConsole = NULL;
59static PVM g_pVM = NULL;
60
61#ifndef RT_OS_WINDOWS
62#define DWORD int
63#define WINAPI
64#endif
65
66static const char* gszVBoxOGLSSMMagic = "***OpenGL state data***";
67#define SHCROGL_SSM_VERSION 19
68
69static DECLCALLBACK(int) svcUnload (void *)
70{
71 int rc = VINF_SUCCESS;
72
73 Log(("SHARED_CROPENGL svcUnload\n"));
74
75 crVBoxServerTearDown();
76
77 return rc;
78}
79
80static DECLCALLBACK(int) svcConnect (void *, uint32_t u32ClientID, void *pvClient)
81{
82 int rc = VINF_SUCCESS;
83
84 NOREF(pvClient);
85
86 Log(("SHARED_CROPENGL svcConnect: u32ClientID = %d\n", u32ClientID));
87
88 rc = crVBoxServerAddClient(u32ClientID);
89
90 return rc;
91}
92
93static DECLCALLBACK(int) svcDisconnect (void *, uint32_t u32ClientID, void *pvClient)
94{
95 int rc = VINF_SUCCESS;
96
97 NOREF(pvClient);
98
99 Log(("SHARED_CROPENGL svcDisconnect: u32ClientID = %d\n", u32ClientID));
100
101 crVBoxServerRemoveClient(u32ClientID);
102
103 return rc;
104}
105
106static DECLCALLBACK(int) svcSaveState(void *, uint32_t u32ClientID, void *pvClient, PSSMHANDLE pSSM)
107{
108 int rc = VINF_SUCCESS;
109
110 NOREF(pvClient);
111
112 Log(("SHARED_CROPENGL svcSaveState: u32ClientID = %d\n", u32ClientID));
113
114 /* Start*/
115 rc = SSMR3PutStrZ(pSSM, gszVBoxOGLSSMMagic);
116 AssertRCReturn(rc, rc);
117
118 /* Version */
119 rc = SSMR3PutU32(pSSM, (uint32_t) SHCROGL_SSM_VERSION);
120 AssertRCReturn(rc, rc);
121
122 /* The state itself */
123 rc = crVBoxServerSaveState(pSSM);
124 AssertRCReturn(rc, rc);
125
126 /* End */
127 rc = SSMR3PutStrZ(pSSM, gszVBoxOGLSSMMagic);
128 AssertRCReturn(rc, rc);
129
130 return VINF_SUCCESS;
131}
132
133static DECLCALLBACK(int) svcLoadState(void *, uint32_t u32ClientID, void *pvClient, PSSMHANDLE pSSM)
134{
135 int rc = VINF_SUCCESS;
136
137 NOREF(pvClient);
138
139 Log(("SHARED_CROPENGL svcLoadState: u32ClientID = %d\n", u32ClientID));
140
141 char psz[2000];
142 uint32_t ui32;
143
144 /* Start of data */
145 rc = SSMR3GetStrZEx(pSSM, psz, 2000, NULL);
146 AssertRCReturn(rc, rc);
147 if (strcmp(gszVBoxOGLSSMMagic, psz))
148 return VERR_SSM_UNEXPECTED_DATA;
149
150 /* Version */
151 rc = SSMR3GetU32(pSSM, &ui32);
152 AssertRCReturn(rc, rc);
153 if ((SHCROGL_SSM_VERSION != ui32)
154 && ((SHCROGL_SSM_VERSION!=4) || (3!=ui32)))
155 {
156 /*@todo: add some warning here*/
157 /*@todo: in many cases saved states would be made without any opengl guest app running.
158 * that means we could safely restore the default context.
159 */
160 rc = SSMR3SkipToEndOfUnit(pSSM);
161 return rc;
162 }
163
164 /* The state itself */
165 rc = crVBoxServerLoadState(pSSM, ui32);
166 AssertRCReturn(rc, rc);
167
168 /* End of data */
169 rc = SSMR3GetStrZEx(pSSM, psz, 2000, NULL);
170 AssertRCReturn(rc, rc);
171 if (strcmp(gszVBoxOGLSSMMagic, psz))
172 return VERR_SSM_UNEXPECTED_DATA;
173
174 return VINF_SUCCESS;
175}
176
177static void svcClientVersionUnsupported(uint32_t minor, uint32_t major)
178{
179 LogRel(("SHARED_CROPENGL: unsupported client version %d.%d\n", minor, major));
180
181 /*MS's opengl32 tryes to load our ICD around 30 times on failure...this is to prevent unnecessary spam*/
182 static int shown = 0;
183
184 if (g_pVM && !shown)
185 {
186 VMSetRuntimeError(g_pVM, VMSETRTERR_FLAGS_NO_WAIT, "3DSupportIncompatibleAdditions",
187 "An attempt by the virtual machine to use hardware 3D acceleration failed. "
188 "The version of the Guest Additions installed in the virtual machine does not match the "
189 "version of VirtualBox on the host. Please install appropriate Guest Additions to fix this issue");
190 shown = 1;
191 }
192}
193
194static DECLCALLBACK(void) svcPresentFBO(void *data, int32_t screenId, int32_t x, int32_t y, uint32_t w, uint32_t h)
195{
196#if 0
197 ComPtr<IDisplay> pDisplay;
198 BYTE *data;
199 int i,j;
200 CHECK_ERROR(g_pConsole, COMGETTER(Display)(pDisplay.asOutParam()));
201
202 data = (BYTE*) RTMemTmpAllocZ(100*100*4);
203
204 for (i=0; i<100; i+=2)
205 {
206 for (j=0; j<100; ++j)
207 {
208 *(data+i*100*4+j*4+0) = 0xFF;
209 *(data+i*100*4+j*4+1) = 0xFF;
210 *(data+i*100*4+j*4+2) = 0xFF;
211 *(data+i*100*4+j*4+3) = 0xFF;
212 }
213 }
214
215 CHECK_ERROR(pDisplay, DrawToScreen(screenId, data, 0, 0, 100, 100));
216
217 RTMemTmpFree(data);
218#endif
219 HRESULT rc;
220 ComPtr<IDisplay> pDisplay;
221
222 CHECK_ERROR(g_pConsole, COMGETTER(Display)(pDisplay.asOutParam()));
223 CHECK_ERROR(pDisplay, DrawToScreen(screenId, (BYTE*)data, x, y, w, h));
224}
225
226static DECLCALLBACK(void) svcCall (void *, VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void *pvClient, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
227{
228 int rc = VINF_SUCCESS;
229
230 NOREF(pvClient);
231
232 Log(("SHARED_CROPENGL svcCall: u32ClientID = %d, fn = %d, cParms = %d, pparms = %d\n", u32ClientID, u32Function, cParms, paParms));
233
234#ifdef DEBUG
235 uint32_t i;
236
237 for (i = 0; i < cParms; i++)
238 {
239 /** @todo parameters other than 32 bit */
240 Log((" pparms[%d]: type %d value %d\n", i, paParms[i].type, paParms[i].u.uint32));
241 }
242#endif
243
244 switch (u32Function)
245 {
246 case SHCRGL_GUEST_FN_WRITE:
247 {
248 Log(("svcCall: SHCRGL_GUEST_FN_WRITE\n"));
249
250 /* Verify parameter count and types. */
251 if (cParms != SHCRGL_CPARMS_WRITE)
252 {
253 rc = VERR_INVALID_PARAMETER;
254 }
255 else
256 if ( paParms[0].type != VBOX_HGCM_SVC_PARM_PTR /* pBuffer */
257 )
258 {
259 rc = VERR_INVALID_PARAMETER;
260 }
261 else
262 {
263 /* Fetch parameters. */
264 uint8_t *pBuffer = (uint8_t *)paParms[0].u.pointer.addr;
265 uint32_t cbBuffer = paParms[0].u.pointer.size;
266
267 /* Execute the function. */
268 rc = crVBoxServerClientWrite(u32ClientID, pBuffer, cbBuffer);
269 if (!RT_SUCCESS(rc))
270 {
271 Assert(VERR_NOT_SUPPORTED==rc);
272 svcClientVersionUnsupported(0, 0);
273 }
274
275 }
276 break;
277 }
278
279 case SHCRGL_GUEST_FN_READ:
280 {
281 Log(("svcCall: SHCRGL_GUEST_FN_READ\n"));
282
283 /* Verify parameter count and types. */
284 if (cParms != SHCRGL_CPARMS_READ)
285 {
286 rc = VERR_INVALID_PARAMETER;
287 }
288 else
289 if ( paParms[0].type != VBOX_HGCM_SVC_PARM_PTR /* pBuffer */
290 || paParms[1].type != VBOX_HGCM_SVC_PARM_32BIT /* cbBuffer */
291 )
292 {
293 rc = VERR_INVALID_PARAMETER;
294 }
295
296 /* Fetch parameters. */
297 uint8_t *pBuffer = (uint8_t *)paParms[0].u.pointer.addr;
298 uint32_t cbBuffer = paParms[0].u.pointer.size;
299
300 /* Execute the function. */
301 rc = crVBoxServerClientRead(u32ClientID, pBuffer, &cbBuffer);
302
303 if (RT_SUCCESS(rc))
304 {
305 /* Update parameters.*/
306 paParms[0].u.pointer.size = cbBuffer; //@todo guest doesn't see this change somehow?
307 } else if (VERR_NOT_SUPPORTED==rc)
308 {
309 svcClientVersionUnsupported(0, 0);
310 }
311
312 /* Return the required buffer size always */
313 paParms[1].u.uint32 = cbBuffer;
314
315 break;
316 }
317
318 case SHCRGL_GUEST_FN_WRITE_READ:
319 {
320 Log(("svcCall: SHCRGL_GUEST_FN_WRITE_READ\n"));
321
322 /* Verify parameter count and types. */
323 if (cParms != SHCRGL_CPARMS_WRITE_READ)
324 {
325 rc = VERR_INVALID_PARAMETER;
326 }
327 else
328 if ( paParms[0].type != VBOX_HGCM_SVC_PARM_PTR /* pBuffer */
329 || paParms[1].type != VBOX_HGCM_SVC_PARM_PTR /* pWriteback */
330 || paParms[2].type != VBOX_HGCM_SVC_PARM_32BIT /* cbWriteback */
331 )
332 {
333 rc = VERR_INVALID_PARAMETER;
334 }
335 else
336 {
337 /* Fetch parameters. */
338 uint8_t *pBuffer = (uint8_t *)paParms[0].u.pointer.addr;
339 uint32_t cbBuffer = paParms[0].u.pointer.size;
340
341 uint8_t *pWriteback = (uint8_t *)paParms[1].u.pointer.addr;
342 uint32_t cbWriteback = paParms[1].u.pointer.size;
343
344 /* Execute the function. */
345 rc = crVBoxServerClientWrite(u32ClientID, pBuffer, cbBuffer);
346 if (!RT_SUCCESS(rc))
347 {
348 Assert(VERR_NOT_SUPPORTED==rc);
349 svcClientVersionUnsupported(0, 0);
350 }
351
352 rc = crVBoxServerClientRead(u32ClientID, pWriteback, &cbWriteback);
353
354 if (RT_SUCCESS(rc))
355 {
356 /* Update parameters.*/
357 paParms[1].u.pointer.size = cbWriteback;
358 }
359 /* Return the required buffer size always */
360 paParms[2].u.uint32 = cbWriteback;
361 }
362
363 break;
364 }
365
366 case SHCRGL_GUEST_FN_SET_VERSION:
367 {
368 Log(("svcCall: SHCRGL_GUEST_FN_SET_VERSION\n"));
369
370 /* Verify parameter count and types. */
371 if (cParms != SHCRGL_CPARMS_SET_VERSION)
372 {
373 rc = VERR_INVALID_PARAMETER;
374 }
375 else
376 if ( paParms[0].type != VBOX_HGCM_SVC_PARM_32BIT /* vMajor */
377 || paParms[1].type != VBOX_HGCM_SVC_PARM_32BIT /* vMinor */
378 )
379 {
380 rc = VERR_INVALID_PARAMETER;
381 }
382 else
383 {
384 /* Fetch parameters. */
385 uint32_t vMajor = paParms[0].u.uint32;
386 uint32_t vMinor = paParms[1].u.uint32;
387
388 /* Execute the function. */
389 rc = crVBoxServerClientSetVersion(u32ClientID, vMajor, vMinor);
390
391 if (!RT_SUCCESS(rc))
392 {
393 svcClientVersionUnsupported(vMajor, vMinor);
394 }
395 }
396
397 break;
398 }
399
400 default:
401 {
402 rc = VERR_NOT_IMPLEMENTED;
403 }
404 }
405
406
407 LogFlow(("svcCall: rc = %Rrc\n", rc));
408
409 g_pHelpers->pfnCallComplete (callHandle, rc);
410}
411
412/*
413 * We differentiate between a function handler for the guest and one for the host.
414 */
415static DECLCALLBACK(int) svcHostCall (void *, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
416{
417 int rc = VINF_SUCCESS;
418
419 Log(("SHARED_CROPENGL svcHostCall: fn = %d, cParms = %d, pparms = %d\n", u32Function, cParms, paParms));
420
421#ifdef DEBUG
422 uint32_t i;
423
424 for (i = 0; i < cParms; i++)
425 {
426 /** @todo parameters other than 32 bit */
427 Log((" pparms[%d]: type %d value %d\n", i, paParms[i].type, paParms[i].u.uint32));
428 }
429#endif
430
431 switch (u32Function)
432 {
433 case SHCRGL_HOST_FN_SET_CONSOLE:
434 {
435 Log(("svcCall: SHCRGL_HOST_FN_SET_DISPLAY\n"));
436
437 /* Verify parameter count and types. */
438 if (cParms != SHCRGL_CPARMS_SET_CONSOLE)
439 {
440 rc = VERR_INVALID_PARAMETER;
441 }
442 else if (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR)
443 {
444 rc = VERR_INVALID_PARAMETER;
445 }
446 else
447 {
448 /* Fetch parameters. */
449 IConsole* pConsole = (IConsole*)paParms[0].u.pointer.addr;
450 uint32_t cbData = paParms[0].u.pointer.size;
451
452 /* Verify parameters values. */
453 if (cbData != sizeof (IConsole*))
454 {
455 rc = VERR_INVALID_PARAMETER;
456 }
457 else if (!pConsole)
458 {
459 rc = VERR_INVALID_PARAMETER;
460 }
461 else /* Execute the function. */
462 {
463 ComPtr<IMachine> pMachine;
464 ComPtr<IDisplay> pDisplay;
465 ComPtr<IFramebuffer> pFramebuffer;
466 LONG xo, yo;
467 ULONG64 winId = 0;
468 ULONG monitorCount, i, w, h;
469
470 CHECK_ERROR_BREAK(pConsole, COMGETTER(Machine)(pMachine.asOutParam()));
471 CHECK_ERROR_BREAK(pMachine, COMGETTER(MonitorCount)(&monitorCount));
472 CHECK_ERROR_BREAK(pConsole, COMGETTER(Display)(pDisplay.asOutParam()));
473
474 rc = crVBoxServerSetScreenCount(monitorCount);
475 AssertRCReturn(rc, rc);
476
477 for (i=0; i<monitorCount; ++i)
478 {
479 CHECK_ERROR_RET(pDisplay, GetFramebuffer(i, pFramebuffer.asOutParam(), &xo, &yo), rc);
480
481 if (!pDisplay)
482 {
483 rc = crVBoxServerUnmapScreen(i);
484 AssertRCReturn(rc, rc);
485 }
486 else
487 {
488 CHECK_ERROR_RET(pFramebuffer, COMGETTER(WinId)(&winId), rc);
489 CHECK_ERROR_RET(pFramebuffer, COMGETTER(Width)(&w), rc);
490 CHECK_ERROR_RET(pFramebuffer, COMGETTER(Height)(&h), rc);
491
492 rc = crVBoxServerMapScreen(i, xo, yo, w, h, winId);
493 AssertRCReturn(rc, rc);
494 }
495 }
496
497 g_pConsole = pConsole;
498 rc = VINF_SUCCESS;
499 }
500 }
501 break;
502 }
503 case SHCRGL_HOST_FN_SET_VM:
504 {
505 Log(("svcCall: SHCRGL_HOST_FN_SET_VM\n"));
506
507 /* Verify parameter count and types. */
508 if (cParms != SHCRGL_CPARMS_SET_VM)
509 {
510 rc = VERR_INVALID_PARAMETER;
511 }
512 else if (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR)
513 {
514 rc = VERR_INVALID_PARAMETER;
515 }
516 else
517 {
518 /* Fetch parameters. */
519 PVM pVM = (PVM)paParms[0].u.pointer.addr;
520 uint32_t cbData = paParms[0].u.pointer.size;
521
522 /* Verify parameters values. */
523 if (cbData != sizeof (PVM))
524 {
525 rc = VERR_INVALID_PARAMETER;
526 }
527 else
528 {
529 /* Execute the function. */
530 g_pVM = pVM;
531 rc = VINF_SUCCESS;
532 }
533 }
534 break;
535 }
536 case SHCRGL_HOST_FN_SET_VISIBLE_REGION:
537 {
538 Log(("svcCall: SHCRGL_HOST_FN_SET_VISIBLE_REGION\n"));
539
540 if (cParms != SHCRGL_CPARMS_SET_VISIBLE_REGION)
541 {
542 rc = VERR_INVALID_PARAMETER;
543 break;
544 }
545
546 if ( paParms[0].type != VBOX_HGCM_SVC_PARM_PTR /* pRects */
547 || paParms[1].type != VBOX_HGCM_SVC_PARM_32BIT /* cRects */
548 )
549 {
550 rc = VERR_INVALID_PARAMETER;
551 break;
552 }
553
554 Assert(sizeof(RTRECT)==4*sizeof(GLint));
555
556 rc = crVBoxServerSetRootVisibleRegion(paParms[1].u.uint32, (GLint*)paParms[0].u.pointer.addr);
557 break;
558 }
559 case SHCRGL_HOST_FN_SCREEN_CHANGED:
560 {
561 Log(("svcCall: SHCRGL_HOST_FN_SCREEN_CHANGED\n"));
562
563 /* Verify parameter count and types. */
564 if (cParms != SHCRGL_CPARMS_SCREEN_CHANGED)
565 {
566 rc = VERR_INVALID_PARAMETER;
567 }
568 else if (paParms[0].type != VBOX_HGCM_SVC_PARM_32BIT)
569 {
570 rc = VERR_INVALID_PARAMETER;
571 }
572 else
573 {
574 /* Fetch parameters. */
575 uint32_t screenId = paParms[0].u.uint32;
576
577 /* Execute the function. */
578 ComPtr<IDisplay> pDisplay;
579 ComPtr<IFramebuffer> pFramebuffer;
580 LONG xo, yo;
581 ULONG64 winId = 0;
582 ULONG w, h;
583
584 Assert(g_pConsole);
585 CHECK_ERROR_RET(g_pConsole, COMGETTER(Display)(pDisplay.asOutParam()), rc);
586 CHECK_ERROR_RET(pDisplay, GetFramebuffer(screenId, pFramebuffer.asOutParam(), &xo, &yo), rc);
587
588 if (!pFramebuffer)
589 {
590 rc = crVBoxServerUnmapScreen(screenId);
591 AssertRCReturn(rc, rc);
592 }
593 else
594 {
595 CHECK_ERROR_RET(pFramebuffer, COMGETTER(WinId)(&winId), rc);
596 CHECK_ERROR_RET(pFramebuffer, COMGETTER(Width)(&w), rc);
597 CHECK_ERROR_RET(pFramebuffer, COMGETTER(Height)(&h), rc);
598
599 rc = crVBoxServerMapScreen(screenId, xo, yo, w, h, winId);
600 AssertRCReturn(rc, rc);
601 }
602
603 rc = VINF_SUCCESS;
604 }
605 break;
606 }
607 default:
608 rc = VERR_NOT_IMPLEMENTED;
609 break;
610 }
611
612 LogFlow(("svcHostCall: rc = %Rrc\n", rc));
613 return rc;
614}
615
616extern "C" DECLCALLBACK(DECLEXPORT(int)) VBoxHGCMSvcLoad (VBOXHGCMSVCFNTABLE *ptable)
617{
618 int rc = VINF_SUCCESS;
619
620 Log(("SHARED_CROPENGL VBoxHGCMSvcLoad: ptable = %p\n", ptable));
621
622 if (!ptable)
623 {
624 rc = VERR_INVALID_PARAMETER;
625 }
626 else
627 {
628 Log(("VBoxHGCMSvcLoad: ptable->cbSize = %d, ptable->u32Version = 0x%08X\n", ptable->cbSize, ptable->u32Version));
629
630 if ( ptable->cbSize != sizeof (VBOXHGCMSVCFNTABLE)
631 || ptable->u32Version != VBOX_HGCM_SVC_VERSION)
632 {
633 rc = VERR_INVALID_PARAMETER;
634 }
635 else
636 {
637 g_pHelpers = ptable->pHelpers;
638
639 ptable->cbClient = sizeof (void*);
640
641 ptable->pfnUnload = svcUnload;
642 ptable->pfnConnect = svcConnect;
643 ptable->pfnDisconnect = svcDisconnect;
644 ptable->pfnCall = svcCall;
645 ptable->pfnHostCall = svcHostCall;
646 ptable->pfnSaveState = svcSaveState;
647 ptable->pfnLoadState = svcLoadState;
648 ptable->pvService = NULL;
649
650 if (!crVBoxServerInit())
651 return VERR_NOT_SUPPORTED;
652
653 crVBoxServerSetPresentFBOCB(svcPresentFBO);
654 }
655 }
656
657 return rc;
658}
659
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