blob: d6f50141a3ebda2495dc0eacfddd0415bcdfac55 [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>
José Fonseca71913ee2011-10-30 13:35:33 +000031
José Fonseca5cc28b02011-10-30 13:38:25 +000032#include "cli.hpp"
José Fonsecab122f502011-12-10 18:07:25 +000033#include "cli_pager.hpp"
José Fonseca822d20a2011-08-20 13:49:40 +010034
José Fonseca7e329022010-11-19 17:05:18 +000035#include "trace_parser.hpp"
José Fonseca946da142011-12-11 14:32:50 +000036#include "trace_dump.hpp"
José Fonseca225193d2012-01-26 19:08:32 +000037#include "trace_callset.hpp"
José Fonseca946da142011-12-11 14:32:50 +000038
José Fonseca7e329022010-11-19 17:05:18 +000039
Carl Worthe525e6f2011-10-20 15:22:09 -070040enum ColorOption {
41 COLOR_OPTION_NEVER = 0,
42 COLOR_OPTION_ALWAYS = 1,
43 COLOR_OPTION_AUTO = -1
44};
45
46static ColorOption color = COLOR_OPTION_AUTO;
José Fonseca822d20a2011-08-20 13:49:40 +010047
José Fonseca340f5692011-11-30 07:04:44 +000048static bool verbose = false;
49
José Fonseca225193d2012-01-26 19:08:32 +000050static trace::CallSet calls(trace::FREQUENCY_ALL);
51
José Fonseca3c167fe2011-10-30 14:21:03 +000052static const char *synopsis = "Dump given trace(s) to standard output.";
53
54static void
55usage(void)
Carl Worthbd5e1642011-10-21 20:40:56 -070056{
José Fonseca3c167fe2011-10-30 14:21:03 +000057 std::cout
José Fonseca689b6b22012-01-28 13:54:52 +000058 << "usage: apitrace dump [OPTIONS] TRACE_FILE...\n"
José Fonseca3c167fe2011-10-30 14:21:03 +000059 << synopsis << "\n"
José Fonseca646a00d2011-10-30 14:07:20 +000060 "\n"
José Fonseca689b6b22012-01-28 13:54:52 +000061 " -h, --help show this help message and exit\n"
62 " -v, --verbose verbose output\n"
63 " --calls=CALLSET only dump specified calls\n"
64 " --color[=WHEN]\n"
65 " --colour[=WHEN] colored syntax highlighting\n"
66 " WHEN is 'auto', 'always', or 'never'\n"
67 " --arg-names[=BOOL] dump argument names [default: yes]\n"
68 " --thread-ids=[=BOOL] dump thread ids [default: no]\n"
69 "\n"
José Fonseca946da142011-12-11 14:32:50 +000070 ;
José Fonseca822d20a2011-08-20 13:49:40 +010071}
72
José Fonseca689b6b22012-01-28 13:54:52 +000073enum {
74 CALLS_OPT = CHAR_MAX + 1,
75 COLOR_OPT,
76 ARG_NAMES_OPT,
77 THREAD_IDS_OPT,
78};
79
80const static char *
81shortOptions = "hv";
82
83const static struct option
84longOptions[] = {
85 {"help", no_argument, 0, 'h'},
86 {"verbose", no_argument, 0, 'v'},
87 {"calls", required_argument, 0, CALLS_OPT},
88 {"colour", optional_argument, 0, COLOR_OPT},
89 {"color", optional_argument, 0, COLOR_OPT},
90 {"arg-names", optional_argument, 0, ARG_NAMES_OPT},
91 {"thread-ids", optional_argument, 0, THREAD_IDS_OPT},
92 {0, 0, 0, 0}
93};
94
95static bool
96boolOption(const char *option, bool default_ = true) {
97 if (!option) {
98 return default_;
99 }
100 if (strcmp(option, "0") == 0 ||
101 strcmp(option, "no") == 0 ||
102 strcmp(option, "false") == 0) {
103 return false;
104 }
105 if (strcmp(option, "0") == 0 ||
106 strcmp(option, "yes") == 0 ||
107 strcmp(option, "true") == 0) {
108 return true;
109 }
110 std::cerr << "error: unexpected bool " << option << "\n";
111 return default_;
112}
113
José Fonseca3c167fe2011-10-30 14:21:03 +0000114static int
José Fonsecaffba5972011-10-30 14:29:28 +0000115command(int argc, char *argv[])
José Fonseca7e329022010-11-19 17:05:18 +0000116{
José Fonseca946da142011-12-11 14:32:50 +0000117 trace::DumpFlags dumpFlags = 0;
José Fonseca0af96702011-12-22 21:35:10 +0000118 bool dumpThreadIds = false;
José Fonseca689b6b22012-01-28 13:54:52 +0000119
José Fonseca689b6b22012-01-28 13:54:52 +0000120 int opt;
121 while ((opt = getopt_long(argc, argv, shortOptions, longOptions, NULL)) != -1) {
122 switch (opt) {
123 case 'h':
José Fonseca3c167fe2011-10-30 14:21:03 +0000124 usage();
José Fonseca646a00d2011-10-30 14:07:20 +0000125 return 0;
José Fonseca689b6b22012-01-28 13:54:52 +0000126 case 'v':
José Fonseca340f5692011-11-30 07:04:44 +0000127 verbose = true;
José Fonseca689b6b22012-01-28 13:54:52 +0000128 break;
129 case CALLS_OPT:
130 calls = trace::CallSet(optarg);
131 break;
132 case COLOR_OPT:
133 if (!optarg ||
134 !strcmp(optarg, "always")) {
135 color = COLOR_OPTION_ALWAYS;
136 } else if (!strcmp(optarg, "auto")) {
137 color = COLOR_OPTION_AUTO;
138 } else if (!strcmp(optarg, "never")) {
139 color = COLOR_OPTION_NEVER;
140 } else {
141 std::cerr << "error: unknown color argument " << optarg << "\n";
142 return 1;
143 }
144 break;
145 case ARG_NAMES_OPT:
146 if (boolOption(optarg)) {
147 dumpFlags &= ~trace::DUMP_FLAG_NO_ARG_NAMES;
148 } else {
149 dumpFlags |= trace::DUMP_FLAG_NO_ARG_NAMES;
150 }
151 break;
152 case THREAD_IDS_OPT:
José Fonseca54f752c2012-01-31 10:02:52 +0000153 dumpThreadIds = boolOption(optarg);
José Fonseca689b6b22012-01-28 13:54:52 +0000154 break;
155 default:
156 std::cerr << "error: unexpected option `" << opt << "`\n";
José Fonseca3c167fe2011-10-30 14:21:03 +0000157 usage();
José Fonseca822d20a2011-08-20 13:49:40 +0100158 return 1;
159 }
160 }
161
Carl Worthe525e6f2011-10-20 15:22:09 -0700162 if (color == COLOR_OPTION_AUTO) {
163#ifdef _WIN32
164 color = COLOR_OPTION_ALWAYS;
165#else
166 color = isatty(1) ? COLOR_OPTION_ALWAYS : COLOR_OPTION_NEVER;
José Fonsecab122f502011-12-10 18:07:25 +0000167 pipepager();
Carl Worthe525e6f2011-10-20 15:22:09 -0700168#endif
169 }
170
José Fonseca946da142011-12-11 14:32:50 +0000171 if (color == COLOR_OPTION_NEVER) {
172 dumpFlags |= trace::DUMP_FLAG_NO_COLOR;
173 }
174
José Fonseca689b6b22012-01-28 13:54:52 +0000175 for (int i = optind; i < argc; ++i) {
José Fonsecab4a3d142011-10-27 07:43:19 +0100176 trace::Parser p;
José Fonseca610ed332011-06-04 22:55:42 +0100177
178 if (!p.open(argv[i])) {
179 std::cerr << "error: failed to open " << argv[i] << "\n";
180 return 1;
181 }
182
José Fonsecab4a3d142011-10-27 07:43:19 +0100183 trace::Call *call;
José Fonseca610ed332011-06-04 22:55:42 +0100184 while ((call = p.parse_call())) {
José Fonseca225193d2012-01-26 19:08:32 +0000185 if (calls.contains(*call)) {
186 if (verbose ||
187 !(call->flags & trace::CALL_FLAG_VERBOSE)) {
188 if (dumpThreadIds) {
189 std::cout << std::hex << call->thread_id << std::dec << " ";
190 }
191 trace::dump(*call, std::cout, dumpFlags);
José Fonseca0af96702011-12-22 21:35:10 +0000192 }
José Fonseca340f5692011-11-30 07:04:44 +0000193 }
José Fonseca610ed332011-06-04 22:55:42 +0100194 delete call;
José Fonseca19828972010-11-29 20:34:32 +0000195 }
196 }
José Fonseca822d20a2011-08-20 13:49:40 +0100197
José Fonseca19828972010-11-29 20:34:32 +0000198 return 0;
José Fonseca7e329022010-11-19 17:05:18 +0000199}
José Fonseca3c167fe2011-10-30 14:21:03 +0000200
Carl Worth68f7c982011-11-01 13:47:26 -0700201const Command dump_command = {
José Fonseca3c167fe2011-10-30 14:21:03 +0000202 "dump",
203 synopsis,
204 usage,
205 command
206};