blob: 62feef982b6bee5a9569b25a1fdc803965eccb9c [file] [log] [blame]
José Fonseca7e329022010-11-19 17:05:18 +00001/**************************************************************************
2 *
José Fonseca946da142011-12-11 14:32:50 +00003 * Copyright 2011 Jose Fonseca
José Fonseca7e329022010-11-19 17:05:18 +00004 * Copyright 2010 VMware, Inc.
5 * All Rights Reserved.
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy
8 * of this software and associated documentation files (the "Software"), to deal
9 * in the Software without restriction, including without limitation the rights
10 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 * copies of the Software, and to permit persons to whom the Software is
12 * furnished to do so, subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in
15 * all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
23 * THE SOFTWARE.
24 *
25 **************************************************************************/
26
José Fonseca946da142011-12-11 14:32:50 +000027
José Fonseca71913ee2011-10-30 13:35:33 +000028#include <string.h>
José Fonseca689b6b22012-01-28 13:54:52 +000029#include <limits.h> // for CHAR_MAX
30#include <getopt.h>
Martin Olssonbdd92812012-03-26 10:40:45 +020031#ifndef _WIN32
32#include <unistd.h> // for isatty()
33#endif
José Fonseca71913ee2011-10-30 13:35:33 +000034
Jose Fonseca5b1d9012016-02-19 11:17:49 +000035#include <memory>
36#include <fstream>
37
Jose Fonseca841f3912016-02-26 07:17:31 +000038#include "cxx_compat.hpp" // for std::to_string
39
José Fonseca5cc28b02011-10-30 13:38:25 +000040#include "cli.hpp"
José Fonsecab122f502011-12-10 18:07:25 +000041#include "cli_pager.hpp"
José Fonseca822d20a2011-08-20 13:49:40 +010042
José Fonseca7e329022010-11-19 17:05:18 +000043#include "trace_parser.hpp"
Jose Fonseca5b1d9012016-02-19 11:17:49 +000044#include "trace_dump_internal.hpp"
José Fonseca225193d2012-01-26 19:08:32 +000045#include "trace_callset.hpp"
Carl Wortha7e7b272012-08-12 16:48:10 -070046#include "trace_option.hpp"
José Fonseca946da142011-12-11 14:32:50 +000047
José Fonseca7e329022010-11-19 17:05:18 +000048
Carl Worthe525e6f2011-10-20 15:22:09 -070049enum ColorOption {
50 COLOR_OPTION_NEVER = 0,
51 COLOR_OPTION_ALWAYS = 1,
52 COLOR_OPTION_AUTO = -1
53};
54
55static ColorOption color = COLOR_OPTION_AUTO;
José Fonseca822d20a2011-08-20 13:49:40 +010056
José Fonseca340f5692011-11-30 07:04:44 +000057static bool verbose = false;
58
José Fonseca225193d2012-01-26 19:08:32 +000059static trace::CallSet calls(trace::FREQUENCY_ALL);
60
José Fonseca3c167fe2011-10-30 14:21:03 +000061static const char *synopsis = "Dump given trace(s) to standard output.";
62
63static void
64usage(void)
Carl Worthbd5e1642011-10-21 20:40:56 -070065{
José Fonseca3c167fe2011-10-30 14:21:03 +000066 std::cout
José Fonseca689b6b22012-01-28 13:54:52 +000067 << "usage: apitrace dump [OPTIONS] TRACE_FILE...\n"
José Fonseca3c167fe2011-10-30 14:21:03 +000068 << synopsis << "\n"
José Fonseca646a00d2011-10-30 14:07:20 +000069 "\n"
José Fonseca689b6b22012-01-28 13:54:52 +000070 " -h, --help show this help message and exit\n"
71 " -v, --verbose verbose output\n"
72 " --calls=CALLSET only dump specified calls\n"
73 " --color[=WHEN]\n"
74 " --colour[=WHEN] colored syntax highlighting\n"
75 " WHEN is 'auto', 'always', or 'never'\n"
José Fonseca689b6b22012-01-28 13:54:52 +000076 " --thread-ids=[=BOOL] dump thread ids [default: no]\n"
José Fonseca0a478812012-03-16 15:40:04 +000077 " --call-nos[=BOOL] dump call numbers[default: yes]\n"
78 " --arg-names[=BOOL] dump argument names [default: yes]\n"
Jose Fonseca5b1d9012016-02-19 11:17:49 +000079 " --blobs dump blobs into files\n"
José Fonseca689b6b22012-01-28 13:54:52 +000080 "\n"
José Fonseca946da142011-12-11 14:32:50 +000081 ;
José Fonseca822d20a2011-08-20 13:49:40 +010082}
83
José Fonseca689b6b22012-01-28 13:54:52 +000084enum {
José Fonseca8a4baf82014-05-15 23:04:58 +010085 CALLS_OPT = CHAR_MAX + 1,
86 COLOR_OPT,
José Fonseca689b6b22012-01-28 13:54:52 +000087 THREAD_IDS_OPT,
José Fonseca0a478812012-03-16 15:40:04 +000088 CALL_NOS_OPT,
89 ARG_NAMES_OPT,
Jose Fonseca5b1d9012016-02-19 11:17:49 +000090 BLOBS_OPT,
José Fonseca689b6b22012-01-28 13:54:52 +000091};
92
93const static char *
94shortOptions = "hv";
95
96const static struct option
97longOptions[] = {
98 {"help", no_argument, 0, 'h'},
99 {"verbose", no_argument, 0, 'v'},
100 {"calls", required_argument, 0, CALLS_OPT},
101 {"colour", optional_argument, 0, COLOR_OPT},
102 {"color", optional_argument, 0, COLOR_OPT},
José Fonseca689b6b22012-01-28 13:54:52 +0000103 {"thread-ids", optional_argument, 0, THREAD_IDS_OPT},
José Fonseca0a478812012-03-16 15:40:04 +0000104 {"call-nos", optional_argument, 0, CALL_NOS_OPT},
105 {"arg-names", optional_argument, 0, ARG_NAMES_OPT},
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000106 {"blobs", no_argument, 0, BLOBS_OPT},
José Fonseca689b6b22012-01-28 13:54:52 +0000107 {0, 0, 0, 0}
108};
109
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000110
111class BlobDumper : public trace::Dumper
112{
113 unsigned callNo = 0;
114 unsigned blobNo = 0;
115public:
116 BlobDumper(std::ostream &_os, trace::DumpFlags _flags) :
117 Dumper(_os, _flags)
118 {
119 }
120
Jose Fonseca6ca93402016-03-05 14:44:48 +0000121 void visit(trace::Blob *blob) override {
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000122 std::string fileName = "blob_call" + std::to_string(callNo);
123 if (blobNo) {
124 fileName += "_";
125 fileName += std::to_string(blobNo);
126 }
127 fileName += ".bin";
128
129 {
130 std::ofstream stream(fileName, std::ofstream::binary);
131 stream.write((const char *)blob->buf, blob->size);
132 stream.close();
133 }
134
135 os << pointer << "blob(\"" << fileName << "\")" << normal;
136 ++blobNo;
137 }
138
Jose Fonseca6ca93402016-03-05 14:44:48 +0000139 void visit(trace::Call *call) override {
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000140 callNo = call->no;
141 blobNo = 0;
142 Dumper::visit(call);
143 }
144};
145
146
José Fonseca3c167fe2011-10-30 14:21:03 +0000147static int
José Fonsecaffba5972011-10-30 14:29:28 +0000148command(int argc, char *argv[])
José Fonseca7e329022010-11-19 17:05:18 +0000149{
José Fonseca946da142011-12-11 14:32:50 +0000150 trace::DumpFlags dumpFlags = 0;
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000151 bool blobs = false;
152
José Fonseca689b6b22012-01-28 13:54:52 +0000153 int opt;
154 while ((opt = getopt_long(argc, argv, shortOptions, longOptions, NULL)) != -1) {
155 switch (opt) {
156 case 'h':
José Fonseca3c167fe2011-10-30 14:21:03 +0000157 usage();
José Fonseca646a00d2011-10-30 14:07:20 +0000158 return 0;
José Fonseca689b6b22012-01-28 13:54:52 +0000159 case 'v':
José Fonseca340f5692011-11-30 07:04:44 +0000160 verbose = true;
José Fonseca689b6b22012-01-28 13:54:52 +0000161 break;
162 case CALLS_OPT:
Lawrence L Love5ab982d2013-12-05 11:31:16 -0800163 calls.merge(optarg);
José Fonseca689b6b22012-01-28 13:54:52 +0000164 break;
165 case COLOR_OPT:
166 if (!optarg ||
167 !strcmp(optarg, "always")) {
168 color = COLOR_OPTION_ALWAYS;
169 } else if (!strcmp(optarg, "auto")) {
170 color = COLOR_OPTION_AUTO;
171 } else if (!strcmp(optarg, "never")) {
172 color = COLOR_OPTION_NEVER;
173 } else {
174 std::cerr << "error: unknown color argument " << optarg << "\n";
175 return 1;
176 }
177 break;
José Fonseca0a478812012-03-16 15:40:04 +0000178 case THREAD_IDS_OPT:
José Fonseca8a4baf82014-05-15 23:04:58 +0100179 if (trace::boolOption(optarg)) {
180 dumpFlags |= trace::DUMP_FLAG_THREAD_IDS;
181 } else {
182 dumpFlags &= ~trace::DUMP_FLAG_THREAD_IDS;
183 }
José Fonseca0a478812012-03-16 15:40:04 +0000184 break;
185 case CALL_NOS_OPT:
Carl Wortha7e7b272012-08-12 16:48:10 -0700186 if (trace::boolOption(optarg)) {
José Fonseca0a478812012-03-16 15:40:04 +0000187 dumpFlags &= ~trace::DUMP_FLAG_NO_CALL_NO;
188 } else {
189 dumpFlags |= trace::DUMP_FLAG_NO_CALL_NO;
190 }
191 break;
José Fonseca689b6b22012-01-28 13:54:52 +0000192 case ARG_NAMES_OPT:
Carl Wortha7e7b272012-08-12 16:48:10 -0700193 if (trace::boolOption(optarg)) {
José Fonseca689b6b22012-01-28 13:54:52 +0000194 dumpFlags &= ~trace::DUMP_FLAG_NO_ARG_NAMES;
195 } else {
196 dumpFlags |= trace::DUMP_FLAG_NO_ARG_NAMES;
197 }
198 break;
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000199 case BLOBS_OPT:
200 blobs = true;
201 break;
José Fonseca689b6b22012-01-28 13:54:52 +0000202 default:
José Fonseca4ce88b82013-10-11 17:24:47 -0700203 std::cerr << "error: unexpected option `" << (char)opt << "`\n";
José Fonseca3c167fe2011-10-30 14:21:03 +0000204 usage();
José Fonseca822d20a2011-08-20 13:49:40 +0100205 return 1;
206 }
207 }
208
Carl Worthe525e6f2011-10-20 15:22:09 -0700209 if (color == COLOR_OPTION_AUTO) {
210#ifdef _WIN32
211 color = COLOR_OPTION_ALWAYS;
212#else
José Fonseca685c5152013-06-09 11:00:20 +0100213 color = isatty(STDOUT_FILENO) ? COLOR_OPTION_ALWAYS : COLOR_OPTION_NEVER;
José Fonsecab122f502011-12-10 18:07:25 +0000214 pipepager();
Carl Worthe525e6f2011-10-20 15:22:09 -0700215#endif
216 }
217
José Fonseca946da142011-12-11 14:32:50 +0000218 if (color == COLOR_OPTION_NEVER) {
219 dumpFlags |= trace::DUMP_FLAG_NO_COLOR;
220 }
221
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000222 std::unique_ptr<trace::Dumper> dumper;
223
224 if (blobs) {
225 dumper = std::unique_ptr<trace::Dumper>(new BlobDumper(std::cout, dumpFlags));
226 } else {
227 dumper = std::unique_ptr<trace::Dumper>(new trace::Dumper(std::cout, dumpFlags));
228 }
229
José Fonseca689b6b22012-01-28 13:54:52 +0000230 for (int i = optind; i < argc; ++i) {
José Fonsecab4a3d142011-10-27 07:43:19 +0100231 trace::Parser p;
José Fonseca610ed332011-06-04 22:55:42 +0100232
233 if (!p.open(argv[i])) {
José Fonseca610ed332011-06-04 22:55:42 +0100234 return 1;
235 }
236
José Fonsecab4a3d142011-10-27 07:43:19 +0100237 trace::Call *call;
José Fonseca610ed332011-06-04 22:55:42 +0100238 while ((call = p.parse_call())) {
José Fonseca225193d2012-01-26 19:08:32 +0000239 if (calls.contains(*call)) {
240 if (verbose ||
241 !(call->flags & trace::CALL_FLAG_VERBOSE)) {
Jose Fonseca5b1d9012016-02-19 11:17:49 +0000242 dumper->visit(call);
José Fonseca0af96702011-12-22 21:35:10 +0000243 }
José Fonseca340f5692011-11-30 07:04:44 +0000244 }
José Fonseca610ed332011-06-04 22:55:42 +0100245 delete call;
José Fonseca19828972010-11-29 20:34:32 +0000246 }
247 }
José Fonseca822d20a2011-08-20 13:49:40 +0100248
José Fonseca19828972010-11-29 20:34:32 +0000249 return 0;
José Fonseca7e329022010-11-19 17:05:18 +0000250}
José Fonseca3c167fe2011-10-30 14:21:03 +0000251
Carl Worth68f7c982011-11-01 13:47:26 -0700252const Command dump_command = {
José Fonseca3c167fe2011-10-30 14:21:03 +0000253 "dump",
254 synopsis,
255 usage,
256 command
257};