iOS底层系列01-- objc4-781源码的编译与调试

  • 本文主要是通过将objc4-781的源码编译成功,然后在源码中新建一个YYTest的Target,然后将其与objc4-781工程源码进行绑定,最后运行YYTest进行底层源码的调试;

工具环境

objc4-781源码运行所需要的依赖文件

编译objc4-781源码工程所遇到的问题及解决方案

问题一: unable to find sdk 'macosx.internal

Snip20210201_4.png
  • 解决方案:选择target -> objc -> Build Settings -> Base SDK -> 选择 macOS【target中的 objc 和 obc-trampolines都需要更改】
Snip20210201_3.png

问题二: 'sys/reason.h' file not found

Snip20210201_6.png
  • 解决方案: 在objc4-781的根目录下新建YYCommon文件夹, 同时在YYCommon文件中创建sys文件夹;
Snip20210202_31.png
  • 将依赖文件xnu-6153.11.26下的reason.h(路径为:xnu-6153.11.26/bsd/sys/reason.h)拷贝到新建YYCommon/sys文件夹中;

  • 设置文件检索路径:选择 target -> objc -> Build Settings,在工程的 Header Serach Paths 中添加搜索路径 $(SRCROOT)/YYCommon

    Snip20210202_32.png

问题三: mach-o/dyld_priv.h' file not found

Snip20210201_9.png
  • 解决方案:在objc4-781的根目录下的YYCommon文件夹下新建mach-o文件夹;
  • 将依赖文件dyld-733.6下的dyld_priv.h(路径为:dyld-733.6/include/mach-o/dyld_priv.h)拷贝到新建YYCommon/mach-o文件夹中;
  • 拷贝到文件后,还需要修改 dyld_priv.h 文件,即在 dyld_priv.h文件顶部加入一下宏:
#define DYLD_MACOSX_VERSION_10_11 0x000A0B00
#define DYLD_MACOSX_VERSION_10_12 0x000A0C00
#define DYLD_MACOSX_VERSION_10_13 0x000A0D00
#define DYLD_MACOSX_VERSION_10_14 0x000A0E00
  • 修改后的dyld_priv.h文件;
Snip20210202_33.png
  • 若dyld_priv.h文件出现bridgeos(3.0)报错,则将参数bridgeos(3.0)去除,去除之后如下所示;
Snip20210202_34.png

问题四: os/lock_private.h' file not found

Snip20210201_10.png
  • 解决方案:在objc4-781的根目录下的YYCommon文件夹下新建os文件夹;
  • 将依赖文件libplatform-220下的lock_private.h和base_private.h(路径为:libplatform-220/private/os/lock_private.h,base_private.h)拷贝到新建YYCommon/os文件夹中;

问题五: 'pthread/tsd_private.h' file not found

Snip20210201_11.png
  • 解决方案:在objc4-781的根目录下的YYCommon文件夹下新建pthread文件夹;
  • 将依赖文件libpthread-416.11.1下的tsd_private.hspinlock_private.h(路径为:libpthread-416.11.1/private/tsd_private.h,spinlock_private.h)拷贝到新建YYCommon/pthread文件夹中;

问题六: System/machine/cpu_capabilities.h' file not found

Snip20210201_12.png
  • 解决方案: 在objc4-781的根目录下的YYCommon文件夹下新建System/machine文件夹;
  • 将依赖文件xnu6153.11.26下的cpu_capabilities.h(路径为:xnu6153.11.26/osfmk/machine/cpu_capabilities.h)拷贝到新建YYCommon/System/machine文件夹中;

问题七: 'os/tsd.h' file not found

Snip20210201_13.png
  • 解决方案:将依赖文件xnu6153.11.26下的tsd.h(路径为:xnu6153.11.26/libsyscall/os/tsd.h)拷贝到新建YYCommon/os文件夹中;

问题八: 'System/pthread_machdep.h' file not found

Snip20210201_14.png
  • 解决方案: 将依赖文件Libc-583下的pthread_machdep.h(路径为:Libc-583/pthreads/pthread_machdep.h)拷贝到新建YYCommon/System文件夹中;

问题九: 'CrashReporterClient.h' file not found

Snip20210202_15.png
  • 解决方案:将依赖文件Libc-825.24下的CrashReporterClient.h(路径为:Libc-825.24/include/CrashReporterClient.h)拷贝到新建YYCommon文件夹中;
  • 工程配置在 Build Settings -> Preprocessor Macros中加入:
    LIBC_NO_LIBCRASHREPORTERCLIENT
    Snip20210202_35.png
  • 如果还是报错CrashReporterClient 的问题,解决方法: 在BuildSetting --> Other Linker Flags中去掉CrashReporterClient

问题十: 'objc-shared-cache.h' file not found

Snip20210202_16.png
  • 解决方案:将依赖文件dyld-733.6下的objc-shared-cache.h(路径为:dyld-733.6/include/objc-shared-cache.h)拷贝到新建YYCommon文件夹中;

问题十一: 'kern/restartable.h' file not found

Snip20210202_17.png
  • 解决方案:在objc4-781的根目录下的YYCommon文件夹下新建kern文件夹;
  • 将依赖文件xnu6153.11.26下的restartable.h(路径为:xnu6153.11.26/osfmk/kern/restartable.h)拷贝到新建YYCommon/kern文件夹中;

问题十二: Mismatch in debug-ness macros

Snip20210202_18.png
  • 解决方案:注释掉objc-runtime.mm中的#error mismatch in debug-ness macros;

问题十三: '_simple.h' file not found

Snip20210202_19.png
  • 解决方案:将依赖文件libplatform-220下的_simple.h(路径为:libplatform-220/include/_simple.h)拷贝到新建YYCommon文件夹中;

问题十四: ''Block_private.h' file not found

Snip20210202_20.png
  • 解决方案:将依赖文件libclosure-74下的Block_private.h(路径为:libclosure-74/Block_private.h)拷贝到新建YYCommon文件夹中;

问题十五: 'can't open order file: /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX10.15.sdk/AppleInternal/OrderFiles/libobjc.order

Snip20210202_21.png
  • 解决方案:在工程中配置: target -> objc -> Build Settings -> Order File中添加路径$(SRCROOT)/libobjc.order
    Snip20210202_36.png

问题十六: library not found for -lCrashReporterClient

Snip20210202_22.png
  • 解决方案:在工程中配置: target -> objc -> Build Settings -> Other Linker Flags Debug与Release模式下的Any macOS SDK 找到lCrashReporterClient删除掉;
Snip20210202_37.png

问题十七: /xcodebuild:1:1: SDK "macosx.internal" cannot be located

解决方案:

  • 在工程中配置: targets -> Build Phases -> Run Script中的macosx.internal改成macosx
Snip20210202_38.png
  • 上面的所有问题解决完,objc4-781源码工程就能编译成功了,完整的工程文件如下:
Snip20210202_41.png

在objc4-781源码工程中新建Target 命名为YYTest,然后配置最后可以进行源码调试了;

  • 新建Target 命名为YYTest;
Snip20210202_23.png
Snip20210202_24.png
Snip20210202_25.png
  • YYTest的工程配置

  • 1.给YYTest绑定依赖关系,YYTest -> Build Phases -> Dependencies 添加objc

  • 2.给YYTest添加库,YYTest -> Build Phases -> Link Binary With Libraries添加libobjc.A.dylib

    Snip20210202_27.png

  • YYTest中的断点调试设置

    1. 打断点调试发现断点无效;


      Snip20210202_39.png
  • 解决方案:YYTest -> Build Phases -> Compile Sources中 将main.m拖到最上面;

Snip20210202_28.png

2.源码中的断点无效;

Snip20210202_40.png
  • 解决方案:Build Setting --> 将Enable Hardened Runtime 置为NO
Snip20210202_30.png

推荐阅读更多精彩内容