OC底层原理-objc818源码编译
objc818编译
一、准备工作
Xcode:Version 12.3
macOS:Version 10.15.7
objc源码:objc4-818.2.tar.gz
二、开始配置
使用Xcode打开解压后的objc源码。直接编译objc,此时需要修改Xcode提示的错误。
-
【错误1】
描述:unable to find sdk 'macosx.internal'。
解决办法:- 设置 “TARGETS->objc->Build Setting->Base SDK”,将其值设置为“macOS”。
- 设置 “TARGETS->objc-trampolines->Build Setting->Base SDK”,将其值设置为“macOS”。
-
【错误2】
描述:“'sys/reason.h' file not found”
解决办法:- 在Apple Source中找到“xnu-7195.50.7.100.1 -> bsd -> sys -> reason.h”文件并下载。
- 在项目中添加自定义文件夹,用于存放该文件。
- 在“TARGETS -> objc -> Build Setting -> Heard Search Path"中添加自定义文件夹的路径。
-
【错误3】
描述:“dyld_priv.h:160:148: Expected ','”
解决办法:将“bridgeos(3.0)”去掉。如下:
//将含有bridgeos(3.0)的代码注释
//extern dyld_platform_t dyld_get_base_platform(dyld_platform_t platform) __API_AVAILABLE(macos(10.14), iOS(12.0), watchos(5.0), tvos(12.0), bridgeos(3.0));
//修改后的代码
extern dyld_platform_t dyld_get_base_platform(dyld_platform_t platform) __API_AVAILABLE(macos(10.14), iOS(12.0), watchos(5.0), tvos(12.0));
-
【错误4】
描述:“'mach-o/dyld_priv.h' file not found“
解决办法:- 在Apple Source中找到“dyld-832.7.1 -> include -> mach-o -> dyld_priv.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误5】
描述:“'os/lock_private.h'/'os/base_private.h' file not found”
解决办法:- 在Apple Source中找到“libplatform-254.40.4 --> private --> os --> lock_private.h 、base_private.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误6】
描述:“'pthread/tsd_private.h'/'pthread/spinlock_private.h' file not found”
解决办法:- 在Apple Source中找到“libplatform-254.40.4 --> private --> tsd_private.h、spinlock_private.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误7】
描述:“'System/machine/cpu_capabilities.h' file not found”
解决办法:- 在Apple Source中找到“libplatform-254.40.4 --> private --> tsd_private.h、spinlock_private.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误8】
描述:“'os/tsd.h' file not found”
解决办法:- 在Apple Source中找到“xnu-7195.50.7.100.1 --> libsyscall --> os --> tsd.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误9】
描述:“'System/pthread_machdep.h' file not found”
解决办法:- 在Apple Source中找到“Libc-583/pthreads/pthread_machdep.h”文件并下载。
- 注意:在最新版的libc中没有这个文件,需要在“Libc-583”中下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误10】
描述:“'CrashReporterClient.h' file not found”
解决办法:- 在Apple Source中找到“Libc-825.24/include/CrashReporterClient.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
- 注意:该文件添加之后仍然报错,主要是因为宏“LIBC_NO_LIBCRASHREPORTERCLIENT”示定义的原因。
- 解决1:在"Build Settings -> Preprocessor Macros"中添加“LIBC_NO_LIBCRASHREPORTERCLIENT”宏定义。
- 解决2:在“CrashReporterClient.h”文件起始处,加入“#define LIBC_NO_LIBCRASHREPORTERCLIENT”。
-
【错误11】
描述:"'os/feature_private.h' file not found"
解决办法:
直接将该引用的头文件注释。 -
【错误12】
描述:“'objc-shared-cache.h' file not found”
解决办法:- 在Apple Source中找到“dyld-832.7.1 --> include --> objc-shared-cache.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
-
【错误13】
描述:“'objc-bp-assist.h' file not found”
解决办法:
直接将该引用的头文件注释。 -
【错误14】
描述:“Use of undeclared identifier 'dyld_platform_version_macOS_10_13'”
解决办法:
直接定位到该段代码,并注释掉。如下所示
// if (!dyld_program_sdk_at_least(dyld_platform_version_macOS_10_13)) {
// DisableInitializeForkSafety = true;
// if (PrintInitializing) {
// _objc_inform("INITIALIZE: disabling +initialize fork "
// "safety enforcement because the app is "
// "too old.)");
// }
// }
- 【错误15】
描述:“'_simple.h' file not found”
解决办法:- 在Apple Source中找到“libplatform-254.40.4 --> private --> _simple.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
- 【错误16】
描述:“'kern/restartable.h' file not found”
解决办法:- 在Apple Source中找到“xnu-7195.50.7.100.1 --> osfmk --> kern -->restartable.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
- 【错误17】
描述:“'Block_private.h' file not found”
解决办法:- 在Apple Source中找到“libclosure-78 --> Block_private.h”文件并下载。
- 将该文件拷贝至项目中的自定义文件夹中。
- 【错误18】
描述:"objc-cache.mm:1122:33: Use of undeclared identifier 'objc_thread_get_rip'"
解决办法:
将代码直接注释,如下所示
//#if TARGET_OS_OSX
// if (oah_is_current_process_translated()) {
// kern_return_t ret = objc_thread_get_rip(threads[count], (uint64_t*)&pc);
// if (ret != KERN_SUCCESS) {
// pc = PC_SENTINEL;
// }
// } else {
// pc = _get_pc_for_thread (threads[count]);
// }
//#else
pc = _get_pc_for_thread (threads[count]);
//#endif
- 【错误19】
描述:"Can't open order file: /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.1.sdk/AppleInternal/OrderFiles/libobjc.order"
解决办法:- 找到“TARGETS -> objc -> Build Setting -> Linking -> Order File”,修改成“$(SRCROOT)/libobjc.order”
- 【错误20】
描述:“Library not found for -lCrashReporterClient”
解决办法:- 找到“TARGETS -> objc -> Build Setting -> Other Link Flags”,删除“-lCrashReporterClient”。
- 【错误21】
描述:“Library not found for -loah”
解决办法:- 找到“TARGETS -> objc -> Build Setting -> Other Link Flags”,删除“-loah”。
- 【错误22】
描述:“ObjectiveC.apinotes 异常”
解决办法:- 选择“TARGETS -> objc -> Build Settings -> Text-Based InstallAPI Verification Model”中选择“Errors Only”。
- 清除“Other Text-Based InstallAPI Flags”中的所有内容。
至此,已经能完全编译成功了。
三、开始调试
-
新建一个 Target : HQObjc
新建项目 -
添加绑定关系
添加绑定关系 -
此时运行代码,就能进入到源码中了。
image.png
注意:如果经过1、2步无法断点到源码中,需要完成以下两步:
* 找到“HQObjc -> Build Setting -> Enable Hardended Runtime”,将其设置为“NO”。
* 确保编译的源文件中,“main.m”处于第一位。
最终编译通过的源码路径为:objc4-818可编译源码