符号与链接 (2)
遇到的问题
- 关于xcconfig
xconfig编写指南
xconfig编写指南
- 关于xcconfig
//绝对+相对
#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文件格式
MachO,前面是配置,后面是代码,通过配置来找到代码
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
代码 -> .0的过程
- 汇编
- 符号归类 -> 重定位符号表 符号表
- 重定位符号表 -> 放置的是.m/.o用到的API
- .o文件 -> 链接 -> 符号表合并到一张表中 -> 可执行文件(exec)
- 链接过程 -> 就是处理目标文件符号的过程
#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
test
查看符号表
// 查看符号表
// objdump --macho --syms ${MACH_PATH}
- 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:调试符号
!
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 [把断点写入一个文件]