Fri, 10 Jun 2011 15:08:36 -0700
6941923: RFE: Handling large log files produced by long running Java Applications
Summary: supply optinal flags to realize gc log rotation
Reviewed-by: ysr, jwilhelm
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 */
25 #include "precompiled.hpp"
27 #ifndef _WINDOWS
29 #include <string.h>
30 #include <stdio.h>
31 #include <limits.h>
32 #include <new>
34 #include "memory/allocation.inline.hpp"
35 #include "utilities/decoder.hpp"
36 #include "utilities/elfFile.hpp"
37 #include "utilities/elfStringTable.hpp"
38 #include "utilities/elfSymbolTable.hpp"
41 ElfFile::ElfFile(const char* filepath) {
42 assert(filepath, "null file path");
43 memset(&m_elfHdr, 0, sizeof(m_elfHdr));
44 m_string_tables = NULL;
45 m_symbol_tables = NULL;
46 m_next = NULL;
47 m_status = Decoder::no_error;
49 int len = strlen(filepath) + 1;
50 m_filepath = (const char*)os::malloc(len * sizeof(char));
51 if (m_filepath != NULL) {
52 strcpy((char*)m_filepath, filepath);
53 m_file = fopen(filepath, "r");
54 if (m_file != NULL) {
55 load_tables();
56 } else {
57 m_status = Decoder::file_not_found;
58 }
59 } else {
60 m_status = Decoder::out_of_memory;
61 }
62 }
64 ElfFile::~ElfFile() {
65 if (m_string_tables != NULL) {
66 delete m_string_tables;
67 }
69 if (m_symbol_tables != NULL) {
70 delete m_symbol_tables;
71 }
73 if (m_file != NULL) {
74 fclose(m_file);
75 }
77 if (m_filepath != NULL) {
78 os::free((void*)m_filepath);
79 }
81 if (m_next != NULL) {
82 delete m_next;
83 }
84 };
87 //Check elf header to ensure the file is valid.
88 bool ElfFile::is_elf_file(Elf_Ehdr& hdr) {
89 return (ELFMAG0 == hdr.e_ident[EI_MAG0] &&
90 ELFMAG1 == hdr.e_ident[EI_MAG1] &&
91 ELFMAG2 == hdr.e_ident[EI_MAG2] &&
92 ELFMAG3 == hdr.e_ident[EI_MAG3] &&
93 ELFCLASSNONE != hdr.e_ident[EI_CLASS] &&
94 ELFDATANONE != hdr.e_ident[EI_DATA]);
95 }
97 bool ElfFile::load_tables() {
98 assert(m_file, "file not open");
99 assert(m_status == Decoder::no_error, "already in error");
101 // read elf file header
102 if (fread(&m_elfHdr, sizeof(m_elfHdr), 1, m_file) != 1) {
103 m_status = Decoder::file_invalid;
104 return false;
105 }
107 if (!is_elf_file(m_elfHdr)) {
108 m_status = Decoder::file_invalid;
109 return false;
110 }
112 // walk elf file's section headers, and load string tables
113 Elf_Shdr shdr;
114 if (!fseek(m_file, m_elfHdr.e_shoff, SEEK_SET)) {
115 if (m_status != Decoder::no_error) return false;
117 for (int index = 0; index < m_elfHdr.e_shnum; index ++) {
118 if (fread((void*)&shdr, sizeof(Elf_Shdr), 1, m_file) != 1) {
119 m_status = Decoder::file_invalid;
120 return false;
121 }
122 // string table
123 if (shdr.sh_type == SHT_STRTAB) {
124 ElfStringTable* table = new (std::nothrow) ElfStringTable(m_file, shdr, index);
125 if (table == NULL) {
126 m_status = Decoder::out_of_memory;
127 return false;
128 }
129 add_string_table(table);
130 } else if (shdr.sh_type == SHT_SYMTAB || shdr.sh_type == SHT_DYNSYM) {
131 ElfSymbolTable* table = new (std::nothrow) ElfSymbolTable(m_file, shdr);
132 if (table == NULL) {
133 m_status = Decoder::out_of_memory;
134 return false;
135 }
136 add_symbol_table(table);
137 }
138 }
139 }
140 return true;
141 }
143 const char* ElfFile::decode(address addr, int* offset) {
144 // something already went wrong, just give up
145 if (m_status != Decoder::no_error) {
146 return NULL;
147 }
149 ElfSymbolTable* symbol_table = m_symbol_tables;
150 int string_table_index;
151 int pos_in_string_table;
152 int off = INT_MAX;
153 bool found_symbol = false;
154 while (symbol_table != NULL) {
155 if (Decoder::no_error == symbol_table->lookup(addr, &string_table_index, &pos_in_string_table, &off)) {
156 found_symbol = true;
157 }
158 symbol_table = symbol_table->m_next;
159 }
160 if (!found_symbol) return NULL;
162 ElfStringTable* string_table = get_string_table(string_table_index);
163 if (string_table == NULL) {
164 m_status = Decoder::file_invalid;
165 return NULL;
166 }
167 if (offset) *offset = off;
168 return string_table->string_at(pos_in_string_table);
169 }
172 void ElfFile::add_symbol_table(ElfSymbolTable* table) {
173 if (m_symbol_tables == NULL) {
174 m_symbol_tables = table;
175 } else {
176 table->m_next = m_symbol_tables;
177 m_symbol_tables = table;
178 }
179 }
181 void ElfFile::add_string_table(ElfStringTable* table) {
182 if (m_string_tables == NULL) {
183 m_string_tables = table;
184 } else {
185 table->m_next = m_string_tables;
186 m_string_tables = table;
187 }
188 }
190 ElfStringTable* ElfFile::get_string_table(int index) {
191 ElfStringTable* p = m_string_tables;
192 while (p != NULL) {
193 if (p->index() == index) return p;
194 p = p->m_next;
195 }
196 return NULL;
197 }
199 #endif // _WINDOWS