iOS-开发进阶02:链接与Symbol(上)

iOS 开发进阶 文章汇总

目录


一、Mach-O与链接器

Mach-O
  • Mach-O(Mach Object)macOS、iOS、 iPadOS存储程序和库的文件格式。对应系统通过应用二进制接口(application binary interface,缩写为ABI)来运行该格式的文件。

  • Mach-O格式用来替代BSD系统的a.out格式。Mach-O文件格式保存了在编译过程链接过程中产生的机器代码数据,从而为静态链接动态链接的代码提供了单一文件格式

Mach-O文件就是一个可读可写二进制文件


可执行文件调用过程:
  1. 调用fork函数,创建一个process进程
  2. 调用execve或其衍生函数,在该进程上加载,执行我们的Mach-O文件

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

  1. 将文件加载到内存
  2. 开始分析Mach-O中的mach header,以确认它是有效的Mach-O文件


通过终端命令查看Mach-O文件:objdump --macho --private-headers 可执行文件地址


通过自己的项目查看Mach-O文件

machoinfo项目是用来查看Mach-O文件的,编译后生成machoinfo的可执行文件可作为命令使用:
拷贝machoinfo可执行文件到桌面


使用machoinfo项目调试Mach-O文件读取过程

通过以下操作将TestCode项目编译的可执行文件路径拖入到machoinfo项目启动参数中,这样TestCode可执行文件路径就会传入到machoinfo项目main函数的argv参数中

main函数中添加断点后运行machoinfo项目就能看到参数传入进来了,然后通过这个路径就能读取到TestCode可执行文件并进行Mach-O文件读取过程


二、符号的种类与作用

Symbol Table
  • Symbol Table:就是用来保存符号。
  • String Table:就是用来保存符号的名称。
  • Indirect Symbol Table:间接符号表。保存使用的外部符号,也就是使用的外部动态库的符号(比如NSLog)。是Symbol Table的子集。

通过编译项目生成可执行文件-->然后在终端使用命令查看Mach-O的中的符号操作比较繁琐,现在通过配置Shell脚本,当项目编译成功后直接在终端执行命令并展示命令的结果。

首先通过重定向在Xcode中让当前终端显示特定内容:

Xcode让终端显示xcconfig文件中的变量:

现在通过xcode_run_cmd.sh脚本执行相关的命令:

#!/bin/sh

RunCommand() {
  #判断全局字符串VERBOSE_SCRIPT_LOGGING是否为空。-n string判断字符串是否非空
  #[[是 bash 程序语言的关键字。用于判断
  if [[ -n "$VERBOSE_SCRIPT_LOGGING" ]]; then
    #作为一个字符串输出所有参数。使用时加引号"$*" 会将所有的参数作为一个整体,以"$1 $2 … $n"的形式输出所有参数
      if [[ -n "$TTY" ]]; then
          echo "♦ $@" 1>$TTY
      else
          echo "♦ $*"
      fi
      echo "------------------------------------------------------------------------------" 1>$TTY
  fi
  #与$*相同。但是使用时加引号,并在引号中返回每个参数。"$@" 会将各个参数分开,以"$1" "$2" … "$n" 的形式输出所有参数
  if [[ -n "$TTY" ]]; then
      echo `$@ &>$TTY`
  else
      "$@"
  fi
  #显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
  return $?
}

EchoError() {
    #在shell脚本中,默认情况下,总是有三个文件处于打开状态,标准输入(键盘输入)、标准输出(输出到屏幕)、标准错误(也是输出到屏幕),它们分别对应的文件描述符是0,1,2
    # >  默认为标准输出重定向,与 1> 相同
    # 2>&1  意思是把 标准错误输出 重定向到 标准输出.
    # &>file  意思是把标准输出 和 标准错误输出 都重定向到文件file中
    # 1>&2 将标准输出重定向到标准错误输出。实际上就是打印所有参数已标准错误格式
    if [[ -n "$TTY" ]]; then
        echo "$@" 1>&2>$TTY
    else
        echo "$@" 1>&2
    fi
    
}

RunCMDToTTY() {
    if [[ ! -e "$TTY" ]]; then
        EchoError "=========================================="
        EchoError "ERROR: Not Config tty to output."
        exit -1
    fi
    # CMD:终端需要运行的命令
    # CMD_FLAG:运行的命令的参数
    # TTY:终端标志
    if [[ -n "$CMD" ]]; then
        RunCommand "$CMD" ${CMD_FLAG}
    else
        EchoError "=========================================="
        EchoError "ERROR:Failed to run CMD. THE CMD must not null"
    fi
}

RunCMDToTTY

xcode_run_cmd.sh脚本需需要三个参数CMD 、CMD_FLAG 、TTY,这三个参数在xcconfig文件中定义就能获取到

// -p:不排序
// -a: 显示除了调试符号的其他所有符号
MACHO_PATH = ${BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME)/$(FULL_PRODUCT_NAME)/$(PRODUCT_NAME)
CMD = nm
CMD_FLAG = -pa ${MACHO_PATH}
TTY = /dev/ttys000

添加shell脚本执行命令/bin/sh "$SRCROOT/xcode_run_cmd.sh"并编译即可在终端看到nm命令执行的结果

编译日志

Xcode编译日志中可以看到项目编译后-签名前执行的shell脚本,因此执行shell脚本时Mach-O文件已经生成了


三、strip命令

strip命令可以用来剥离Mach-O文件中的符号,比如调试符号等。strip命令修改的是Symbol Table符号表、不能修改Indirect Symbol Table间接符号表。
Xcode默认会在Release编译情况下剥离所有符号,但是Debug编译情况下不会剥离符号。

设置Debug编译情况下剥离调试符号:

  • 从编译日志中也可以看到,Xcodestrip命令是在shell脚本之后执行的
  • 在实际开发项目中测试Xcodestrip,没有剥离符号的Mach-O大小为34M,剥离符号后大小为20.8M,可见剥离符号对于瘦包还是非常有用的


现在不使用Xcodestrip命令,改在shell脚本中执行strip命令

因为Xcodestrip使用的是clang的命令,shell脚本使用的是ld链接器的命令,所以可以在终端查看ld链接器的参数,终端输入命令man ld回车后输入/-S进行搜索:

xcconfig文件添加ld的参数

OTHER_LDFLAGS = -Xlinker -S

XcodeBuild Sttings中可以看到添加成功了:

编译后可发现终端输出中少了很多调试符号。

strip 参数如下:

  • -x: Non-Global
  • 无参数: All Symbol
  • -S: 调试符号

四、在LLVM项目中调试nm命令

LLVM项目下载安装参考:iOS-底层探索29:自定义Clang插件

填入启动参数和machoinfo项目查看Mach-O文件添加启动参数的方法相同:

运行LLVM项目(也就是运行LLVM项目中llvm-nm Schemellvm-nm Target),控制台打印如下,可以看到和shell中使用nm命令输出到终端的信息相同:

此外在llvm-nm.cpp源码的main函数中添加断点后运行项目即可断点调试llvm-nm命令的源码,从llvm-nm的源码中我们就能看到nm命令是如何读取Mach-O文件的。


五、总结

通过对符号的strip不仅可以减少ipa包体积还可以减少动态库、静态库的体积

ipa包瘦身主要有以下操作:

  • 编译时期:-O0、-Os生成目标文件
  • 链接时期:dead code strip死代码剥离(也是剥离符号)
  • 生成Mach-O后:strip剥离符号,对Mach-O文件进行修改


快捷键

Command + K清空终端显示内容


参考

iOS-底层探索28:LLVM入门

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

推荐阅读更多精彩内容