1
/* Copyright (C) 2000 MySQL AB
3
This program is free software; you can redistribute it and/or modify
4
it under the terms of the GNU General Public License as published by
5
the Free Software Foundation; version 2 of the License.
7
This program is distributed in the hope that it will be useful,
8
but WITHOUT ANY WARRANTY; without even the implied warranty of
9
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
GNU General Public License for more details.
12
You should have received a copy of the GNU General Public License
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
16
/* Resolve numeric stack dump produced by drizzled 3.23.30 and later
17
versions into symbolic names. By Sasha Pachev <sasha@mysql.com>
25
#include "drizzled/charset_info.h"
26
#include "drizzled/internal/my_sys.h"
27
#include "drizzled/internal/m_string.h"
28
#include "drizzled/option.h"
29
#include <boost/program_options.hpp>
32
namespace po= boost::program_options;
33
using namespace drizzled;
35
#define INIT_SYM_TABLE 4096
36
#define INC_SYM_TABLE 4096
37
#define MAX_SYM_SIZE 128
38
#define DUMP_VERSION "1.4"
39
#define HEX_INVALID (unsigned char)255
41
typedef struct sym_entry
43
char symbol[MAX_SYM_SIZE];
48
std::string dump_fname, sym_fname;
49
static DYNAMIC_ARRAY sym_table; /* how do you like this , static DYNAMIC ? */
50
static FILE* fp_dump, *fp_sym = 0, *fp_out;
52
static void verify_sort(void);
54
static void die(const char* fmt, ...)
58
fprintf(stderr, "%s: ", internal::my_progname);
59
vfprintf(stderr, fmt, args);
60
fprintf(stderr, "\n");
65
static void open_files(void)
70
if (! dump_fname.empty() && !(fp_dump= fopen(dump_fname.c_str(), "r")))
71
die("Could not open %s", dump_fname.c_str());
72
/* if name not given, assume stdin*/
74
if (sym_fname.empty())
75
die("Please run nm --numeric-sort on drizzled binary that produced stack "
76
"trace dump and specify the path to it with -s or --symbols-file");
77
if (!(fp_sym= fopen(sym_fname.c_str(), "r")))
78
die("Could not open %s", sym_fname.c_str());
82
static unsigned char hex_val(char c)
85
if (my_isdigit(&my_charset_utf8_general_ci,c))
87
l = my_tolower(&my_charset_utf8_general_ci,c);
88
if (l < 'a' || l > 'f')
90
return (unsigned char)10 + ((unsigned char)c - (unsigned char)'a');
93
static unsigned long read_addr(char** buf)
97
unsigned long addr = 0;
99
while((c = hex_val(*p++)) != HEX_INVALID)
100
addr = (addr << 4) + c;
106
static int init_sym_entry(SYM_ENTRY* se, char* buf)
109
se->addr = (unsigned char*)read_addr(&buf);
113
while (my_isspace(&my_charset_utf8_general_ci,*buf++))
116
while (my_isspace(&my_charset_utf8_general_ci,*buf++))
117
/* empty - skip more space */;
119
/* now we are on the symbol */
120
for (p = se->symbol, p_end = se->symbol + sizeof(se->symbol) - 1;
121
*buf != '\n' && *buf && p < p_end; ++buf,++p)
124
if (!strcmp(se->symbol, "gcc2_compiled."))
129
static void init_sym_table(void)
132
if (my_init_dynamic_array(&sym_table, sizeof(SYM_ENTRY), INIT_SYM_TABLE,
134
die("Failed in my_init_dynamic_array() -- looks like out of memory problem");
136
while (fgets(buf, sizeof(buf), fp_sym))
139
if (init_sym_entry(&se, buf))
141
if (insert_dynamic(&sym_table, (unsigned char*)&se))
142
die("insert_dynamic() failed - looks like we are out of memory");
148
static void clean_up(void)
150
delete_dynamic(&sym_table);
153
static void verify_sort()
156
unsigned char* last = 0;
158
for (i = 0; i < sym_table.elements; i++)
161
get_dynamic(&sym_table, (unsigned char*)&se, i);
163
die("sym table does not appear to be sorted, did you forget "
164
"--numeric-sort arg to nm? trouble addr = %p, last = %p",
171
static SYM_ENTRY* resolve_addr(unsigned char* addr, SYM_ENTRY* se)
174
get_dynamic(&sym_table, (unsigned char*)se, 0);
178
for (i = 1; i < sym_table.elements; i++)
180
get_dynamic(&sym_table, (unsigned char*)se, i);
183
get_dynamic(&sym_table, (unsigned char*)se, i - 1);
192
static void do_resolve(void)
195
while (fgets(buf, sizeof(buf), fp_dump))
199
while (my_isspace(&my_charset_utf8_general_ci,*p))
202
if (*p++ == '0' && *p++ == 'x')
205
unsigned char* addr = (unsigned char*)read_addr(&p);
206
if (resolve_addr(addr, &se))
207
fprintf(fp_out, "%p %s + %d\n", addr, se.symbol,
208
(int) (addr - se.addr));
210
fprintf(fp_out, "%p (?)\n", addr);
222
int main(int argc, char** argv)
228
po::options_description commandline_options("Options specific to the commandline");
229
commandline_options.add_options()
230
("help,h", "Display this help and exit.")
231
("version,V", "Output version information and exit.")
234
po::options_description positional_options("Positional Options");
235
positional_options.add_options()
236
("symbols-file,s", po::value<string>(),
237
"Use specified symbols file.")
238
("numeric-dump-file,n", po::value<string>(),
239
"Read the dump from specified file.")
242
po::options_description long_options("Allowed Options");
243
long_options.add(commandline_options).add(positional_options);
245
po::positional_options_description p;
246
p.add("symbols-file,s", 1);
247
p.add("numeric-dump-file,n",1);
249
po::variables_map vm;
250
po::store(po::command_line_parser(argc, argv).options(long_options).positional(p).run(), vm);
253
if (vm.count("help"))
255
printf("%s Ver %s Distrib %s, for %s-%s (%s)\n",internal::my_progname,DUMP_VERSION,
256
VERSION,HOST_VENDOR,HOST_OS,HOST_CPU);
257
printf("MySQL AB, by Sasha Pachev\n");
258
printf("This software comes with ABSOLUTELY NO WARRANTY\n\n");
259
printf("Resolve numeric stack strace dump into symbols.\n\n");
260
printf("Usage: %s [OPTIONS] symbols-file [numeric-dump-file]\n",
261
internal::my_progname);
262
cout << long_options << endl;
264
"The symbols-file should include the output from: \n"
265
" 'nm --numeric-sort drizzled'.\n"
266
"The numeric-dump-file should contain a numeric stack trace "
268
"If the numeric-dump-file is not given, the stack trace is "
269
"read from stdin.\n");
272
if (vm.count("version"))
274
printf("%s Ver %s Distrib %s, for %s-%s (%s)\n",internal::my_progname,DUMP_VERSION,
275
VERSION,HOST_VENDOR,HOST_OS,HOST_CPU);
278
if (vm.count("symbols-file") && vm.count("numeric-file-dump"))
280
sym_fname= vm["symbols-file"].as<string>();
281
dump_fname= vm["numeric-dump-file"].as<string>();
289
catch(exception &err)
291
cerr << err.what() << endl;