blob: c7f0a8f0d3d0e0be6ca5456dc2aba5c409eb5e15 [file] [log] [blame]
Eli Bendersky3edefab2011-09-16 14:52:54 +03001#!/usr/bin/env python
Eli Bendersky933f6992011-09-09 08:11:06 +03002#-------------------------------------------------------------------------------
3# readelf.py
4#
5# A clone of 'readelf' in Python, based on the pyelftools library
6#
7# Eli Bendersky (eliben@gmail.com)
8# This code is in the public domain
9#-------------------------------------------------------------------------------
10import sys
11from optparse import OptionParser
12
13# If elftools is not installed, maybe we're running from the root or scripts
14# dir of the source distribution
15#
16try:
17 import elftools
18except ImportError:
19 sys.path.extend(['.', '..'])
20
21from elftools.common.exceptions import ELFError
22from elftools.elf.elffile import ELFFile
Eli Bendersky3f4de3e2011-09-14 05:58:06 +030023from elftools.elf.segments import InterpSegment
Eli Bendersky3edefab2011-09-16 14:52:54 +030024from elftools.elf.sections import SymbolTableSection
Eli Bendersky933f6992011-09-09 08:11:06 +030025from elftools.elf.descriptions import (
26 describe_ei_class, describe_ei_data, describe_ei_version,
Eli Benderskyde8d71e2011-09-09 08:22:35 +030027 describe_ei_osabi, describe_e_type, describe_e_machine,
Eli Bendersky26de2ac2011-09-13 06:50:28 +030028 describe_e_version_numeric, describe_p_type, describe_p_flags,
Eli Bendersky377bd862011-09-16 11:10:44 +030029 describe_sh_type, describe_sh_flags,
Eli Bendersky3edefab2011-09-16 14:52:54 +030030 describe_symbol_type, describe_symbol_bind, describe_symbol_visibility,
31 describe_symbol_shndx,
Eli Bendersky933f6992011-09-09 08:11:06 +030032 )
33
34
35class ReadElf(object):
36 """ display_* methods are used to emit output into the output stream
37 """
38 def __init__(self, file, output):
39 """ file:
40 stream object with the ELF file to read
41
42 output:
43 output stream to write to
44 """
45 self.elffile = ELFFile(file)
46 self.output = output
47
48 def display_file_header(self):
49 """ Display the ELF file header
50 """
51 self._emitline('ELF Header:')
52 self._emit(' Magic: ')
53 self._emitline(' '.join('%2.2x' % ord(b)
54 for b in self.elffile.e_ident_raw))
55 header = self.elffile.header
56 e_ident = header['e_ident']
57 self._emitline(' Class: %s' %
58 describe_ei_class(e_ident['EI_CLASS']))
59 self._emitline(' Data: %s' %
60 describe_ei_data(e_ident['EI_DATA']))
61 self._emitline(' Version: %s' %
62 describe_ei_version(e_ident['EI_VERSION']))
63 self._emitline(' OS/ABI: %s' %
64 describe_ei_osabi(e_ident['EI_OSABI']))
65 self._emitline(' ABI Version: %d' %
66 e_ident['EI_ABIVERSION'])
67 self._emitline(' Type: %s' %
68 describe_e_type(header['e_type']))
Eli Benderskyde8d71e2011-09-09 08:22:35 +030069 self._emitline(' Machine: %s' %
70 describe_e_machine(header['e_machine']))
71 self._emitline(' Version: %s' %
72 describe_e_version_numeric(header['e_version']))
Eli Benderskyd62928d2011-09-09 10:05:57 +030073 self._emitline(' Entry point address: %s' %
Eli Bendersky26de2ac2011-09-13 06:50:28 +030074 self._format_hex(header['e_entry']))
Eli Benderskyd62928d2011-09-09 10:05:57 +030075 self._emit(' Start of program headers %s' %
76 header['e_phoff'])
77 self._emitline(' (bytes into file)')
78 self._emit(' Start of section headers %s' %
79 header['e_shoff'])
80 self._emitline(' (bytes into file)')
81 self._emitline(' Flags: %s' %
Eli Bendersky26de2ac2011-09-13 06:50:28 +030082 self._format_hex(header['e_flags']))
Eli Benderskyd62928d2011-09-09 10:05:57 +030083 self._emitline(' Size of this header: %s (bytes)' %
84 header['e_ehsize'])
85 self._emitline(' Size of program headers: %s (bytes)' %
86 header['e_phentsize'])
87 self._emitline(' Number of program headers: %s' %
88 header['e_phnum'])
89 self._emitline(' Size of section headers: %s (bytes)' %
90 header['e_shentsize'])
91 self._emitline(' Number of section headers: %s' %
92 header['e_shnum'])
93 self._emitline(' Section header string table index: %s' %
94 header['e_shstrndx'])
Eli Bendersky933f6992011-09-09 08:11:06 +030095
Eli Bendersky26de2ac2011-09-13 06:50:28 +030096 def display_program_headers(self):
97 """ Display the ELF program headers
98 """
99 self._emitline()
100 elfheader = self.elffile.header
101 self._emitline('Elf file type is %s' %
Eli Bendersky377bd862011-09-16 11:10:44 +0300102 describe_e_type(elfheader['e_type']))
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300103 self._emitline('Entry point is %s' %
Eli Bendersky377bd862011-09-16 11:10:44 +0300104 self._format_hex(elfheader['e_entry']))
105 # readelf weirness - why isn't e_phoff printed as hex? (for section
106 # headers, it is...)
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300107 self._emitline('There are %s program headers, starting at offset %s' % (
Eli Bendersky377bd862011-09-16 11:10:44 +0300108 elfheader['e_phnum'], elfheader['e_phoff']))
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300109
110 self._emitline('\nProgram headers:')
111
112 # Now comes the table of program headers with their attributes. Note
113 # that due to different formatting constraints of 32-bit and 64-bit
114 # addresses, there are some conditions on elfclass here.
115 #
116 # First comes the table heading
117 #
118 if self.elffile.elfclass == 32:
119 self._emitline(' Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align')
120 else:
121 self._emitline(' Type Offset VirtAddr PhysAddr')
122 self._emitline(' FileSiz MemSiz Flags Align')
123
124 # Now the entries
125 #
126 for segment in self.elffile.iter_segments():
127 self._emit(' %-14s ' % describe_p_type(segment['p_type']))
128
129 if self.elffile.elfclass == 32:
130 self._emitline('%s %s %s %s %s %-3s %s' % (
131 self._format_hex(segment['p_offset'], fieldsize=6),
132 self._format_hex(segment['p_vaddr'], fullhex=True),
133 self._format_hex(segment['p_paddr'], fullhex=True),
134 self._format_hex(segment['p_filesz'], fieldsize=5),
135 self._format_hex(segment['p_memsz'], fieldsize=5),
136 describe_p_flags(segment['p_flags']),
137 self._format_hex(segment['p_align'])))
Eli Benderskya41c3c02011-09-14 06:18:28 +0300138 else: # 64
139 self._emitline('%s %s %s' % (
140 self._format_hex(segment['p_offset'], fullhex=True),
141 self._format_hex(segment['p_vaddr'], fullhex=True),
142 self._format_hex(segment['p_paddr'], fullhex=True)))
143 self._emitline(' %s %s %-3s %s' % (
144 self._format_hex(segment['p_filesz'], fullhex=True),
145 self._format_hex(segment['p_memsz'], fullhex=True),
146 describe_p_flags(segment['p_flags']),
147 # lead0x set to False for p_align, to mimic readelf.
148 # No idea why the difference from 32-bit mode :-|
149 self._format_hex(segment['p_align'], lead0x=False)))
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300150
Eli Bendersky3f4de3e2011-09-14 05:58:06 +0300151 if isinstance(segment, InterpSegment):
152 self._emitline(' [Requesting program interpreter: %s]' %
153 segment.get_interp_name())
154
Eli Bendersky58585b02011-09-15 07:07:54 +0300155 # Sections to segments mapping
156 #
157 if self.elffile.num_sections() == 0:
158 # No sections? We're done
159 return
160
161 self._emitline('\n Section to Segment mapping:')
162 self._emitline(' Segment Sections...\n')
163
164 for nseg, segment in enumerate(self.elffile.iter_segments()):
165 self._emit(' %2.2d ' % nseg)
166
167 for section in self.elffile.iter_sections():
168 if ( not section.is_null() and
169 segment.section_in_segment(section)):
170 self._emit('%s ' % section.name)
171
172 self._emitline('')
173
Eli Bendersky377bd862011-09-16 11:10:44 +0300174 def display_section_headers(self):
175 """ Display the ELF section headers
176 """
177 elfheader = self.elffile.header
178 self._emitline('There are %s section headers, starting at offset %s' % (
179 elfheader['e_shnum'], self._format_hex(elfheader['e_shoff'])))
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300180
Eli Bendersky377bd862011-09-16 11:10:44 +0300181 self._emitline('\nSection header%s:' % (
182 's' if elfheader['e_shnum'] > 1 else ''))
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300183
Eli Bendersky377bd862011-09-16 11:10:44 +0300184 # Different formatting constraints of 32-bit and 64-bit addresses
185 #
186 if self.elffile.elfclass == 32:
187 self._emitline(' [Nr] Name Type Addr Off Size ES Flg Lk Inf Al')
188 else:
189 self._emitline(' [Nr] Name Type Address Offset')
190 self._emitline(' Size EntSize Flags Link Info Align')
191
192 # Now the entries
193 #
194 for nsec, section in enumerate(self.elffile.iter_sections()):
195 self._emit(' [%2u] %-17.17s %-15.15s ' % (
196 nsec, section.name, describe_sh_type(section['sh_type'])))
197
198 if self.elffile.elfclass == 32:
199 self._emitline('%s %s %s %s %3s %2s %3s %2s' % (
200 self._format_hex(section['sh_addr'], fieldsize=8, lead0x=False),
201 self._format_hex(section['sh_offset'], fieldsize=6, lead0x=False),
202 self._format_hex(section['sh_size'], fieldsize=6, lead0x=False),
203 self._format_hex(section['sh_entsize'], fieldsize=2, lead0x=False),
204 describe_sh_flags(section['sh_flags']),
205 section['sh_link'], section['sh_info'],
206 section['sh_addralign']))
207 else: # 64
208 self._emitline(' %s %s' % (
209 self._format_hex(section['sh_addr'], fullhex=True, lead0x=False),
210 self._format_hex(section['sh_offset'],
211 fieldsize=16 if section['sh_offset'] > 0xffffffff else 8,
212 lead0x=False)))
213 self._emitline(' %s %s %3s %2s %3s %s' % (
214 self._format_hex(section['sh_size'], fullhex=True, lead0x=False),
215 self._format_hex(section['sh_entsize'], fullhex=True, lead0x=False),
216 describe_sh_flags(section['sh_flags']),
217 section['sh_link'], section['sh_info'],
218 section['sh_addralign']))
219
220 self._emitline('Key to Flags:')
221 self._emitline(' W (write), A (alloc), X (execute), M (merge), S (strings)')
222 self._emitline(' I (info), L (link order), G (group), x (unknown)')
223 self._emitline(' O (extra OS processing required) o (OS specific), p (processor specific)')
224
Eli Bendersky3edefab2011-09-16 14:52:54 +0300225 def display_symbol_tables(self):
226 """ Display the symbol tables contained in the file
227 """
228 for section in self.elffile.iter_sections():
229 if not isinstance(section, SymbolTableSection):
230 continue
231
232 if section['sh_entsize'] == 0:
233 self._emitline("\nSymbol table '%s' has a sh_entsize of zero!" % (
234 section.name))
235 continue
236
237 self._emitline("\nSymbol table '%s' contains %s entries:" % (
238 section.name, section.num_symbols()))
239
240 if self.elffile.elfclass == 32:
241 self._emitline(' Num: Value Size Type Bind Vis Ndx Name')
242 else: # 64
243 self._emitline(' Num: Value Size Type Bind Vis Ndx Name')
244
245 for nsym, symbol in enumerate(section.iter_symbols()):
Eli Benderskyb6fa3652011-09-16 15:20:20 +0300246 # symbol names are truncated to 25 chars, similarly to readelf
247 self._emitline('%6d: %s %5d %-7s %-6s %-7s %4s %.25s' % (
Eli Bendersky3edefab2011-09-16 14:52:54 +0300248 nsym,
249 self._format_hex(symbol['st_value'], fullhex=True, lead0x=False),
250 symbol['st_size'],
251 describe_symbol_type(symbol['st_info']['type']),
252 describe_symbol_bind(symbol['st_info']['bind']),
253 describe_symbol_visibility(symbol['st_other']['visibility']),
254 describe_symbol_shndx(symbol['st_shndx']),
255 symbol.name))
256
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300257 def _format_hex(self, addr, fieldsize=None, fullhex=False, lead0x=True):
Eli Bendersky6a12cde2011-09-09 10:23:16 +0300258 """ Format an address into a hexadecimal string.
Eli Benderskyd62928d2011-09-09 10:05:57 +0300259
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300260 fieldsize:
261 Size of the hexadecimal field (with leading zeros to fit the
262 address into. For example with fieldsize=8, the format will
263 be %08x
264 If None, the minimal required field size will be used.
265
Eli Bendersky6a12cde2011-09-09 10:23:16 +0300266 fullhex:
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300267 If True, override fieldsize to set it to the maximal size
268 needed for the elfclass
Eli Bendersky6a12cde2011-09-09 10:23:16 +0300269
270 lead0x:
271 If True, leading 0x is added
Eli Benderskyd62928d2011-09-09 10:05:57 +0300272 """
Eli Bendersky6a12cde2011-09-09 10:23:16 +0300273 s = '0x' if lead0x else ''
274 if fullhex:
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300275 fieldsize = 8 if self.elffile.elfclass == 32 else 16
276 if fieldsize is None:
277 field = '%x'
Eli Bendersky6a12cde2011-09-09 10:23:16 +0300278 else:
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300279 field = '%' + '0%sx' % fieldsize
280 return s + field % addr
Eli Benderskyd62928d2011-09-09 10:05:57 +0300281
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300282 def _emit(self, s=''):
Eli Bendersky933f6992011-09-09 08:11:06 +0300283 """ Emit an object to output
284 """
285 self.output.write(str(s))
Eli Benderskyd62928d2011-09-09 10:05:57 +0300286
Eli Bendersky26de2ac2011-09-13 06:50:28 +0300287 def _emitline(self, s=''):
Eli Bendersky933f6992011-09-09 08:11:06 +0300288 """ Emit an object to output, followed by a newline
289 """
290 self.output.write(str(s) + '\n')
291
292
293def main():
294 optparser = OptionParser()
295 options, args = optparser.parse_args()
296
297 with open(args[0], 'rb') as file:
298 try:
299 readelf = ReadElf(file, sys.stdout)
Eli Bendersky3edefab2011-09-16 14:52:54 +0300300 #readelf.display_file_header()
Eli Bendersky377bd862011-09-16 11:10:44 +0300301 #readelf.display_program_headers()
Eli Bendersky3edefab2011-09-16 14:52:54 +0300302 #readelf.display_section_headers()
303 readelf.display_symbol_tables()
Eli Bendersky933f6992011-09-09 08:11:06 +0300304 except ELFError as ex:
305 sys.stderr.write('ELF read error: %s\n' % ex)
306 sys.exit(1)
307
308
309#-------------------------------------------------------------------------------
310if __name__ == '__main__':
311 main()
312