blob: 27f9a56a420ff962100b770b41454b5179d857ef [file] [log] [blame]
Jose Fonsecae1dd9152016-03-23 11:01:39 +00001
José Fonseca7e329022010-11-19 17:05:18 +00002/**************************************************************************
3 *
José Fonseca946da142011-12-11 14:32:50 +00004 * Copyright 2011 Jose Fonseca
José Fonseca7e329022010-11-19 17:05:18 +00005 * Copyright 2010 VMware, Inc.
6 * All Rights Reserved.
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy
9 * of this software and associated documentation files (the "Software"), to deal
10 * in the Software without restriction, including without limitation the rights
11 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 * copies of the Software, and to permit persons to whom the Software is
13 * furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 * THE SOFTWARE.
25 *
26 **************************************************************************/
27
José Fonseca946da142011-12-11 14:32:50 +000028
José Fonseca71913ee2011-10-30 13:35:33 +000029#include <string.h>
José Fonseca689b6b22012-01-28 13:54:52 +000030#include <limits.h> // for CHAR_MAX
31#include <getopt.h>
Martin Olssonbdd92812012-03-26 10:40:45 +020032#ifndef _WIN32
33#include <unistd.h> // for isatty()
34#endif
José Fonseca71913ee2011-10-30 13:35:33 +000035
Jose Fonseca5b1d9012016-02-19 11:17:49 +000036#include <memory>
37#include <fstream>
Jose Fonsecae1dd9152016-03-23 11:01:39 +000038#include <string>
Jose Fonsecabc9c2e72018-01-12 21:03:31 +000039#include <regex>
Jose Fonseca5b1d9012016-02-19 11:17:49 +000040
Jose Fonseca05d2a582016-03-31 12:44:56 +010041#include "cxx_compat.hpp" // for std::to_string, std::make_unique
Jose Fonseca841f3912016-02-26 07:17:31 +000042
José Fonseca5cc28b02011-10-30 13:38:25 +000043#include "cli.hpp"
José Fonsecab122f502011-12-10 18:07:25 +000044#include "cli_pager.hpp"
José Fonseca822d20a2011-08-20 13:49:40 +010045
José Fonseca7e329022010-11-19 17:05:18 +000046#include "trace_parser.hpp"
Jose Fonseca5b1d9012016-02-19 11:17:49 +000047#include "trace_dump_internal.hpp"
José Fonseca225193d2012-01-26 19:08:32 +000048#include "trace_callset.hpp"
Carl Wortha7e7b272012-08-12 16:48:10 -070049#include "trace_option.hpp"
José Fonseca946da142011-12-11 14:32:50 +000050
José Fonseca7e329022010-11-19 17:05:18 +000051
Carl Worthe525e6f2011-10-20 15:22:09 -070052enum ColorOption {
53 COLOR_OPTION_NEVER = 0,
54 COLOR_OPTION_ALWAYS = 1,
55 COLOR_OPTION_AUTO = -1
56};
57
58static ColorOption color = COLOR_OPTION_AUTO;
José Fonseca822d20a2011-08-20 13:49:40 +010059
José Fonseca340f5692011-11-30 07:04:44 +000060static bool verbose = false;
61
José Fonseca225193d2012-01-26 19:08:32 +000062static trace::CallSet calls(trace::FREQUENCY_ALL);
63
José Fonseca3c167fe2011-10-30 14:21:03 +000064static const char *synopsis = "Dump given trace(s) to standard output.";
65
66static void
67usage(void)
Carl Worthbd5e1642011-10-21 20:40:56 -070068{
José Fonseca3c167fe2011-10-30 14:21:03 +000069 std::cout
José Fonseca689b6b22012-01-28 13:54:52 +000070 << "usage: apitrace dump [OPTIONS] TRACE_FILE...\n"
José Fonseca3c167fe2011-10-30 14:21:03 +000071 << synopsis << "\n"
José Fonseca646a00d2011-10-30 14:07:20 +000072 "\n"
José Fonseca689b6b22012-01-28 13:54:52 +000073 " -h, --help show this help message and exit\n"
74 " -v, --verbose verbose output\n"
75 " --calls=CALLSET only dump specified calls\n"
76 " --color[=WHEN]\n"
77 " --colour[=WHEN] colored syntax highlighting\n"
78 " WHEN is 'auto', 'always', or 'never'\n"
Jose Fonsecabc9c2e72018-01-12 21:03:31 +000079 " --grep[=REGEX] dump only calls whose function names match regex\n"
José Fonseca689b6b22012-01-28 13:54:52 +000080 " --thread-ids=[=BOOL] dump thread ids [default: no]\n"
José Fonseca0a478812012-03-16 15:40:04 +000081 " --call-nos[=BOOL] dump call numbers[default: yes]\n"
82 " --arg-names[=BOOL] dump argument names [default: yes]\n"
Jose Fonseca5b1d9012016-02-19 11:17:49 +000083 " --blobs dump blobs into files\n"
Jose Fonseca3ac81962017-10-10 18:40:21 +010084 " --multiline[=BOOL] dump newline in strings literally [default: yes]\n"
José Fonseca689b6b22012-01-28 13:54:52 +000085 "\n"
José Fonseca946da142011-12-11 14:32:50 +000086 ;
José Fonseca822d20a2011-08-20 13:49:40 +010087}
88
José Fonseca689b6b22012-01-28 13:54:52 +000089enum {
José Fonseca8a4baf82014-05-15 23:04:58 +010090 CALLS_OPT = CHAR_MAX + 1,
91 COLOR_OPT,
Jose Fonsecabc9c2e72018-01-12 21:03:31 +000092 GREP_OPT,
José Fonseca689b6b22012-01-28 13:54:52 +000093 THREAD_IDS_OPT,
José Fonseca0a478812012-03-16 15:40:04 +000094 CALL_NOS_OPT,
95 ARG_NAMES_OPT,
Jose Fonseca5b1d9012016-02-19 11:17:49 +000096 BLOBS_OPT,
Jose Fonseca3ac81962017-10-10 18:40:21 +010097 MULTILINE_OPT,
José Fonseca689b6b22012-01-28 13:54:52 +000098};
99
100const static char *
101shortOptions = "hv";
102
103const static struct option
104longOptions[] = {
105 {"help", no_argument, 0, 'h'},
106 {"verbose", no_argument, 0, 'v'},
107 {"calls", required_argument, 0, CALLS_OPT},
108 {"colour", optional_argument, 0, COLOR_OPT},
109 {"color", optional_argument, 0, COLOR_OPT},
Jose Fonsecabc9c2e72018-01-12 21:03:31 +0000110 {"grep", required_argument, 0, GREP_OPT},
José Fonseca689b6b22012-01-28 13:54:52 +0000111 {"thread-ids", optional_argument, 0, THREAD_IDS_OPT},
José Fonseca0a478812012-03-16 15:40:04 +0000112 {"call-nos", optional_argument, 0, CALL_NOS_OPT},
113 {"arg-names", optional_argument, 0, ARG_NAMES_OPT},
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000114 {"blobs", no_argument, 0, BLOBS_OPT},
Jose Fonseca3ac81962017-10-10 18:40:21 +0100115 {"multiline", optional_argument, 0, MULTILINE_OPT},
José Fonseca689b6b22012-01-28 13:54:52 +0000116 {0, 0, 0, 0}
117};
118
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000119
120class BlobDumper : public trace::Dumper
121{
122 unsigned callNo = 0;
123 unsigned blobNo = 0;
124public:
125 BlobDumper(std::ostream &_os, trace::DumpFlags _flags) :
126 Dumper(_os, _flags)
127 {
128 }
129
Jose Fonseca6ca93402016-03-05 14:44:48 +0000130 void visit(trace::Blob *blob) override {
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000131 std::string fileName = "blob_call" + std::to_string(callNo);
132 if (blobNo) {
133 fileName += "_";
134 fileName += std::to_string(blobNo);
135 }
136 fileName += ".bin";
137
138 {
139 std::ofstream stream(fileName, std::ofstream::binary);
140 stream.write((const char *)blob->buf, blob->size);
141 stream.close();
142 }
143
144 os << pointer << "blob(\"" << fileName << "\")" << normal;
145 ++blobNo;
146 }
147
Jose Fonseca6ca93402016-03-05 14:44:48 +0000148 void visit(trace::Call *call) override {
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000149 callNo = call->no;
150 blobNo = 0;
151 Dumper::visit(call);
152 }
153};
154
155
José Fonseca3c167fe2011-10-30 14:21:03 +0000156static int
José Fonsecaffba5972011-10-30 14:29:28 +0000157command(int argc, char *argv[])
José Fonseca7e329022010-11-19 17:05:18 +0000158{
José Fonseca946da142011-12-11 14:32:50 +0000159 trace::DumpFlags dumpFlags = 0;
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000160 bool blobs = false;
Jose Fonsecabc9c2e72018-01-12 21:03:31 +0000161 bool grep = false;
162 std::regex grepRegex;
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000163
José Fonseca689b6b22012-01-28 13:54:52 +0000164 int opt;
165 while ((opt = getopt_long(argc, argv, shortOptions, longOptions, NULL)) != -1) {
166 switch (opt) {
167 case 'h':
José Fonseca3c167fe2011-10-30 14:21:03 +0000168 usage();
José Fonseca646a00d2011-10-30 14:07:20 +0000169 return 0;
José Fonseca689b6b22012-01-28 13:54:52 +0000170 case 'v':
José Fonseca340f5692011-11-30 07:04:44 +0000171 verbose = true;
José Fonseca689b6b22012-01-28 13:54:52 +0000172 break;
173 case CALLS_OPT:
Lawrence L Love5ab982d2013-12-05 11:31:16 -0800174 calls.merge(optarg);
José Fonseca689b6b22012-01-28 13:54:52 +0000175 break;
176 case COLOR_OPT:
177 if (!optarg ||
178 !strcmp(optarg, "always")) {
179 color = COLOR_OPTION_ALWAYS;
180 } else if (!strcmp(optarg, "auto")) {
181 color = COLOR_OPTION_AUTO;
182 } else if (!strcmp(optarg, "never")) {
183 color = COLOR_OPTION_NEVER;
184 } else {
185 std::cerr << "error: unknown color argument " << optarg << "\n";
186 return 1;
187 }
188 break;
Jose Fonsecabc9c2e72018-01-12 21:03:31 +0000189 case GREP_OPT:
190 grepRegex = std::regex(optarg);
191 grep = true;
192 break;
José Fonseca0a478812012-03-16 15:40:04 +0000193 case THREAD_IDS_OPT:
José Fonseca8a4baf82014-05-15 23:04:58 +0100194 if (trace::boolOption(optarg)) {
195 dumpFlags |= trace::DUMP_FLAG_THREAD_IDS;
196 } else {
197 dumpFlags &= ~trace::DUMP_FLAG_THREAD_IDS;
198 }
José Fonseca0a478812012-03-16 15:40:04 +0000199 break;
200 case CALL_NOS_OPT:
Carl Wortha7e7b272012-08-12 16:48:10 -0700201 if (trace::boolOption(optarg)) {
José Fonseca0a478812012-03-16 15:40:04 +0000202 dumpFlags &= ~trace::DUMP_FLAG_NO_CALL_NO;
203 } else {
204 dumpFlags |= trace::DUMP_FLAG_NO_CALL_NO;
205 }
206 break;
José Fonseca689b6b22012-01-28 13:54:52 +0000207 case ARG_NAMES_OPT:
Carl Wortha7e7b272012-08-12 16:48:10 -0700208 if (trace::boolOption(optarg)) {
José Fonseca689b6b22012-01-28 13:54:52 +0000209 dumpFlags &= ~trace::DUMP_FLAG_NO_ARG_NAMES;
210 } else {
211 dumpFlags |= trace::DUMP_FLAG_NO_ARG_NAMES;
212 }
213 break;
Jose Fonseca3ac81962017-10-10 18:40:21 +0100214 case MULTILINE_OPT:
215 if (trace::boolOption(optarg)) {
216 dumpFlags &= ~trace::DUMP_FLAG_NO_MULTILINE;
217 } else {
218 dumpFlags |= trace::DUMP_FLAG_NO_MULTILINE;
219 }
220 break;
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000221 case BLOBS_OPT:
222 blobs = true;
223 break;
José Fonseca689b6b22012-01-28 13:54:52 +0000224 default:
José Fonseca4ce88b82013-10-11 17:24:47 -0700225 std::cerr << "error: unexpected option `" << (char)opt << "`\n";
José Fonseca3c167fe2011-10-30 14:21:03 +0000226 usage();
José Fonseca822d20a2011-08-20 13:49:40 +0100227 return 1;
228 }
229 }
230
Carl Worthe525e6f2011-10-20 15:22:09 -0700231 if (color == COLOR_OPTION_AUTO) {
232#ifdef _WIN32
233 color = COLOR_OPTION_ALWAYS;
234#else
José Fonseca685c5152013-06-09 11:00:20 +0100235 color = isatty(STDOUT_FILENO) ? COLOR_OPTION_ALWAYS : COLOR_OPTION_NEVER;
José Fonsecab122f502011-12-10 18:07:25 +0000236 pipepager();
Carl Worthe525e6f2011-10-20 15:22:09 -0700237#endif
238 }
239
José Fonseca946da142011-12-11 14:32:50 +0000240 if (color == COLOR_OPTION_NEVER) {
241 dumpFlags |= trace::DUMP_FLAG_NO_COLOR;
242 }
243
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000244 std::unique_ptr<trace::Dumper> dumper;
245
246 if (blobs) {
Jose Fonseca05d2a582016-03-31 12:44:56 +0100247 dumper = std::make_unique<BlobDumper>(std::cout, dumpFlags);
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000248 } else {
Jose Fonseca05d2a582016-03-31 12:44:56 +0100249 dumper = std::make_unique<trace::Dumper>(std::cout, dumpFlags);
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000250 }
251
José Fonseca689b6b22012-01-28 13:54:52 +0000252 for (int i = optind; i < argc; ++i) {
José Fonsecab4a3d142011-10-27 07:43:19 +0100253 trace::Parser p;
José Fonseca610ed332011-06-04 22:55:42 +0100254
255 if (!p.open(argv[i])) {
José Fonseca610ed332011-06-04 22:55:42 +0100256 return 1;
257 }
258
Jose Fonseca4b530662019-01-15 14:53:08 +0000259 if (!grep) {
260 const trace::Properties &properties = p.getProperties();
261 for (auto & kv : properties) {
262 std::cout << "// " << kv.first << " = ";
263 dumper->visitString(kv.second.c_str());
264 std::cout << std::endl;
265 }
Jose Fonsecac110dbc2017-06-23 13:04:17 +0100266 }
267
José Fonsecab4a3d142011-10-27 07:43:19 +0100268 trace::Call *call;
José Fonseca610ed332011-06-04 22:55:42 +0100269 while ((call = p.parse_call())) {
Jose Fonsecabc9c2e72018-01-12 21:03:31 +0000270 if (call->no > calls.getLast()) {
271 delete call;
272 break;
273 }
274 if (calls.contains(*call) &&
275 (!grep ||
276 std::regex_search(call->sig->name, grepRegex))) {
José Fonseca225193d2012-01-26 19:08:32 +0000277 if (verbose ||
278 !(call->flags & trace::CALL_FLAG_VERBOSE)) {
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000279 dumper->visit(call);
Jose Fonsecabc9c2e72018-01-12 21:03:31 +0000280 if (dumpFlags & trace::DUMP_FLAG_NO_MULTILINE) {
281 std::cout << '\n';
282 }
283 if (grep) {
284 std::cout << std::flush;
285 }
José Fonseca0af96702011-12-22 21:35:10 +0000286 }
José Fonseca340f5692011-11-30 07:04:44 +0000287 }
José Fonseca610ed332011-06-04 22:55:42 +0100288 delete call;
José Fonseca19828972010-11-29 20:34:32 +0000289 }
290 }
José Fonseca822d20a2011-08-20 13:49:40 +0100291
José Fonseca19828972010-11-29 20:34:32 +0000292 return 0;
José Fonseca7e329022010-11-19 17:05:18 +0000293}
José Fonseca3c167fe2011-10-30 14:21:03 +0000294
Carl Worth68f7c982011-11-01 13:47:26 -0700295const Command dump_command = {
José Fonseca3c167fe2011-10-30 14:21:03 +0000296 "dump",
297 synopsis,
298 usage,
299 command
300};