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可编译源码

你可能感兴趣的:(OC底层原理-objc818源码编译)