VirtualBox

source: vbox/trunk/src/VBox/Additions/common/crOpenGL/load.c@ 47631

Last change on this file since 47631 was 47161, checked in by vboxsync, 12 years ago

crOpenGL: fix wddm init & win state handling

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 37.9 KB
Line 
1/* Copyright (c) 2001, Stanford University
2 * All rights reserved
3 *
4 * See the file LICENSE.txt for information on redistributing this software.
5 */
6
7#include "cr_spu.h"
8#include "cr_net.h"
9#include "cr_error.h"
10#include "cr_mem.h"
11#include "cr_string.h"
12#include "cr_net.h"
13#include "cr_environment.h"
14#include "cr_process.h"
15#include "cr_rand.h"
16#include "cr_netserver.h"
17#include "stub.h"
18#include <stdlib.h>
19#include <string.h>
20#include <signal.h>
21#include <iprt/initterm.h>
22#include <iprt/thread.h>
23#include <iprt/err.h>
24#include <iprt/asm.h>
25#ifndef WINDOWS
26# include <sys/types.h>
27# include <unistd.h>
28#endif
29
30#ifdef VBOX_WITH_WDDM
31#include <d3d9types.h>
32#include <D3dumddi.h>
33#include "../../WINNT/Graphics/Video/common/wddm/VBoxMPIf.h"
34#include "../../WINNT/Graphics/Video/disp/wddm/VBoxDispMp.h"
35#endif
36
37/**
38 * If you change this, see the comments in tilesortspu_context.c
39 */
40#define MAGIC_CONTEXT_BASE 500
41
42#define CONFIG_LOOKUP_FILE ".crconfigs"
43
44#ifdef WINDOWS
45#define PYTHON_EXE "python.exe"
46#else
47#define PYTHON_EXE "python"
48#endif
49
50static bool stub_initialized = 0;
51#ifdef WINDOWS
52static CRmutex stub_init_mutex;
53#define STUB_INIT_LOCK() do { crLockMutex(&stub_init_mutex); } while (0)
54#define STUB_INIT_UNLOCK() do { crUnlockMutex(&stub_init_mutex); } while (0)
55#else
56#define STUB_INIT_LOCK() do { } while (0)
57#define STUB_INIT_UNLOCK() do { } while (0)
58#endif
59
60/* NOTE: 'SPUDispatchTable glim' is declared in NULLfuncs.py now */
61/* NOTE: 'SPUDispatchTable stubThreadsafeDispatch' is declared in tsfuncs.c */
62Stub stub;
63#ifdef CHROMIUM_THREADSAFE
64static bool g_stubIsCurrentContextTSDInited;
65CRtsd g_stubCurrentContextTSD;
66#endif
67
68
69static void stubInitNativeDispatch( void )
70{
71#define MAX_FUNCS 1000
72 SPUNamedFunctionTable gl_funcs[MAX_FUNCS];
73 int numFuncs;
74
75 numFuncs = crLoadOpenGL( &stub.wsInterface, gl_funcs );
76
77 stub.haveNativeOpenGL = (numFuncs > 0);
78
79 /* XXX call this after context binding */
80 numFuncs += crLoadOpenGLExtensions( &stub.wsInterface, gl_funcs + numFuncs );
81
82 CRASSERT(numFuncs < MAX_FUNCS);
83
84 crSPUInitDispatchTable( &stub.nativeDispatch );
85 crSPUInitDispatch( &stub.nativeDispatch, gl_funcs );
86 crSPUInitDispatchNops( &stub.nativeDispatch );
87#undef MAX_FUNCS
88}
89
90
91/** Pointer to the SPU's real glClear and glViewport functions */
92static ClearFunc_t origClear;
93static ViewportFunc_t origViewport;
94static SwapBuffersFunc_t origSwapBuffers;
95static DrawBufferFunc_t origDrawBuffer;
96static ScissorFunc_t origScissor;
97
98static void stubCheckWindowState(WindowInfo *window, GLboolean bFlushOnChange)
99{
100 bool bForceUpdate = false;
101 bool bChanged = false;
102
103#ifdef WINDOWS
104 /* @todo install hook and track for WM_DISPLAYCHANGE */
105 {
106 DEVMODE devMode;
107
108 devMode.dmSize = sizeof(DEVMODE);
109 EnumDisplaySettings(NULL, ENUM_CURRENT_SETTINGS, &devMode);
110
111 if (devMode.dmPelsWidth!=window->dmPelsWidth || devMode.dmPelsHeight!=window->dmPelsHeight)
112 {
113 crDebug("Resolution changed(%d,%d), forcing window Pos/Size update", devMode.dmPelsWidth, devMode.dmPelsHeight);
114 window->dmPelsWidth = devMode.dmPelsWidth;
115 window->dmPelsHeight = devMode.dmPelsHeight;
116 bForceUpdate = true;
117 }
118 }
119#endif
120
121 bChanged = stubUpdateWindowGeometry(window, bForceUpdate) || bForceUpdate;
122
123#if defined(GLX) || defined (WINDOWS)
124 if (stub.trackWindowVisibleRgn)
125 {
126 bChanged = stubUpdateWindowVisibileRegions(window) || bChanged;
127 }
128#endif
129
130 if (stub.trackWindowVisibility && window->type == CHROMIUM && window->drawable) {
131 const int mapped = stubIsWindowVisible(window);
132 if (mapped != window->mapped) {
133 crDebug("Dispatched: WindowShow(%i, %i)", window->spuWindow, mapped);
134 stub.spu->dispatch_table.WindowShow(window->spuWindow, mapped);
135 window->mapped = mapped;
136 bChanged = true;
137 }
138 }
139
140 if (bFlushOnChange && bChanged)
141 {
142 stub.spu->dispatch_table.Flush();
143 }
144}
145
146static bool stubSystemWindowExist(WindowInfo *pWindow)
147{
148#ifdef WINDOWS
149 if (pWindow->hWnd!=WindowFromDC(pWindow->drawable))
150 {
151 return false;
152 }
153#else
154 Window root;
155 int x, y;
156 unsigned int border, depth, w, h;
157 Display *dpy;
158
159 dpy = stubGetWindowDisplay(pWindow);
160
161 XLOCK(dpy);
162 if (!XGetGeometry(dpy, pWindow->drawable, &root, &x, &y, &w, &h, &border, &depth))
163 {
164 XUNLOCK(dpy);
165 return false;
166 }
167 XUNLOCK(dpy);
168#endif
169
170 return true;
171}
172
173static void stubCheckWindowsCB(unsigned long key, void *data1, void *data2)
174{
175 WindowInfo *pWindow = (WindowInfo *) data1;
176 ContextInfo *pCtx = (ContextInfo *) data2;
177
178 if (pWindow == pCtx->currentDrawable
179 || pWindow->type!=CHROMIUM
180 || pWindow->pOwner!=pCtx)
181 {
182 return;
183 }
184
185 if (!stubSystemWindowExist(pWindow))
186 {
187#ifdef WINDOWS
188 stubDestroyWindow(CR_CTX_CON(pCtx), (GLint)pWindow->hWnd);
189#else
190 stubDestroyWindow(CR_CTX_CON(pCtx), (GLint)pWindow->drawable);
191#endif
192 return;
193 }
194
195 stubCheckWindowState(pWindow, GL_FALSE);
196}
197
198static void stubCheckWindowsState(void)
199{
200 ContextInfo *context = stubGetCurrentContext();
201
202 CRASSERT(stub.trackWindowSize || stub.trackWindowPos);
203
204 if (!context)
205 return;
206
207#if defined(WINDOWS) && defined(VBOX_WITH_WDDM)
208 if (stub.bRunningUnderWDDM)
209 return;
210#endif
211
212#if defined(CR_NEWWINTRACK) && !defined(WINDOWS)
213 crLockMutex(&stub.mutex);
214#endif
215
216 stubCheckWindowState(context->currentDrawable, GL_TRUE);
217 crHashtableWalk(stub.windowTable, stubCheckWindowsCB, context);
218
219#if defined(CR_NEWWINTRACK) && !defined(WINDOWS)
220 crUnlockMutex(&stub.mutex);
221#endif
222}
223
224
225/**
226 * Override the head SPU's glClear function.
227 * We're basically trapping this function so that we can poll the
228 * application window size at a regular interval.
229 */
230static void SPU_APIENTRY trapClear(GLbitfield mask)
231{
232 stubCheckWindowsState();
233 /* call the original SPU glClear function */
234 origClear(mask);
235}
236
237/**
238 * As above, but for glViewport. Most apps call glViewport before
239 * glClear when a window is resized.
240 */
241static void SPU_APIENTRY trapViewport(GLint x, GLint y, GLsizei w, GLsizei h)
242{
243 stubCheckWindowsState();
244 /* call the original SPU glViewport function */
245 origViewport(x, y, w, h);
246}
247
248static void SPU_APIENTRY trapSwapBuffers(GLint window, GLint flags)
249{
250 stubCheckWindowsState();
251 origSwapBuffers(window, flags);
252}
253
254static void SPU_APIENTRY trapDrawBuffer(GLenum buf)
255{
256 stubCheckWindowsState();
257 origDrawBuffer(buf);
258}
259
260static void SPU_APIENTRY trapScissor(GLint x, GLint y, GLsizei w, GLsizei h)
261{
262 int winX, winY;
263 unsigned int winW, winH;
264 WindowInfo *pWindow;
265 ContextInfo *context = stubGetCurrentContext();
266 pWindow = context->currentDrawable;
267 stubGetWindowGeometry(pWindow, &winX, &winY, &winW, &winH);
268 origScissor(0, 0, winW, winH);
269}
270
271/**
272 * Use the GL function pointers in <spu> to initialize the static glim
273 * dispatch table.
274 */
275static void stubInitSPUDispatch(SPU *spu)
276{
277 crSPUInitDispatchTable( &stub.spuDispatch );
278 crSPUCopyDispatchTable( &stub.spuDispatch, &(spu->dispatch_table) );
279
280 if (stub.trackWindowSize || stub.trackWindowPos || stub.trackWindowVisibleRgn) {
281 /* patch-in special glClear/Viewport function to track window sizing */
282 origClear = stub.spuDispatch.Clear;
283 origViewport = stub.spuDispatch.Viewport;
284 origSwapBuffers = stub.spuDispatch.SwapBuffers;
285 origDrawBuffer = stub.spuDispatch.DrawBuffer;
286 origScissor = stub.spuDispatch.Scissor;
287 stub.spuDispatch.Clear = trapClear;
288 stub.spuDispatch.Viewport = trapViewport;
289
290 /*stub.spuDispatch.SwapBuffers = trapSwapBuffers;
291 stub.spuDispatch.DrawBuffer = trapDrawBuffer;*/
292 }
293
294 crSPUCopyDispatchTable( &glim, &stub.spuDispatch );
295}
296
297// Callback function, used to destroy all created contexts
298static void hsWalkStubDestroyContexts(unsigned long key, void *data1, void *data2)
299{
300 stubDestroyContext(key);
301}
302
303/**
304 * This is called when we exit.
305 * We call all the SPU's cleanup functions.
306 */
307static void stubSPUTearDownLocked(void)
308{
309 crDebug("stubSPUTearDownLocked");
310
311#ifdef WINDOWS
312# ifndef CR_NEWWINTRACK
313 stubUninstallWindowMessageHook();
314# endif
315#endif
316
317#ifdef CR_NEWWINTRACK
318 ASMAtomicWriteBool(&stub.bShutdownSyncThread, true);
319#endif
320
321 //delete all created contexts
322 stubMakeCurrent( NULL, NULL);
323
324 /* the lock order is windowTable->contextTable (see wglMakeCurrent_prox, glXMakeCurrent)
325 * this is why we need to take a windowTable lock since we will later do stub.windowTable access & locking */
326 crHashtableLock(stub.windowTable);
327 crHashtableWalk(stub.contextTable, hsWalkStubDestroyContexts, NULL);
328 crHashtableUnlock(stub.windowTable);
329
330 /* shutdown, now trap any calls to a NULL dispatcher */
331 crSPUCopyDispatchTable(&glim, &stubNULLDispatch);
332
333 crSPUUnloadChain(stub.spu);
334 stub.spu = NULL;
335
336#ifndef Linux
337 crUnloadOpenGL();
338#endif
339
340#ifndef WINDOWS
341 crNetTearDown();
342#endif
343
344#ifdef GLX
345 if (stub.xshmSI.shmid>=0)
346 {
347 shmctl(stub.xshmSI.shmid, IPC_RMID, 0);
348 shmdt(stub.xshmSI.shmaddr);
349 }
350 crFreeHashtable(stub.pGLXPixmapsHash, crFree);
351#endif
352
353 crFreeHashtable(stub.windowTable, crFree);
354 crFreeHashtable(stub.contextTable, NULL);
355
356 crMemset(&stub, 0, sizeof(stub));
357
358}
359
360/**
361 * This is called when we exit.
362 * We call all the SPU's cleanup functions.
363 */
364static void stubSPUTearDown(void)
365{
366 STUB_INIT_LOCK();
367 if (stub_initialized)
368 {
369 stubSPUTearDownLocked();
370 stub_initialized = 0;
371 }
372 STUB_INIT_UNLOCK();
373}
374
375static void stubSPUSafeTearDown(void)
376{
377#ifdef CHROMIUM_THREADSAFE
378 CRmutex *mutex;
379#endif
380
381 if (!stub_initialized) return;
382 stub_initialized = 0;
383
384#ifdef CHROMIUM_THREADSAFE
385 mutex = &stub.mutex;
386 crLockMutex(mutex);
387#endif
388 crDebug("stubSPUSafeTearDown");
389
390#ifdef WINDOWS
391# ifndef CR_NEWWINTRACK
392 stubUninstallWindowMessageHook();
393# endif
394#endif
395
396#if defined(CR_NEWWINTRACK)
397 crUnlockMutex(mutex);
398# if defined(WINDOWS)
399 if (stub.hSyncThread && RTThreadGetState(stub.hSyncThread)!=RTTHREADSTATE_TERMINATED)
400 {
401 HANDLE hNative;
402 DWORD ec=0;
403
404 hNative = OpenThread(SYNCHRONIZE|THREAD_QUERY_INFORMATION|THREAD_TERMINATE,
405 false, RTThreadGetNative(stub.hSyncThread));
406 if (!hNative)
407 {
408 crWarning("Failed to get handle for sync thread(%#x)", GetLastError());
409 }
410 else
411 {
412 crDebug("Got handle %p for thread %#x", hNative, RTThreadGetNative(stub.hSyncThread));
413 }
414
415 ASMAtomicWriteBool(&stub.bShutdownSyncThread, true);
416
417 if (PostThreadMessage(RTThreadGetNative(stub.hSyncThread), WM_QUIT, 0, 0))
418 {
419 RTThreadWait(stub.hSyncThread, 1000, NULL);
420
421 /*Same issue as on linux, RTThreadWait exits before system thread is terminated, which leads
422 * to issues as our dll goes to be unloaded.
423 *@todo
424 *We usually call this function from DllMain which seems to be holding some lock and thus we have to
425 * kill thread via TerminateThread.
426 */
427 if (WaitForSingleObject(hNative, 100)==WAIT_TIMEOUT)
428 {
429 crDebug("Wait failed, terminating");
430 if (!TerminateThread(hNative, 1))
431 {
432 crDebug("TerminateThread failed");
433 }
434 }
435 if (GetExitCodeThread(hNative, &ec))
436 {
437 crDebug("Thread %p exited with ec=%i", hNative, ec);
438 }
439 else
440 {
441 crDebug("GetExitCodeThread failed(%#x)", GetLastError());
442 }
443 }
444 else
445 {
446 crDebug("Sync thread killed before DLL_PROCESS_DETACH");
447 }
448
449 if (hNative)
450 {
451 CloseHandle(hNative);
452 }
453 }
454#else
455 if (stub.hSyncThread!=NIL_RTTHREAD)
456 {
457 ASMAtomicWriteBool(&stub.bShutdownSyncThread, true);
458 {
459 /*RTThreadWait might return too early, which cause our code being unloaded while RT thread wrapper is still running*/
460 int rc = pthread_join(RTThreadGetNative(stub.hSyncThread), NULL);
461 if (!rc)
462 {
463 crDebug("pthread_join failed %i", rc);
464 }
465 }
466 }
467#endif
468 crLockMutex(mutex);
469#endif
470
471#ifndef WINDOWS
472 crNetTearDown();
473#endif
474
475#ifdef CHROMIUM_THREADSAFE
476 crUnlockMutex(mutex);
477 crFreeMutex(mutex);
478#endif
479 crMemset(&stub, 0, sizeof(stub));
480}
481
482
483static void stubExitHandler(void)
484{
485 stubSPUSafeTearDown();
486}
487
488/**
489 * Called when we receive a SIGTERM signal.
490 */
491static void stubSignalHandler(int signo)
492{
493 stubSPUSafeTearDown();
494 exit(0); /* this causes stubExitHandler() to be called */
495}
496
497#ifndef RT_OS_WINDOWS
498# ifdef CHROMIUM_THREADSAFE
499static DECLCALLBACK(void) stubThreadTlsDtor(void *pvValue)
500{
501 ContextInfo *pCtx = (ContextInfo*)pvValue;
502 VBoxTlsRefRelease(pCtx);
503}
504# endif
505#endif
506
507
508/**
509 * Init variables in the stub structure, install signal handler.
510 */
511static void stubInitVars(void)
512{
513 WindowInfo *defaultWin;
514
515#ifdef CHROMIUM_THREADSAFE
516 crInitMutex(&stub.mutex);
517#endif
518
519 /* At the very least we want CR_RGB_BIT. */
520 stub.haveNativeOpenGL = GL_FALSE;
521 stub.spu = NULL;
522 stub.appDrawCursor = 0;
523 stub.minChromiumWindowWidth = 0;
524 stub.minChromiumWindowHeight = 0;
525 stub.maxChromiumWindowWidth = 0;
526 stub.maxChromiumWindowHeight = 0;
527 stub.matchChromiumWindowCount = 0;
528 stub.matchChromiumWindowID = NULL;
529 stub.matchWindowTitle = NULL;
530 stub.ignoreFreeglutMenus = 0;
531 stub.threadSafe = GL_FALSE;
532 stub.trackWindowSize = 0;
533 stub.trackWindowPos = 0;
534 stub.trackWindowVisibility = 0;
535 stub.trackWindowVisibleRgn = 0;
536 stub.mothershipPID = 0;
537 stub.spu_dir = NULL;
538
539 stub.freeContextNumber = MAGIC_CONTEXT_BASE;
540 stub.contextTable = crAllocHashtable();
541#ifndef RT_OS_WINDOWS
542# ifdef CHROMIUM_THREADSAFE
543 if (!g_stubIsCurrentContextTSDInited)
544 {
545 crInitTSDF(&g_stubCurrentContextTSD, stubThreadTlsDtor);
546 g_stubIsCurrentContextTSDInited = true;
547 }
548# endif
549#endif
550 stubSetCurrentContext(NULL);
551
552 stub.windowTable = crAllocHashtable();
553
554#ifdef CR_NEWWINTRACK
555 stub.bShutdownSyncThread = false;
556 stub.hSyncThread = NIL_RTTHREAD;
557#endif
558
559 defaultWin = (WindowInfo *) crCalloc(sizeof(WindowInfo));
560 defaultWin->type = CHROMIUM;
561 defaultWin->spuWindow = 0; /* window 0 always exists */
562#ifdef WINDOWS
563 defaultWin->hVisibleRegion = INVALID_HANDLE_VALUE;
564#elif defined(GLX)
565 defaultWin->pVisibleRegions = NULL;
566 defaultWin->cVisibleRegions = 0;
567#endif
568 crHashtableAdd(stub.windowTable, 0, defaultWin);
569
570#if 1
571 atexit(stubExitHandler);
572 signal(SIGTERM, stubSignalHandler);
573 signal(SIGINT, stubSignalHandler);
574#ifndef WINDOWS
575 signal(SIGPIPE, SIG_IGN); /* the networking code should catch this */
576#endif
577#else
578 (void) stubExitHandler;
579 (void) stubSignalHandler;
580#endif
581}
582
583
584/**
585 * Return a free port number for the mothership to use, or -1 if we
586 * can't find one.
587 */
588static int
589GenerateMothershipPort(void)
590{
591 const int MAX_PORT = 10100;
592 unsigned short port;
593
594 /* generate initial port number randomly */
595 crRandAutoSeed();
596 port = (unsigned short) crRandInt(10001, MAX_PORT);
597
598#ifdef WINDOWS
599 /* XXX should implement a free port check here */
600 return port;
601#else
602 /*
603 * See if this port number really is free, try another if needed.
604 */
605 {
606 struct sockaddr_in servaddr;
607 int so_reuseaddr = 1;
608 int sock, k;
609
610 /* create socket */
611 sock = socket(AF_INET, SOCK_STREAM, 0);
612 CRASSERT(sock > 2);
613
614 /* deallocate socket/port when we exit */
615 k = setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
616 (char *) &so_reuseaddr, sizeof(so_reuseaddr));
617 CRASSERT(k == 0);
618
619 /* initialize the servaddr struct */
620 crMemset(&servaddr, 0, sizeof(servaddr) );
621 servaddr.sin_family = AF_INET;
622 servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
623
624 while (port < MAX_PORT) {
625 /* Bind to the given port number, return -1 if we fail */
626 servaddr.sin_port = htons((unsigned short) port);
627 k = bind(sock, (struct sockaddr *) &servaddr, sizeof(servaddr));
628 if (k) {
629 /* failed to create port. try next one. */
630 port++;
631 }
632 else {
633 /* free the socket/port now so mothership can make it */
634 close(sock);
635 return port;
636 }
637 }
638 }
639#endif /* WINDOWS */
640 return -1;
641}
642
643
644/**
645 * Try to determine which mothership configuration to use for this program.
646 */
647static char **
648LookupMothershipConfig(const char *procName)
649{
650 const int procNameLen = crStrlen(procName);
651 FILE *f;
652 const char *home;
653 char configPath[1000];
654
655 /* first, check if the CR_CONFIG env var is set */
656 {
657 const char *conf = crGetenv("CR_CONFIG");
658 if (conf && crStrlen(conf) > 0)
659 return crStrSplit(conf, " ");
660 }
661
662 /* second, look up config name from config file */
663 home = crGetenv("HOME");
664 if (home)
665 sprintf(configPath, "%s/%s", home, CONFIG_LOOKUP_FILE);
666 else
667 crStrcpy(configPath, CONFIG_LOOKUP_FILE); /* from current dir */
668 /* Check if the CR_CONFIG_PATH env var is set. */
669 {
670 const char *conf = crGetenv("CR_CONFIG_PATH");
671 if (conf)
672 crStrcpy(configPath, conf); /* from env var */
673 }
674
675 f = fopen(configPath, "r");
676 if (!f) {
677 return NULL;
678 }
679
680 while (!feof(f)) {
681 char line[1000];
682 char **args;
683 fgets(line, 999, f);
684 line[crStrlen(line) - 1] = 0; /* remove trailing newline */
685 if (crStrncmp(line, procName, procNameLen) == 0 &&
686 (line[procNameLen] == ' ' || line[procNameLen] == '\t'))
687 {
688 crWarning("Using Chromium configuration for %s from %s",
689 procName, configPath);
690 args = crStrSplit(line + procNameLen + 1, " ");
691 return args;
692 }
693 }
694 fclose(f);
695 return NULL;
696}
697
698
699static int Mothership_Awake = 0;
700
701
702/**
703 * Signal handler to determine when mothership is ready.
704 */
705static void
706MothershipPhoneHome(int signo)
707{
708 crDebug("Got signal %d: mothership is awake!", signo);
709 Mothership_Awake = 1;
710}
711
712void stubSetDefaultConfigurationOptions(void)
713{
714 unsigned char key[16]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
715
716 stub.appDrawCursor = 0;
717 stub.minChromiumWindowWidth = 0;
718 stub.minChromiumWindowHeight = 0;
719 stub.maxChromiumWindowWidth = 0;
720 stub.maxChromiumWindowHeight = 0;
721 stub.matchChromiumWindowID = NULL;
722 stub.numIgnoreWindowID = 0;
723 stub.matchWindowTitle = NULL;
724 stub.ignoreFreeglutMenus = 0;
725 stub.trackWindowSize = 1;
726 stub.trackWindowPos = 1;
727 stub.trackWindowVisibility = 1;
728 stub.trackWindowVisibleRgn = 1;
729 stub.matchChromiumWindowCount = 0;
730 stub.spu_dir = NULL;
731 crNetSetRank(0);
732 crNetSetContextRange(32, 35);
733 crNetSetNodeRange("iam0", "iamvis20");
734 crNetSetKey(key,sizeof(key));
735 stub.force_pbuffers = 0;
736
737#ifdef WINDOWS
738# ifdef VBOX_WITH_WDDM
739 stub.bRunningUnderWDDM = false;
740# endif
741#endif
742}
743
744#ifdef CR_NEWWINTRACK
745# ifdef VBOX_WITH_WDDM
746static stubDispatchVisibleRegions(WindowInfo *pWindow)
747{
748 DWORD dwCount;
749 LPRGNDATA lpRgnData;
750
751 dwCount = GetRegionData(pWindow->hVisibleRegion, 0, NULL);
752 lpRgnData = crAlloc(dwCount);
753
754 if (lpRgnData)
755 {
756 GetRegionData(pWindow->hVisibleRegion, dwCount, lpRgnData);
757 crDebug("Dispatched WindowVisibleRegion (%i, cRects=%i)", pWindow->spuWindow, lpRgnData->rdh.nCount);
758 stub.spuDispatch.WindowVisibleRegion(pWindow->spuWindow, lpRgnData->rdh.nCount, (GLint*) lpRgnData->Buffer);
759 crFree(lpRgnData);
760 }
761 else crWarning("GetRegionData failed, VisibleRegions update failed");
762}
763
764static HRGN stubMakeRegionFromRects(PVBOXVIDEOCM_CMD_RECTS pRegions, uint32_t start)
765{
766 HRGN hRgn, hTmpRgn;
767 uint32_t i;
768
769 if (pRegions->RectsInfo.cRects<=start)
770 {
771 return INVALID_HANDLE_VALUE;
772 }
773
774 hRgn = CreateRectRgn(0, 0, 0, 0);
775 for (i=start; i<pRegions->RectsInfo.cRects; ++i)
776 {
777 hTmpRgn = CreateRectRgnIndirect(&pRegions->RectsInfo.aRects[i]);
778 CombineRgn(hRgn, hRgn, hTmpRgn, RGN_OR);
779 DeleteObject(hTmpRgn);
780 }
781 return hRgn;
782}
783
784# endif /* VBOX_WITH_WDDM */
785
786static void stubSyncTrCheckWindowsCB(unsigned long key, void *data1, void *data2)
787{
788 WindowInfo *pWindow = (WindowInfo *) data1;
789 (void) data2;
790
791 if (pWindow->type!=CHROMIUM || pWindow->spuWindow==0)
792 {
793 return;
794 }
795
796 stub.spu->dispatch_table.VBoxPackSetInjectID(pWindow->u32ClientID);
797
798 if (!stubSystemWindowExist(pWindow))
799 {
800#ifdef WINDOWS
801 stubDestroyWindow(0, (GLint)pWindow->hWnd);
802#else
803 stubDestroyWindow(0, (GLint)pWindow->drawable);
804#endif
805 /*No need to flush here as crWindowDestroy does it*/
806 return;
807 }
808
809#if defined(WINDOWS) && defined(VBOX_WITH_WDDM)
810 if (stub.bRunningUnderWDDM)
811 return;
812#endif
813 stubCheckWindowState(pWindow, GL_TRUE);
814}
815
816static DECLCALLBACK(int) stubSyncThreadProc(RTTHREAD ThreadSelf, void *pvUser)
817{
818#ifdef WINDOWS
819 MSG msg;
820# ifdef VBOX_WITH_WDDM
821 HMODULE hVBoxD3D = NULL;
822 GLint spuConnection = 0;
823# endif
824#endif
825
826 (void) pvUser;
827
828 crDebug("Sync thread started");
829#ifdef WINDOWS
830 PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE);
831# ifdef VBOX_WITH_WDDM
832 hVBoxD3D = NULL;
833 if (!GetModuleHandleEx(0, VBOX_MODNAME_DISPD3D, &hVBoxD3D))
834 {
835 crDebug("GetModuleHandleEx failed err %d", GetLastError());
836 hVBoxD3D = NULL;
837 }
838
839 if (hVBoxD3D)
840 {
841 crDebug("running with " VBOX_MODNAME_DISPD3D);
842 stub.trackWindowVisibleRgn = 0;
843 stub.bRunningUnderWDDM = true;
844 }
845# endif /* VBOX_WITH_WDDM */
846#endif /* WINDOWS */
847
848 crLockMutex(&stub.mutex);
849#if defined(WINDOWS) && defined(VBOX_WITH_WDDM)
850 spuConnection =
851#endif
852 stub.spu->dispatch_table.VBoxPackSetInjectThread(NULL);
853#if defined(WINDOWS) && defined(VBOX_WITH_WDDM)
854 if (stub.bRunningUnderWDDM && !spuConnection)
855 {
856 crError("VBoxPackSetInjectThread failed!");
857 }
858#endif
859 crUnlockMutex(&stub.mutex);
860
861 RTThreadUserSignal(ThreadSelf);
862
863 while(!stub.bShutdownSyncThread)
864 {
865#ifdef WINDOWS
866 if (!PeekMessage(&msg, 0, 0, 0, PM_REMOVE))
867 {
868# ifdef VBOX_WITH_WDDM
869 if (stub.bRunningUnderWDDM)
870 {
871
872 }
873 else
874# endif
875 {
876 crHashtableWalk(stub.windowTable, stubSyncTrCheckWindowsCB, NULL);
877 RTThreadSleep(50);
878 }
879 }
880 else
881 {
882 if (WM_QUIT==msg.message)
883 {
884 crDebug("Sync thread got WM_QUIT");
885 break;
886 }
887 else
888 {
889 TranslateMessage(&msg);
890 DispatchMessage(&msg);
891 }
892 }
893#else
894 crLockMutex(&stub.mutex);
895 crHashtableWalk(stub.windowTable, stubSyncTrCheckWindowsCB, NULL);
896 crUnlockMutex(&stub.mutex);
897 RTThreadSleep(50);
898#endif
899 }
900
901#ifdef VBOX_WITH_WDDM
902 if (spuConnection)
903 {
904 stub.spu->dispatch_table.VBoxConDestroy(spuConnection);
905 }
906 if (hVBoxD3D)
907 {
908 FreeLibrary(hVBoxD3D);
909 }
910#endif
911 crDebug("Sync thread stopped");
912 return 0;
913}
914#endif /* CR_NEWWINTRACK */
915
916/**
917 * Do one-time initializations for the faker.
918 * Returns TRUE on success, FALSE otherwise.
919 */
920static bool
921stubInitLocked(void)
922{
923 /* Here is where we contact the mothership to find out what we're supposed
924 * to be doing. Networking code in a DLL initializer. I sure hope this
925 * works :)
926 *
927 * HOW can I pass the mothership address to this if I already know it?
928 */
929
930 CRConnection *conn = NULL;
931 char response[1024];
932 char **spuchain;
933 int num_spus;
934 int *spu_ids;
935 char **spu_names;
936 const char *app_id;
937 int i;
938 int disable_sync = 0;
939#if defined(WINDOWS) && defined(VBOX_WITH_WDDM)
940 HMODULE hVBoxD3D = NULL;
941#endif
942
943 stubInitVars();
944
945 crGetProcName(response, 1024);
946 crDebug("Stub launched for %s", response);
947
948#if defined(CR_NEWWINTRACK) && !defined(WINDOWS)
949 /*@todo when vm boots with compiz turned on, new code causes hang in xcb_wait_for_reply in the sync thread
950 * as at the start compiz runs our code under XGrabServer.
951 */
952 if (!crStrcmp(response, "compiz") || !crStrcmp(response, "compiz_real") || !crStrcmp(response, "compiz.real")
953 || !crStrcmp(response, "compiz-bin"))
954 {
955 disable_sync = 1;
956 }
957#endif
958
959 /* @todo check if it'd be of any use on other than guests, no use for windows */
960 app_id = crGetenv( "CR_APPLICATION_ID_NUMBER" );
961
962 crNetInit( NULL, NULL );
963
964#ifndef WINDOWS
965 {
966 CRNetServer ns;
967
968 ns.name = "vboxhgcm://host:0";
969 ns.buffer_size = 1024;
970 crNetServerConnect(&ns
971#if defined(VBOX_WITH_CRHGSMI) && defined(IN_GUEST)
972 , NULL
973#endif
974 );
975 if (!ns.conn)
976 {
977 crWarning("Failed to connect to host. Make sure 3D acceleration is enabled for this VM.");
978 return false;
979 }
980 else
981 {
982 crNetFreeConnection(ns.conn);
983 }
984#if 0 && defined(CR_NEWWINTRACK)
985 {
986 Status st = XInitThreads();
987 if (st==0)
988 {
989 crWarning("XInitThreads returned %i", (int)st);
990 }
991 }
992#endif
993 }
994#endif
995
996 strcpy(response, "2 0 feedback 1 pack");
997 spuchain = crStrSplit( response, " " );
998 num_spus = crStrToInt( spuchain[0] );
999 spu_ids = (int *) crAlloc( num_spus * sizeof( *spu_ids ) );
1000 spu_names = (char **) crAlloc( num_spus * sizeof( *spu_names ) );
1001 for (i = 0 ; i < num_spus ; i++)
1002 {
1003 spu_ids[i] = crStrToInt( spuchain[2*i+1] );
1004 spu_names[i] = crStrdup( spuchain[2*i+2] );
1005 crDebug( "SPU %d/%d: (%d) \"%s\"", i+1, num_spus, spu_ids[i], spu_names[i] );
1006 }
1007
1008 stubSetDefaultConfigurationOptions();
1009
1010#if defined(WINDOWS) && defined(VBOX_WITH_WDDM)
1011 hVBoxD3D = NULL;
1012 if (!GetModuleHandleEx(0, VBOX_MODNAME_DISPD3D, &hVBoxD3D))
1013 {
1014 crDebug("GetModuleHandleEx failed err %d", GetLastError());
1015 hVBoxD3D = NULL;
1016 }
1017
1018 if (hVBoxD3D)
1019 {
1020 disable_sync = 1;
1021 crDebug("running with %s", VBOX_MODNAME_DISPD3D);
1022 stub.trackWindowVisibleRgn = 0;
1023 /* @todo: should we enable that? */
1024 stub.trackWindowSize = 0;
1025 stub.trackWindowPos = 0;
1026 stub.trackWindowVisibility = 0;
1027 stub.bRunningUnderWDDM = true;
1028 }
1029#endif
1030
1031 stub.spu = crSPULoadChain( num_spus, spu_ids, spu_names, stub.spu_dir, NULL );
1032
1033 crFree( spuchain );
1034 crFree( spu_ids );
1035 for (i = 0; i < num_spus; ++i)
1036 crFree(spu_names[i]);
1037 crFree( spu_names );
1038
1039 // spu chain load failed somewhere
1040 if (!stub.spu) {
1041 return false;
1042 }
1043
1044 crSPUInitDispatchTable( &glim );
1045
1046 /* This is unlikely to change -- We still want to initialize our dispatch
1047 * table with the functions of the first SPU in the chain. */
1048 stubInitSPUDispatch( stub.spu );
1049
1050 /* we need to plug one special stub function into the dispatch table */
1051 glim.GetChromiumParametervCR = stub_GetChromiumParametervCR;
1052
1053#if !defined(VBOX_NO_NATIVEGL)
1054 /* Load pointers to native OpenGL functions into stub.nativeDispatch */
1055 stubInitNativeDispatch();
1056#endif
1057
1058/*crDebug("stub init");
1059raise(SIGINT);*/
1060
1061#ifdef WINDOWS
1062# ifndef CR_NEWWINTRACK
1063 stubInstallWindowMessageHook();
1064# endif
1065#endif
1066
1067#ifdef CR_NEWWINTRACK
1068 {
1069 int rc;
1070
1071 RTR3InitDll(RTR3INIT_FLAGS_UNOBTRUSIVE);
1072
1073 if (!disable_sync)
1074 {
1075 crDebug("Starting sync thread");
1076
1077 rc = RTThreadCreate(&stub.hSyncThread, stubSyncThreadProc, NULL, 0, RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, "Sync");
1078 if (RT_FAILURE(rc))
1079 {
1080 crError("Failed to start sync thread! (%x)", rc);
1081 }
1082 RTThreadUserWait(stub.hSyncThread, 60 * 1000);
1083 RTThreadUserReset(stub.hSyncThread);
1084
1085 crDebug("Going on");
1086 }
1087 }
1088#endif
1089
1090#ifdef GLX
1091 stub.xshmSI.shmid = -1;
1092 stub.bShmInitFailed = GL_FALSE;
1093 stub.pGLXPixmapsHash = crAllocHashtable();
1094
1095 stub.bXExtensionsChecked = GL_FALSE;
1096 stub.bHaveXComposite = GL_FALSE;
1097 stub.bHaveXFixes = GL_FALSE;
1098#endif
1099
1100 return true;
1101}
1102
1103/**
1104 * Do one-time initializations for the faker.
1105 * Returns TRUE on success, FALSE otherwise.
1106 */
1107bool
1108stubInit(void)
1109{
1110 bool bRc = true;
1111 /* we need to serialize the initialization, otherwise racing is possible
1112 * for XPDM-based d3d when a d3d switcher is testing the gl lib in two or more threads
1113 * NOTE: the STUB_INIT_LOCK/UNLOCK is a NOP for non-win currently */
1114 STUB_INIT_LOCK();
1115 if (!stub_initialized)
1116 bRc = stub_initialized = stubInitLocked();
1117 STUB_INIT_UNLOCK();
1118 return bRc;
1119}
1120
1121/* Sigh -- we can't do initialization at load time, since Windows forbids
1122 * the loading of other libraries from DLLMain. */
1123
1124#ifdef LINUX
1125/* GCC crap
1126 *void (*stub_init_ptr)(void) __attribute__((section(".ctors"))) = __stubInit; */
1127#endif
1128
1129#ifdef WINDOWS
1130#define WIN32_LEAN_AND_MEAN
1131#include <windows.h>
1132
1133#if 1//def DEBUG_misha
1134 /* debugging: this is to be able to catch first-chance notifications
1135 * for exceptions other than EXCEPTION_BREAKPOINT in kernel debugger */
1136# define VDBG_VEHANDLER
1137#endif
1138
1139#ifdef VDBG_VEHANDLER
1140# include <dbghelp.h>
1141static PVOID g_VBoxVehHandler = NULL;
1142static DWORD g_VBoxVehEnable = 0;
1143
1144/* generate a crash dump on exception */
1145#define VBOXVEH_F_DUMP 0x00000001
1146/* generate a debugger breakpoint exception */
1147#define VBOXVEH_F_BREAK 0x00000002
1148/* exit on exception */
1149#define VBOXVEH_F_EXIT 0x00000004
1150
1151static DWORD g_VBoxVehFlags = VBOXVEH_F_DUMP
1152#ifdef DEBUG_misha
1153 | VBOXVEH_F_BREAK
1154#endif
1155 ;
1156
1157typedef BOOL WINAPI FNVBOXDBG_MINIDUMPWRITEDUMP(HANDLE hProcess,
1158 DWORD ProcessId,
1159 HANDLE hFile,
1160 MINIDUMP_TYPE DumpType,
1161 PMINIDUMP_EXCEPTION_INFORMATION ExceptionParam,
1162 PMINIDUMP_USER_STREAM_INFORMATION UserStreamParam,
1163 PMINIDUMP_CALLBACK_INFORMATION CallbackParam);
1164typedef FNVBOXDBG_MINIDUMPWRITEDUMP *PFNVBOXDBG_MINIDUMPWRITEDUMP;
1165
1166static HMODULE g_hVBoxMdDbgHelp = NULL;
1167static PFNVBOXDBG_MINIDUMPWRITEDUMP g_pfnVBoxMdMiniDumpWriteDump = NULL;
1168static uint32_t g_cVBoxMdFilePrefixLen = 0;
1169static WCHAR g_aszwVBoxMdFilePrefix[MAX_PATH];
1170static WCHAR g_aszwVBoxMdDumpCount = 0;
1171static MINIDUMP_TYPE g_enmVBoxMdDumpType = MiniDumpNormal
1172 | MiniDumpWithDataSegs
1173 | MiniDumpWithFullMemory
1174 | MiniDumpWithHandleData
1175//// | MiniDumpFilterMemory
1176//// | MiniDumpScanMemory
1177// | MiniDumpWithUnloadedModules
1178//// | MiniDumpWithIndirectlyReferencedMemory
1179//// | MiniDumpFilterModulePaths
1180// | MiniDumpWithProcessThreadData
1181// | MiniDumpWithPrivateReadWriteMemory
1182//// | MiniDumpWithoutOptionalData
1183// | MiniDumpWithFullMemoryInfo
1184// | MiniDumpWithThreadInfo
1185// | MiniDumpWithCodeSegs
1186// | MiniDumpWithFullAuxiliaryState
1187// | MiniDumpWithPrivateWriteCopyMemory
1188// | MiniDumpIgnoreInaccessibleMemory
1189// | MiniDumpWithTokenInformation
1190//// | MiniDumpWithModuleHeaders
1191//// | MiniDumpFilterTriage
1192 ;
1193
1194
1195
1196#define VBOXMD_DUMP_DIR_PREFIX_DEFAULT L"C:\\dumps\\vboxdmp"
1197
1198static HMODULE loadSystemDll(const char *pszName)
1199{
1200 char szPath[MAX_PATH];
1201 UINT cchPath = GetSystemDirectoryA(szPath, sizeof(szPath));
1202 size_t cbName = strlen(pszName) + 1;
1203 if (cchPath + 1 + cbName > sizeof(szPath))
1204 {
1205 SetLastError(ERROR_FILENAME_EXCED_RANGE);
1206 return NULL;
1207 }
1208 szPath[cchPath] = '\\';
1209 memcpy(&szPath[cchPath + 1], pszName, cbName);
1210 return LoadLibraryA(szPath);
1211}
1212
1213static DWORD vboxMdMinidumpCreate(struct _EXCEPTION_POINTERS *pExceptionInfo)
1214{
1215 WCHAR aszwMdFileName[MAX_PATH];
1216 HANDLE hProcess = GetCurrentProcess();
1217 DWORD ProcessId = GetCurrentProcessId();
1218 MINIDUMP_EXCEPTION_INFORMATION ExceptionInfo;
1219 HANDLE hFile;
1220 DWORD winErr = ERROR_SUCCESS;
1221
1222 if (!g_pfnVBoxMdMiniDumpWriteDump)
1223 {
1224 if (!g_hVBoxMdDbgHelp)
1225 {
1226 g_hVBoxMdDbgHelp = loadSystemDll("DbgHelp.dll");
1227 if (!g_hVBoxMdDbgHelp)
1228 return GetLastError();
1229 }
1230
1231 g_pfnVBoxMdMiniDumpWriteDump = (PFNVBOXDBG_MINIDUMPWRITEDUMP)GetProcAddress(g_hVBoxMdDbgHelp, "MiniDumpWriteDump");
1232 if (!g_pfnVBoxMdMiniDumpWriteDump)
1233 return GetLastError();
1234 }
1235
1236 /* @todo: this is a tmp stuff until we get that info from the settings properly */
1237 if (!g_cVBoxMdFilePrefixLen)
1238 {
1239 g_cVBoxMdFilePrefixLen = sizeof (VBOXMD_DUMP_DIR_PREFIX_DEFAULT)/sizeof (g_aszwVBoxMdFilePrefix[0]) - 1 /* <- don't include nul terminator */;
1240 memcpy(g_aszwVBoxMdFilePrefix, VBOXMD_DUMP_DIR_PREFIX_DEFAULT, sizeof (VBOXMD_DUMP_DIR_PREFIX_DEFAULT));
1241 }
1242
1243
1244 if (RT_ELEMENTS(aszwMdFileName) <= g_cVBoxMdFilePrefixLen)
1245 {
1246 return ERROR_INVALID_STATE;
1247 }
1248
1249 ++g_aszwVBoxMdDumpCount;
1250
1251 memcpy(aszwMdFileName, g_aszwVBoxMdFilePrefix, g_cVBoxMdFilePrefixLen * sizeof (g_aszwVBoxMdFilePrefix[0]));
1252 swprintf(aszwMdFileName + g_cVBoxMdFilePrefixLen, RT_ELEMENTS(aszwMdFileName) - g_cVBoxMdFilePrefixLen, L"%d_%d.dmp", ProcessId, g_aszwVBoxMdDumpCount);
1253
1254 hFile = CreateFileW(aszwMdFileName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
1255 if (hFile == INVALID_HANDLE_VALUE)
1256 return GetLastError();
1257
1258 ExceptionInfo.ThreadId = GetCurrentThreadId();
1259 ExceptionInfo.ExceptionPointers = pExceptionInfo;
1260 ExceptionInfo.ClientPointers = FALSE;
1261
1262 if (!g_pfnVBoxMdMiniDumpWriteDump(hProcess, ProcessId, hFile, g_enmVBoxMdDumpType, &ExceptionInfo, NULL, NULL))
1263 winErr = GetLastError();
1264
1265 CloseHandle(hFile);
1266 return winErr;
1267}
1268
1269LONG WINAPI vboxVDbgVectoredHandler(struct _EXCEPTION_POINTERS *pExceptionInfo)
1270{
1271 PEXCEPTION_RECORD pExceptionRecord = pExceptionInfo->ExceptionRecord;
1272 PCONTEXT pContextRecord = pExceptionInfo->ContextRecord;
1273 switch (pExceptionRecord->ExceptionCode)
1274 {
1275 case EXCEPTION_BREAKPOINT:
1276 case EXCEPTION_ACCESS_VIOLATION:
1277 case EXCEPTION_STACK_OVERFLOW:
1278 case EXCEPTION_ARRAY_BOUNDS_EXCEEDED:
1279 case EXCEPTION_FLT_DIVIDE_BY_ZERO:
1280 case EXCEPTION_FLT_INVALID_OPERATION:
1281 case EXCEPTION_INT_DIVIDE_BY_ZERO:
1282 case EXCEPTION_ILLEGAL_INSTRUCTION:
1283 if (g_VBoxVehFlags & VBOXVEH_F_BREAK)
1284 {
1285 BOOL fBreak = TRUE;
1286 if (pExceptionRecord->ExceptionCode == EXCEPTION_BREAKPOINT)
1287 {
1288 HANDLE hProcess = GetCurrentProcess();
1289 BOOL fDebuggerPresent = FALSE;
1290 /* we do not want to generate breakpoint exceptions recursively, so do it only when running under debugger */
1291 if (CheckRemoteDebuggerPresent(hProcess, &fDebuggerPresent))
1292 fBreak = !!fDebuggerPresent;
1293 else
1294 fBreak = FALSE; /* <- the function has failed, don't break for sanity */
1295 }
1296
1297 if (fBreak)
1298 {
1299 RT_BREAKPOINT();
1300 }
1301 }
1302
1303 if (g_VBoxVehFlags & VBOXVEH_F_DUMP)
1304 vboxMdMinidumpCreate(pExceptionInfo);
1305
1306 if (g_VBoxVehFlags & VBOXVEH_F_EXIT)
1307 exit(1);
1308 break;
1309 default:
1310 break;
1311 }
1312 return EXCEPTION_CONTINUE_SEARCH;
1313}
1314
1315void vboxVDbgVEHandlerRegister()
1316{
1317 CRASSERT(!g_VBoxVehHandler);
1318 g_VBoxVehHandler = AddVectoredExceptionHandler(1,vboxVDbgVectoredHandler);
1319 CRASSERT(g_VBoxVehHandler);
1320}
1321
1322void vboxVDbgVEHandlerUnregister()
1323{
1324 ULONG uResult;
1325 if (g_VBoxVehHandler)
1326 {
1327 uResult = RemoveVectoredExceptionHandler(g_VBoxVehHandler);
1328 CRASSERT(uResult);
1329 g_VBoxVehHandler = NULL;
1330 }
1331}
1332#endif
1333
1334/* Windows crap */
1335BOOL WINAPI DllMain(HINSTANCE hDLLInst, DWORD fdwReason, LPVOID lpvReserved)
1336{
1337 (void) lpvReserved;
1338
1339 switch (fdwReason)
1340 {
1341 case DLL_PROCESS_ATTACH:
1342 {
1343 CRNetServer ns;
1344
1345#ifdef CHROMIUM_THREADSAFE
1346 crInitTSD(&g_stubCurrentContextTSD);
1347#endif
1348
1349 crInitMutex(&stub_init_mutex);
1350
1351#ifdef VDBG_VEHANDLER
1352 g_VBoxVehEnable = !!crGetenv("CR_DBG_VEH_ENABLE");
1353 if (g_VBoxVehEnable)
1354 vboxVDbgVEHandlerRegister();
1355#endif
1356
1357 crNetInit(NULL, NULL);
1358 ns.name = "vboxhgcm://host:0";
1359 ns.buffer_size = 1024;
1360 crNetServerConnect(&ns
1361#if defined(VBOX_WITH_CRHGSMI) && defined(IN_GUEST)
1362 , NULL
1363#endif
1364);
1365 if (!ns.conn)
1366 {
1367 crDebug("Failed to connect to host (is guest 3d acceleration enabled?), aborting ICD load.");
1368#ifdef VDBG_VEHANDLER
1369 if (g_VBoxVehEnable)
1370 vboxVDbgVEHandlerUnregister();
1371#endif
1372 return FALSE;
1373 }
1374 else
1375 {
1376 crNetFreeConnection(ns.conn);
1377 }
1378
1379 break;
1380 }
1381
1382 case DLL_PROCESS_DETACH:
1383 {
1384 /* do exactly the same thing as for DLL_THREAD_DETACH since
1385 * DLL_THREAD_DETACH is not called for the thread doing DLL_PROCESS_DETACH according to msdn docs */
1386 stubSetCurrentContext(NULL);
1387 if (stub_initialized)
1388 {
1389 CRASSERT(stub.spu);
1390 stub.spu->dispatch_table.VBoxDetachThread();
1391 }
1392
1393 stubSPUSafeTearDown();
1394
1395#ifdef CHROMIUM_THREADSAFE
1396 crFreeTSD(&g_stubCurrentContextTSD);
1397#endif
1398
1399#ifdef VDBG_VEHANDLER
1400 if (g_VBoxVehEnable)
1401 vboxVDbgVEHandlerUnregister();
1402#endif
1403 break;
1404 }
1405
1406 case DLL_THREAD_ATTACH:
1407 {
1408 if (stub_initialized)
1409 {
1410 CRASSERT(stub.spu);
1411 stub.spu->dispatch_table.VBoxAttachThread();
1412 }
1413 break;
1414 }
1415
1416 case DLL_THREAD_DETACH:
1417 {
1418 stubSetCurrentContext(NULL);
1419 if (stub_initialized)
1420 {
1421 CRASSERT(stub.spu);
1422 stub.spu->dispatch_table.VBoxDetachThread();
1423 }
1424 break;
1425 }
1426
1427 default:
1428 break;
1429 }
1430
1431 return TRUE;
1432}
1433#endif
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