blob: 427ee8f80abd9346a72f52f48d2334c0347c2b3d [file] [log] [blame]
Eli Bendersky933f6992011-09-09 08:11:06 +03001#-------------------------------------------------------------------------------
2# readelf.py
3#
4# A clone of 'readelf' in Python, based on the pyelftools library
5#
6# Eli Bendersky (eliben@gmail.com)
7# This code is in the public domain
8#-------------------------------------------------------------------------------
9import sys
10from optparse import OptionParser
11
12# If elftools is not installed, maybe we're running from the root or scripts
13# dir of the source distribution
14#
15try:
16 import elftools
17except ImportError:
18 sys.path.extend(['.', '..'])
19
20from elftools.common.exceptions import ELFError
21from elftools.elf.elffile import ELFFile
22from elftools.elf.descriptions import (
23 describe_ei_class, describe_ei_data, describe_ei_version,
Eli Benderskyde8d71e2011-09-09 08:22:35 +030024 describe_ei_osabi, describe_e_type, describe_e_machine,
25 describe_e_version_numeric,
Eli Bendersky933f6992011-09-09 08:11:06 +030026 )
27
28
29class ReadElf(object):
30 """ display_* methods are used to emit output into the output stream
31 """
32 def __init__(self, file, output):
33 """ file:
34 stream object with the ELF file to read
35
36 output:
37 output stream to write to
38 """
39 self.elffile = ELFFile(file)
40 self.output = output
41
42 def display_file_header(self):
43 """ Display the ELF file header
44 """
45 self._emitline('ELF Header:')
46 self._emit(' Magic: ')
47 self._emitline(' '.join('%2.2x' % ord(b)
48 for b in self.elffile.e_ident_raw))
49 header = self.elffile.header
50 e_ident = header['e_ident']
51 self._emitline(' Class: %s' %
52 describe_ei_class(e_ident['EI_CLASS']))
53 self._emitline(' Data: %s' %
54 describe_ei_data(e_ident['EI_DATA']))
55 self._emitline(' Version: %s' %
56 describe_ei_version(e_ident['EI_VERSION']))
57 self._emitline(' OS/ABI: %s' %
58 describe_ei_osabi(e_ident['EI_OSABI']))
59 self._emitline(' ABI Version: %d' %
60 e_ident['EI_ABIVERSION'])
61 self._emitline(' Type: %s' %
62 describe_e_type(header['e_type']))
Eli Benderskyde8d71e2011-09-09 08:22:35 +030063 self._emitline(' Machine: %s' %
64 describe_e_machine(header['e_machine']))
65 self._emitline(' Version: %s' %
66 describe_e_version_numeric(header['e_version']))
Eli Benderskyd62928d2011-09-09 10:05:57 +030067 self._emitline(' Entry point address: %s' %
68 self._format_addr(header['e_entry']))
69 self._emit(' Start of program headers %s' %
70 header['e_phoff'])
71 self._emitline(' (bytes into file)')
72 self._emit(' Start of section headers %s' %
73 header['e_shoff'])
74 self._emitline(' (bytes into file)')
75 self._emitline(' Flags: %s' %
76 self._format_addr(header['e_flags']))
77 self._emitline(' Size of this header: %s (bytes)' %
78 header['e_ehsize'])
79 self._emitline(' Size of program headers: %s (bytes)' %
80 header['e_phentsize'])
81 self._emitline(' Number of program headers: %s' %
82 header['e_phnum'])
83 self._emitline(' Size of section headers: %s (bytes)' %
84 header['e_shentsize'])
85 self._emitline(' Number of section headers: %s' %
86 header['e_shnum'])
87 self._emitline(' Section header string table index: %s' %
88 header['e_shstrndx'])
Eli Bendersky933f6992011-09-09 08:11:06 +030089
Eli Bendersky6a12cde2011-09-09 10:23:16 +030090 def _format_addr(self, addr, fullhex=False, lead0x=True):
91 """ Format an address into a hexadecimal string.
Eli Benderskyd62928d2011-09-09 10:05:57 +030092
Eli Bendersky6a12cde2011-09-09 10:23:16 +030093 fullhex:
94 If True, include leading zeros in the address, adjusted for
95 the elfclass.
96
97 lead0x:
98 If True, leading 0x is added
Eli Benderskyd62928d2011-09-09 10:05:57 +030099 """
Eli Bendersky6a12cde2011-09-09 10:23:16 +0300100 s = '0x' if lead0x else ''
101 if fullhex:
102 if self.elffile.elfclass == 32:
103 return s + '%08x' % addr
104 else:
105 return s + '%016x' % addr
106 else:
107 return s + '%x' % addr
Eli Benderskyd62928d2011-09-09 10:05:57 +0300108
Eli Bendersky933f6992011-09-09 08:11:06 +0300109 def _emit(self, s):
110 """ Emit an object to output
111 """
112 self.output.write(str(s))
Eli Benderskyd62928d2011-09-09 10:05:57 +0300113
Eli Bendersky933f6992011-09-09 08:11:06 +0300114 def _emitline(self, s):
115 """ Emit an object to output, followed by a newline
116 """
117 self.output.write(str(s) + '\n')
118
119
120def main():
121 optparser = OptionParser()
122 options, args = optparser.parse_args()
123
124 with open(args[0], 'rb') as file:
125 try:
126 readelf = ReadElf(file, sys.stdout)
127 readelf.display_file_header()
128 except ELFError as ex:
129 sys.stderr.write('ELF read error: %s\n' % ex)
130 sys.exit(1)
131
132
133#-------------------------------------------------------------------------------
134if __name__ == '__main__':
135 main()
136