blob: dc7b8d672abc1af80d79d2959863897cbc37f93b [file] [log] [blame]
José Fonseca7e329022010-11-19 17:05:18 +00001/**************************************************************************
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#ifndef _TRACE_PARSER_HPP_
27#define _TRACE_PARSER_HPP_
28
29
30#include <cassert>
31
32#include <iostream>
33
34#include <zlib.h>
35
36#include "trace_format.hpp"
37#include "trace_model.hpp"
38
39
40namespace Trace {
41
42
43class Parser
44{
45protected:
46 gzFile file;
47public:
48 Parser() {
49 file = NULL;
50 }
51
José Fonseca6f51d3b2010-11-22 19:56:19 +000052 ~Parser() {
53 close();
54 }
55
56 bool open(const char *filename) {
José Fonseca7e329022010-11-19 17:05:18 +000057 unsigned long long version;
58
59 file = gzopen(filename, "rb");
60 if (!file) {
61 return false;
62 }
63
64 version = read_uint();
65 if (version != TRACE_VERSION) {
José Fonseca501f2862010-11-19 20:41:18 +000066 std::cerr << "error: unsupported format version" << version << "\n";
José Fonseca7e329022010-11-19 17:05:18 +000067 return false;
68 }
69
José Fonseca7e329022010-11-19 17:05:18 +000070 return true;
71 }
72
José Fonseca6f51d3b2010-11-22 19:56:19 +000073 void close(void) {
74 if (file) {
75 gzclose(file);
76 file = NULL;
77 }
78 }
79
80 Call *parse_call(void) {
81 Call *call = new Call;
82 call->name = read_string();
José Fonseca7e329022010-11-19 17:05:18 +000083 do {
José Fonseca6f51d3b2010-11-22 19:56:19 +000084 int c = read_byte();
José Fonseca7e329022010-11-19 17:05:18 +000085 switch(c) {
86 case Trace::CALL_END:
José Fonseca6f51d3b2010-11-22 19:56:19 +000087 return call;
José Fonseca7e329022010-11-19 17:05:18 +000088 case Trace::CALL_ARG:
José Fonseca6f51d3b2010-11-22 19:56:19 +000089 call->args.push_back(parse_arg());
José Fonseca7e329022010-11-19 17:05:18 +000090 break;
91 case Trace::CALL_RET:
José Fonseca6f51d3b2010-11-22 19:56:19 +000092 call->ret = parse_value();
José Fonseca7e329022010-11-19 17:05:18 +000093 break;
94 default:
José Fonseca501f2862010-11-19 20:41:18 +000095 std::cerr << "error: unknown call detail " << c << "\n";
José Fonseca885f2652010-11-20 11:22:25 +000096 assert(0);
José Fonseca6f51d3b2010-11-22 19:56:19 +000097 /* fallthrough */
98 case -1:
99 delete call;
100 return NULL;
José Fonseca7e329022010-11-19 17:05:18 +0000101 }
102 } while(true);
José Fonseca7e329022010-11-19 17:05:18 +0000103 }
104
José Fonseca7e329022010-11-19 17:05:18 +0000105 Arg parse_arg(void) {
106 std::string name = read_string();
107 Value *value = parse_value();
108 return Arg(name, value);
109 }
110
111 Value *parse_value(void) {
112 int c;
José Fonseca5cb3e172010-11-22 16:59:34 +0000113 c = read_byte();
José Fonseca7e329022010-11-19 17:05:18 +0000114 switch(c) {
José Fonsecaf6592d72010-11-21 12:44:41 +0000115 case Trace::TYPE_NULL:
116 return new Null;
José Fonsecab1887f92010-11-21 02:33:38 +0000117 case Trace::TYPE_FALSE:
118 return new Bool(false);
119 case Trace::TYPE_TRUE:
120 return new Bool(true);
José Fonseca7e329022010-11-19 17:05:18 +0000121 case Trace::TYPE_SINT:
122 return parse_sint();
123 case Trace::TYPE_UINT:
124 return parse_uint();
125 case Trace::TYPE_FLOAT:
126 return parse_float();
127 case Trace::TYPE_DOUBLE:
128 return parse_double();
129 case Trace::TYPE_STRING:
130 return parse_string();
131 case Trace::TYPE_CONST:
132 return parse_const();
133 case Trace::TYPE_BITMASK:
134 return parse_bitmask();
135 case Trace::TYPE_ARRAY:
136 return parse_array();
José Fonseca5cb3e172010-11-22 16:59:34 +0000137 case Trace::TYPE_STRUCT:
138 return parse_struct();
José Fonseca885f2652010-11-20 11:22:25 +0000139 case Trace::TYPE_BLOB:
140 return parse_blob();
José Fonseca7e329022010-11-19 17:05:18 +0000141 case Trace::TYPE_POINTER:
142 return parse_pointer();
José Fonsecaf6592d72010-11-21 12:44:41 +0000143 case Trace::TYPE_OPAQUE:
144 return parse_opaque();
José Fonseca7e329022010-11-19 17:05:18 +0000145 default:
José Fonseca501f2862010-11-19 20:41:18 +0000146 std::cerr << "error: unknown type " << c << "\n";
José Fonseca7e329022010-11-19 17:05:18 +0000147 assert(0);
148 return NULL;
149 }
150 }
151
José Fonseca7e329022010-11-19 17:05:18 +0000152 Value *parse_sint() {
153 return new SInt(-read_uint());
154 }
155
156 Value *parse_uint() {
157 return new UInt(read_uint());
158 }
159
160 Value *parse_float() {
161 float value;
162 gzread(file, &value, sizeof value);
163 return new Float(value);
164 }
165
166 Value *parse_double() {
167 double value;
168 gzread(file, &value, sizeof value);
169 return new Float(value);
170 }
171
172 Value *parse_string() {
173 return new String(read_string());
174 }
175
176 Value *parse_const() {
177 std::string name = read_string();
178 Value *value = parse_value();
179 return new Const(name, value);
180 }
181
182 Value *parse_bitmask() {
183 unsigned long long value = 0;
184 int c;
185 do {
José Fonseca5cb3e172010-11-22 16:59:34 +0000186 c = read_byte();
José Fonseca7e329022010-11-19 17:05:18 +0000187 switch(c) {
188 case Trace::TYPE_SINT:
189 value |= -read_uint();
190 break;
191 case Trace::TYPE_UINT:
192 value |= read_uint();
193 break;
194 case Trace::TYPE_CONST:
195 read_string();
196 break;
José Fonsecab1887f92010-11-21 02:33:38 +0000197 case Trace::TYPE_NULL:
José Fonseca7e329022010-11-19 17:05:18 +0000198 goto done;
199 default:
José Fonseca501f2862010-11-19 20:41:18 +0000200 std::cerr << "error: uexpected type " << c << "\n";
José Fonseca7e329022010-11-19 17:05:18 +0000201 assert(0);
202 return NULL;
203 }
204 } while(true);
205done:
206 return new UInt(value);
207 }
208
José Fonseca885f2652010-11-20 11:22:25 +0000209 Value *parse_array(void) {
José Fonseca7e329022010-11-19 17:05:18 +0000210 size_t len = read_uint();
211 Array *array = new Array(len);
212 for (size_t i = 0; i < len; ++i) {
213 array->values[i] = parse_value();
214 }
215 return array;
216 }
217
José Fonseca885f2652010-11-20 11:22:25 +0000218 Value *parse_blob(void) {
219 size_t size = read_uint();
220 Blob *blob = new Blob(size);
221 if (size) {
222 gzread(file, blob->buf, size);
223 }
224 return blob;
225 }
226
José Fonseca7e329022010-11-19 17:05:18 +0000227 Value *parse_pointer() {
228 unsigned long long addr;
229 Value *value;
230 addr = read_uint();
231 value = parse_value();
232 if (!value)
233 value = new UInt(addr);
234 return value;
235 }
236
José Fonseca5cb3e172010-11-22 16:59:34 +0000237 Value *parse_struct() {
238 std::string name;
239 /* XXX */
240 name = read_string();
241 while(name.length()) {
242 Value *value = parse_value();
243 std::cout << " " << name << " = " << value << "\n";
244 name = read_string();
245 }
246 return NULL;
247 }
248
José Fonsecaf6592d72010-11-21 12:44:41 +0000249 Value *parse_opaque() {
250 unsigned long long addr;
251 addr = read_uint();
252 /* XXX */
253 return new UInt(addr);
254 }
255
José Fonseca7e329022010-11-19 17:05:18 +0000256 std::string read_string(void) {
257 size_t len = read_uint();
José Fonseca5cb3e172010-11-22 16:59:34 +0000258 if (!len) {
259 return std::string();
260 }
José Fonseca7e329022010-11-19 17:05:18 +0000261 char * buf = new char[len];
262 gzread(file, buf, len);
263 std::string value(buf, len);
264 delete [] buf;
José Fonseca5cb3e172010-11-22 16:59:34 +0000265#ifdef TRACE_VERBOSE
266 std::cerr << '"' << value << '"' << "\n";
267#endif
José Fonseca7e329022010-11-19 17:05:18 +0000268 return value;
269 }
270
271 unsigned long long read_uint(void) {
272 unsigned long long value = 0;
273 int c;
274 unsigned shift = 0;
275 do {
276 c = gzgetc(file);
277 if (c == -1) {
278 break;
279 }
280 value |= (unsigned long long)(c & 0x7f) << shift;
281 shift += 7;
282 } while(c & 0x80);
José Fonseca5cb3e172010-11-22 16:59:34 +0000283#ifdef TRACE_VERBOSE
284 std::cerr << value << "\n";
285#endif
José Fonseca7e329022010-11-19 17:05:18 +0000286 return value;
287 }
José Fonseca5cb3e172010-11-22 16:59:34 +0000288
289 int read_byte(void) {
290 int c = gzgetc(file);
291#ifdef TRACE_VERBOSE
292 if (c < 0)
293 std::cerr << "EOF" << "\n";
294 else
295 std::cerr << "0x" << std::hex << c << "\n";
296#endif
297 return c;
298 }
José Fonseca7e329022010-11-19 17:05:18 +0000299};
300
301
302} /* namespace Trace */
303
304#endif /* _TRACE_PARSER_HPP_ */