blob: 9e671d6353836575cd20571b672c9de6aa8b686e [file] [log] [blame]
José Fonseca8fbdd3a2010-11-23 20:55:07 +00001##########################################################################
2#
José Fonseca99771fc2010-11-25 20:32:59 +00003# Copyright 2008-2010 VMware, Inc.
José Fonseca8fbdd3a2010-11-23 20:55:07 +00004# All Rights Reserved.
5#
6# Permission is hereby granted, free of charge, to any person obtaining a copy
7# of this software and associated documentation files (the "Software"), to deal
8# in the Software without restriction, including without limitation the rights
9# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10# copies of the Software, and to permit persons to whom the Software is
11# furnished to do so, subject to the following conditions:
12#
13# The above copyright notice and this permission notice shall be included in
14# all copies or substantial portions of the Software.
15#
16# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22# THE SOFTWARE.
23#
24##########################################################################/
25
26
José Fonseca4a826ed2010-11-30 16:58:22 +000027"""GLX tracing generator."""
28
29
José Fonseca68ec4122011-02-20 11:25:25 +000030from stdapi import API
31from glapi import glapi
José Fonseca99771fc2010-11-25 20:32:59 +000032from glxapi import glxapi
José Fonseca669b1222011-02-20 09:05:10 +000033from gltrace import GlTracer
José Fonseca669b2002011-02-20 13:32:19 +000034from dispatch import function_pointer_type, function_pointer_value
José Fonseca8fbdd3a2010-11-23 20:55:07 +000035
36
José Fonseca669b1222011-02-20 09:05:10 +000037class GlxTracer(GlTracer):
José Fonseca8fbdd3a2010-11-23 20:55:07 +000038
39 def get_function_address(self, function):
José Fonseca68ec4122011-02-20 11:25:25 +000040 return '__%s' % (function.name,)
José Fonseca8fbdd3a2010-11-23 20:55:07 +000041
42 def wrap_ret(self, function, instance):
José Fonseca14c21bc2011-02-20 23:32:22 +000043 if function.name in ("glXGetProcAddress", "glXGetProcAddressARB"):
44 print ' %s = __unwrap_proc_addr(procName, %s);' % (instance, instance)
José Fonseca8fbdd3a2010-11-23 20:55:07 +000045
46
47if __name__ == '__main__':
48 print
49 print '#include <stdlib.h>'
50 print '#include <string.h>'
José Fonseca85a20682011-04-16 13:41:24 +010051 print
52 print '#ifndef _GNU_SOURCE'
53 print '#define _GNU_SOURCE // for dladdr'
54 print '#endif'
José Fonseca8fbdd3a2010-11-23 20:55:07 +000055 print '#include <dlfcn.h>'
José Fonsecad3b80592010-11-29 11:51:23 +000056 print
José Fonsecaf2efcea2010-11-26 11:35:54 +000057 print '#include "trace_write.hpp"'
José Fonseca68ec4122011-02-20 11:25:25 +000058 print
59 print '#include "glproc.hpp"'
José Fonseca48e37fe2010-11-25 12:15:17 +000060 print '#include "glsize.hpp"'
José Fonseca8fbdd3a2010-11-23 20:55:07 +000061 print
62 print 'extern "C" {'
63 print
José Fonseca14c21bc2011-02-20 23:32:22 +000064 print 'static __GLXextFuncPtr __unwrap_proc_addr(const GLubyte * procName, __GLXextFuncPtr procPtr);'
65 print
66
José Fonseca68ec4122011-02-20 11:25:25 +000067 api = API()
68 api.add_api(glxapi)
69 api.add_api(glapi)
José Fonseca8fbdd3a2010-11-23 20:55:07 +000070 tracer = GlxTracer()
José Fonseca68ec4122011-02-20 11:25:25 +000071 tracer.trace_api(api)
José Fonseca14c21bc2011-02-20 23:32:22 +000072
73 print 'static __GLXextFuncPtr __unwrap_proc_addr(const GLubyte * procName, __GLXextFuncPtr procPtr) {'
74 print ' if (!procPtr) {'
75 print ' return procPtr;'
76 print ' }'
77 for f in api.functions:
78 ptype = function_pointer_type(f)
79 pvalue = function_pointer_value(f)
José Fonsecadbaae492011-04-21 09:28:10 +010080 print ' if (!strcmp("%s", (const char *)procName)) {' % f.name
José Fonseca14c21bc2011-02-20 23:32:22 +000081 print ' %s = (%s)procPtr;' % (pvalue, ptype)
82 print ' return (__GLXextFuncPtr)&%s;' % (f.name,)
83 print ' }'
84 print ' return procPtr;'
85 print '}'
José Fonseca8fbdd3a2010-11-23 20:55:07 +000086 print
José Fonseca85a20682011-04-16 13:41:24 +010087 print r'''
88
José Fonseca68b14052011-04-21 10:32:45 +010089
90static void *libgl_handle = NULL;
91
92
José Fonseca85a20682011-04-16 13:41:24 +010093/*
José Fonseca68b14052011-04-21 10:32:45 +010094 * Invoke the true dlopen() function.
José Fonseca85a20682011-04-16 13:41:24 +010095 */
José Fonseca68b14052011-04-21 10:32:45 +010096static void *__dlopen(const char *filename, int flag)
José Fonseca85a20682011-04-16 13:41:24 +010097{
98 typedef void * (*PFNDLOPEN)(const char *, int);
99 static PFNDLOPEN dlopen_ptr = NULL;
José Fonseca85a20682011-04-16 13:41:24 +0100100
101 if (!dlopen_ptr) {
102 dlopen_ptr = (PFNDLOPEN)dlsym(RTLD_NEXT, "dlopen");
103 if (!dlopen_ptr) {
104 OS::DebugMessage("error: dlsym(RTLD_NEXT, \"dlopen\") failed\n");
105 return NULL;
106 }
107 }
108
José Fonseca68b14052011-04-21 10:32:45 +0100109 return dlopen_ptr(filename, flag);
110}
111
112
113/*
114 * Several applications, such as Quake3, use dlopen("libGL.so.1"), but
115 * LD_PRELOAD does not intercept symbols obtained via dlopen/dlsym, therefore
116 * we need to intercept the dlopen() call here, and redirect to our wrapper
117 * shared object.
118 */
119void *dlopen(const char *filename, int flag)
120{
121 void *handle;
122
123 handle = __dlopen(filename, flag);
José Fonseca85a20682011-04-16 13:41:24 +0100124
125 if (filename && handle) {
126 if (0) {
127 OS::DebugMessage("warning: dlopen(\"%s\", 0x%x)\n", filename, flag);
128 }
129
130 // FIXME: handle absolute paths and other versions
José Fonseca94215532011-04-17 22:30:26 +0100131 if (strcmp(filename, "libGL.so") == 0 ||
132 strcmp(filename, "libGL.so.1") == 0) {
José Fonseca85a20682011-04-16 13:41:24 +0100133 // Use the true libGL.so handle instead of RTLD_NEXT from now on
134 libgl_handle = handle;
135
136 // Get the file path for our shared object, and use it instead
137 static int dummy = 0xdeedbeef;
138 Dl_info info;
139 if (dladdr(&dummy, &info)) {
140 OS::DebugMessage("apitrace: redirecting dlopen(\"%s\", 0x%x)\n", filename, flag);
José Fonseca68b14052011-04-21 10:32:45 +0100141 handle = __dlopen(info.dli_fname, flag);
José Fonseca85a20682011-04-16 13:41:24 +0100142 } else {
143 OS::DebugMessage("warning: dladdr() failed\n");
144 }
145 }
146 }
147
148 return handle;
149}
150
José Fonseca68b14052011-04-21 10:32:45 +0100151} /* extern "C" */
152
153
154/*
155 * Lookup a libGL symbol
156 */
157static void * __dlsym(const char *symbol)
158{
159 void *result;
160
161 if (!libgl_handle) {
162 /*
163 * Try to use whatever libGL.so the library is linked against.
164 */
165 result = dlsym(RTLD_NEXT, symbol);
166 if (result) {
167 libgl_handle = RTLD_NEXT;
168 return result;
169 }
170
171 /*
172 * The app doesn't directly link against libGL.so, nor does it directly
173 * dlopen it. So we have to load it ourselves.
174 */
175 libgl_handle = __dlopen("libGL.so", RTLD_LAZY | RTLD_LOCAL);
176 if (!libgl_handle) {
177 OS::DebugMessage("error: couldn't find libGL.so\n");
178 return NULL;
179 }
180 }
181
182 return dlsym(libgl_handle, symbol);
183}
184
185
José Fonseca85a20682011-04-16 13:41:24 +0100186'''