LIEF/examples/python/elf_reader.py

553 lines
19 KiB
Python
Raw Normal View History

2017-03-30 16:56:49 +02:00
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Description
# -----------
# Print information about an ELF binary
import lief
from lief import ELF
import sys
import os
2017-07-07 16:25:14 +02:00
import traceback
2018-03-07 09:37:42 +01:00
import textwrap
2017-03-30 16:56:49 +02:00
from lief import Logger
Logger.set_level(lief.LOGGING_LEVEL.INFO)
2017-03-30 16:56:49 +02:00
from optparse import OptionParser
terminal_rows, terminal_columns = 100, 100
try:
terminal_rows, terminal_columns = os.popen('stty size', 'r').read().split()
except ValueError:
pass
terminal_columns = int(terminal_columns)
terminal_rows = int(terminal_rows)
2017-07-07 16:25:14 +02:00
class exceptions_handler(object):
func = None
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
def __init__(self, exceptions, on_except_callback=None):
self.exceptions = exceptions
self.on_except_callback = on_except_callback
def __call__(self, *args, **kwargs):
if self.func is None:
self.func = args[0]
return self
try:
return self.func(*args, **kwargs)
except self.exceptions as e:
if self.on_except_callback is not None:
self.on_except_callback(e)
else:
print("-" * 60)
print("Exception in {}: {}".format(self.func.__name__, e))
exc_type, exc_value, exc_traceback = sys.exc_info()
traceback.print_tb(exc_traceback)
print("-" * 60)
@exceptions_handler(Exception)
2017-03-30 16:56:49 +02:00
def print_header(binary):
header = binary.header
identity = header.identity
print("== Header ==\n")
format_str = "{:<30} {:<30}"
format_hex = "{:<30} 0x{:<13x}"
2017-03-30 16:56:49 +02:00
format_dec = "{:<30} {:<30d}"
format_ide = "{:<30} {:<02x} {:<02x} {:<02x} {:<02x}"
eflags_str = ""
if header.machine_type == lief.ELF.ARCH.ARM:
eflags_str = " - ".join([str(s).split(".")[-1] for s in header.arm_flags_list])
if header.machine_type in [lief.ELF.ARCH.MIPS, lief.ELF.ARCH.MIPS_RS3_LE, lief.ELF.ARCH.MIPS_X]:
eflags_str = " - ".join([str(s).split(".")[-1] for s in header.mips_flags_list])
if header.machine_type == lief.ELF.ARCH.PPC64:
eflags_str = " - ".join([str(s).split(".")[-1] for s in header.ppc64_flags_list])
if header.machine_type == lief.ELF.ARCH.HEXAGON:
eflags_str = " - ".join([str(s).split(".")[-1] for s in header.hexagon_flags_list])
2017-03-30 16:56:49 +02:00
print(format_ide.format("Magic:", identity[0], identity[1], identity[2], identity[3]))
print(format_str.format("Class:", str(header.identity_class).split(".")[-1]))
print(format_str.format("Endianness:", str(header.identity_data).split(".")[-1]))
print(format_str.format("Version:", str(header.identity_version).split(".")[-1]))
print(format_str.format("OS/ABI:", str(header.identity_os_abi).split(".")[-1]))
print(format_str.format("File Type:", str(header.file_type).split(".")[-1]))
print(format_str.format("Machine Type:", str(header.machine_type).split(".")[-1]))
print(format_str.format("Object File Version:", str(header.object_file_version).split(".")[-1]))
print(format_hex.format("Entry Point:", header.entrypoint))
print(format_hex.format("Program Header Offset:", header.program_header_offset))
print(format_hex.format("Section Header Offset:", header.section_header_offset))
print(format_hex.format("Processor flags:", header.processor_flag) + eflags_str)
2017-03-30 16:56:49 +02:00
print(format_dec.format("Header Size:", header.header_size))
print(format_dec.format("Program Header Size:", header.program_header_size))
print(format_dec.format("Section Header Size:", header.section_header_size))
2017-03-30 16:56:49 +02:00
print(format_dec.format("Number of segments:", header.numberof_segments))
print(format_dec.format("Number of sections:", header.numberof_sections))
print("")
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-03-30 16:56:49 +02:00
def print_sections(binary):
sections = binary.sections
if len(sections) > 0:
print("== Sections ==\n")
f_title = "|{:<30} | {:<12}| {:<17}| {:<12}| {:<10}| {:<8}| {:<8}|"
f_value = "|{:<30} | {:<12}| 0x{:<14x} | 0x{:<10x}| 0x{:<8x}| {:<8.2f}| {:<10}"
print(f_title.format("Name", "Type", "Virtual address", "File offset", "Size", "Entropy", "Segment(s)"))
for section in sections:
segments_str = " - ".join([str(s.type).split(".")[-1] for s in section.segments])
print(f_value.format(
section.name,
str(section.type).split(".")[-1],
section.virtual_address,
section.file_offset,
section.size,
abs(section.entropy),
segments_str))
print("")
else:
print("No sections")
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-03-30 16:56:49 +02:00
def print_segments(binary):
segments = binary.segments
# Segments
if len(segments) > 0:
print("== Segments ==\n")
f_title = "|{:<30} | {:<10}| {:<18}| {:<17}| {:<17}| {:<17}| {:<19}|"
f_value = "|{:<30} | {:<10}| 0x{:<16x}| 0x{:<15x}| 0x{:<15x}| 0x{:<15x}| {}"
print(f_title.format("Type",
"Flags", "File offset", "Virtual Address", "Virtual Size", "Size", "Sections"))
2017-03-30 16:56:49 +02:00
for segment in segments:
sections = segment.sections
s = ", ".join([section.name for section in sections])
flags_str = ["-"] * 3
2017-08-02 07:53:23 +02:00
if ELF.SEGMENT_FLAGS.R in segment:
flags_str[0] = "r"
2017-08-02 07:53:23 +02:00
if ELF.SEGMENT_FLAGS.W in segment:
flags_str[1] = "w"
2017-08-02 07:53:23 +02:00
if ELF.SEGMENT_FLAGS.X in segment:
flags_str[2] = "x"
flags_str = "".join(flags_str)
2017-03-30 16:56:49 +02:00
print(f_value.format(
str(segment.type).split(".")[-1],
flags_str,
2017-03-30 16:56:49 +02:00
segment.file_offset,
segment.virtual_address,
segment.virtual_size,
segment.physical_size, s))
print("")
else:
print("No segments")
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-03-30 16:56:49 +02:00
def print_dynamic_entries(binary):
dynamic_entries = binary.dynamic_entries
2017-03-30 16:56:49 +02:00
# Dynamic entries
if len(dynamic_entries) == 0:
return
print("== Dynamic entries ==\n")
f_title = "|{:<16} | {:<10}| {:<20}|"
f_value = "|{:<16} | 0x{:<8x}| {:<20}|"
print(f_title.format("Tag", "Value", "Info"))
for entry in dynamic_entries:
if entry.tag == ELF.DYNAMIC_TAGS.NULL:
continue
if entry.tag in [ELF.DYNAMIC_TAGS.SONAME, ELF.DYNAMIC_TAGS.NEEDED, ELF.DYNAMIC_TAGS.RUNPATH, ELF.DYNAMIC_TAGS.RPATH]:
print(f_value.format(str(entry.tag).split(".")[-1], entry.value, entry.name))
elif type(entry) is ELF.DynamicEntryArray: # [ELF.DYNAMIC_TAGS.INIT_ARRAY,ELF.DYNAMIC_TAGS.FINI_ARRAY]:
print(f_value.format(str(entry.tag).split(".")[-1], entry.value, ", ".join(map(hex, entry.array))))
elif entry.tag == ELF.DYNAMIC_TAGS.FLAGS:
flags_str = " - ".join([str(ELF.DYNAMIC_FLAGS(s)).split(".")[-1] for s in entry.flags])
print(f_value.format(str(entry.tag).split(".")[-1], entry.value, flags_str))
elif entry.tag == ELF.DYNAMIC_TAGS.FLAGS_1:
flags_str = " - ".join([str(ELF.DYNAMIC_FLAGS_1(s)).split(".")[-1] for s in entry.flags])
print(f_value.format(str(entry.tag).split(".")[-1], entry.value, flags_str))
else:
print(f_value.format(str(entry.tag).split(".")[-1], entry.value, ""))
2017-03-30 16:56:49 +02:00
print("")
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2018-03-07 09:37:42 +01:00
def print_symbols(symbols, no_trunc):
try:
maxsize = max([len(symbol.demangled_name) for symbol in symbols])
except:
maxsize = max([len(symbol.name) for symbol in symbols])
2017-03-30 16:56:49 +02:00
2018-03-07 09:37:42 +01:00
SIZE = 70
maxsize = min(maxsize, terminal_columns - SIZE) if terminal_columns > SIZE else terminal_columns
print(maxsize, terminal_columns)
2017-03-30 16:56:49 +02:00
2018-03-07 09:37:42 +01:00
f_title = "|{:<" + str(maxsize) + "} | {:<7}| {:<8}| {:<10}| {:<8}| {:<4}| {:<14}|"
f_value = "|{:<" + str(maxsize) + "} | {:<7}| {:<8x}| {:<10}| {:<8}| {:<4}| {:<14}|"
print(f_title.format("Name", "Type", "Value", "Visibility", "Binding", "I/E", "Version"))
for symbol in symbols:
symbol_version = symbol.symbol_version if symbol.has_version else ""
import_export = ""
if symbol.imported:
import_export = "I"
2017-03-30 16:56:49 +02:00
if symbol.exported:
import_export = "E"
2017-03-30 16:56:49 +02:00
try:
symbol_name = symbol.demangled_name
2017-03-30 16:56:49 +02:00
except:
symbol_name = symbol.name
2018-03-07 09:37:42 +01:00
wrapped = textwrap.wrap(symbol_name, maxsize)
if len(wrapped) <= 1 or no_trunc:
symbol_name = symbol_name
else:
symbol_name = wrapped[0][:-3] + "..."
print(f_value.format(
symbol_name,
str(symbol.type).split(".")[-1],
symbol.value,
2018-03-07 09:37:42 +01:00
str(symbol.visibility).split(".")[-1],
str(symbol.binding).split(".")[-1],
import_export,
str(symbol_version)
))
2017-03-30 16:56:49 +02:00
@exceptions_handler(Exception)
2018-03-07 09:37:42 +01:00
def print_dynamic_symbols(binary, args):
print("== Dynamic symbols ==\n")
2018-03-07 09:37:42 +01:00
print_symbols(binary.dynamic_symbols, args.no_trunc)
2017-03-30 16:56:49 +02:00
@exceptions_handler(Exception)
2018-03-07 09:37:42 +01:00
def print_static_symbols(binary, args):
print("== Static symbols ==\n")
2018-03-07 09:37:42 +01:00
print_symbols(binary.static_symbols, args.no_trunc)
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
def print_relocations(binary, relocations):
2017-08-28 07:18:52 +02:00
f_title = "|{:<10} | {:<10}| {:<8}| {:<8}| {:<15}| {:<30} |"
f_value = "|0x{:<8x} | {:<10}| {:<8d}| {:<8x}| {:<15}| {:<30} |"
2017-03-30 16:56:49 +02:00
2017-08-28 07:18:52 +02:00
print(f_title.format("Address", "Type", "Size", "Addend", "Purpose", "Symbol"))
2017-03-30 16:56:49 +02:00
for relocation in relocations:
type = str(relocation.type)
if binary.header.machine_type == ELF.ARCH.x86_64:
type = str(ELF.RELOCATION_X86_64(relocation.type))
elif binary.header.machine_type == ELF.ARCH.i386:
type = str(ELF.RELOCATION_i386(relocation.type))
elif binary.header.machine_type == ELF.ARCH.ARM:
type = str(ELF.RELOCATION_ARM(relocation.type))
elif binary.header.machine_type == ELF.ARCH.AARCH64:
type = str(ELF.RELOCATION_AARCH64(relocation.type))
2017-03-30 16:56:49 +02:00
symbol_name = str(relocation.symbol.name) if relocation.has_symbol else ""
2017-06-28 17:45:12 +02:00
print(f_value.format(
relocation.address,
type.split(".")[-1],
relocation.size,
2017-08-28 07:18:52 +02:00
relocation.addend,
str(relocation.purpose).split(".")[-1],
symbol_name))
2017-06-28 17:45:12 +02:00
@exceptions_handler(Exception)
def print_all_relocations(binary):
dynamicrelocations = binary.dynamic_relocations
pltgot_relocations = binary.pltgot_relocations
object_relocations = binary.object_relocations
if len(dynamicrelocations) > 0:
print("== Dynamic Relocations ==\n")
print_relocations(binary, dynamicrelocations)
2017-03-30 16:56:49 +02:00
if len(pltgot_relocations) > 0:
print("== PLT/GOT Relocations ==\n")
print_relocations(binary, pltgot_relocations)
if len(object_relocations) > 0:
print("== Object Relocations ==\n")
print_relocations(binary, object_relocations)
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2018-03-07 09:37:42 +01:00
def print_exported_symbols(binary, args):
2017-03-30 16:56:49 +02:00
symbols = binary.exported_symbols
2018-03-07 09:37:42 +01:00
print("== Exported symbols ==\n")
if len(symbols) == 0:
print("No exports!")
return
print_symbols(symbols, args.no_trunc)
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2018-03-07 09:37:42 +01:00
def print_imported_symbols(binary, args):
2017-03-30 16:56:49 +02:00
symbols = binary.imported_symbols
print("== Imported symbols ==\n")
2018-03-07 09:37:42 +01:00
if len(symbols) == 0:
print("No imports!")
return
print_symbols(symbols, args.no_trunc)
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-06-23 17:35:48 +02:00
def print_information(binary):
print("== Information ==\n")
2017-03-30 16:56:49 +02:00
format_str = "{:<30} {:<30}"
format_hex = "{:<30} 0x{:<28x}"
format_dec = "{:<30} {:<30d}"
print(format_str.format("Name:", binary.name))
print(format_hex.format("Address base:", binary.imagebase))
print(format_hex.format("Virtual size:", binary.virtual_size))
2017-08-01 17:57:38 +02:00
print(format_str.format("PIE:", str(binary.is_pie)))
print(format_str.format("NX:", str(binary.has_nx)))
2017-03-30 16:56:49 +02:00
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-03-30 16:56:49 +02:00
def print_gnu_hash(binary):
print("== GNU Hash ==\n")
2017-06-10 08:28:22 +02:00
if not binary.use_gnu_hash:
return
2017-03-30 16:56:49 +02:00
gnu_hash = binary.gnu_hash
format_str = "{:<30} {}"
format_hex = "{:<30} 0x{:<28x}"
format_dec = "{:<30} {:<30d}"
print(format_dec.format("Number of buckets:", gnu_hash.nb_buckets))
print(format_dec.format("First symbol index:", gnu_hash.symbol_index))
print(format_hex.format("Shift Count:", gnu_hash.shift2))
print(format_str.format("Bloom filters:", gnu_hash.bloom_filters))
print(format_str.format("Buckets:", gnu_hash.buckets))
print(format_str.format("Hash values:", gnu_hash.hash_values))
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-06-10 08:28:22 +02:00
def print_sysv_hash(binary):
print("== SYSV Hash ==\n")
if not binary.use_sysv_hash:
return
sysv_hash = binary.sysv_hash
format_str = "{:<30} {}"
format_hex = "{:<30} 0x{:<28x}"
format_dec = "{:<30} {:<30d}"
print(format_dec.format("Number of buckets:", sysv_hash.nbucket))
print(format_dec.format("Number of chains:", sysv_hash.nchain))
print(format_str.format("Buckets:", sysv_hash.buckets))
print(format_str.format("Chains:", sysv_hash.chains))
2017-07-07 16:25:14 +02:00
@exceptions_handler(Exception)
2017-06-11 09:13:52 +02:00
def print_notes(binary):
print("== Notes ==\n")
format_str = "{:<19} {}"
format_hex = "{:<19} 0x{:<28x}"
format_dec = "{:<19} {:<30d}"
notes = binary.notes
for idx, note in enumerate(notes):
description = note.description
description_str = " ".join(map(lambda e : "{:02x}".format(e), description[:16]))
if len(description) > 16:
description_str += " ..."
print("Note #{:d}".format(idx))
print(format_str.format("Name:", note.name))
2017-06-28 17:45:12 +02:00
print(format_str.format("Type:", str(ELF.NOTE_TYPES(note.type)).split(".")[-1]))
2017-06-11 09:13:52 +02:00
print(format_str.format("Description:", description_str))
if ELF.NOTE_TYPES(note.type) == ELF.NOTE_TYPES.ABI_TAG:
if type(note) == lief.ELF.AndroidNote:
print(format_dec.format("SDK Version:", note.sdk_version))
print(format_str.format("NDK Version:", note.ndk_version))
print(format_str.format("NDK build number:", note.ndk_build_number))
else:
version = note.version
2017-06-28 17:45:12 +02:00
version_str = "{:d}.{:d}.{:d}".format(version[0], version[1], version[2])
print(format_str.format("ABI:", note.abi))
print(format_str.format("Version:", version_str))
if ELF.NOTE_TYPES(note.type) == ELF.NOTE_TYPES.GOLD_VERSION:
print(format_str.format("Version:", "".join(map(chr, note.description))))
2017-06-11 09:13:52 +02:00
print("\n")
2017-03-30 16:56:49 +02:00
def main():
optparser = OptionParser(
usage='Usage: %prog [options] <elf-file>',
add_help_option=False, # -h is a real option of readelf
prog=sys.argv[0])
optparser.add_option('-a', '--all',
action='store_true', dest='show_all',
help='Equivalent to: -h -l -S -s -r -d -V')
optparser.add_option('-d', '--dynamic',
action='store_true', dest='show_dynamic_tags',
help='Display the dynamic section')
optparser.add_option('-H', '--help',
action='store_true', dest='help',
help='Display this information')
optparser.add_option('-h', '--file-header',
action='store_true', dest='show_file_header',
help='Display the ELF file header')
optparser.add_option('-i', '--imported',
action='store_true', dest='show_imported_symbols',
help='Display imported symbols')
optparser.add_option('-l', '--program-headers', '--segments',
action='store_true', dest='show_program_header',
help='Display the program headers')
optparser.add_option('-S', '--section-headers', '--sections',
action='store_true', dest='show_section_header',
help="Display the sections' headers")
optparser.add_option('-e', '--headers',
action='store_true', dest='show_all_headers',
help='Equivalent to: -h -l -S')
optparser.add_option('-s', '--symbols', '--syms',
action='store_true', dest='show_symbols',
help='Display the symbol table')
optparser.add_option('--dynamic-symbols', '--dsyms',
action='store_true', dest='show_dynamic_symbols',
help='Display the dynamic symbols')
optparser.add_option('--static-symbols', '--ssyms',
action='store_true', dest='show_static_symbols',
help='Display the static symbols')
2017-03-30 16:56:49 +02:00
optparser.add_option('-r', '--relocs',
action='store_true', dest='show_relocs',
help='Display the relocations (if present)')
optparser.add_option('-V', '--version-info',
action='store_true', dest='show_version_info',
help='Display the version sections (if present)')
optparser.add_option('-x', '--exported',
action='store_true', dest='show_exported_symbols',
help='Display exported symbols')
optparser.add_option('--gnu-hash',
action='store_true', dest='show_gnu_hash',
help='Display GNU Hash')
2017-06-10 08:28:22 +02:00
optparser.add_option('--sysv-hash',
action='store_true', dest='show_sysv_hash',
help='Display SYSV Hash')
2017-06-11 09:13:52 +02:00
optparser.add_option('-n', '--notes',
action='store_true', dest='show_notes',
help='Display Notes')
2018-03-07 09:37:42 +01:00
optparser.add_option('--no-trunc',
action='store_true', dest='no_trunc',
default=False,
help='Do not trunc symbol names ...')
2017-03-30 16:56:49 +02:00
options, args = optparser.parse_args()
if options.help or len(args) == 0:
optparser.print_help()
sys.exit(0)
binary = ELF.parse(args[0])
2017-06-23 17:35:48 +02:00
print_information(binary)
2017-03-30 16:56:49 +02:00
if options.show_all:
do_file_header = do_section_header = do_program_header = True
if options.show_all_headers:
do_file_header = do_section_header = do_program_header = True
else:
do_file_header = options.show_file_header
do_section_header = options.show_section_header
do_program_header = options.show_program_header
if do_file_header or options.show_all:
print_header(binary)
if do_section_header or options.show_all:
print_sections(binary)
if do_program_header or options.show_all:
print_segments(binary)
if options.show_dynamic_tags or options.show_all:
print_dynamic_entries(binary)
if (options.show_symbols or options.show_all or options.show_dynamic_symbols) and len(binary.dynamic_symbols) > 0:
2018-03-07 09:37:42 +01:00
print_dynamic_symbols(binary, options)
if (options.show_symbols or options.show_all or options.show_static_symbols) and len(binary.static_symbols) > 0:
2018-03-07 09:37:42 +01:00
print_static_symbols(binary, options)
2017-03-30 16:56:49 +02:00
if options.show_relocs or options.show_all:
print_all_relocations(binary)
2017-03-30 16:56:49 +02:00
if options.show_imported_symbols or options.show_all:
2018-03-07 09:37:42 +01:00
print_imported_symbols(binary, options)
2017-03-30 16:56:49 +02:00
if options.show_exported_symbols or options.show_all:
2018-03-07 09:37:42 +01:00
print_exported_symbols(binary, options)
2017-03-30 16:56:49 +02:00
if (options.show_gnu_hash or options.show_all) and binary.use_gnu_hash:
2017-03-30 16:56:49 +02:00
print_gnu_hash(binary)
if (options.show_sysv_hash or options.show_all) and binary.use_sysv_hash:
2017-06-10 08:28:22 +02:00
print_sysv_hash(binary)
2017-06-11 09:13:52 +02:00
if options.show_notes or options.show_all:
print_notes(binary)
2017-03-30 16:56:49 +02:00
if __name__ == "__main__":
main()