Mercurial > hg > truffle
comparison src/share/vm/utilities/elfFile.cpp @ 2022:2d4762ec74af
7003748: Decode C stack frames when symbols are presented (PhoneHome project)
Summary: Implemented in-process C native stack frame decoding when symbols are available.
Reviewed-by: coleenp, never
author | zgu |
---|---|
date | Sat, 11 Dec 2010 13:20:56 -0500 |
parents | |
children | 2a3da7eaf4a6 |
comparison
equal
deleted
inserted
replaced
1972:f95d63e2154a | 2022:2d4762ec74af |
---|---|
1 /* | |
2 * Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved. | |
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. | |
4 * | |
5 * This code is free software; you can redistribute it and/or modify it | |
6 * under the terms of the GNU General Public License version 2 only, as | |
7 * published by the Free Software Foundation. | |
8 * | |
9 * This code is distributed in the hope that it will be useful, but WITHOUT | |
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | |
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License | |
12 * version 2 for more details (a copy is included in the LICENSE file that | |
13 * accompanied this code). | |
14 * | |
15 * You should have received a copy of the GNU General Public License version | |
16 * 2 along with this work; if not, write to the Free Software Foundation, | |
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. | |
18 * | |
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA | |
20 * or visit www.oracle.com if you need additional information or have any | |
21 * questions. | |
22 * | |
23 */ | |
24 | |
25 #include "precompiled.hpp" | |
26 | |
27 #ifndef _WINDOWS | |
28 | |
29 #include <string.h> | |
30 #include <stdio.h> | |
31 #include <limits.h> | |
32 | |
33 #include "memory/allocation.inline.hpp" | |
34 #include "utilities/decoder.hpp" | |
35 #include "utilities/elfFile.hpp" | |
36 #include "utilities/elfStringTable.hpp" | |
37 #include "utilities/elfSymbolTable.hpp" | |
38 | |
39 | |
40 ElfFile::ElfFile(const char* filepath) { | |
41 assert(filepath, "null file path"); | |
42 memset(&m_elfHdr, 0, sizeof(m_elfHdr)); | |
43 m_string_tables = NULL; | |
44 m_symbol_tables = NULL; | |
45 m_next = NULL; | |
46 m_status = Decoder::no_error; | |
47 | |
48 int len = strlen(filepath) + 1; | |
49 m_filepath = NEW_C_HEAP_ARRAY(char, len); | |
50 if (m_filepath != NULL) { | |
51 strcpy((char*)m_filepath, filepath); | |
52 m_file = fopen(filepath, "r"); | |
53 if (m_file != NULL) { | |
54 load_tables(); | |
55 } else { | |
56 m_status = Decoder::file_not_found; | |
57 } | |
58 } else { | |
59 m_status = Decoder::out_of_memory; | |
60 } | |
61 } | |
62 | |
63 ElfFile::~ElfFile() { | |
64 if (m_string_tables != NULL) { | |
65 delete m_string_tables; | |
66 } | |
67 | |
68 if (m_symbol_tables != NULL) { | |
69 delete m_symbol_tables; | |
70 } | |
71 | |
72 if (m_file != NULL) { | |
73 fclose(m_file); | |
74 } | |
75 | |
76 if (m_filepath != NULL) { | |
77 FREE_C_HEAP_ARRAY(char, m_filepath); | |
78 } | |
79 | |
80 if (m_next != NULL) { | |
81 delete m_next; | |
82 } | |
83 }; | |
84 | |
85 | |
86 //Check elf header to ensure the file is valid. | |
87 bool ElfFile::is_elf_file(Elf_Ehdr& hdr) { | |
88 return (ELFMAG0 == hdr.e_ident[EI_MAG0] && | |
89 ELFMAG1 == hdr.e_ident[EI_MAG1] && | |
90 ELFMAG2 == hdr.e_ident[EI_MAG2] && | |
91 ELFMAG3 == hdr.e_ident[EI_MAG3] && | |
92 ELFCLASSNONE != hdr.e_ident[EI_CLASS] && | |
93 ELFDATANONE != hdr.e_ident[EI_DATA]); | |
94 } | |
95 | |
96 bool ElfFile::load_tables() { | |
97 assert(m_file, "file not open"); | |
98 assert(m_status == Decoder::no_error, "already in error"); | |
99 | |
100 // read elf file header | |
101 if (fread(&m_elfHdr, sizeof(m_elfHdr), 1, m_file) != 1) { | |
102 m_status = Decoder::file_invalid; | |
103 return false; | |
104 } | |
105 | |
106 if (!is_elf_file(m_elfHdr)) { | |
107 m_status = Decoder::file_invalid; | |
108 return false; | |
109 } | |
110 | |
111 // walk elf file's section headers, and load string tables | |
112 Elf_Shdr shdr; | |
113 if (!fseek(m_file, m_elfHdr.e_shoff, SEEK_SET)) { | |
114 if (m_status != Decoder::no_error) return false; | |
115 | |
116 for (int index = 0; index < m_elfHdr.e_shnum; index ++) { | |
117 if (fread((void*)&shdr, sizeof(Elf_Shdr), 1, m_file) != 1) { | |
118 m_status = Decoder::file_invalid; | |
119 return false; | |
120 } | |
121 // string table | |
122 if (shdr.sh_type == SHT_STRTAB) { | |
123 ElfStringTable* table = new ElfStringTable(m_file, shdr, index); | |
124 if (table == NULL) { | |
125 m_status = Decoder::out_of_memory; | |
126 return false; | |
127 } | |
128 add_string_table(table); | |
129 } else if (shdr.sh_type == SHT_SYMTAB || shdr.sh_type == SHT_DYNSYM) { | |
130 ElfSymbolTable* table = new ElfSymbolTable(m_file, shdr); | |
131 if (table == NULL) { | |
132 m_status = Decoder::out_of_memory; | |
133 return false; | |
134 } | |
135 add_symbol_table(table); | |
136 } | |
137 } | |
138 } | |
139 return true; | |
140 } | |
141 | |
142 const char* ElfFile::decode(address addr, int* offset) { | |
143 // something already went wrong, just give up | |
144 if (m_status != Decoder::no_error) { | |
145 return NULL; | |
146 } | |
147 | |
148 ElfSymbolTable* symbol_table = m_symbol_tables; | |
149 int string_table_index; | |
150 int pos_in_string_table; | |
151 int off = INT_MAX; | |
152 bool found_symbol = false; | |
153 while (symbol_table != NULL) { | |
154 if (Decoder::no_error == symbol_table->lookup(addr, &string_table_index, &pos_in_string_table, &off)) { | |
155 found_symbol = true; | |
156 } | |
157 symbol_table = symbol_table->m_next; | |
158 } | |
159 if (!found_symbol) return NULL; | |
160 | |
161 ElfStringTable* string_table = get_string_table(string_table_index); | |
162 if (string_table == NULL) { | |
163 m_status = Decoder::file_invalid; | |
164 return NULL; | |
165 } | |
166 if (offset) *offset = off; | |
167 return string_table->string_at(pos_in_string_table); | |
168 } | |
169 | |
170 | |
171 void ElfFile::add_symbol_table(ElfSymbolTable* table) { | |
172 if (m_symbol_tables == NULL) { | |
173 m_symbol_tables = table; | |
174 } else { | |
175 table->m_next = m_symbol_tables; | |
176 m_symbol_tables = table; | |
177 } | |
178 } | |
179 | |
180 void ElfFile::add_string_table(ElfStringTable* table) { | |
181 if (m_string_tables == NULL) { | |
182 m_string_tables = table; | |
183 } else { | |
184 table->m_next = m_string_tables; | |
185 m_string_tables = table; | |
186 } | |
187 } | |
188 | |
189 ElfStringTable* ElfFile::get_string_table(int index) { | |
190 ElfStringTable* p = m_string_tables; | |
191 while (p != NULL) { | |
192 if (p->index() == index) return p; | |
193 p = p->m_next; | |
194 } | |
195 return NULL; | |
196 } | |
197 | |
198 #endif // _WINDOWS |