1*4f1223e8SApple OSS Distributionsimport macholib 2*4f1223e8SApple OSS Distributionsfrom macholib import MachO as macho 3*4f1223e8SApple OSS Distributionsfrom collections import namedtuple 4*4f1223e8SApple OSS Distributionsimport uuid 5*4f1223e8SApple OSS Distributionsimport sys 6*4f1223e8SApple OSS Distributions 7*4f1223e8SApple OSS Distributions 8*4f1223e8SApple OSS Distributions# 9*4f1223e8SApple OSS Distributions# Defines segment/section descriptions that can be used by external users 10*4f1223e8SApple OSS Distributions# like kext management to keep track about memory layout. To avoid the need 11*4f1223e8SApple OSS Distributions# to keep full Mach-O instance around. 12*4f1223e8SApple OSS Distributions# 13*4f1223e8SApple OSS Distributions 14*4f1223e8SApple OSS DistributionsMachOSegment = namedtuple( 15*4f1223e8SApple OSS Distributions 'MachOSegment', 16*4f1223e8SApple OSS Distributions 'name vmaddr vmsize fileoff filesize sections' 17*4f1223e8SApple OSS Distributions) 18*4f1223e8SApple OSS Distributions 19*4f1223e8SApple OSS DistributionsMachOSection = namedtuple( 20*4f1223e8SApple OSS Distributions 'MachOSection', 21*4f1223e8SApple OSS Distributions 'sectname addr size fileoff' 22*4f1223e8SApple OSS Distributions) 23*4f1223e8SApple OSS Distributions 24*4f1223e8SApple OSS Distributions 25*4f1223e8SApple OSS Distributions# 26*4f1223e8SApple OSS Distributions# The Mach-O library loads data for each section in a Mach-O. 27*4f1223e8SApple OSS Distributions# This breaks our macros in few ways: 28*4f1223e8SApple OSS Distributions# - It is slow and no one is really using it. 29*4f1223e8SApple OSS Distributions# - File offsets in fileset KC points outside of the file window reported 30*4f1223e8SApple OSS Distributions# by OSkext API. 31*4f1223e8SApple OSS Distributions# 32*4f1223e8SApple OSS Distributions# Until macholib gets some optin to avoid reading section data we have to 33*4f1223e8SApple OSS Distributions# patch it here. 34*4f1223e8SApple OSS Distributions# 35*4f1223e8SApple OSS Distributions# !!! Note. This works only with the latest lib 1.15.1 !!! 36*4f1223e8SApple OSS Distributions 37*4f1223e8SApple OSS Distributionsif macholib.__version__ == "1.15.2": 38*4f1223e8SApple OSS Distributions from macholib.mach_o import ( 39*4f1223e8SApple OSS Distributions LC_ID_DYLIB, 40*4f1223e8SApple OSS Distributions LC_REGISTRY, 41*4f1223e8SApple OSS Distributions LC_SEGMENT, 42*4f1223e8SApple OSS Distributions LC_SEGMENT_64, 43*4f1223e8SApple OSS Distributions S_ZEROFILL, 44*4f1223e8SApple OSS Distributions load_command, 45*4f1223e8SApple OSS Distributions section, 46*4f1223e8SApple OSS Distributions section_64, 47*4f1223e8SApple OSS Distributions ) 48*4f1223e8SApple OSS Distributions from macholib.ptypes import sizeof 49*4f1223e8SApple OSS Distributions from macholib.util import fileview 50*4f1223e8SApple OSS Distributions 51*4f1223e8SApple OSS Distributions 52*4f1223e8SApple OSS Distributions # !!! This is the actual patch for macholib 1.15.2 !!! 53*4f1223e8SApple OSS Distributions # 54*4f1223e8SApple OSS Distributions # 1. MemMachOHeader subclasses macho.MachOHeader 55*4f1223e8SApple OSS Distributions # 2. Overloaded load() method is copy/paste of the original load() with 56*4f1223e8SApple OSS Distributions # small patch added that disables section contents loading. 57*4f1223e8SApple OSS Distributions # 3. The new MemMachOHeader is injected back into library and used 58*4f1223e8SApple OSS Distributions # in place of macho.MachOHeader. 59*4f1223e8SApple OSS Distributions # 60*4f1223e8SApple OSS Distributions # This code should not ever exist in the first place. So the plan is to 61*4f1223e8SApple OSS Distributions # remove it when macholib gets improved or abandoned by our own 62*4f1223e8SApple OSS Distributions # implementation. 63*4f1223e8SApple OSS Distributions class MemMachOHeader(macho.MachOHeader): 64*4f1223e8SApple OSS Distributions """ Mach-O header parser that does not try to load section data. """ 65*4f1223e8SApple OSS Distributions 66*4f1223e8SApple OSS Distributions def load(self, fh): 67*4f1223e8SApple OSS Distributions fh = fileview(fh, self.offset, self.size) 68*4f1223e8SApple OSS Distributions fh.seek(0) 69*4f1223e8SApple OSS Distributions 70*4f1223e8SApple OSS Distributions self.sizediff = 0 71*4f1223e8SApple OSS Distributions kw = {"_endian_": self.endian} 72*4f1223e8SApple OSS Distributions header = self.mach_header.from_fileobj(fh, **kw) 73*4f1223e8SApple OSS Distributions self.header = header 74*4f1223e8SApple OSS Distributions # If header.magic != self.MH_MAGIC: 75*4f1223e8SApple OSS Distributions # raise ValueError("header has magic %08x, expecting %08x" % ( 76*4f1223e8SApple OSS Distributions # header.magic, self.MH_MAGIC)) 77*4f1223e8SApple OSS Distributions 78*4f1223e8SApple OSS Distributions cmd = self.commands = [] 79*4f1223e8SApple OSS Distributions 80*4f1223e8SApple OSS Distributions self.filetype = self.get_filetype_shortname(header.filetype) 81*4f1223e8SApple OSS Distributions 82*4f1223e8SApple OSS Distributions read_bytes = 0 83*4f1223e8SApple OSS Distributions low_offset = sys.maxsize 84*4f1223e8SApple OSS Distributions for i in range(header.ncmds): 85*4f1223e8SApple OSS Distributions # read the load command 86*4f1223e8SApple OSS Distributions cmd_load = load_command.from_fileobj(fh, **kw) 87*4f1223e8SApple OSS Distributions 88*4f1223e8SApple OSS Distributions # read the specific command 89*4f1223e8SApple OSS Distributions klass = LC_REGISTRY.get(cmd_load.cmd, None) 90*4f1223e8SApple OSS Distributions if klass is None: 91*4f1223e8SApple OSS Distributions raise ValueError("Unknown load command: %d" % (cmd_load.cmd,)) 92*4f1223e8SApple OSS Distributions cmd_cmd = klass.from_fileobj(fh, **kw) 93*4f1223e8SApple OSS Distributions 94*4f1223e8SApple OSS Distributions if cmd_load.cmd == LC_ID_DYLIB: 95*4f1223e8SApple OSS Distributions # remember where this command was 96*4f1223e8SApple OSS Distributions if self.id_cmd is not None: 97*4f1223e8SApple OSS Distributions raise ValueError("This dylib already has an id") 98*4f1223e8SApple OSS Distributions self.id_cmd = i 99*4f1223e8SApple OSS Distributions 100*4f1223e8SApple OSS Distributions if cmd_load.cmd in (LC_SEGMENT, LC_SEGMENT_64): 101*4f1223e8SApple OSS Distributions # for segment commands, read the list of segments 102*4f1223e8SApple OSS Distributions segs = [] 103*4f1223e8SApple OSS Distributions # assert that the size makes sense 104*4f1223e8SApple OSS Distributions if cmd_load.cmd == LC_SEGMENT: 105*4f1223e8SApple OSS Distributions section_cls = section 106*4f1223e8SApple OSS Distributions else: # LC_SEGMENT_64 107*4f1223e8SApple OSS Distributions section_cls = section_64 108*4f1223e8SApple OSS Distributions 109*4f1223e8SApple OSS Distributions expected_size = ( 110*4f1223e8SApple OSS Distributions sizeof(klass) 111*4f1223e8SApple OSS Distributions + sizeof(load_command) 112*4f1223e8SApple OSS Distributions + (sizeof(section_cls) * cmd_cmd.nsects) 113*4f1223e8SApple OSS Distributions ) 114*4f1223e8SApple OSS Distributions if cmd_load.cmdsize != expected_size: 115*4f1223e8SApple OSS Distributions raise ValueError("Segment size mismatch") 116*4f1223e8SApple OSS Distributions # this is a zero block or something 117*4f1223e8SApple OSS Distributions # so the beginning is wherever the fileoff of this command is 118*4f1223e8SApple OSS Distributions if cmd_cmd.nsects == 0: 119*4f1223e8SApple OSS Distributions if cmd_cmd.filesize != 0: 120*4f1223e8SApple OSS Distributions low_offset = min(low_offset, cmd_cmd.fileoff) 121*4f1223e8SApple OSS Distributions else: 122*4f1223e8SApple OSS Distributions # this one has multiple segments 123*4f1223e8SApple OSS Distributions for _j in range(cmd_cmd.nsects): 124*4f1223e8SApple OSS Distributions # read the segment 125*4f1223e8SApple OSS Distributions seg = section_cls.from_fileobj(fh, **kw) 126*4f1223e8SApple OSS Distributions # If the segment has a size and is not zero filled 127*4f1223e8SApple OSS Distributions # then its beginning is the offset of this segment 128*4f1223e8SApple OSS Distributions not_zerofill = (seg.flags & S_ZEROFILL) != S_ZEROFILL 129*4f1223e8SApple OSS Distributions if seg.offset > 0 and seg.size > 0 and not_zerofill: 130*4f1223e8SApple OSS Distributions low_offset = min(low_offset, seg.offset) 131*4f1223e8SApple OSS Distributions 132*4f1223e8SApple OSS Distributions # Do NOT read section data. It is not required and 133*4f1223e8SApple OSS Distributions # does not work well with filset KC offsets. 134*4f1223e8SApple OSS Distributions """ 135*4f1223e8SApple OSS Distributions if not_zerofill: 136*4f1223e8SApple OSS Distributions c = fh.tell() 137*4f1223e8SApple OSS Distributions fh.seek(seg.offset) 138*4f1223e8SApple OSS Distributions sd = fh.read(seg.size) 139*4f1223e8SApple OSS Distributions seg.add_section_data(sd) 140*4f1223e8SApple OSS Distributions fh.seek(c) 141*4f1223e8SApple OSS Distributions """ 142*4f1223e8SApple OSS Distributions segs.append(seg) 143*4f1223e8SApple OSS Distributions # data is a list of segments 144*4f1223e8SApple OSS Distributions cmd_data = segs 145*4f1223e8SApple OSS Distributions 146*4f1223e8SApple OSS Distributions else: 147*4f1223e8SApple OSS Distributions # data is a raw str 148*4f1223e8SApple OSS Distributions data_size = cmd_load.cmdsize - sizeof(klass) - sizeof(load_command) 149*4f1223e8SApple OSS Distributions cmd_data = fh.read(data_size) 150*4f1223e8SApple OSS Distributions cmd.append((cmd_load, cmd_cmd, cmd_data)) 151*4f1223e8SApple OSS Distributions read_bytes += cmd_load.cmdsize 152*4f1223e8SApple OSS Distributions 153*4f1223e8SApple OSS Distributions # make sure the header made sense 154*4f1223e8SApple OSS Distributions if read_bytes != header.sizeofcmds: 155*4f1223e8SApple OSS Distributions raise ValueError( 156*4f1223e8SApple OSS Distributions "Read %d bytes, header reports %d bytes" 157*4f1223e8SApple OSS Distributions % (read_bytes, header.sizeofcmds) 158*4f1223e8SApple OSS Distributions ) 159*4f1223e8SApple OSS Distributions self.total_size = sizeof(self.mach_header) + read_bytes 160*4f1223e8SApple OSS Distributions self.low_offset = low_offset 161*4f1223e8SApple OSS Distributions 162*4f1223e8SApple OSS Distributions 163*4f1223e8SApple OSS Distributions # Patch the library to use our own header class instead. 164*4f1223e8SApple OSS Distributions macho.MachOHeader = MemMachOHeader 165*4f1223e8SApple OSS Distributions 166*4f1223e8SApple OSS Distributions 167*4f1223e8SApple OSS Distributionsclass MemMachO(macho.MachO): 168*4f1223e8SApple OSS Distributions """ Mach-O implementation that accepts I/O stream instead of file. """ 169*4f1223e8SApple OSS Distributions 170*4f1223e8SApple OSS Distributions def __init__(self, file): 171*4f1223e8SApple OSS Distributions """ Creates Mach-O parser on top of provided I/O. """ 172*4f1223e8SApple OSS Distributions 173*4f1223e8SApple OSS Distributions # Figured out file size from the I/O. 174*4f1223e8SApple OSS Distributions file.seek(0, 2) 175*4f1223e8SApple OSS Distributions size = file.tell() 176*4f1223e8SApple OSS Distributions file.seek(0, 0) 177*4f1223e8SApple OSS Distributions 178*4f1223e8SApple OSS Distributions # supports the ObjectGraph protocol 179*4f1223e8SApple OSS Distributions self.graphident = 'mem:%d//'.format(size) 180*4f1223e8SApple OSS Distributions self.filename = 'mem:%d//'.format(size) 181*4f1223e8SApple OSS Distributions self.loader_path = "<no-path>" 182*4f1223e8SApple OSS Distributions 183*4f1223e8SApple OSS Distributions # initialized by load 184*4f1223e8SApple OSS Distributions self.fat = None 185*4f1223e8SApple OSS Distributions self.headers = [] 186*4f1223e8SApple OSS Distributions 187*4f1223e8SApple OSS Distributions self.load(file) 188*4f1223e8SApple OSS Distributions 189*4f1223e8SApple OSS Distributions @staticmethod 190*4f1223e8SApple OSS Distributions def make_seg(seg, sects): 191*4f1223e8SApple OSS Distributions """ Constructs MachOSegment from input. """ 192*4f1223e8SApple OSS Distributions 193*4f1223e8SApple OSS Distributions # Wrap all sections in MachOSection tuple. 194*4f1223e8SApple OSS Distributions segsec = [ 195*4f1223e8SApple OSS Distributions MachOSection( 196*4f1223e8SApple OSS Distributions sectname = s.segname[:s.segname.find(b'\x00')].decode(), 197*4f1223e8SApple OSS Distributions addr = s.addr, 198*4f1223e8SApple OSS Distributions fileoff = s.offset, 199*4f1223e8SApple OSS Distributions size = s.size 200*4f1223e8SApple OSS Distributions ) 201*4f1223e8SApple OSS Distributions for s in sects 202*4f1223e8SApple OSS Distributions ] 203*4f1223e8SApple OSS Distributions 204*4f1223e8SApple OSS Distributions # Return MachOSegment 205*4f1223e8SApple OSS Distributions return MachOSegment( 206*4f1223e8SApple OSS Distributions name=seg.segname[:seg.segname.find(b'\x00')].decode(), 207*4f1223e8SApple OSS Distributions vmaddr = seg.vmaddr, 208*4f1223e8SApple OSS Distributions vmsize = seg.vmsize, 209*4f1223e8SApple OSS Distributions fileoff = seg.fileoff, 210*4f1223e8SApple OSS Distributions filesize = seg.filesize, 211*4f1223e8SApple OSS Distributions sections = segsec 212*4f1223e8SApple OSS Distributions ) 213*4f1223e8SApple OSS Distributions 214*4f1223e8SApple OSS Distributions @property 215*4f1223e8SApple OSS Distributions def segments(self): 216*4f1223e8SApple OSS Distributions """ Constructs section/segment descriptors. 217*4f1223e8SApple OSS Distributions 218*4f1223e8SApple OSS Distributions Values are cached in an instance attribute. 219*4f1223e8SApple OSS Distributions """ 220*4f1223e8SApple OSS Distributions if hasattr(self, '_segments'): 221*4f1223e8SApple OSS Distributions return self._segments 222*4f1223e8SApple OSS Distributions 223*4f1223e8SApple OSS Distributions # Wrap all segments/sections into a MachOSegment/MachOSection. 224*4f1223e8SApple OSS Distributions self._segments = [ 225*4f1223e8SApple OSS Distributions self.make_seg(seg, sec) 226*4f1223e8SApple OSS Distributions for h in self.headers 227*4f1223e8SApple OSS Distributions for _, seg, sec in h.commands 228*4f1223e8SApple OSS Distributions if isinstance(seg, SEGMENT_TYPES) 229*4f1223e8SApple OSS Distributions ] 230*4f1223e8SApple OSS Distributions 231*4f1223e8SApple OSS Distributions return self._segments 232*4f1223e8SApple OSS Distributions 233*4f1223e8SApple OSS Distributions @property 234*4f1223e8SApple OSS Distributions def uuid(self): 235*4f1223e8SApple OSS Distributions """ Returns UUID of the Mach-O. """ 236*4f1223e8SApple OSS Distributions if hasattr(self, '_uuid'): 237*4f1223e8SApple OSS Distributions return self._uuid 238*4f1223e8SApple OSS Distributions 239*4f1223e8SApple OSS Distributions for h in self.headers: 240*4f1223e8SApple OSS Distributions for cmd in h.commands: 241*4f1223e8SApple OSS Distributions # cmds is [(load_command, segment, [sections..])] 242*4f1223e8SApple OSS Distributions (_, segment, _) = cmd 243*4f1223e8SApple OSS Distributions if isinstance(segment, macholib.mach_o.uuid_command): 244*4f1223e8SApple OSS Distributions self._uuid = str(uuid.UUID(bytes=segment.uuid)).upper() 245*4f1223e8SApple OSS Distributions return self._uuid 246*4f1223e8SApple OSS Distributions 247*4f1223e8SApple OSS Distributions 248*4f1223e8SApple OSS Distributions# some fixups in macholib that are required for kext support 249*4f1223e8SApple OSS Distributionsmacholib.mach_o.MH_KEXT_BUNDLE = 0xB 250*4f1223e8SApple OSS Distributions 251*4f1223e8SApple OSS Distributionsmacholib.mach_o.MH_FILETYPE_NAMES[macholib.mach_o.MH_KEXT_BUNDLE] = "kext bundle" 252*4f1223e8SApple OSS Distributionsmacholib.mach_o.MH_FILETYPE_SHORTNAMES[macholib.mach_o.MH_KEXT_BUNDLE] = "kext" 253*4f1223e8SApple OSS Distributions 254*4f1223e8SApple OSS DistributionsSEGMENT_TYPES = (macholib.mach_o.segment_command_64, macholib.mach_o.segment_command) 255*4f1223e8SApple OSS Distributions 256*4f1223e8SApple OSS Distributionsdef get_load_command_human_name(lc): 257*4f1223e8SApple OSS Distributions return lc.get_cmd_name() 258*4f1223e8SApple OSS Distributions 259*4f1223e8SApple OSS Distributions 260*4f1223e8SApple OSS Distributionsclass VisualMachoMap(object): 261*4f1223e8SApple OSS Distributions KB_1 = 1024 262*4f1223e8SApple OSS Distributions KB_16 = 16 * 1024 263*4f1223e8SApple OSS Distributions MB_1 = 1 * 1024 * 1024 264*4f1223e8SApple OSS Distributions GB_1 = 1 * 1024 * 1024 * 1024 265*4f1223e8SApple OSS Distributions 266*4f1223e8SApple OSS Distributions def __init__(self, name, width=40): 267*4f1223e8SApple OSS Distributions self.name = name 268*4f1223e8SApple OSS Distributions self.width = 40 269*4f1223e8SApple OSS Distributions self.default_side_padding = 2 270*4f1223e8SApple OSS Distributions 271*4f1223e8SApple OSS Distributions def get_header_line(self): 272*4f1223e8SApple OSS Distributions return '+' + '-' * (self.width - 2) + '+' 273*4f1223e8SApple OSS Distributions 274*4f1223e8SApple OSS Distributions def get_space_line(self): 275*4f1223e8SApple OSS Distributions return '|' + ' ' * (self.width - 2) + '|' 276*4f1223e8SApple OSS Distributions 277*4f1223e8SApple OSS Distributions def get_dashed_line(self): 278*4f1223e8SApple OSS Distributions return '|' + '-' * (self.width - 2) + '|' 279*4f1223e8SApple OSS Distributions 280*4f1223e8SApple OSS Distributions def get_dotted_line(self): 281*4f1223e8SApple OSS Distributions return '|' + '.' * (self.width - 2) + '|' 282*4f1223e8SApple OSS Distributions 283*4f1223e8SApple OSS Distributions def center_text_in_line(self, line, text): 284*4f1223e8SApple OSS Distributions even_length = bool(len(text) % 2 == 0) 285*4f1223e8SApple OSS Distributions if len(text) > len(line) - 2: 286*4f1223e8SApple OSS Distributions raise ValueError("text is larger than line of text") 287*4f1223e8SApple OSS Distributions 288*4f1223e8SApple OSS Distributions lbreak_pos = (len(line) // 2) - (len(text) // 2) 289*4f1223e8SApple OSS Distributions if not even_length: 290*4f1223e8SApple OSS Distributions lbreak_pos -= 1 291*4f1223e8SApple OSS Distributions out = line[:lbreak_pos] + text 292*4f1223e8SApple OSS Distributions return out + line[len(out):] 293*4f1223e8SApple OSS Distributions 294*4f1223e8SApple OSS Distributions def get_separator_lines(self): 295*4f1223e8SApple OSS Distributions return ['/' + ' ' * (self.width - 2) + '/', '/' + ' ' * (self.width - 2) + '/'] 296*4f1223e8SApple OSS Distributions 297*4f1223e8SApple OSS Distributions def printMachoMap(self, mobj): 298*4f1223e8SApple OSS Distributions MapBlock = namedtuple('MapBlock', 'name vmaddr vmsize fileoff filesize extra_info is_segment') 299*4f1223e8SApple OSS Distributions outstr = self.name + '\n' 300*4f1223e8SApple OSS Distributions other_cmds = '' 301*4f1223e8SApple OSS Distributions blocks = [] 302*4f1223e8SApple OSS Distributions for hdr in mobj.headers: 303*4f1223e8SApple OSS Distributions cmd_index = 0 304*4f1223e8SApple OSS Distributions for cmd in hdr.commands: 305*4f1223e8SApple OSS Distributions # cmds is [(load_command, segment, [sections..])] 306*4f1223e8SApple OSS Distributions (lc, segment, sections) = cmd 307*4f1223e8SApple OSS Distributions lc_cmd_str = get_load_command_human_name(lc) 308*4f1223e8SApple OSS Distributions lc_str_rep = "\n\t LC: {:s} size:{:d} nsects:{:d}".format(lc_cmd_str, lc.cmdsize, len(sections)) 309*4f1223e8SApple OSS Distributions # print lc_str_rep 310*4f1223e8SApple OSS Distributions if isinstance(segment, SEGMENT_TYPES): 311*4f1223e8SApple OSS Distributions segname = segment.segname[:segment.segname.find(b'\x00')].decode() 312*4f1223e8SApple OSS Distributions # print "\tsegment: {:s} vmaddr: {:x} vmsize:{:d} fileoff: {:x} filesize: {:d}".format( 313*4f1223e8SApple OSS Distributions # segname, segment.vmaddr, segment.vmsize, segment.fileoff, segment.filesize) 314*4f1223e8SApple OSS Distributions blocks.append(MapBlock(segname, segment.vmaddr, segment.vmsize, segment.fileoff, segment.filesize, 315*4f1223e8SApple OSS Distributions ' LC:{} : {} init:{:#0X} max:{:#0X}'.format(lc_cmd_str, segname, segment.initprot, segment.maxprot), 316*4f1223e8SApple OSS Distributions True)) 317*4f1223e8SApple OSS Distributions for section in sections: 318*4f1223e8SApple OSS Distributions section_name = section.sectname[:section.sectname.find(b'\x00')].decode() 319*4f1223e8SApple OSS Distributions blocks.append(MapBlock(section_name, section.addr, section.size, section.offset, 320*4f1223e8SApple OSS Distributions section.size, 'al:{} flags:{:#0X}'.format(section.align, section.flags), False)) 321*4f1223e8SApple OSS Distributions #print "\t\tsection:{:s} addr:{:x} off:{:x} size:{:d}".format(section_name, section.addr, section.offset, section.size) 322*4f1223e8SApple OSS Distributions elif isinstance(segment, macholib.mach_o.uuid_command): 323*4f1223e8SApple OSS Distributions other_cmds += "\n\t uuid: {:s}".format(str(uuid.UUID(bytes=segment.uuid)).upper()) 324*4f1223e8SApple OSS Distributions elif isinstance(segment, macholib.mach_o.rpath_command): 325*4f1223e8SApple OSS Distributions other_cmds += "\n\t rpath: {:s}".format(segment.path) 326*4f1223e8SApple OSS Distributions elif isinstance(segment, macholib.mach_o.dylib_command): 327*4f1223e8SApple OSS Distributions other_cmds += "\n\t dylib: {:s} ({:s})".format(str(sections[:sections.find(b'\x00')]), str(segment.current_version)) 328*4f1223e8SApple OSS Distributions else: 329*4f1223e8SApple OSS Distributions other_cmds += lc_str_rep 330*4f1223e8SApple OSS Distributions cmd_index += 1 331*4f1223e8SApple OSS Distributions 332*4f1223e8SApple OSS Distributions # fixup the self.width param 333*4f1223e8SApple OSS Distributions for _b in blocks: 334*4f1223e8SApple OSS Distributions if self.default_side_padding + len(_b.name) + 2 > self.width: 335*4f1223e8SApple OSS Distributions self.width = self.default_side_padding + len(_b.name) + 2 336*4f1223e8SApple OSS Distributions if self.width % 2 != 0: 337*4f1223e8SApple OSS Distributions self.width += 1 338*4f1223e8SApple OSS Distributions 339*4f1223e8SApple OSS Distributions sorted_blocks = sorted(blocks, key=lambda b: b.vmaddr) 340*4f1223e8SApple OSS Distributions mstr = [self.get_header_line()] 341*4f1223e8SApple OSS Distributions prev_block = MapBlock('', 0, 0, 0, 0, '', False) 342*4f1223e8SApple OSS Distributions for b in sorted_blocks: 343*4f1223e8SApple OSS Distributions # TODO add separator blocks if vmaddr is large from prev_block 344*4f1223e8SApple OSS Distributions if b.is_segment: 345*4f1223e8SApple OSS Distributions s = self.get_dashed_line() 346*4f1223e8SApple OSS Distributions else: 347*4f1223e8SApple OSS Distributions s = self.get_dotted_line() 348*4f1223e8SApple OSS Distributions s = self.center_text_in_line(s, b.name) 349*4f1223e8SApple OSS Distributions line = "{:s} {: <#020X} ({: <10d}) floff:{: <#08x} {}".format(s, b.vmaddr, b.vmsize, b.fileoff, b.extra_info) 350*4f1223e8SApple OSS Distributions if (b.vmaddr - prev_block.vmaddr) > VisualMachoMap.KB_16: 351*4f1223e8SApple OSS Distributions mstr.append(self.get_space_line()) 352*4f1223e8SApple OSS Distributions mstr.append(self.get_space_line()) 353*4f1223e8SApple OSS Distributions 354*4f1223e8SApple OSS Distributions mstr.append(line) 355*4f1223e8SApple OSS Distributions 356*4f1223e8SApple OSS Distributions if b.vmsize > VisualMachoMap.MB_1: 357*4f1223e8SApple OSS Distributions mstr.append(self.get_space_line()) 358*4f1223e8SApple OSS Distributions mstr.extend(self.get_separator_lines()) 359*4f1223e8SApple OSS Distributions mstr.append(self.get_space_line()) 360*4f1223e8SApple OSS Distributions #mstr.append(self.get_space_line()) 361*4f1223e8SApple OSS Distributions prev_block = b 362*4f1223e8SApple OSS Distributions mstr.append(self.get_space_line()) 363*4f1223e8SApple OSS Distributions if prev_block.vmsize > VisualMachoMap.KB_16: 364*4f1223e8SApple OSS Distributions mstr.append(self.get_space_line()) 365*4f1223e8SApple OSS Distributions mstr.append(self.get_header_line()) 366*4f1223e8SApple OSS Distributions print(outstr) 367*4f1223e8SApple OSS Distributions print("\n".join(mstr)) 368*4f1223e8SApple OSS Distributions print("\n\n=============== Other Load Commands ===============") 369*4f1223e8SApple OSS Distributions print(other_cmds) 370*4f1223e8SApple OSS Distributions 371*4f1223e8SApple OSS Distributions 372*4f1223e8SApple OSS Distributionsif __name__ == '__main__': 373*4f1223e8SApple OSS Distributions import sys 374*4f1223e8SApple OSS Distributions if len(sys.argv) < 2: 375*4f1223e8SApple OSS Distributions print("Usage: {} /path/to/macho_binary".format(sys.argv[0])) 376*4f1223e8SApple OSS Distributions sys.exit(1) 377*4f1223e8SApple OSS Distributions with open(sys.argv[-1], 'rb') as fp: 378*4f1223e8SApple OSS Distributions mobject = MemMachO(fp) 379*4f1223e8SApple OSS Distributions 380*4f1223e8SApple OSS Distributions p = VisualMachoMap(sys.argv[-1]) 381*4f1223e8SApple OSS Distributions p.printMachoMap(mobject) 382*4f1223e8SApple OSS Distributions sys.exit(0) 383