]> git.cworth.org Git - apitrace/blob - retrace/retrace.hpp
9a1d5ccc2832c7e5c704b7bac9c42edf5014da13
[apitrace] / retrace / retrace.hpp
1 /**************************************************************************
2  *
3  * Copyright 2011-2012 Jose Fonseca
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 #ifndef _RETRACE_HPP_
27 #define _RETRACE_HPP_
28
29 #include <assert.h>
30 #include <string.h>
31 #include <stdint.h>
32
33 #include <list>
34 #include <map>
35 #include <ostream>
36
37 #include "trace_model.hpp"
38 #include "trace_parser.hpp"
39
40
41 namespace image {
42     class Image;
43 }
44
45
46 namespace retrace {
47
48
49 extern trace::Parser parser;
50
51
52 /**
53  * Handle map.
54  *
55  * It is just like a regular std::map<T, T> container, but lookups of missing
56  * keys return the key instead of default constructor.
57  *
58  * This is necessary for several GL named objects, where one can either request
59  * the implementation to generate an unique name, or pick a value never used
60  * before.
61  *
62  * XXX: In some cases, instead of returning the key, it would make more sense
63  * to return an unused data value (e.g., container count).
64  */
65 template <class T>
66 class map
67 {
68 private:
69     typedef std::map<T, T> base_type;
70     base_type base;
71
72 public:
73
74     T & operator[] (const T &key) {
75         typename base_type::iterator it;
76         it = base.find(key);
77         if (it == base.end()) {
78             return (base[key] = key);
79         }
80         return it->second;
81     }
82     
83     const T & operator[] (const T &key) const {
84         typename base_type::const_iterator it;
85         it = base.find(key);
86         if (it == base.end()) {
87             return (base[key] = key);
88         }
89         return it->second;
90     }
91 };
92
93
94 /**
95  * Similar to alloca(), but implemented with malloc.
96  */
97 class ScopedAllocator
98 {
99 private:
100     uintptr_t next;
101
102 public:
103     ScopedAllocator() :
104         next(0) {
105     }
106
107     inline void *
108     alloc(size_t size) {
109         /* Always return valid address, even when size is zero */
110         size = std::max(size, sizeof(uintptr_t));
111
112         uintptr_t * buf = static_cast<uintptr_t *>(malloc(sizeof(uintptr_t) + size));
113         if (!buf) {
114             return NULL;
115         }
116
117         *buf = next;
118         next = reinterpret_cast<uintptr_t>(buf);
119         assert((next & 1) == 0);
120
121         return static_cast<void *>(&buf[1]);
122     }
123
124     template< class T >
125     inline T *
126     alloc(size_t n = 1) {
127         return static_cast<T *>(alloc(sizeof(T) * n));
128     }
129
130     /**
131      * Allocate an array with the same dimensions as the specified value.
132      */
133     template< class T >
134     inline T *
135     alloc(const trace::Value *value) {
136         const trace::Array *array = dynamic_cast<const trace::Array *>(value);
137         if (array) {
138             return alloc<T>(array->size());
139         }
140         const trace::Null *null = dynamic_cast<const trace::Null *>(value);
141         if (null) {
142             return NULL;
143         }
144         assert(0);
145         return NULL;
146     }
147
148     /**
149      * Prevent this pointer from being automatically freed.
150      */
151     template< class T >
152     inline void
153     bind(T *ptr) {
154         if (ptr) {
155             reinterpret_cast<uintptr_t *>(ptr)[-1] |= 1;
156         }
157     }
158
159     inline
160     ~ScopedAllocator() {
161         while (next) {
162             uintptr_t temp = *reinterpret_cast<uintptr_t *>(next);
163
164             bool bind = temp & 1;
165             temp &= ~1;
166
167             if (!bind) {
168                 free(reinterpret_cast<void *>(next));
169             }
170
171             next = temp;
172         }
173     }
174 };
175
176
177 void
178 addRegion(unsigned long long address, void *buffer, unsigned long long size);
179
180 void
181 delRegionByPointer(void *ptr);
182
183 void *
184 toPointer(trace::Value &value, bool bind = false);
185
186
187 /**
188  * Output verbosity when retracing files.
189  */
190 extern int verbosity;
191
192 /**
193  * Debugging checks.
194  */
195 extern bool debug;
196
197 /**
198  * Add profiling data to the dump when retracing.
199  */
200 extern bool profiling;
201
202 /**
203  * State dumping.
204  */
205 extern bool dumpingState;
206
207
208 extern bool doubleBuffer;
209 extern bool coreProfile;
210
211
212 std::ostream &warning(trace::Call &call);
213
214
215 void ignore(trace::Call &call);
216 void unsupported(trace::Call &call);
217
218
219 typedef void (*Callback)(trace::Call &call);
220
221 struct Entry {
222     const char *name;
223     Callback callback;
224 };
225
226
227 struct stringComparer {
228   bool operator() (const char *a, const  char *b) const {
229     return strcmp(a, b) < 0;
230   }
231 };
232
233
234 extern const Entry stdc_callbacks[];
235
236
237 class Retracer
238 {
239     typedef std::map<const char *, Callback, stringComparer> Map;
240     Map map;
241
242     std::vector<Callback> callbacks;
243
244 public:
245     Retracer() {
246         addCallbacks(stdc_callbacks);
247     }
248
249     virtual ~Retracer() {}
250
251     void addCallback(const Entry *entry);
252     void addCallbacks(const Entry *entries);
253
254     void retrace(trace::Call &call);
255 };
256
257
258 void
259 setUp(void);
260
261 void
262 addCallbacks(retrace::Retracer &retracer);
263
264 void
265 frameComplete(trace::Call &call);
266
267 image::Image *
268 getSnapshot(void);
269
270 bool
271 dumpState(std::ostream &os);
272
273 void
274 flushRendering(void);
275
276 void
277 waitForInput(void);
278
279 void
280 cleanUp(void);
281
282
283 } /* namespace retrace */
284
285 #endif /* _RETRACE_HPP_ */