annotate agent/src/os/linux/salibelf.c @ 3011:f00918f35c7f

inlining and runtime interface related changes: added codeSize() and compilerStorage() to RiMethod HotSpotMethodResolved uses reflective methods instead of vmIds and survives compilations HotSpotResolvedType.isInitialized not represented as field (can change) inlining stores graphs into method objects and reuses them
author Lukas Stadler <lukas.stadler@jku.at>
date Thu, 16 Jun 2011 20:36:17 +0200
parents c18cbe5936b8
children f2512d89ad0c
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 0
diff changeset
2 * Copyright (c) 2003, 2006, Oracle and/or its affiliates. All rights reserved.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 0
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 0
diff changeset
20 * or visit www.oracle.com if you need additional information or have any
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 0
diff changeset
21 * questions.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24
a61af66fc99e Initial load
duke
parents:
diff changeset
25 #include "salibelf.h"
a61af66fc99e Initial load
duke
parents:
diff changeset
26 #include <stdlib.h>
a61af66fc99e Initial load
duke
parents:
diff changeset
27 #include <unistd.h>
a61af66fc99e Initial load
duke
parents:
diff changeset
28
a61af66fc99e Initial load
duke
parents:
diff changeset
29 extern void print_debug(const char*,...);
a61af66fc99e Initial load
duke
parents:
diff changeset
30
a61af66fc99e Initial load
duke
parents:
diff changeset
31 // ELF file parsing helpers. Note that we do *not* use libelf here.
a61af66fc99e Initial load
duke
parents:
diff changeset
32 int read_elf_header(int fd, ELF_EHDR* ehdr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
33 if (pread(fd, ehdr, sizeof (ELF_EHDR), 0) != sizeof (ELF_EHDR) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
34 memcmp(&ehdr->e_ident[EI_MAG0], ELFMAG, SELFMAG) != 0 ||
a61af66fc99e Initial load
duke
parents:
diff changeset
35 ehdr->e_version != EV_CURRENT) {
a61af66fc99e Initial load
duke
parents:
diff changeset
36 return 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
37 }
a61af66fc99e Initial load
duke
parents:
diff changeset
38 return 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
39 }
a61af66fc99e Initial load
duke
parents:
diff changeset
40
a61af66fc99e Initial load
duke
parents:
diff changeset
41 bool is_elf_file(int fd) {
a61af66fc99e Initial load
duke
parents:
diff changeset
42 ELF_EHDR ehdr;
a61af66fc99e Initial load
duke
parents:
diff changeset
43 return read_elf_header(fd, &ehdr);
a61af66fc99e Initial load
duke
parents:
diff changeset
44 }
a61af66fc99e Initial load
duke
parents:
diff changeset
45
a61af66fc99e Initial load
duke
parents:
diff changeset
46 // read program header table of an ELF file
a61af66fc99e Initial load
duke
parents:
diff changeset
47 ELF_PHDR* read_program_header_table(int fd, ELF_EHDR* hdr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
48 ELF_PHDR* phbuf = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
49 // allocate memory for program header table
a61af66fc99e Initial load
duke
parents:
diff changeset
50 size_t nbytes = hdr->e_phnum * hdr->e_phentsize;
a61af66fc99e Initial load
duke
parents:
diff changeset
51
a61af66fc99e Initial load
duke
parents:
diff changeset
52 if ((phbuf = (ELF_PHDR*) malloc(nbytes)) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
53 print_debug("can't allocate memory for reading program header table\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
54 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
55 }
a61af66fc99e Initial load
duke
parents:
diff changeset
56
a61af66fc99e Initial load
duke
parents:
diff changeset
57 if (pread(fd, phbuf, nbytes, hdr->e_phoff) != nbytes) {
a61af66fc99e Initial load
duke
parents:
diff changeset
58 print_debug("ELF file is truncated! can't read program header table\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
59 free(phbuf);
a61af66fc99e Initial load
duke
parents:
diff changeset
60 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
61 }
a61af66fc99e Initial load
duke
parents:
diff changeset
62
a61af66fc99e Initial load
duke
parents:
diff changeset
63 return phbuf;
a61af66fc99e Initial load
duke
parents:
diff changeset
64 }
a61af66fc99e Initial load
duke
parents:
diff changeset
65
a61af66fc99e Initial load
duke
parents:
diff changeset
66 // read section header table of an ELF file
a61af66fc99e Initial load
duke
parents:
diff changeset
67 ELF_SHDR* read_section_header_table(int fd, ELF_EHDR* hdr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
68 ELF_SHDR* shbuf = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
69 // allocate memory for section header table
a61af66fc99e Initial load
duke
parents:
diff changeset
70 size_t nbytes = hdr->e_shnum * hdr->e_shentsize;
a61af66fc99e Initial load
duke
parents:
diff changeset
71
a61af66fc99e Initial load
duke
parents:
diff changeset
72 if ((shbuf = (ELF_SHDR*) malloc(nbytes)) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
73 print_debug("can't allocate memory for reading section header table\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
74 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
75 }
a61af66fc99e Initial load
duke
parents:
diff changeset
76
a61af66fc99e Initial load
duke
parents:
diff changeset
77 if (pread(fd, shbuf, nbytes, hdr->e_shoff) != nbytes) {
a61af66fc99e Initial load
duke
parents:
diff changeset
78 print_debug("ELF file is truncated! can't read section header table\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
79 free(shbuf);
a61af66fc99e Initial load
duke
parents:
diff changeset
80 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
81 }
a61af66fc99e Initial load
duke
parents:
diff changeset
82
a61af66fc99e Initial load
duke
parents:
diff changeset
83 return shbuf;
a61af66fc99e Initial load
duke
parents:
diff changeset
84 }
a61af66fc99e Initial load
duke
parents:
diff changeset
85
a61af66fc99e Initial load
duke
parents:
diff changeset
86 // read a particular section's data
a61af66fc99e Initial load
duke
parents:
diff changeset
87 void* read_section_data(int fd, ELF_EHDR* ehdr, ELF_SHDR* shdr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
88 void *buf = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
89 if (shdr->sh_type == SHT_NOBITS || shdr->sh_size == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
90 return buf;
a61af66fc99e Initial load
duke
parents:
diff changeset
91 }
a61af66fc99e Initial load
duke
parents:
diff changeset
92 if ((buf = calloc(shdr->sh_size, 1)) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
93 print_debug("can't allocate memory for reading section data\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
94 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
95 }
a61af66fc99e Initial load
duke
parents:
diff changeset
96 if (pread(fd, buf, shdr->sh_size, shdr->sh_offset) != shdr->sh_size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
97 free(buf);
a61af66fc99e Initial load
duke
parents:
diff changeset
98 print_debug("section data read failed\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
99 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
100 }
a61af66fc99e Initial load
duke
parents:
diff changeset
101 return buf;
a61af66fc99e Initial load
duke
parents:
diff changeset
102 }
a61af66fc99e Initial load
duke
parents:
diff changeset
103
a61af66fc99e Initial load
duke
parents:
diff changeset
104 uintptr_t find_base_address(int fd, ELF_EHDR* ehdr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
105 uintptr_t baseaddr = (uintptr_t)-1;
a61af66fc99e Initial load
duke
parents:
diff changeset
106 int cnt;
a61af66fc99e Initial load
duke
parents:
diff changeset
107 ELF_PHDR *phbuf, *phdr;
a61af66fc99e Initial load
duke
parents:
diff changeset
108
a61af66fc99e Initial load
duke
parents:
diff changeset
109 // read program header table
a61af66fc99e Initial load
duke
parents:
diff changeset
110 if ((phbuf = read_program_header_table(fd, ehdr)) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
111 goto quit;
a61af66fc99e Initial load
duke
parents:
diff changeset
112 }
a61af66fc99e Initial load
duke
parents:
diff changeset
113
a61af66fc99e Initial load
duke
parents:
diff changeset
114 // the base address of a shared object is the lowest vaddr of
a61af66fc99e Initial load
duke
parents:
diff changeset
115 // its loadable segments (PT_LOAD)
a61af66fc99e Initial load
duke
parents:
diff changeset
116 for (phdr = phbuf, cnt = 0; cnt < ehdr->e_phnum; cnt++, phdr++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
117 if (phdr->p_type == PT_LOAD && phdr->p_vaddr < baseaddr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
118 baseaddr = phdr->p_vaddr;
a61af66fc99e Initial load
duke
parents:
diff changeset
119 }
a61af66fc99e Initial load
duke
parents:
diff changeset
120 }
a61af66fc99e Initial load
duke
parents:
diff changeset
121
a61af66fc99e Initial load
duke
parents:
diff changeset
122 quit:
a61af66fc99e Initial load
duke
parents:
diff changeset
123 if (phbuf) free(phbuf);
a61af66fc99e Initial load
duke
parents:
diff changeset
124 return baseaddr;
a61af66fc99e Initial load
duke
parents:
diff changeset
125 }