Mac系统下lipo, ar, nm等工具的使用简介
引言
开发第三方库时, 如果没有进行特殊处理, 很容易把其他第三方库的符号暴露出来, 导致链接时产生符号重复. 如下图所示

如果用户链接了其他版本的libjpeg, 会因为入口地址不正确让程序直接崩溃


本文就从这个问题入手, 简要介绍Mac OS X系统下几个常用二进制文件修改工具的使用
概述
我们都知道, 代码到可执行文件需要经过编译(compile)和链接(link)两个主要步骤. 编译是把程序语言转换为机器指令, 这个不在本文的讨论范围.
链接是把分块编译的对象文件(obj)合并成一个完整的程序, 主要解决函数入口重定向的问题. 其功能主要就是把所有的对象文件打个包, 生成一个导出符号表, 让其他程序可以知道本文件的结构.
而对于苹果系统来说, 丰富的设备包含了多种架构平台, armv7, arm64, i386, x86_64是最常见的指令集, 一个程序库为了兼容各个平台, 常常要把不同平台编译的程序合并起来, 生成所谓的胖文件(Fat File), 这样子开发者就不需要专门准备一套真机版本库和模拟器版本库了(在早期, 很多第三方库确实是提供了不同版本的二进制文件来减少库文件大小).
因此对于一个可执行文件, 其实包含了3层结构: Universal Fat File -> Single Architecture Binary File -> Mach-O Object.
本文将要介绍的lipo, ar, nm, strip, ld等工具的功能就是对这三层结构进行转换和修改.
本文以高德地图iOS SDK MAMapKit.framework 4.0.3为例, 演示如何从这个库文件里剔除暴露的png符号.
工具介绍
lipo
lipo是管理Fat File的工具, 可以查看平台列表, 提取特定平台, 重新打包.
首先运行 lipo -info MAMapKit

可以看出这个文件包含了4个平台的代码. 接下来的所有操作都是要针对单一平台进行的, 因此先提取出来armv7平台,
lipo -thin armv7 MAMapKit -output MAMapKit.armv7

可以看出单平台的程序文件要小得多.
接下来我们来查看一下这个文件的符号表
nm
nm用来显示一个程序包的符号表, 默认会显示入口地址, 符号类型, 符号名.
nm -j MAMapKit.armv7 | grep png > symbols 可以获得所有的libpng导出符号, 存入到symbols文件, 为接下来的工作做准备. -j 选项控制只输出符号名.
strip
strip用来删除程序里的符号表. -R 用来指定一个要删除的符号列表, 使用上述生成的symbols文件. 添加 -S 选项来保留其他符号.
strip -S -R symbols MAMapKit.armv7 -o MAMapKit.armv7.strip
可以验证生成的新文件已经没有了png符号.
我们用这个方法应用到其他所有平台.
ar
我们用上述方法处理arm64时遇到了问题

这个文件里的一些符号用作了重定向入口, strip命令不允许删除, 这时就需要更强力的工具ld登场了. ld 其实苹果系统下的链接器, 可以更精确的控制符号表的导出.
ld的操作对象是obj, 因此我们需要先用到ar打包工具. ar可以查看一个程序包里的对象文件列表, 解压出其中的对象文件并重新打包.
ar -t MAMapKit.arm64

可以看到整个包就包含了一个主对象文件.
使用 -x 解压出来, 接下来就要轮到 ld 上场了.
ld
本问题需要的ld功能和strip基本一致, 使用下述命令
ld -x -r -unexported_symbols_list symbols MAMapKit-arm64-master.o -o MAMapKit-arm64-master.o.strip
由此生成对象文件就剔除了png符号表
接下来要做的就是上述逆过程, 对象文件合并成程序文件
ar -r MAMapKit.arm64 MAMapKit-arm64-master.o.strip Pods-MAMapKit-dummy.o
最后是把各个平台的程序打包成Fat File即可.
终极大招
2016-09-02 更新
使用 Xcode 编译选项即可以完成上述任务!
1. Perform Single-Object Prelink
这个参数设置为 Yes
这一步, 把所有的object文件合并成一个object文件, 相当于进行预编译
这是最核心的操作, 因为该命令会触发Xcode调用 ld 命令

2. Single-Object Prelink Flags
这个参数设置 -unexported_symbols_list $(PROJECT_DIR)/symbol.txt
这就相当于给 ld 命令传递参数, 正式我们上文提到的操作
通过这两个选项的配置, 我们就在Xcode里设置好了符号删除的任务, 不用再在编译完成后手动删除.
其他的一些参数, 比如 Strip Style, 可以根据需要手动决定, 因为删除后就不能再断点调试了!
Mac系统下lipo, ar, nm等工具的使用简介的更多相关文章
- 在Linux系统下运行微信Web开发者工具
		
微信Web开发者工具只有window版本和mac版本,如果想要在Linux系统下运行微信Web开发者工具,需要花费很大周折. 注:带 * 的步骤或文件为不确定是否管用的步骤或文件.本人系统为Linux ...
 - Mac系统下React Native环境搭建
		
这里记录一下在Mac系统下搭建React Native开发环境的过程: 1. 安装HomeBrew: /usr/bin/ruby -e "$(curl -fsSL https://raw.g ...
 - (转载)Mac系统下利用ADB命令连接android手机并进行文件操作
		
Mac系统下利用ADB命令连接android手机并进行文件操作 标签: Mac adb android 2016-03-14 10:09 5470人阅读 评论(1) 收藏 举报 分类: Androi ...
 - MAC系统下,删除.svn文件
		
MAC系统下,.svn文件是隐藏的. 如果项目是非export导出的,那么项目中会有很多的.svn文件. 如果项目的体积非常庞大,我们如何快速的批量删除.svn文件呢?下面是操作方法: 打开终端,cd ...
 - Mac系统下使用VirtualBox虚拟机安装win7--第一步 安装vbox虚拟机
		
Mac系统下使用VirtualBox虚拟机安装win7操作步骤: 第一步 安装vbox虚拟机 1.先下载vbox,下载地址:: https://www.virtualbox.org/wiki/Down ...
 - 曲线救国,解决Mac系统下,Android sdk下载失败的问题
		
Mac下翻_墙的问题 话说GFW屏蔽谷歌已经有一阵子了,最近打算在Mac系统下折腾个Android应用,备好了IDE,只欠SDK,无奈下载时因为GFW的缘故,总是失败,我心痛哉! 由于本人偏爱Mac系 ...
 - MAC系统下Sublime Text3 配置Python3详细教程
		
MAC系统下Sublime Text3 配置Python3详细教程(亲测有效) https://blog.csdn.net/weixin_41768008/article/details/798590 ...
 - Mac系统下编译支持Android平台的最新X264编码器
		
Mac系统下编译支持Android平台的最新X264编码器 原文来自 http://www.mingjianhua.com,转载请注明出处 1.首先去官网下载最新的x264源代码,解压到任意目录 ht ...
 - mac系统下ionic环境配置
		
本人是在mac环境下进行配置的: 下载nodejs:https://nodejs.org/download/ 并双击安装 Cordova and Ionic command-line tools 安装 ...
 
随机推荐
- js jquery 扩展方法
			
//扩展Array,增加IsInAyyay函数.函数功能:判断数组是否包含某元素 Array.prototype.IsInAyyay=function(e) { for (var i=0;i<t ...
 - 用定时器令P0(或其它IO口)产生多路方波
			
void Timer0_isr(void) interrupt 1 using 1{ static unsigned char i; //重新赋值 12M晶振计算,指令周期1uS,500x2=1mS ...
 - linux 无法解压过大文件解决
			
[root@vmbbak yum]# unzip RHEL_5.7\ x86_64\ DVD-1.zip error: Zip file too big (greater than 429495910 ...
 - 关于Datagridview控件用法的一些总结(设置列chicun)
			
1. 关于Datagridview控件用法的一些总结:http://www.cnblogs.com/mingjiatang/p/4968049.html
 - JDK的安装及部署配置(配图解)
			
JDK的安装及部署配置 双击安装文件,出现如下界面 点击[下一步]出现如下界面,更改安装路径(建议安装至D盘), 点击[下一步],出现如下界面,修改文件夹名. 点击[确定],耐心等待 直至出现如下界面 ...
 - linux服务之X windows
			
gnome display manager :gdm window manager :metacity x-server x-client [root@localhost gdm]# xlsclien ...
 - Centos7 mysql-community-5.7.11编译安装
			
安装环境 [root@localhost ~]# cat /etc/centos-release CentOS Linux release 7.0.1406 (Core) 0x01 准备工作 1.到m ...
 - javascript学习第三课引用类型object
			
主要内容: 1.object 是所有类型的基类 实例化对象: 1. var obj = new Object(); 2. var obj = {}; 设置对象属性和方法: obj.name = 'he ...
 - try catch finally return之间的关系
			
一.try catch finally return之间的关系: 正在写dsoFramer的时候,同事突然说面试的时候问的一个问题,catch和return那个先执行,我瞬间迷茫了,然后整理了整理,稍 ...
 - iOS 微信支付
			
相关资料 SDK下载:https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=11_1 APP端开发步骤说明:https://pay.weixin ...