forked from nico/demumble
-
Notifications
You must be signed in to change notification settings - Fork 0
/
demumble.cc
147 lines (133 loc) · 4.16 KB
/
demumble.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <algorithm>
extern "C" {
char* __cxa_demangle(const char* mangled_name,
char* buf,
size_t* n,
int* status);
typedef void* (*malloc_func_t)(size_t);
typedef void (*free_func_t)(void*);
char* __unDName(char* buffer,
const char* mangled,
int buflen,
malloc_func_t memget,
free_func_t memfree,
unsigned short int flags);
}
const char kDemumbleVersion[] = "1.0.0.git";
static void print_help(FILE* out) {
fprintf(out,
"usage: demumble [options] [symbols...]\n"
"\n"
"if symbols are unspecified, reads from stdin.\n"
"\n"
"options:\n"
" -m only print mangled names that were demangled, omit other output\n"
" --version print demumble version (\"%s\")\n", kDemumbleVersion);
}
static bool starts_with(const char* s, const char* prefix) {
return strncmp(s, prefix, strlen(prefix)) == 0;
}
static void print_demangled(const char* s) {
const char* cxa_in = s;
if (starts_with(s, "__Z") || starts_with(s, "____Z"))
cxa_in += 1;
if (char* itanium = __cxa_demangle(cxa_in, NULL, NULL, NULL)) {
printf("%s", itanium);
free(itanium);
} else if (char* ms = __unDName(NULL, s, 0, &malloc, &free, 0)) {
printf("%s", ms);
free(ms);
} else {
printf("%s", s);
}
}
static bool is_mangle_char_posix(char c) {
return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') ||
(c >= '0' && c <= '9') || c == '_';
}
static bool is_mangle_char_win(char c) {
return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') ||
(c >= '0' && c <= '9') || strchr("?_@$", c);
}
static bool is_plausible_itanium_prefix(char* s) {
// Itanium symbols start with 1-4 underscores followed by Z.
// strnstr() is BSD, so use a small local buffer and strstr().
const int N = 5; // == strlen("____Z")
char prefix[N + 1];
strncpy(prefix, s, N); prefix[N] = '\0';
return strstr(prefix, "_Z");
}
static char buf[8192];
int main(int argc, char* argv[]) {
enum { kPrintAll, kPrintMatching } print_mode = kPrintAll;
while (argc > 1 && argv[1][0] == '-') {
if (strcmp(argv[1], "-h") == 0 || strcmp(argv[1], "--help") == 0) {
print_help(stdout);
return 0;
} else if (strcmp(argv[1], "-m") == 0) {
print_mode = kPrintMatching;
} else if (strcmp(argv[1], "--version") == 0) {
printf("%s\n", kDemumbleVersion);
return 0;
} else if (strcmp(argv[1], "--") == 0) {
--argc;
++argv;
break;
} else {
fprintf(stderr, "demumble: unrecognized option `%s'\n", argv[1]);
print_help(stderr);
return 1;
}
--argc;
++argv;
}
for (int i = 1; i < argc; ++i) {
print_demangled(argv[i]);
printf("\n");
}
if (argc == 1) { // Read stdin instead.
// By default, don't demangle types. Mangled function names are unlikely
// to appear in text for since they start with _Z (or ___Z) or ?? / ?$ / ?@.
// But type manglings can be regular words ("Pi" is "int*").
// (For command-line args, do try to demangle types though.)
while (fgets(buf, sizeof(buf), stdin)) {
bool need_separator = false;
char* cur = buf;
char* end = cur + strlen(cur);
while (cur != end) {
size_t special = strcspn(cur, "_?");
if (print_mode == kPrintAll)
printf("%.*s", static_cast<int>(special), cur);
else if (need_separator)
printf("\n");
need_separator = false;
cur += special;
if (cur == end)
break;
size_t n_sym = 0;
if (*cur == '?')
while (cur + n_sym != end && is_mangle_char_win(cur[n_sym]))
++n_sym;
else if (is_plausible_itanium_prefix(cur))
while (cur + n_sym != end && is_mangle_char_posix(cur[n_sym]))
++n_sym;
else {
if (print_mode == kPrintAll)
printf("_");
++cur;
continue;
}
char tmp = cur[n_sym];
cur[n_sym] = '\0';
print_demangled(cur);
need_separator = true;
cur[n_sym] = tmp;
cur += n_sym;
}
}
}
}