macho-dump: Basic Mach 64 support.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@98453 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Daniel Dunbar 2010-03-13 22:10:11 +00:00
parent 1b2eb0e8a6
commit f0a0be47ed

View File

@ -12,12 +12,10 @@ class Reader:
else: else:
self.file = open(path,'rb') self.file = open(path,'rb')
self.isLSB = None self.isLSB = None
self.is64Bit = None
self.string_table = None self.string_table = None
def setLSB(self, isLSB):
self.isLSB = bool(isLSB)
def tell(self): def tell(self):
return self.file.tell() return self.file.tell()
@ -42,6 +40,9 @@ class Reader:
Value = struct.unpack('><'[self.isLSB] + 'I', self.read(4))[0] Value = struct.unpack('><'[self.isLSB] + 'I', self.read(4))[0]
return int(Value) return int(Value)
def read64(self):
return struct.unpack('><'[self.isLSB] + 'Q', self.read(8))[0]
def registerStringTable(self, strings): def registerStringTable(self, strings):
if self.string_table is not None: if self.string_table is not None:
raise ValueError,"%s: warning: multiple string tables" % sys.argv[0] raise ValueError,"%s: warning: multiple string tables" % sys.argv[0]
@ -60,9 +61,13 @@ def dumpmacho(path, opts):
magic = f.read(4) magic = f.read(4)
if magic == '\xFE\xED\xFA\xCE': if magic == '\xFE\xED\xFA\xCE':
f.setLSB(False) f.isLSB, f.is64Bit = False, False
elif magic == '\xCE\xFA\xED\xFE': elif magic == '\xCE\xFA\xED\xFE':
f.setLSB(True) f.isLSB, f.is64Bit = True, False
elif magic == '\xFE\xED\xFA\xCF':
f.isLSB, f.is64Bit = False, True
elif magic == '\xCF\xFA\xED\xFE':
f.isLSB, f.is64Bit = True, True
else: else:
raise ValueError,"Not a Mach-O object file: %r (bad magic)" % path raise ValueError,"Not a Mach-O object file: %r (bad magic)" % path
@ -79,6 +84,9 @@ def dumpmacho(path, opts):
print "('flag', %r)" % f.read32() print "('flag', %r)" % f.read32()
if f.is64Bit:
print "('reserved', %r)" % f.read32()
start = f.tell() start = f.tell()
print "('load_commands', [" print "('load_commands', ["
@ -87,7 +95,8 @@ def dumpmacho(path, opts):
print "])" print "])"
if f.tell() - start != loadCommandsSize: if f.tell() - start != loadCommandsSize:
raise ValueError,"%s: warning: invalid load commands size: %r" % (sys.argv[0], loadCommandsSize) raise ValueError,"%s: warning: invalid load commands size: %r" % (
sys.argv[0], loadCommandsSize)
def dumpLoadCommand(f, i, opts): def dumpLoadCommand(f, i, opts):
start = f.tell() start = f.tell()
@ -99,24 +108,34 @@ def dumpLoadCommand(f, i, opts):
print " ('size', %r)" % cmdSize print " ('size', %r)" % cmdSize
if cmd == 1: if cmd == 1:
dumpSegmentLoadCommand32(f, opts) dumpSegmentLoadCommand(f, opts, False)
elif cmd == 2: elif cmd == 2:
dumpSymtabCommand(f, opts) dumpSymtabCommand(f, opts)
elif cmd == 11: elif cmd == 11:
dumpDysymtabCommand(f, opts) dumpDysymtabCommand(f, opts)
elif cmd == 25:
dumpSegmentLoadCommand(f, opts, True)
elif cmd == 27: elif cmd == 27:
import uuid import uuid
print " ('uuid', %s)" % uuid.UUID(bytes=f.read(16)) print " ('uuid', %s)" % uuid.UUID(bytes=f.read(16))
else: else:
print >>sys.stderr,"%s: warning: unknown load command: %r" % (sys.argv[0], cmd) print >>sys.stderr,"%s: warning: unknown load command: %r" % (
sys.argv[0], cmd)
f.read(cmdSize - 8) f.read(cmdSize - 8)
print " )," print " ),"
if f.tell() - start != cmdSize: if f.tell() - start != cmdSize:
raise ValueError,"%s: warning: invalid load command size: %r" % (sys.argv[0], cmdSize) raise ValueError,"%s: warning: invalid load command size: %r" % (
sys.argv[0], cmdSize)
def dumpSegmentLoadCommand32(f, opts): def dumpSegmentLoadCommand(f, opts, is64Bit):
print " ('segment_name', %r)" % f.read(16) print " ('segment_name', %r)" % f.read(16)
if is64Bit:
print " ('vm_addr', %r)" % f.read64()
print " ('vm_size', %r)" % f.read64()
print " ('file_offset', %r)" % f.read64()
print " ('file_size', %r)" % f.read64()
else:
print " ('vm_addr', %r)" % f.read32() print " ('vm_addr', %r)" % f.read32()
print " ('vm_size', %r)" % f.read32() print " ('vm_size', %r)" % f.read32()
print " ('file_offset', %r)" % f.read32() print " ('file_offset', %r)" % f.read32()
@ -129,7 +148,7 @@ def dumpSegmentLoadCommand32(f, opts):
print " ('sections', [" print " ('sections', ["
for i in range(numSections): for i in range(numSections):
dumpSection32(f, i, opts) dumpSection(f, i, opts, is64Bit)
print " ])" print " ])"
def dumpSymtabCommand(f, opts): def dumpSymtabCommand(f, opts):
@ -206,10 +225,15 @@ def dumpDysymtabCommand(f, opts):
f.seek(prev_pos) f.seek(prev_pos)
def dumpSection32(f, i, opts): def dumpSection(f, i, opts, is64Bit):
print " # Section %r" % i print " # Section %r" % i
print " (('section_name', %r)" % f.read(16) print " (('section_name', %r)" % f.read(16)
print " ('segment_name', %r)" % f.read(16) print " ('segment_name', %r)" % f.read(16)
if is64Bit:
print " ('address', %r)" % f.read64()
size = f.read64()
print " ('size', %r)" % size
else:
print " ('address', %r)" % f.read32() print " ('address', %r)" % f.read32()
size = f.read32() size = f.read32()
print " ('size', %r)" % size print " ('size', %r)" % size
@ -223,6 +247,8 @@ def dumpSection32(f, i, opts):
print " ('flags', %#x)" % f.read32() print " ('flags', %#x)" % f.read32()
print " ('reserved1', %r)" % f.read32() print " ('reserved1', %r)" % f.read32()
print " ('reserved2', %r)" % f.read32() print " ('reserved2', %r)" % f.read32()
if is64Bit:
print " ('reserved3', %r)" % f.read32()
print " )," print " ),"
prev_pos = f.tell() prev_pos = f.tell()