iOS高级强化--002:ABI Mach-O

Mach-OMach Object)是macOSiOSiPadOS存储程序和库的⽂件格式。对应系统通过应⽤⼆进制接⼝(application binary interface,缩写为ABI)来运⾏该格式的⽂件。

Mach-O格式⽤来替代BSD系统的a.out格式。Mach-O⽂件格式保存了在编译过程和链接过程中产⽣的机器代码和数据,从⽽为静态链接和动态链接的代码提供了单⼀⽂件格式。

苹果很多文件都采用Mach-O格式,最常见的就是可执行文件和动态库。当然,还有.o的目标文件、.a.framework的静态库以及动态连接器dyld等等。

项目运行时,IPA包里的可执行文件会被加载

可执⾏⽂件调⽤过程:

  • 调⽤fork函数,创建⼀个process(进程)
  • 调⽤execve或其衍⽣函数,在该进程上加载,执⾏我们的Mach-O⽂件

当调⽤execve(程序加载器)时,内核实际上在执⾏以下操作:

  • 将⽂件加载到内存
  • 开始分析Mach-O中的mach_header,以确认它是有效的Mach-O⽂件

Mach-O文件结构

Header包含该二进制文件的一般信息

  • 字节顺序、架构类型、加载指令的数量等
  • 使得可以快速确认一些信息,比如当前文件用于32位还是64位,对应的处理器是什么、文件类型是什么

Load commands一张包含很多内容的表

  • 内容包括区域的位置、符号表、动态符号表等

Data通常是对象文件中最大的部分

  • 包含Segement的具体数据

Header包含Mach HeaderSegmentSections三种类型

  • Header内的Sections描述了对应的⼆进制信息
  • Mach Header属于Header的⼀部分,它包含了整个⽂件的信息和Segment信息
  • Segmentssegment commands):指定操作系统应该将Segments加载到内存中的什么位置,以及为该Segments分配的字节数。还指定⽂件中的哪些字节属于该Segments,以及⽂件包含多少sections。之前段始终是4096字节4KB的倍数,其中4096字节是最小大小。现在段是16KB的倍数,在macOS_x86_64上是16KB,在iOS上是32KBSegments名称的约定是使⽤全⼤写字⺟,后跟双下划线(例如__TEXT
  • Section:所有sections都在每个segment之后⼀个接⼀个地描述。sections⾥⾯定义其名称,在内存中的地址、⼤⼩、⽂件中section数据的偏移量和segment名称。Section的名称约定是使⽤全⼩写字⺟,再加上双下划线(例如__text

Header的最开始是Magic Number,表示这是一个Mach-O文件

使用objdump --macho -private-header 【Mach-O路径】命令,查看Mach Header信息

也可以使用otool -h 【Mach-O路径】命令,这种方式打印的更像是原始数据,开发者难以阅读

Mach-OLoad CommandsData是分开的

  • Load Commands:二进制⽂件加载进内存要执⾏的⼀些指令
    这⾥的指令主要在负责APP对应进程的创建和基本设置(分配虚拟内
    存,创建主线程,处理代码签名/加密的⼯作),然后对动态链接库(.dylib
    系统库和自定义的动态库)进⾏库加载和符号解析的⼯作
  • Data:由多个⼆进制组成,逐一排列。包含__TEXT代码__DATA代码符号表,存储实际的代码和数据
    __TEXT段:只读区域,包含可执⾏代码和常量数据
    __DATA段:读/写,包含初始化和未初始化数据和⼀些动态链接专属数据
  • 例如:Load Command __TEXT中,记录了代码起始位置和大小等配置信息,dyld根据这些配置信息读取__TEXT代码段的实际代码
  • Mach-O中都是二进制数据,根据结构体内存对齐规则,数据会按照指定方式在二进制中进行排列,dyld按照相同方式逐一读取Load Command
  • Mach-O中保存了执行过程中的所有信息:
    Load Command __LINKEDIT:动态连接器需要的信息
    Load Command LC_LOAD_DYLINKER:链接器的位置
    Load Command LC_UUIDMach-O的唯一标识符
    Load Command LC_BUILD_VERSIONMach-O的编译信息
    Load Command LC_MAIN:入口函数,默认为main,也可以修改
    Load Command LC_LOAD_DYLIB:加载动态库的信息

使用objdump命令,查看Mach-O文件

objdump --macho --private-headers 【Mach-O路径】

查看Load Commands中入口函数的配置(LC_MAIN

objdump --macho --private-headers 【Mach-O路径】 | grep 'LC_MAIN' -A 3

__TEXT代码段的格式:

main函数为例,在x86架构下通过clang编译出来的并不是汇编代码,而是机器指令,这些机器指令对应每一条汇编代码都是固定不变的

使用objdump --macho -d 【Mach-O路径】命令,查看__TEXT代码段信息

  • __TEXT段__text section中,main函数以100003fa0地址开始,至100003fb5地址结束。中间列是机器码,最后列的是汇编代码
  • 汇编代码转换机器码在底层有类似字典的映射关系;汇编是给开发者提供的,而机器只需要读取最原始的机器码即可

通过代码读取Mach-O文件

machoinfo是一个读取Mach-O文件的项目,通过main函数接收参数

编译后生成Mach-O文件

打开终端,进入Mach-O所在目录,使用./machoinfo作为命令执行

./machoinfo 【将要读取的Mach-O文件路径】

machoinfo项目进行调试

打开Edit Scheme...配置默认参数

main函数设置断点并运行项目,可以接收到配置的默认参数

通过lldb可以查看结果

总结
  • Mach-O本质是一个二进制文件,数据按照指定方式在二进制中进行排列,dyld按照相同方式进行读取
  • Mach-O文件包含了文件配置 + 二进制代码
  • Mach-O文件可读可写Mach-O文件在苹果签名前、签名后均可修改,但签名后修改需要进行重签名
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,744评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,505评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,105评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,242评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,269评论 6 389
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,215评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,096评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,939评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,354评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,573评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,745评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,448评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,048评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,683评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,838评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,776评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,652评论 2 354

推荐阅读更多精彩内容