符号与链接 (2)

符号与链接 (2)

遇到的问题

    1. 关于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"
    1. 关于多Target, 每个Target可以指定编译的文件
    1. 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. 汇编
  2. 符号归类 -> 重定位符号表 符号表
    1. 重定位符号表 -> 放置的是.m/.o用到的API
  3. .o文件 -> 链接 -> 符号表合并到一张表中 -> 可执行文件(exec)
    1. 链接过程 -> 就是处理目标文件符号的过程
#import 
#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

16189259162526.jpg

查看符号表

// 查看符号表
// objdump --macho --syms ${MACH_PATH}
16189266407232.jpg
  1. OTHER_LDFLAGS=$(inherited) -Xlinker -S 去除调试符号
    1. -S 去除调试符号
    2. DWARF调试文件 -> __DWARF 段 / .o
    3. 链接 -> DWARF -> 符号表 -> exec
  2. 没有static开头的 -> 全局符号
  3. 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

!

16189321050427.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 [把断点写入一个文件]

strip_lldb

你可能感兴趣的:(符号与链接 (2))