author | coleenp |
Thu, 10 Jan 2019 15:13:51 -0500 | |
changeset 53244 | 9807daeb47c4 |
parent 49364 | 601146c66cad |
permissions | -rw-r--r-- |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
1 |
/* |
53244
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
49364
diff
changeset
|
2 |
* Copyright (c) 1997, 2019, Oracle and/or its affiliates. All rights reserved. |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
3 |
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
4 |
* |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
5 |
* This code is free software; you can redistribute it and/or modify it |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
6 |
* under the terms of the GNU General Public License version 2 only, as |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
7 |
* published by the Free Software Foundation. |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
8 |
* |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
9 |
* This code is distributed in the hope that it will be useful, but WITHOUT |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
10 |
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
11 |
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
12 |
* version 2 for more details (a copy is included in the LICENSE file that |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
13 |
* accompanied this code). |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
14 |
* |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
15 |
* You should have received a copy of the GNU General Public License version |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
16 |
* 2 along with this work; if not, write to the Free Software Foundation, |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
17 |
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
18 |
* |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
19 |
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
20 |
* or visit www.oracle.com if you need additional information or have any |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
21 |
* questions. |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
22 |
* |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
23 |
*/ |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
24 |
|
53244
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
49364
diff
changeset
|
25 |
#ifndef SHARE_UTILITIES_ELFFILE_HPP |
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
49364
diff
changeset
|
26 |
#define SHARE_UTILITIES_ELFFILE_HPP |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
27 |
|
48988
c2cd23e1d9cb
8198275: AIX build broken after latest whitebox.cpp changes
mbaesken
parents:
48975
diff
changeset
|
28 |
#if !defined(_WINDOWS) && !defined(__APPLE__) && !defined(_AIX) |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
29 |
|
10565 | 30 |
#if defined(__OpenBSD__) |
31 |
#include <sys/exec_elf.h> |
|
32 |
#else |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
33 |
#include <elf.h> |
10565 | 34 |
#endif |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
35 |
#include <stdio.h> |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
36 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
37 |
#ifdef _LP64 |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
38 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
39 |
typedef Elf64_Half Elf_Half; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
40 |
typedef Elf64_Word Elf_Word; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
41 |
typedef Elf64_Off Elf_Off; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
42 |
typedef Elf64_Addr Elf_Addr; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
43 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
44 |
typedef Elf64_Ehdr Elf_Ehdr; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
45 |
typedef Elf64_Shdr Elf_Shdr; |
15926
8e87d545195f
7107135: Stack guard pages are no more protected after loading a shared library with executable stack
iklam
parents:
13963
diff
changeset
|
46 |
typedef Elf64_Phdr Elf_Phdr; |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
47 |
typedef Elf64_Sym Elf_Sym; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
48 |
|
10565 | 49 |
#if !defined(_ALLBSD_SOURCE) || defined(__APPLE__) |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
50 |
#define ELF_ST_TYPE ELF64_ST_TYPE |
10565 | 51 |
#endif |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
52 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
53 |
#else |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
54 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
55 |
typedef Elf32_Half Elf_Half; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
56 |
typedef Elf32_Word Elf_Word; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
57 |
typedef Elf32_Off Elf_Off; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
58 |
typedef Elf32_Addr Elf_Addr; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
59 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
60 |
typedef Elf32_Ehdr Elf_Ehdr; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
61 |
typedef Elf32_Shdr Elf_Shdr; |
15926
8e87d545195f
7107135: Stack guard pages are no more protected after loading a shared library with executable stack
iklam
parents:
13963
diff
changeset
|
62 |
typedef Elf32_Phdr Elf_Phdr; |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
63 |
typedef Elf32_Sym Elf_Sym; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
64 |
|
10565 | 65 |
#if !defined(_ALLBSD_SOURCE) || defined(__APPLE__) |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
66 |
#define ELF_ST_TYPE ELF32_ST_TYPE |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
67 |
#endif |
10565 | 68 |
#endif |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
69 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
70 |
#include "globalDefinitions.hpp" |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
71 |
#include "memory/allocation.hpp" |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
72 |
#include "utilities/decoder.hpp" |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
73 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
74 |
class ElfStringTable; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
75 |
class ElfSymbolTable; |
22857
2167396cfc83
8019929: PPC64 (part 107): Extend ELF-decoder to support PPC64 function descriptor tables
simonis
parents:
15926
diff
changeset
|
76 |
class ElfFuncDescTable; |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
77 |
|
48975 | 78 |
// ELF section, may or may not have cached data |
49364
601146c66cad
8173070: Remove ValueObj class for allocation subclassing for runtime code
coleenp
parents:
48988
diff
changeset
|
79 |
class ElfSection { |
48975 | 80 |
private: |
81 |
Elf_Shdr _section_hdr; |
|
82 |
void* _section_data; |
|
83 |
NullDecoder::decoder_status _stat; |
|
84 |
public: |
|
85 |
ElfSection(FILE* fd, const Elf_Shdr& hdr); |
|
86 |
~ElfSection(); |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
87 |
|
48975 | 88 |
NullDecoder::decoder_status status() const { return _stat; } |
89 |
||
90 |
const Elf_Shdr* section_header() const { return &_section_hdr; } |
|
91 |
const void* section_data() const { return (const void*)_section_data; } |
|
92 |
private: |
|
93 |
// load this section. |
|
94 |
// it return no_error, when it fails to cache the section data due to lack of memory |
|
95 |
NullDecoder::decoder_status load_section(FILE* const file, const Elf_Shdr& hdr); |
|
96 |
}; |
|
97 |
||
98 |
class FileReader : public StackObj { |
|
99 |
protected: |
|
100 |
FILE* const _fd; |
|
101 |
public: |
|
102 |
FileReader(FILE* const fd) : _fd(fd) {}; |
|
103 |
bool read(void* buf, size_t size); |
|
104 |
int read_buffer(void* buf, size_t size); |
|
105 |
bool set_position(long offset); |
|
106 |
}; |
|
107 |
||
108 |
// Mark current position, so we can get back to it after |
|
109 |
// reads. |
|
110 |
class MarkedFileReader : public FileReader { |
|
111 |
private: |
|
112 |
long _marked_pos; |
|
113 |
public: |
|
114 |
MarkedFileReader(FILE* const fd); |
|
115 |
~MarkedFileReader(); |
|
116 |
||
117 |
bool has_mark() const { return _marked_pos >= 0; } |
|
118 |
}; |
|
119 |
||
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
120 |
// ElfFile is basically an elf file parser, which can lookup the symbol |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
121 |
// that is the nearest to the given address. |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
122 |
// Beware, this code is called from vm error reporting code, when vm is already |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
123 |
// in "error" state, so there are scenarios, lookup will fail. We want this |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
124 |
// part of code to be very defensive, and bait out if anything went wrong. |
13195 | 125 |
class ElfFile: public CHeapObj<mtInternal> { |
11483 | 126 |
friend class ElfDecoder; |
48975 | 127 |
|
128 |
private: |
|
129 |
// link ElfFiles |
|
130 |
ElfFile* _next; |
|
131 |
||
132 |
// Elf file |
|
133 |
char* _filepath; |
|
134 |
FILE* _file; |
|
135 |
||
136 |
// Elf header |
|
137 |
Elf_Ehdr _elfHdr; |
|
138 |
||
139 |
// symbol tables |
|
140 |
ElfSymbolTable* _symbol_tables; |
|
141 |
||
142 |
// regular string tables |
|
143 |
ElfStringTable* _string_tables; |
|
144 |
||
145 |
// section header string table, used for finding section name |
|
146 |
ElfStringTable* _shdr_string_table; |
|
147 |
||
148 |
// function descriptors table |
|
149 |
ElfFuncDescTable* _funcDesc_table; |
|
150 |
||
151 |
NullDecoder::decoder_status _status; |
|
152 |
||
153 |
public: |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
154 |
ElfFile(const char* filepath); |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
155 |
~ElfFile(); |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
156 |
|
11483 | 157 |
bool decode(address addr, char* buf, int buflen, int* offset); |
48975 | 158 |
|
159 |
const char* filepath() const { |
|
160 |
return _filepath; |
|
161 |
} |
|
162 |
||
163 |
bool same_elf_file(const char* filepath) const { |
|
164 |
assert(filepath != NULL, "null file path"); |
|
165 |
return (_filepath != NULL && !strcmp(filepath, _filepath)); |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
166 |
} |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
167 |
|
48975 | 168 |
NullDecoder::decoder_status get_status() const { |
169 |
return _status; |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
170 |
} |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
171 |
|
48975 | 172 |
// Returns true if the elf file is marked NOT to require an executable stack, |
173 |
// or if the file could not be opened. |
|
174 |
// Returns false if the elf file requires an executable stack, the stack flag |
|
175 |
// is not set at all, or if the file can not be read. |
|
176 |
// On systems other than linux it always returns false. |
|
177 |
static bool specifies_noexecstack(const char* filepath) NOT_LINUX({ return false; }); |
|
178 |
private: |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
179 |
// sanity check, if the file is a real elf file |
46629
8eeacdc76bf2
8183262: noexecstack check in os::dll_load on Linux is too expensive
kvn
parents:
22872
diff
changeset
|
180 |
static bool is_elf_file(Elf_Ehdr&); |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
181 |
|
48975 | 182 |
// parse this elf file |
183 |
NullDecoder::decoder_status parse_elf(const char* filename); |
|
184 |
||
185 |
// load string, symbol and function descriptor tables from the elf file |
|
186 |
NullDecoder::decoder_status load_tables(); |
|
187 |
||
188 |
ElfFile* next() const { return _next; } |
|
189 |
void set_next(ElfFile* file) { _next = file; } |
|
190 |
||
191 |
// find a section by name, return section index |
|
192 |
// if there is no such section, return -1 |
|
193 |
int section_by_name(const char* name, Elf_Shdr& hdr); |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
194 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
195 |
// string tables are stored in a linked list |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
196 |
void add_string_table(ElfStringTable* table); |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
197 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
198 |
// symbol tables are stored in a linked list |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
199 |
void add_symbol_table(ElfSymbolTable* table); |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
200 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
201 |
// return a string table at specified section index |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
202 |
ElfStringTable* get_string_table(int index); |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
203 |
|
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
204 |
|
48975 | 205 |
FILE* const fd() const { return _file; } |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
206 |
|
48975 | 207 |
// Cleanup string, symbol and function descriptor tables |
208 |
void cleanup_tables(); |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
209 |
|
48975 | 210 |
public: |
211 |
// For whitebox test |
|
212 |
static bool _do_not_cache_elf_section; |
|
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
213 |
}; |
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
214 |
|
22857
2167396cfc83
8019929: PPC64 (part 107): Extend ELF-decoder to support PPC64 function descriptor tables
simonis
parents:
15926
diff
changeset
|
215 |
#endif // !_WINDOWS && !__APPLE__ |
7447
32c42d627f41
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
zgu
parents:
diff
changeset
|
216 |
|
53244
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
49364
diff
changeset
|
217 |
#endif // SHARE_UTILITIES_ELFFILE_HPP |