]> git.notmuchmail.org Git - apitrace/blob - retrace/retrace.hpp
Share more code between glretrace and d3dretrace.
[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 extern bool doubleBuffer;
204 extern bool coreProfile;
205
206
207 std::ostream &warning(trace::Call &call);
208
209
210 void ignore(trace::Call &call);
211 void unsupported(trace::Call &call);
212
213
214 typedef void (*Callback)(trace::Call &call);
215
216 struct Entry {
217     const char *name;
218     Callback callback;
219 };
220
221
222 struct stringComparer {
223   bool operator() (const char *a, const  char *b) const {
224     return strcmp(a, b) < 0;
225   }
226 };
227
228
229 extern const Entry stdc_callbacks[];
230
231
232 class Retracer
233 {
234     typedef std::map<const char *, Callback, stringComparer> Map;
235     Map map;
236
237     std::vector<Callback> callbacks;
238
239 public:
240     Retracer() {
241         addCallbacks(stdc_callbacks);
242     }
243
244     virtual ~Retracer() {}
245
246     void addCallback(const Entry *entry);
247     void addCallbacks(const Entry *entries);
248
249     void retrace(trace::Call &call);
250 };
251
252
253 void
254 setUp(void);
255
256 void
257 addCallbacks(retrace::Retracer &retracer);
258
259 void
260 frameComplete(trace::Call &call);
261
262 image::Image *
263 getSnapshot(void);
264
265 bool
266 dumpState(std::ostream &os);
267
268 void
269 flushRendering(void);
270
271 void
272 waitForInput(void);
273
274 void
275 cleanUp(void);
276
277
278 } /* namespace retrace */
279
280 #endif /* _RETRACE_HPP_ */