]> git.cworth.org Git - apitrace/blob - retrace.py
Merge branch 'master' into multi-context
[apitrace] / retrace.py
1 ##########################################################################
2 #
3 # Copyright 2010 VMware, Inc.
4 # 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
27 """Generic retracing code generator."""
28
29 import stdapi
30 import glapi
31 from codegen import *
32
33
34 class ConstRemover(stdapi.Rebuilder):
35
36     def visit_const(self, const):
37         return const.type
38
39     def visit_opaque(self, opaque):
40         return opaque
41
42
43 def handle_entry(handle, value):
44     if handle.key is None:
45         return "__%s_map[%s]" % (handle.name, value)
46     else:
47         key_name, key_type = handle.key
48         return "__%s_map[%s][%s]" % (handle.name, key_name, value)
49
50
51 class ValueExtractor(stdapi.Visitor):
52
53     def visit_literal(self, literal, lvalue, rvalue):
54         #if literal.format in ('Bool', 'UInt'):
55         print '    %s = (%s).to%s();' % (lvalue, rvalue, literal.format)
56
57     def visit_const(self, const, lvalue, rvalue):
58         self.visit(const.type, lvalue, rvalue)
59
60     def visit_alias(self, alias, lvalue, rvalue):
61         self.visit(alias.type, lvalue, rvalue)
62     
63     def visit_enum(self, enum, lvalue, rvalue):
64         print '    %s = (%s).toSInt();' % (lvalue, rvalue)
65
66     def visit_bitmask(self, bitmask, lvalue, rvalue):
67         self.visit(bitmask.type, lvalue, rvalue)
68
69     def visit_array(self, array, lvalue, rvalue):
70         print '    const Trace::Array *__a%s = dynamic_cast<const Trace::Array *>(&%s);' % (array.id, rvalue)
71         print '    if (__a%s) {' % (array.id)
72         length = '__a%s->values.size()' % array.id
73         print '        %s = new %s[%s];' % (lvalue, array.type, length)
74         index = '__j' + array.id
75         print '        for (size_t {i} = 0; {i} < {length}; ++{i}) {{'.format(i = index, length = length)
76         try:
77             self.visit(array.type, '%s[%s]' % (lvalue, index), '*__a%s->values[%s]' % (array.id, index))
78         finally:
79             print '        }'
80             print '    } else {'
81             print '        %s = NULL;' % lvalue
82             print '    }'
83     
84     def visit_pointer(self, pointer, lvalue, rvalue):
85         print '    const Trace::Array *__a%s = dynamic_cast<const Trace::Array *>(&%s);' % (pointer.id, rvalue)
86         print '    if (__a%s) {' % (pointer.id)
87         print '        %s = new %s;' % (lvalue, pointer.type)
88         try:
89             self.visit(pointer.type, '%s[0]' % (lvalue,), '*__a%s->values[0]' % (pointer.id,))
90         finally:
91             print '    } else {'
92             print '        %s = NULL;' % lvalue
93             print '    }'
94
95     def visit_handle(self, handle, lvalue, rvalue):
96         OpaqueValueExtractor().visit(handle.type, lvalue, rvalue);
97         new_lvalue = handle_entry(handle, lvalue)
98         print '    if (retrace::verbosity >= 2)'
99         print '        std::cout << "%s " << size_t(%s) << " <- " << size_t(%s) << "\\n";' % (handle.name, lvalue, new_lvalue)
100         print '    %s = %s;' % (lvalue, new_lvalue)
101     
102     def visit_blob(self, blob, lvalue, rvalue):
103         print '    %s = static_cast<%s>((%s).toPointer());' % (lvalue, blob, rvalue)
104     
105     def visit_string(self, string, lvalue, rvalue):
106         print '    %s = (%s)((%s).toString());' % (lvalue, string.expr, rvalue)
107
108
109 class OpaqueValueExtractor(ValueExtractor):
110     '''Value extractor that also understands opaque values.
111
112     Normally opaque values can't be retraced, unless they are being extracted
113     in the context of handles.'''
114
115     def visit_opaque(self, opaque, lvalue, rvalue):
116         print '    %s = static_cast<%s>((%s).toPointer());' % (lvalue, opaque, rvalue)
117
118
119 class ValueWrapper(stdapi.Visitor):
120
121     def visit_literal(self, literal, lvalue, rvalue):
122         pass
123
124     def visit_alias(self, alias, lvalue, rvalue):
125         self.visit(alias.type, lvalue, rvalue)
126     
127     def visit_enum(self, enum, lvalue, rvalue):
128         pass
129
130     def visit_bitmask(self, bitmask, lvalue, rvalue):
131         pass
132
133     def visit_array(self, array, lvalue, rvalue):
134         print '    const Trace::Array *__a%s = dynamic_cast<const Trace::Array *>(&%s);' % (array.id, rvalue)
135         print '    if (__a%s) {' % (array.id)
136         length = '__a%s->values.size()' % array.id
137         index = '__j' + array.id
138         print '        for (size_t {i} = 0; {i} < {length}; ++{i}) {{'.format(i = index, length = length)
139         try:
140             self.visit(array.type, '%s[%s]' % (lvalue, index), '*__a%s->values[%s]' % (array.id, index))
141         finally:
142             print '        }'
143             print '    }'
144     
145     def visit_pointer(self, pointer, lvalue, rvalue):
146         print '    const Trace::Array *__a%s = dynamic_cast<const Trace::Array *>(&%s);' % (pointer.id, rvalue)
147         print '    if (__a%s) {' % (pointer.id)
148         try:
149             self.visit(pointer.type, '%s[0]' % (lvalue,), '*__a%s->values[0]' % (pointer.id,))
150         finally:
151             print '    }'
152     
153     def visit_handle(self, handle, lvalue, rvalue):
154         print '    %s __orig_result;' % handle.type
155         OpaqueValueExtractor().visit(handle.type, '__orig_result', rvalue);
156         if handle.range is None:
157             rvalue = "__orig_result"
158             entry = handle_entry(handle, rvalue) 
159             print "    %s = %s;" % (entry, lvalue)
160             print '    if (retrace::verbosity >= 2)'
161             print '        std::cout << "{handle.name} " << {rvalue} << " -> " << {lvalue} << "\\n";'.format(**locals())
162         else:
163             i = '__h' + handle.id
164             lvalue = "%s + %s" % (lvalue, i)
165             rvalue = "__orig_result + %s" % (i,)
166             entry = handle_entry(handle, rvalue) 
167             print '    for ({handle.type} {i} = 0; {i} < {handle.range}; ++{i}) {{'.format(**locals())
168             print '        {entry} = {lvalue};'.format(**locals())
169             print '        if (retrace::verbosity >= 2)'
170             print '            std::cout << "{handle.name} " << ({rvalue}) << " -> " << ({lvalue}) << "\\n";'.format(**locals())
171             print '    }'
172     
173     def visit_blob(self, blob, lvalue, rvalue):
174         pass
175     
176     def visit_string(self, string, lvalue, rvalue):
177         pass
178
179
180 class Retracer:
181
182     def retrace_function(self, function):
183         print 'static void retrace_%s(Trace::Call &call) {' % function.name
184         self.retrace_function_body(function)
185         print '}'
186         print
187
188     def retrace_function_body(self, function):
189         if not function.sideeffects:
190             return
191
192         success = True
193         for arg in function.args:
194             arg_type = ConstRemover().visit(arg.type)
195             #print '    // %s ->  %s' % (arg.type, arg_type)
196             print '    %s %s;' % (arg_type, arg.name)
197             rvalue = 'call.arg(%u)' % (arg.index,)
198             lvalue = arg.name
199             try:
200                 self.extract_arg(function, arg, arg_type, lvalue, rvalue)
201             except NotImplementedError:
202                 success = False
203                 print '    %s = 0; // FIXME' % arg.name
204         if not success:
205             self.fail_function(function)
206         self.call_function(function)
207         for arg in function.args:
208             if arg.output:
209                 arg_type = ConstRemover().visit(arg.type)
210                 rvalue = 'call.arg(%u)' % (arg.index,)
211                 lvalue = arg.name
212                 try:
213                     ValueWrapper().visit(arg_type, lvalue, rvalue)
214                 except NotImplementedError:
215                     print '   // FIXME: %s' % arg.name
216         if function.type is not stdapi.Void:
217             rvalue = '*call.ret'
218             lvalue = '__result'
219             try:
220                 ValueWrapper().visit(function.type, lvalue, rvalue)
221             except NotImplementedError:
222                 print '   // FIXME: result'
223
224     def fail_function(self, function):
225         print '    if (retrace::verbosity >= 0)'
226         print '        std::cerr << "warning: unsupported call %s\\n";' % function.name
227         print '    return;'
228
229     def extract_arg(self, function, arg, arg_type, lvalue, rvalue):
230         ValueExtractor().visit(arg_type, lvalue, rvalue)
231
232     def call_function(self, function):
233         arg_names = ", ".join([arg.name for arg in function.args])
234         if function.type is not stdapi.Void:
235             print '    %s __result;' % (function.type)
236             print '    __result = %s(%s);' % (function.name, arg_names)
237         else:
238             print '    %s(%s);' % (function.name, arg_names)
239
240     def filter_function(self, function):
241         return True
242
243     def retrace_functions(self, functions):
244         functions = filter(self.filter_function, functions)
245
246         for function in functions:
247             self.retrace_function(function)
248
249         print 'void retrace::retrace_call(Trace::Call &call) {'
250         print '    const char *name = call.name().c_str();'
251         print
252
253         func_dict = dict([(function.name, function) for function in functions])
254
255         def handle_case(function_name):
256             function = func_dict[function_name]
257             print '        retrace_%s(call);' % function.name
258             print '        return;'
259     
260         string_switch('name', func_dict.keys(), handle_case)
261
262         print '    retrace_unknown(call);'
263         print '}'
264         print
265
266
267     def retrace_api(self, api):
268
269         print '#include "trace_parser.hpp"'
270         print '#include "retrace.hpp"'
271         print
272
273         types = api.all_types()
274         handles = [type for type in types if isinstance(type, stdapi.Handle)]
275         handle_names = set()
276         for handle in handles:
277             if handle.name not in handle_names:
278                 if handle.key is None:
279                     print 'static retrace::map<%s> __%s_map;' % (handle.type, handle.name)
280                 else:
281                     key_name, key_type = handle.key
282                     print 'static std::map<%s, retrace::map<%s> > __%s_map;' % (key_type, handle.type, handle.name)
283                 handle_names.add(handle.name)
284         print
285
286         self.retrace_functions(api.functions)
287