符号与链接 (2)
遇到的问题
- 关于xcconfig
xconfig编写指南
xconfig编写指南
//绝对+相对 #include "Pods/Target Support Files/Pods-LoginApp/Pods-LoginApp.debug.xcconfig" #include "LoginApp/PP/Test.xcconfig" SLASH=/ //自定义变量的用法,下面两种都可以 // ${SLASH} // $(SLASH) HOST_URL = http:${SLASH}/127.0.0.1 // Debug iphonesimulator* x86_64 // 可以设置条件, Cat这个库本来不存在 // scheme 为 Debug生效 // iphonesimulator* 模拟器状态下生效 OTHER_LDFLAGS[config=Debug][sdk=iphonesimulator*][arch=x86_64]=-framework "Cat"
- 关于多Target, 每个Target可以指定编译的文件
- Swift宏
- 定义宏时, 需要前面加-D,
- 项目中有swift文件, Target编译里面有swift文件
build setting -> swift -> Other Swift Flags -> Debug -> -D DEV
- 然后项目中就可以使用DEV来判断了
静态链接与动态链接
MachO文件格式
16189242255752.jpg
MachO,前面是配置,后面是代码,通过配置来找到代码
16189248548725.jpg
VERBOSE_SCRIPT_LOGGING=-v MACH_PATH=${BUILD_DIR}/$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME)/${PRODUCT_NAME} // 查看mach-header // objdump --macho -private-header ${MACH_PATH} // otool -h ${MACH_PATH} // 查看__TEXT (查看下面_main函数图片) // objdump --macho -d ${MACH_PATH} // 查看符号表 // objdump --macho --syms ${MACH_PATH} // 查看导出符号 // objdump --macho --exports-trie ${MACH_PATH} // 查看间接符号表 // objdump --macho --indirect-symbols ${MACH_PATH} // nm -m ${MACH_PATH} CMD = objdump --macho --syms ${MACH_PATH} TTY=/dev/ttys002 // mach-o + 签名 -> 苹果就认 // mach-o __TEXT.__text只读 // 2016 7、8年 60m 500m __TEXT.__text
16189251106976.jpg
代码 -> .0的过程
- 汇编
- 符号归类 -> 重定位符号表 符号表
1.重定位符号表 -> 放置的是.m/.o用到的API
- .o文件 -> 链接 -> 符号表合并到一张表中 -> 可执行文件(exec)
1.链接过程 -> 就是处理目标文件符号的过程
#import <Foundation/Foundation.h> #import "WeakImportSymbol.h" //#import "LGOneObject.h" // 全局变量 int global_uninit_value; int global_init_value = 10; double default_x __attribute__((visibility("hidden"))); // 静态变量 -> 本地变量 // 导入 导出 static int static_init_value = 9; static int static_uninit_value; void weak_function(void) { NSLog(@"weak_function"); } // .o -》 虚拟 -〉NSLog // 1. 汇编 // 2. 符号归类 -》 重定位符号表 // 3. 重定位符号表 -〉.m/.o 用到的API // .o -> 链接 -> 一张表 -> exec // 链接 -》 处理目标文件符号 int main(int argc, char *argv[]) { static_uninit_value = 10; // 导入了NSLog // Foundation 导出了 NSLog // 动态库 -> 符号 // 间接符号表 -> 动态库符号 // 间接符号表 // oc动态库 -> 体积 -> // re导出 NSLog(@"%d", static_init_value); // API // 动态库 -》 weak引用 if (weak_import_function) { weak_import_function(); } // LGOneObject *one = [LGOneObject new]; // [one testOneObject]; // NSLog(@"%@", one); return 0; } // 动态库 // 全局符号 -》 导出符号 -〉strip 动态库 不是全局符号的符号 // App -> 本地 + 全局 = 间接符号表中的符号 // 静态库 = .o 合集 + 重定位符号表 = .o 合集 调试符号 // tdb // App -》静态库 -〉 App -》strip -〉 间接符号表 // App -》动态库 -〉 App -》 间接符号表 strip -〉 动态库 // dead strip
重定位符号
objdump --macho --reloc test.o
16189259162526.jpg
查看符号表
// 查看符号表 // objdump --macho --syms ${MACH_PATH}
16189266407232.jpg
- OTHER_LDFLAGS=$(inherited) -Xlinker -S 去除调试符号
-S 去除调试符号 DWARF调试文件 -> __DWARF 段 / .o 链接 -> DWARF -> 符号表 -> exec
- 没有static开头的 -> 全局符号
- static -> 本地符号
#include "MachOAndSymbol.RunCMD.xcconfig" // -S 去除调试符号 // DWARF调试文件 -> __DWARF 段 / .o // 链接 -》 DWARF -〉 符号表 -》exec // -map OTHER_LDFLAGS=$(inherited) -Xlinker -S -Xlinker -map -Xlinker /Users/ws/Desktop/VIP课程/第一节、符号与链接(下)/上课代码/Symbol.text OTHER_LDFLAGS=$(inherited) -Xlinker -unexported_symbol -Xlinker _OBJC_CLASS_$_LGOneObject OTHER_LDFLAGS=$(inherited) -Xlinker -unexported_symbol -Xlinker _OBJC_METACLASS_$_LGOneObject OTHER_LDFLAGS=$(inherited) -Xlinker -U -Xlinker _weak_import_function // 间接符号 符号 别名 OTHER_LDFLAGS=$(inherited) -Xlinker -alias -Xlinker _NSLog -Xlinker Cat_NSLog
隐藏全局符号
第一种
// visibility属性,控制文件导出符号,限制符号可见性 /** -fvisibility:clang参数 default:用它定义的符号将被导出。 hidden:用它定义的符号将不被导出。 */ // 隐藏 -> 本地 int hidden_y __attribute__((visibility("hidden"))) = 99; // 符号 double default_y __attribute__((visibility("default"))) = 100; //double protected_y __attribute__((visibility("protected"))) = 120;
第二种
double default_x __attribute__((visibility("hidden")));
attribute -> 最常用于标记一个方法已经废弃
二级命名空间 two_levelnamespace & flat_namespace
当全局符号声明的方法冲突时,运行并没有崩溃 -> 二级命名空间
二级命名空间与一级命名空间. 链接器默认采用二级命名空间, 也就是除了会记录符号名称, 还会记录符号属于哪个Mach-O的, 比如会记录下来_NSLog来自Foundtion.
符号的种类与作用
导入符号&导出符号
• 导入了NSLog • Foundation 导出了 NSLog (全局符号) • 动态库 -> 提供符号 • 间接符号表 -> 保存了动态库符号 • 全局符号 -> 可以变成导出符号 -> strip 动态库不是全局符号的符号
// 查看导出符号 // objdump --macho --exports-trie ${MACH_PATH} // 查看间接符号表 // objdump --macho --indirect-symbols ${MACH_PATH}
OC 代码默认是导出符号(就算是私有方法)
- oc动态库 -> 体积变小 -> 隐藏OC符号
//隐藏OC符号 OTHER_LDFLAGS=$(inherited) -Xlinker -unexported_symbol -Xlinker _OBJC_CLASS_$_LGOneObject OTHER_LDFLAGS=$(inherited) -Xlinker -unexported_symbol -Xlinker _OBJC_METACLASS_$_LGOneObject //查看当前项目符号情况&以及连接的外部符号的情况 OTHER_LDFLAGS=$(inherited) -Xlinker -S -Xlinker -map -Xlinker /Users/ws/Desktop/VIP课程/第一节、符号与链接(下)/上课代码/Symbol.text
弱引用符号
Weak Symbol:
- Weak Reference Symbol: 表示此未定义符号是弱引用. 如果动态链接器找不到该符号的定义, 则将其置为0. 链接器会将此符号设置弱链接标志.
- Weak defintion Symbol: 表示此符号为弱定义符号. 如果静态链接器或动态链接器为此符号找到另一个(非弱)定义, 则弱定义将被忽略. 只能将合并部分中的符号标记为弱定义.
// weak def -> 导出符号 // weak def -> 解决全局符号冲突 // 定义为弱定义并不会影响导出 void weak_function(void) __attribute__((weak)); // weak 本地符号 void weak_hidden_function(void) __attribute__((weak, visibility("hidden"))); // 弱引用 // 本地其他文件使用还是需要导入头文件 void weak_import_function(void) __attribute__((weak_import)); //也可以通过 -U 声明一个符号是未定义, 需要链接时动态查找 OTHER_LDFLAGS=$(inherited) -Xlinker -U -Xlinker _weak_import_function // API // 可以将动态库 -> 整个声明成weak引用
重新导出符号
// 间接符号 符号 别名 OTHER_LDFLAGS=$(inherited) -Xlinker -alias -Xlinker _NSLog -Xlinker Cat_NSLog //查看符号表命令 CMD = nm -m ${MACH_PATH} | grep 'Cat'
可以让一个隐藏的动态库, 对另一个可见
Swift符号
记得添加Swift文件并编译 -> 编译型语言,不是动态的
strip命令与MachO体积
移出或者修改符号表中的符号 代码优化
- 动态库
- 全局符号 -> 导出符号 -> strip 动态库 不是全局符号的符号
- App
- 本地 + 全局 = 间接符号表中的符号
- 静态库
- .o 合集 + 重定位符号表 = .o 合集 调试符号
// App -> 静态库 -> App符号表 -> strip -> 间接符号表 // App -> 动态库 -> App -> 间接符号表 strip -> 动态库 // dead strip
要看PPT整理strip
Strip Style:
1. Debugging Symbols (.o静态库/可执行文件 动态库)
2. All Symbols
3. Non-Global Symbols
Strip
1. -x: non_global
2. 无参数: 代表全部符号
3. -S:调试符号
16189320434439.jpg
16189320689828.jpg
16189320862983.jpg
!
16189321250766.jpg
strip的源码调试
下面是探索strip的源码过程, 可以根据上面的图片, 来进行相应的探索.
• target 里搜 strip -> llvm-strip • 查看脚本, 原理如果是Debug模式, 那么llvm-objcopy 进行符号链接生成 -> llvm-strip • llvm-objcopy -> 右键替身 -> 命名为strip • 如果名字为strip -> strip • 如果名字为llvm-objcopy -> llvm-objcopy • 去编译文件里面找llvm-objcopy文件, 显示到源文件, Mian函数打断点 • 去scheme添加strip • product -> perform Action -> Run Without Building • 断到断点之后 -> 控制台输入br read -f [strip_lldb路径] • br read -f [strip_lldb路径]从一个文件中, 把断点信息读进来 • br list strip -> 断点默认没有启用 -> br enable strip • scheme -> argments -> 添加可执行文件路径 以及参数 -> 去调试 • -x: non_global • 无参数: 代表全部符号 • -S:调试符号
扩展:
- br write -f [把断点写入一个文件]