blob: b638aa1104d6555c99c979f77642c00ac46501d9 [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
Jose Fonseca9653f952015-05-19 16:32:43 +010026#pragma once
José Fonseca7e329022010-11-19 17:05:18 +000027
28
José Fonseca7e329022010-11-19 17:05:18 +000029#include <iostream>
José Fonseca89851d02010-11-28 12:16:52 +000030#include <list>
José Fonseca7e329022010-11-19 17:05:18 +000031
Zack Rusin712429a2011-08-25 23:22:30 -040032#include "trace_file.hpp"
José Fonseca7e329022010-11-19 17:05:18 +000033#include "trace_format.hpp"
34#include "trace_model.hpp"
José Fonseca67964382012-03-27 23:54:30 +010035#include "trace_api.hpp"
José Fonseca7e329022010-11-19 17:05:18 +000036
37
José Fonsecab4a3d142011-10-27 07:43:19 +010038namespace trace {
José Fonseca7e329022010-11-19 17:05:18 +000039
José Fonseca61e61f72011-09-11 16:53:34 +010040
41struct ParseBookmark
42{
43 File::Offset offset;
44 unsigned next_call_no;
45};
46
47
Jon Ashburn9459abf2014-09-04 23:03:37 +010048// Parser interface
49class AbstractParser
50{
51public:
52 virtual ~AbstractParser() {}
53 virtual Call *parse_call(void) = 0;
54 virtual void getBookmark(ParseBookmark &bookmark) = 0;
55 virtual void setBookmark(const ParseBookmark &bookmark) = 0;
56 virtual bool open(const char *filename) = 0;
57 virtual void close(void) = 0;
58 virtual unsigned long long getVersion(void) const = 0;
59};
60
61
62class Parser: public AbstractParser
José Fonseca7e329022010-11-19 17:05:18 +000063{
64protected:
Zack Rusin5ce45e72011-08-05 13:43:46 -040065 File *file;
José Fonsecafa922142010-11-25 09:36:04 +000066
José Fonsecae2d65222011-09-11 22:04:07 +010067 enum Mode {
68 FULL = 0,
69 SCAN,
70 SKIP
71 };
72
José Fonseca19828972010-11-29 20:34:32 +000073 typedef std::list<Call *> CallList;
74 CallList calls;
José Fonseca34957132010-11-25 16:14:45 +000075
José Fonseca340f5692011-11-30 07:04:44 +000076 struct FunctionSigFlags : public FunctionSig {
77 CallFlags flags;
78 };
79
José Fonseca6a6d3e42011-09-11 14:14:21 +010080 // Helper template that extends a base signature structure, with additional
81 // parsing information.
82 template< class T >
83 struct SigState : public T {
84 // Offset in the file of where signature was defined. It is used when
85 // reparsing to determine whether the signature definition is to be
86 // expected next or not.
José Fonseca9c25e062013-05-06 07:25:40 +010087 File::Offset fileOffset;
José Fonseca6a6d3e42011-09-11 14:14:21 +010088 };
89
José Fonseca340f5692011-11-30 07:04:44 +000090 typedef SigState<FunctionSigFlags> FunctionSigState;
José Fonseca6a6d3e42011-09-11 14:14:21 +010091 typedef SigState<StructSig> StructSigState;
92 typedef SigState<EnumSig> EnumSigState;
93 typedef SigState<BitmaskSig> BitmaskSigState;
José Fonseca9c25e062013-05-06 07:25:40 +010094 typedef SigState<StackFrame> StackFrameState;
José Fonseca6a6d3e42011-09-11 14:14:21 +010095
96 typedef std::vector<FunctionSigState *> FunctionMap;
97 typedef std::vector<StructSigState *> StructMap;
98 typedef std::vector<EnumSigState *> EnumMap;
99 typedef std::vector<BitmaskSigState *> BitmaskMap;
José Fonseca9c25e062013-05-06 07:25:40 +0100100 typedef std::vector<StackFrameState *> StackFrameMap;
José Fonseca6a6d3e42011-09-11 14:14:21 +0100101
José Fonseca19828972010-11-29 20:34:32 +0000102 FunctionMap functions;
José Fonseca19828972010-11-29 20:34:32 +0000103 StructMap structs;
José Fonseca19828972010-11-29 20:34:32 +0000104 EnumMap enums;
José Fonseca19828972010-11-29 20:34:32 +0000105 BitmaskMap bitmasks;
José Fonseca9c25e062013-05-06 07:25:40 +0100106 StackFrameMap frames;
José Fonsecad35973c2010-11-26 14:14:45 +0000107
José Fonseca340f5692011-11-30 07:04:44 +0000108 FunctionSig *glGetErrorSig;
109
José Fonseca19828972010-11-29 20:34:32 +0000110 unsigned next_call_no;
José Fonseca34957132010-11-25 16:14:45 +0000111
José Fonseca6117e312011-04-15 23:52:58 +0100112 unsigned long long version;
Jon Ashburn9459abf2014-09-04 23:03:37 +0100113public:
José Fonseca67964382012-03-27 23:54:30 +0100114 API api;
José Fonseca99221832011-03-22 22:15:46 +0000115
José Fonseca57dbaf52011-04-10 14:45:43 +0100116 Parser();
José Fonseca7e329022010-11-19 17:05:18 +0000117
José Fonseca57dbaf52011-04-10 14:45:43 +0100118 ~Parser();
José Fonseca6f51d3b2010-11-22 19:56:19 +0000119
José Fonseca57dbaf52011-04-10 14:45:43 +0100120 bool open(const char *filename);
José Fonseca7e329022010-11-19 17:05:18 +0000121
José Fonseca57dbaf52011-04-10 14:45:43 +0100122 void close(void);
José Fonseca7e329022010-11-19 17:05:18 +0000123
José Fonsecae2d65222011-09-11 22:04:07 +0100124 Call *parse_call(void) {
125 return parse_call(FULL);
126 }
José Fonseca7e329022010-11-19 17:05:18 +0000127
Zack Rusin712429a2011-08-25 23:22:30 -0400128 bool supportsOffsets() const
129 {
Zack Rusin7c767752011-09-01 11:33:51 -0400130 return file->supportsOffsets();
Zack Rusin712429a2011-08-25 23:22:30 -0400131 }
132
José Fonseca42b10e42011-09-11 21:09:08 +0100133 void getBookmark(ParseBookmark &bookmark);
Zack Rusin712429a2011-08-25 23:22:30 -0400134
José Fonseca42b10e42011-09-11 21:09:08 +0100135 void setBookmark(const ParseBookmark &bookmark);
Zack Rusine0df9522011-09-01 01:50:56 -0400136
Jon Ashburn9459abf2014-09-04 23:03:37 +0100137 unsigned long long getVersion(void) const {
138 return version;
139 }
140
Zack Rusin2b1bd4f2011-09-04 16:14:22 -0400141 int percentRead()
142 {
143 return file->percentRead();
144 }
145
José Fonsecae2d65222011-09-11 22:04:07 +0100146 Call *scan_call() {
147 return parse_call(SCAN);
148 }
Zack Rusin46c4a322011-09-02 01:08:49 -0400149
José Fonseca57dbaf52011-04-10 14:45:43 +0100150protected:
José Fonsecae2d65222011-09-11 22:04:07 +0100151 Call *parse_call(Mode mode);
152
José Fonseca340f5692011-11-30 07:04:44 +0000153 FunctionSigFlags *parse_function_sig(void);
José Fonseca6a6d3e42011-09-11 14:14:21 +0100154 StructSig *parse_struct_sig();
José Fonsecaeb644512011-12-11 10:33:55 +0000155 EnumSig *parse_old_enum_sig();
José Fonseca6a6d3e42011-09-11 14:14:21 +0100156 EnumSig *parse_enum_sig();
157 BitmaskSig *parse_bitmask_sig();
José Fonsecae2d65222011-09-11 22:04:07 +0100158
Jose Fonsecade9621f2015-08-14 00:10:21 +0100159public:
José Fonseca340f5692011-11-30 07:04:44 +0000160 static CallFlags
161 lookupCallFlags(const char *name);
162
Jose Fonsecade9621f2015-08-14 00:10:21 +0100163protected:
José Fonsecae2d65222011-09-11 22:04:07 +0100164 Call *parse_Call(Mode mode);
José Fonseca6a6d3e42011-09-11 14:14:21 +0100165
José Fonsecae2d65222011-09-11 22:04:07 +0100166 void parse_enter(Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400167
José Fonsecae2d65222011-09-11 22:04:07 +0100168 Call *parse_leave(Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400169
José Fonsecae2d65222011-09-11 22:04:07 +0100170 bool parse_call_details(Call *call, Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400171
Eugene Velesevichaa1b2132013-04-23 17:56:14 +0400172 bool parse_call_backtrace(Call *call, Mode mode);
José Fonseca9c25e062013-05-06 07:25:40 +0100173 StackFrame * parse_backtrace_frame(Mode mode);
Eugene Velesevichaa1b2132013-04-23 17:56:14 +0400174
José Fonseca340f5692011-11-30 07:04:44 +0000175 void adjust_call_flags(Call *call);
176
José Fonsecae2d65222011-09-11 22:04:07 +0100177 void parse_arg(Call *call, Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400178
José Fonseca838decf2011-09-11 14:44:41 +0100179 Value *parse_value(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400180 void scan_value(void);
José Fonsecae2d65222011-09-11 22:04:07 +0100181 inline Value *parse_value(Mode mode) {
182 if (mode == FULL) {
183 return parse_value();
184 } else {
185 scan_value();
186 return NULL;
187 }
188 }
Zack Rusin46c4a322011-09-02 01:08:49 -0400189
José Fonseca838decf2011-09-11 14:44:41 +0100190 Value *parse_sint();
Zack Rusin46c4a322011-09-02 01:08:49 -0400191 void scan_sint();
192
José Fonseca838decf2011-09-11 14:44:41 +0100193 Value *parse_uint();
Zack Rusin46c4a322011-09-02 01:08:49 -0400194 void scan_uint();
195
José Fonseca838decf2011-09-11 14:44:41 +0100196 Value *parse_float();
Zack Rusin46c4a322011-09-02 01:08:49 -0400197 void scan_float();
198
José Fonseca838decf2011-09-11 14:44:41 +0100199 Value *parse_double();
Zack Rusin46c4a322011-09-02 01:08:49 -0400200 void scan_double();
201
José Fonseca838decf2011-09-11 14:44:41 +0100202 Value *parse_string();
Zack Rusin46c4a322011-09-02 01:08:49 -0400203 void scan_string();
204
José Fonseca838decf2011-09-11 14:44:41 +0100205 Value *parse_enum();
Zack Rusin46c4a322011-09-02 01:08:49 -0400206 void scan_enum();
207
José Fonseca838decf2011-09-11 14:44:41 +0100208 Value *parse_bitmask();
Zack Rusin46c4a322011-09-02 01:08:49 -0400209 void scan_bitmask();
210
José Fonseca838decf2011-09-11 14:44:41 +0100211 Value *parse_array(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400212 void scan_array(void);
213
José Fonseca838decf2011-09-11 14:44:41 +0100214 Value *parse_blob(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400215 void scan_blob(void);
216
José Fonseca838decf2011-09-11 14:44:41 +0100217 Value *parse_struct();
Zack Rusin46c4a322011-09-02 01:08:49 -0400218 void scan_struct();
219
José Fonseca838decf2011-09-11 14:44:41 +0100220 Value *parse_opaque();
Zack Rusin46c4a322011-09-02 01:08:49 -0400221 void scan_opaque();
222
José Fonseca112a1322012-04-27 17:15:32 +0100223 Value *parse_repr();
224 void scan_repr();
225
José Fonsecad5cda7c2014-09-25 15:19:09 +0100226 Value *parse_wstring();
227 void scan_wstring();
228
José Fonseca838decf2011-09-11 14:44:41 +0100229 const char * read_string(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400230 void skip_string(void);
231
José Fonsecaeb644512011-12-11 10:33:55 +0000232 signed long long read_sint(void);
233 void skip_sint(void);
234
José Fonseca838decf2011-09-11 14:44:41 +0100235 unsigned long long read_uint(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400236 void skip_uint(void);
237
José Fonseca838decf2011-09-11 14:44:41 +0100238 inline int read_byte(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400239 inline void skip_byte(void);
José Fonseca7e329022010-11-19 17:05:18 +0000240};
241
242
Jose Fonseca0c287892015-10-14 14:19:01 +0100243AbstractParser *
244lastFrameLoopParser(AbstractParser *parser, int loopCount);
Jon Ashburn9459abf2014-09-04 23:03:37 +0100245
246
José Fonsecab4a3d142011-10-27 07:43:19 +0100247} /* namespace trace */
José Fonseca7e329022010-11-19 17:05:18 +0000248