blob: 25d81213ce8e9e4eff72aa5e2b05b87901ff73d2 [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:
Jose Fonsecae8a5de72017-06-22 15:18:52 +010065 File *file = nullptr;
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
Jose Fonsecae8a5de72017-06-22 15:18:52 +0100109 FunctionSig *glGetErrorSig = nullptr;
José Fonseca34957132010-11-25 16:14:45 +0000110
Jose Fonsecae8a5de72017-06-22 15:18:52 +0100111 int next_event_type = -1;
112 unsigned next_call_no = 0;
113
114 unsigned long long version = 0;
115 unsigned long long semanticVersion = 0;
116
Jon Ashburn9459abf2014-09-04 23:03:37 +0100117public:
Jose Fonsecae8a5de72017-06-22 15:18:52 +0100118 API api = API_UNKNOWN;
José Fonseca99221832011-03-22 22:15:46 +0000119
José Fonseca57dbaf52011-04-10 14:45:43 +0100120 Parser();
José Fonseca7e329022010-11-19 17:05:18 +0000121
José Fonseca57dbaf52011-04-10 14:45:43 +0100122 ~Parser();
José Fonseca6f51d3b2010-11-22 19:56:19 +0000123
Jose Fonseca697391f2016-03-05 14:44:37 +0000124 bool open(const char *filename) override;
José Fonseca7e329022010-11-19 17:05:18 +0000125
Jose Fonseca697391f2016-03-05 14:44:37 +0000126 void close(void) override;
José Fonseca7e329022010-11-19 17:05:18 +0000127
Jose Fonseca697391f2016-03-05 14:44:37 +0000128 Call *parse_call(void) override {
José Fonsecae2d65222011-09-11 22:04:07 +0100129 return parse_call(FULL);
130 }
José Fonseca7e329022010-11-19 17:05:18 +0000131
Zack Rusin712429a2011-08-25 23:22:30 -0400132 bool supportsOffsets() const
133 {
Zack Rusin7c767752011-09-01 11:33:51 -0400134 return file->supportsOffsets();
Zack Rusin712429a2011-08-25 23:22:30 -0400135 }
136
Jose Fonseca697391f2016-03-05 14:44:37 +0000137 void getBookmark(ParseBookmark &bookmark) override;
Zack Rusin712429a2011-08-25 23:22:30 -0400138
Jose Fonseca697391f2016-03-05 14:44:37 +0000139 void setBookmark(const ParseBookmark &bookmark) override;
Zack Rusine0df9522011-09-01 01:50:56 -0400140
Jose Fonseca697391f2016-03-05 14:44:37 +0000141 unsigned long long getVersion(void) const override {
Jose Fonsecae8a5de72017-06-22 15:18:52 +0100142 return semanticVersion;
Jon Ashburn9459abf2014-09-04 23:03:37 +0100143 }
144
Zack Rusin2b1bd4f2011-09-04 16:14:22 -0400145 int percentRead()
146 {
147 return file->percentRead();
148 }
149
José Fonsecae2d65222011-09-11 22:04:07 +0100150 Call *scan_call() {
151 return parse_call(SCAN);
152 }
Zack Rusin46c4a322011-09-02 01:08:49 -0400153
José Fonseca57dbaf52011-04-10 14:45:43 +0100154protected:
José Fonsecae2d65222011-09-11 22:04:07 +0100155 Call *parse_call(Mode mode);
156
José Fonseca340f5692011-11-30 07:04:44 +0000157 FunctionSigFlags *parse_function_sig(void);
José Fonseca6a6d3e42011-09-11 14:14:21 +0100158 StructSig *parse_struct_sig();
José Fonsecaeb644512011-12-11 10:33:55 +0000159 EnumSig *parse_old_enum_sig();
José Fonseca6a6d3e42011-09-11 14:14:21 +0100160 EnumSig *parse_enum_sig();
161 BitmaskSig *parse_bitmask_sig();
José Fonsecae2d65222011-09-11 22:04:07 +0100162
Jose Fonsecade9621f2015-08-14 00:10:21 +0100163public:
José Fonseca340f5692011-11-30 07:04:44 +0000164 static CallFlags
165 lookupCallFlags(const char *name);
166
Jose Fonsecade9621f2015-08-14 00:10:21 +0100167protected:
José Fonsecae2d65222011-09-11 22:04:07 +0100168 Call *parse_Call(Mode mode);
José Fonseca6a6d3e42011-09-11 14:14:21 +0100169
José Fonsecae2d65222011-09-11 22:04:07 +0100170 void parse_enter(Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400171
José Fonsecae2d65222011-09-11 22:04:07 +0100172 Call *parse_leave(Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400173
José Fonsecae2d65222011-09-11 22:04:07 +0100174 bool parse_call_details(Call *call, Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400175
Eugene Velesevichaa1b2132013-04-23 17:56:14 +0400176 bool parse_call_backtrace(Call *call, Mode mode);
José Fonseca9c25e062013-05-06 07:25:40 +0100177 StackFrame * parse_backtrace_frame(Mode mode);
Eugene Velesevichaa1b2132013-04-23 17:56:14 +0400178
José Fonseca340f5692011-11-30 07:04:44 +0000179 void adjust_call_flags(Call *call);
180
José Fonsecae2d65222011-09-11 22:04:07 +0100181 void parse_arg(Call *call, Mode mode);
Zack Rusin46c4a322011-09-02 01:08:49 -0400182
José Fonseca838decf2011-09-11 14:44:41 +0100183 Value *parse_value(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400184 void scan_value(void);
José Fonsecae2d65222011-09-11 22:04:07 +0100185 inline Value *parse_value(Mode mode) {
186 if (mode == FULL) {
187 return parse_value();
188 } else {
189 scan_value();
190 return NULL;
191 }
192 }
Zack Rusin46c4a322011-09-02 01:08:49 -0400193
José Fonseca838decf2011-09-11 14:44:41 +0100194 Value *parse_sint();
Zack Rusin46c4a322011-09-02 01:08:49 -0400195 void scan_sint();
196
José Fonseca838decf2011-09-11 14:44:41 +0100197 Value *parse_uint();
Zack Rusin46c4a322011-09-02 01:08:49 -0400198 void scan_uint();
199
José Fonseca838decf2011-09-11 14:44:41 +0100200 Value *parse_float();
Zack Rusin46c4a322011-09-02 01:08:49 -0400201 void scan_float();
202
José Fonseca838decf2011-09-11 14:44:41 +0100203 Value *parse_double();
Zack Rusin46c4a322011-09-02 01:08:49 -0400204 void scan_double();
205
José Fonseca838decf2011-09-11 14:44:41 +0100206 Value *parse_string();
Zack Rusin46c4a322011-09-02 01:08:49 -0400207 void scan_string();
208
José Fonseca838decf2011-09-11 14:44:41 +0100209 Value *parse_enum();
Zack Rusin46c4a322011-09-02 01:08:49 -0400210 void scan_enum();
211
José Fonseca838decf2011-09-11 14:44:41 +0100212 Value *parse_bitmask();
Zack Rusin46c4a322011-09-02 01:08:49 -0400213 void scan_bitmask();
214
José Fonseca838decf2011-09-11 14:44:41 +0100215 Value *parse_array(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400216 void scan_array(void);
217
José Fonseca838decf2011-09-11 14:44:41 +0100218 Value *parse_blob(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400219 void scan_blob(void);
220
José Fonseca838decf2011-09-11 14:44:41 +0100221 Value *parse_struct();
Zack Rusin46c4a322011-09-02 01:08:49 -0400222 void scan_struct();
223
José Fonseca838decf2011-09-11 14:44:41 +0100224 Value *parse_opaque();
Zack Rusin46c4a322011-09-02 01:08:49 -0400225 void scan_opaque();
226
José Fonseca112a1322012-04-27 17:15:32 +0100227 Value *parse_repr();
228 void scan_repr();
229
José Fonsecad5cda7c2014-09-25 15:19:09 +0100230 Value *parse_wstring();
231 void scan_wstring();
232
José Fonseca838decf2011-09-11 14:44:41 +0100233 const char * read_string(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400234 void skip_string(void);
235
José Fonsecaeb644512011-12-11 10:33:55 +0000236 signed long long read_sint(void);
237 void skip_sint(void);
238
José Fonseca838decf2011-09-11 14:44:41 +0100239 unsigned long long read_uint(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400240 void skip_uint(void);
241
José Fonseca838decf2011-09-11 14:44:41 +0100242 inline int read_byte(void);
Zack Rusin46c4a322011-09-02 01:08:49 -0400243 inline void skip_byte(void);
José Fonseca7e329022010-11-19 17:05:18 +0000244};
245
246
Jose Fonseca0c287892015-10-14 14:19:01 +0100247AbstractParser *
248lastFrameLoopParser(AbstractParser *parser, int loopCount);
Jon Ashburn9459abf2014-09-04 23:03:37 +0100249
250
José Fonsecab4a3d142011-10-27 07:43:19 +0100251} /* namespace trace */
José Fonseca7e329022010-11-19 17:05:18 +0000252