VirtualBox

source: vbox/trunk/src/VBox/Additions/common/crOpenGL/fakedri_drv.c@ 21094

Last change on this file since 21094 was 20509, checked in by vboxsync, 16 years ago

OSE headers

File size: 15.9 KB
Line 
1/* $Id$ */
2
3/** @file
4 * VBox OpenGL DRI driver functions
5 */
6
7/*
8 * Copyright (C) 2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#define _GNU_SOURCE 1
24
25#include "cr_error.h"
26#include "cr_gl.h"
27#include "cr_mem.h"
28#include "stub.h"
29#include "fakedri_drv.h"
30#include "dri_glx.h"
31#include "iprt/mem.h"
32#include "iprt/err.h"
33#include <dlfcn.h>
34#include <elf.h>
35#include <unistd.h>
36#include "xf86.h"
37
38#define VBOX_NO_MESA_PATCH_REPORTS
39
40//#define DEBUG_DRI_CALLS
41
42//@todo this could be different...
43#ifdef RT_ARCH_AMD64
44# define DRI_DEFAULT_DRIVER_DIR "/usr/lib64/dri:/usr/lib/dri"
45# define DRI_XORG_DRV_DIR "/usr/lib/xorg/modules/drivers/"
46#else
47# define DRI_DEFAULT_DRIVER_DIR "/usr/lib/dri"
48# define DRI_XORG_DRV_DIR "/usr/lib/xorg/modules/drivers/"
49#endif
50
51#ifdef DEBUG_DRI_CALLS
52 #define SWDRI_SHOWNAME(pext, func) \
53 crDebug("SWDRI: sc %s->%s", #pext, #func)
54#else
55 #define SWDRI_SHOWNAME(pext, func)
56#endif
57
58#define SWDRI_SAFECALL(pext, func, ...) \
59 SWDRI_SHOWNAME(pext, func); \
60 if (pext && pext->func){ \
61 (*pext->func)(__VA_ARGS__); \
62 } else { \
63 crDebug("swcore_call NULL for "#func); \
64 }
65
66#define SWDRI_SAFERET(pext, func, ...) \
67 SWDRI_SHOWNAME(pext, func); \
68 if (pext && pext->func){ \
69 return (*pext->func)(__VA_ARGS__); \
70 } else { \
71 crDebug("swcore_call NULL for "#func); \
72 return 0; \
73 }
74
75#define SWDRI_SAFERET_CORE(func, ...) SWDRI_SAFERET(gpSwDriCoreExternsion, func, __VA_ARGS__)
76#define SWDRI_SAFECALL_CORE(func, ...) SWDRI_SAFECALL(gpSwDriCoreExternsion, func, __VA_ARGS__)
77#define SWDRI_SAFERET_SWRAST(func, ...) SWDRI_SAFERET(gpSwDriSwrastExtension, func, __VA_ARGS__)
78#define SWDRI_SAFECALL_SWRAST(func, ...) SWDRI_SAFECALL(gpSwDriSwrastExtension, func, __VA_ARGS__)
79
80#ifndef PAGESIZE
81#define PAGESIZE 4096
82#endif
83
84static struct _glapi_table vbox_glapi_table;
85fakedri_glxapi_table glxim;
86
87static const __DRIextension **gppSwDriExternsion = NULL;
88static const __DRIcoreExtension *gpSwDriCoreExternsion = NULL;
89static const __DRIswrastExtension *gpSwDriSwrastExtension = NULL;
90
91extern const __DRIextension * __driDriverExtensions[];
92
93#define GLAPI_ENTRY(Func) pGLTable->Func = cr_gl##Func;
94static void
95vboxFillMesaGLAPITable(struct _glapi_table *pGLTable)
96{
97 #include "fakedri_glfuncsList.h"
98
99 pGLTable->SampleMaskSGIS = cr_glSampleMaskEXT;
100 pGLTable->SamplePatternSGIS = cr_glSamplePatternEXT;
101 pGLTable->WindowPos2dMESA = cr_glWindowPos2d;
102 pGLTable->WindowPos2dvMESA = cr_glWindowPos2dv;
103 pGLTable->WindowPos2fMESA = cr_glWindowPos2f;
104 pGLTable->WindowPos2fvMESA = cr_glWindowPos2fv;
105 pGLTable->WindowPos2iMESA = cr_glWindowPos2i;
106 pGLTable->WindowPos2ivMESA = cr_glWindowPos2iv;
107 pGLTable->WindowPos2sMESA = cr_glWindowPos2s;
108 pGLTable->WindowPos2svMESA = cr_glWindowPos2sv;
109 pGLTable->WindowPos3dMESA = cr_glWindowPos3d;
110 pGLTable->WindowPos3dvMESA = cr_glWindowPos3dv;
111 pGLTable->WindowPos3fMESA = cr_glWindowPos3f;
112 pGLTable->WindowPos3fvMESA = cr_glWindowPos3fv;
113 pGLTable->WindowPos3iMESA = cr_glWindowPos3i;
114 pGLTable->WindowPos3ivMESA = cr_glWindowPos3iv;
115 pGLTable->WindowPos3sMESA = cr_glWindowPos3s;
116 pGLTable->WindowPos3svMESA = cr_glWindowPos3sv;
117};
118#undef GLAPI_ENTRY
119
120#define GLXAPI_ENTRY(Func) pGLXTable->Func = VBOXGLXTAG(glX##Func);
121static void
122vboxFillGLXAPITable(fakedri_glxapi_table *pGLXTable)
123{
124 #include "fakedri_glxfuncsList.h"
125}
126#undef GLXAPI_ENTRY
127
128static void
129vboxPatchMesaExport(const char* psFuncName, const void *pStart, const void *pEnd)
130{
131 Dl_info dlip;
132 Elf32_Sym* sym=0;
133 int rv;
134 void *alPatch;
135 void *pMesaEntry;
136
137#ifndef VBOX_NO_MESA_PATCH_REPORTS
138 crDebug("vboxPatchMesaExport: %s", psFuncName);
139#endif
140
141 pMesaEntry = dlsym(RTLD_DEFAULT, psFuncName);
142
143 if (!pMesaEntry)
144 {
145 crDebug("%s not defined in current scope, are we being loaded by mesa's libGL.so?", psFuncName);
146 return;
147 }
148
149 rv = dladdr1(pMesaEntry, &dlip, (void**)&sym, RTLD_DL_SYMENT);
150 if (!rv || !sym)
151 {
152 crError("Failed to get size for %p(%s)", pMesaEntry, psFuncName);
153 return;
154 }
155
156#if VBOX_OGL_GLX_USE_CSTUBS
157 {
158 Dl_info dlip1;
159 Elf32_Sym* sym1=0;
160 int rv;
161
162 rv = dladdr1(pStart, &dlip1, (void**)&sym1, RTLD_DL_SYMENT);
163 if (!rv || !sym)
164 {
165 crError("Failed to get size for %p", pStart);
166 return;
167 }
168
169 pEnd = pStart + sym1->st_size;
170 crDebug("VBox Entry: %p, start: %p(%s:%s), size: %i", pStart, dlip1.dli_saddr, dlip1.dli_fname, dlip1.dli_sname, sym1->st_size);
171 }
172#endif
173
174#ifndef VBOX_NO_MESA_PATCH_REPORTS
175 crDebug("Mesa Entry: %p, start: %p(%s:%s), size: %i", pMesaEntry, dlip.dli_saddr, dlip.dli_fname, dlip.dli_sname, sym->st_size);
176 crDebug("Vbox code: start: %p, end %p, size: %i", pStart, pEnd, pEnd-pStart);
177#endif
178
179 if (sym->st_size<(pEnd-pStart))
180 {
181 crDebug("Can't patch size too small.(%s)", psFuncName);
182 return;
183 }
184
185 /* Get aligned start adress we're going to patch*/
186 alPatch = (void*) ((uintptr_t)dlip.dli_saddr & ~(uintptr_t)(PAGESIZE-1));
187
188#ifndef VBOX_NO_MESA_PATCH_REPORTS
189 crDebug("MProtecting: %p, %i", alPatch, dlip.dli_saddr-alPatch+pEnd-pStart);
190#endif
191
192 /* Get write access to mesa functions */
193 rv = RTMemProtect(alPatch, dlip.dli_saddr-alPatch+pEnd-pStart,
194 RTMEM_PROT_READ|RTMEM_PROT_WRITE|RTMEM_PROT_EXEC);
195 if (RT_FAILURE(rv))
196 {
197 crError("mprotect failed with %x (%s)", rv, psFuncName);
198 }
199
200#ifndef VBOX_NO_MESA_PATCH_REPORTS
201 crDebug("Writing %i bytes to %p from %p", pEnd-pStart, dlip.dli_saddr, pStart);
202#endif
203
204 crMemcpy(dlip.dli_saddr, pStart, pEnd-pStart);
205
206 /*@todo Restore the protection, probably have to check what was it before us...*/
207 rv = RTMemProtect(alPatch, dlip.dli_saddr-alPatch+pEnd-pStart,
208 RTMEM_PROT_READ|RTMEM_PROT_EXEC);
209 if (RT_FAILURE(rv))
210 {
211 crError("mprotect2 failed with %x (%s)", rv, psFuncName);
212 }
213}
214
215#ifdef VBOX_OGL_GLX_USE_CSTUBS
216static void
217# define GLXAPI_ENTRY(Func) vboxPatchMesaExport("glX"#Func, &vbox_glX##Func, NULL);
218vboxPatchMesaExports()
219#else
220static void
221# define GLXAPI_ENTRY(Func) vboxPatchMesaExport("glX"#Func, &vbox_glX##Func, &vbox_glX##Func##_EndProc);
222vboxPatchMesaExports()
223#endif
224{
225 crDebug("Patching mesa glx entries");
226 #include "fakedri_glxfuncsList.h"
227}
228#undef GLXAPI_ENTRY
229
230bool vbox_load_sw_dri()
231{
232 const char *libPaths, *p, *next;;
233 char realDriverName[200];
234 void *handle;
235 int len, i;
236
237 /*code from Mesa-7.2/src/glx/x11/dri_common.c:driOpenDriver*/
238
239 libPaths = NULL;
240 if (geteuid() == getuid()) {
241 /* don't allow setuid apps to use LIBGL_DRIVERS_PATH */
242 libPaths = getenv("LIBGL_DRIVERS_PATH");
243 if (!libPaths)
244 libPaths = getenv("LIBGL_DRIVERS_DIR"); /* deprecated */
245 }
246 if (libPaths == NULL)
247 libPaths = DRI_DEFAULT_DRIVER_DIR;
248
249 handle = NULL;
250 for (p = libPaths; *p; p = next)
251 {
252 next = strchr(p, ':');
253 if (next == NULL)
254 {
255 len = strlen(p);
256 next = p + len;
257 }
258 else
259 {
260 len = next - p;
261 next++;
262 }
263
264 snprintf(realDriverName, sizeof realDriverName, "%.*s/%s_dri.so", len, p, "swrast");
265 crDebug("trying %s", realDriverName);
266 handle = dlopen(realDriverName, RTLD_NOW | RTLD_LOCAL);
267 if (handle) break;
268 }
269
270 /*end code*/
271
272 if (handle) gppSwDriExternsion = dlsym(handle, "__driDriverExtensions");
273
274 if (!gppSwDriExternsion)
275 {
276 crDebug("%s doesn't export __driDriverExtensions", realDriverName);
277 return false;
278 }
279 crDebug("loaded %s", realDriverName);
280
281 for (i = 0; gppSwDriExternsion[i]; i++)
282 {
283 if (strcmp(gppSwDriExternsion[i]->name, __DRI_CORE) == 0)
284 gpSwDriCoreExternsion = (__DRIcoreExtension *) gppSwDriExternsion[i];
285 if (strcmp(gppSwDriExternsion[i]->name, __DRI_SWRAST) == 0)
286 gpSwDriSwrastExtension = (__DRIswrastExtension *) gppSwDriExternsion[i];
287 }
288
289 return gpSwDriCoreExternsion && gpSwDriSwrastExtension;
290}
291
292void __attribute__ ((constructor)) vbox_install_into_mesa(void)
293{
294 if (!stubInit())
295 {
296 crDebug("vboxdriInitScreen: stubInit failed");
297 return;
298 }
299
300 {
301 void (*pxf86Msg)(MessageType type, const char *format, ...) _printf_attribute(2,3);
302
303 pxf86Msg = dlsym(RTLD_DEFAULT, "xf86Msg");
304 if (pxf86Msg)
305 {
306 pxf86Msg(X_INFO, "Next line is added to allow vboxvideo_drv.so to appear as whitelisted driver\n");
307 pxf86Msg(X_INFO, "The file referenced, is *NOT* loaded\n");
308 pxf86Msg(X_INFO, "Loading %s/ati_drv.so\n", DRI_XORG_DRV_DIR);
309
310 /* we're failing to proxy software dri driver calls for certain xservers, so just make sure we're unloaded for now */
311 __driDriverExtensions[0] = NULL;
312 return;
313 }
314 }
315
316 /* Load swrast_dri.so to proxy dri related calls there. */
317 if (!vbox_load_sw_dri())
318 {
319 crDebug("vboxdriInitScreen: vbox_load_sw_dri failed...going to fail badly");
320 return;
321 }
322
323 /* Handle gl api.
324 * In the end application call would look like this:
325 * app call glFoo->(mesa asm dispatch stub)->cr_glFoo(vbox asm dispatch stub)->SPU Foo function(packspuFoo or alike)
326 * Note, we don't need to install extension functions via _glapi_add_dispatch, because we'd override glXGetProcAddress.
327 */
328 /* We don't support all mesa's functions. Initialize our table to mesa dispatch first*/
329 crMemcpy(&vbox_glapi_table, _glapi_get_dispatch(), sizeof(struct _glapi_table));
330 /* Now install our assembly dispatch entries into table */
331 vboxFillMesaGLAPITable(&vbox_glapi_table);
332 /* Install our dispatch table into mesa */
333 _glapi_set_dispatch(&vbox_glapi_table);
334
335 /* Handle glx api.
336 * In the end application call would look like this:
337 * app call glxFoo->(mesa asm dispatch stub patched with vbox_glXFoo:jmp glxim[Foo's index])->VBOXGLXTAG(glxFoo)
338 */
339 /* Fill structure used by our assembly stubs */
340 vboxFillGLXAPITable(&glxim);
341 /* Now patch functions exported by libGL.so */
342 vboxPatchMesaExports();
343}
344
345/*
346 * @todo we're missing first glx related call from the client application.
347 * Luckily, this doesn't add much problems, except for some cases.
348 */
349
350/* __DRIcoreExtension */
351
352static __DRIscreen *
353vboxdriCreateNewScreen(int screen, int fd, unsigned int sarea_handle,
354 const __DRIextension **extensions, const __DRIconfig ***driverConfigs,
355 void *loaderPrivate)
356{
357 (void) fd;
358 (void) sarea_handle;
359 SWDRI_SAFERET_SWRAST(createNewScreen, screen, extensions, driverConfigs, loaderPrivate);
360}
361
362static void
363vboxdriDestroyScreen(__DRIscreen *screen)
364{
365 SWDRI_SAFECALL_CORE(destroyScreen, screen);
366}
367
368static const __DRIextension **
369vboxdriGetExtensions(__DRIscreen *screen)
370{
371 SWDRI_SAFERET_CORE(getExtensions, screen);
372}
373
374static int
375vboxdriGetConfigAttrib(const __DRIconfig *config,
376 unsigned int attrib,
377 unsigned int *value)
378{
379 SWDRI_SAFERET_CORE(getConfigAttrib, config, attrib, value);
380}
381
382static int
383vboxdriIndexConfigAttrib(const __DRIconfig *config, int index,
384 unsigned int *attrib, unsigned int *value)
385{
386 SWDRI_SAFERET_CORE(indexConfigAttrib, config, index, attrib, value);
387}
388
389static __DRIdrawable *
390vboxdriCreateNewDrawable(__DRIscreen *screen,
391 const __DRIconfig *config,
392 unsigned int drawable_id,
393 unsigned int head,
394 void *loaderPrivate)
395{
396 (void) drawable_id;
397 (void) head;
398 SWDRI_SAFERET_SWRAST(createNewDrawable, screen, config, loaderPrivate);
399}
400
401static void
402vboxdriDestroyDrawable(__DRIdrawable *drawable)
403{
404 SWDRI_SAFECALL_CORE(destroyDrawable, drawable);
405}
406
407static void
408vboxdriSwapBuffers(__DRIdrawable *drawable)
409{
410 SWDRI_SAFECALL_CORE(swapBuffers, drawable);
411}
412
413static __DRIcontext *
414vboxdriCreateNewContext(__DRIscreen *screen,
415 const __DRIconfig *config,
416 __DRIcontext *shared,
417 void *loaderPrivate)
418{
419 SWDRI_SAFERET_CORE(createNewContext, screen, config, shared, loaderPrivate);
420}
421
422static int
423vboxdriCopyContext(__DRIcontext *dest,
424 __DRIcontext *src,
425 unsigned long mask)
426{
427 SWDRI_SAFERET_CORE(copyContext, dest, src, mask);
428}
429
430static void
431vboxdriDestroyContext(__DRIcontext *context)
432{
433 SWDRI_SAFECALL_CORE(destroyContext, context);
434}
435
436static int
437vboxdriBindContext(__DRIcontext *ctx,
438 __DRIdrawable *pdraw,
439 __DRIdrawable *pread)
440{
441 SWDRI_SAFERET_CORE(bindContext, ctx, pdraw, pread);
442}
443
444static int
445vboxdriUnbindContext(__DRIcontext *ctx)
446{
447 SWDRI_SAFERET_CORE(unbindContext, ctx)
448}
449
450/* __DRIlegacyExtension */
451
452static __DRIscreen *
453vboxdriCreateNewScreen_Legacy(int scrn,
454 const __DRIversion *ddx_version,
455 const __DRIversion *dri_version,
456 const __DRIversion *drm_version,
457 const __DRIframebuffer *frame_buffer,
458 drmAddress pSAREA, int fd,
459 const __DRIextension **extensions,
460 const __DRIconfig ***driver_modes,
461 void *loaderPrivate)
462{
463 (void) ddx_version;
464 (void) dri_version;
465 (void) frame_buffer;
466 (void) pSAREA;
467 (void) fd;
468 SWDRI_SAFERET_SWRAST(createNewScreen, scrn, extensions, driver_modes, loaderPrivate);
469}
470
471static __DRIdrawable *
472vboxdriCreateNewDrawable_Legacy(__DRIscreen *psp, const __DRIconfig *config,
473 drm_drawable_t hwDrawable, int renderType,
474 const int *attrs, void *data)
475{
476 (void) hwDrawable;
477 (void) renderType;
478 (void) attrs;
479 (void) data;
480 SWDRI_SAFERET_SWRAST(createNewDrawable, psp, config, data);
481}
482
483static __DRIcontext *
484vboxdriCreateNewContext_Legacy(__DRIscreen *psp, const __DRIconfig *config,
485 int render_type, __DRIcontext *shared,
486 drm_context_t hwContext, void *data)
487{
488 (void) render_type;
489 (void) hwContext;
490 return vboxdriCreateNewContext(psp, config, shared, data);
491}
492
493
494static const __DRIlegacyExtension vboxdriLegacyExtension = {
495 { __DRI_LEGACY, __DRI_LEGACY_VERSION },
496 vboxdriCreateNewScreen_Legacy,
497 vboxdriCreateNewDrawable_Legacy,
498 vboxdriCreateNewContext_Legacy
499};
500
501static const __DRIcoreExtension vboxdriCoreExtension = {
502 { __DRI_CORE, __DRI_CORE_VERSION },
503 vboxdriCreateNewScreen, /* driCreateNewScreen */
504 vboxdriDestroyScreen,
505 vboxdriGetExtensions,
506 vboxdriGetConfigAttrib,
507 vboxdriIndexConfigAttrib,
508 vboxdriCreateNewDrawable, /* driCreateNewDrawable */
509 vboxdriDestroyDrawable,
510 vboxdriSwapBuffers,
511 vboxdriCreateNewContext,
512 vboxdriCopyContext,
513 vboxdriDestroyContext,
514 vboxdriBindContext,
515 vboxdriUnbindContext
516};
517
518/* This structure is used by dri_util from mesa, don't rename it! */
519DECLEXPORT(const __DRIextension *) __driDriverExtensions[] = {
520 &vboxdriLegacyExtension.base,
521 &vboxdriCoreExtension.base,
522 NULL
523};
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