美文网首页源码解析iOS开发iOS
最新Runtime源码objc4-750编译

最新Runtime源码objc4-750编译

作者: 逆风g | 来源:发表于2018-12-17 21:43 被阅读17次

    最新版本的Runtime源码已经出来了,是不急不可耐的想用用它呢?在这里我将一步步教大家如何编译它,首先贴个自己的环境配置:

    • mac OS 10.14
    • Xcode 10.1
    • objc4-750

    首先给出我已编译好的objc4-750地址,可以直接使用。

    Runtime源码地址

    • 苹果开源网站上可以下载到很多开源项目,可以看到当前最新mac OS系统为10.14.1,最新的Xcode版本为10,安装Xcode 10.0提示mac OS系统需要10.13.6:
    • 通常所说的Runtime源码就是objc4文件,由于iOS中开源项目非常少,所以选择最新mac OS系统10.14.1,command+f键在浏览器页面右上角输入objc4:
    • 可以看到最新的objc4文件为objc4-750.1,点击右边的下载按钮可以下载压缩包:
    • 也可以点击objc4-750.1,可以看到包里的具体内容,用Xcode可以打开它:


    • 此时显示的网页地址为https://opensource.apple.com/source/objc4/objc4-750.1/
    • 去掉最后的objc4-750.1/路径,进入网页地址https://opensource.apple.com/source/objc4/,可以看到以往objc4历史版本:
    • 其中source替换成tarballs,http://opensource.apple.com/tarballs/objc4/,就可以下载自己想要的objc4版本:

    Runtime源码编译

    下载好源码之后用Xcode打开是这个样子:


    这里的libobjc.A.dylib就是我们要编译的目标-Runtime库,编译好之后自己可以再添加一个Target用于测试里面的Runtime源码,但是现在编译会报错,大部分错误是缺少头文件,这些头文件都在苹果开源的其它项目里。接下来依依解决这些问题:
    1. 准备工作
      进入苹果开源网站,下载依赖的开源项目:
    • Libc-825.40.1.tar.gz
    • dyld-551.3.tar.gz
    • libauto-187.tar.gz
    • libclosure-73.tar.gz
    • libdispatch-1008.220.2.tar.gz
    • xnu-4903.221.2.tar.gz
    • libpthread-330.220.2.tar.gz
    • launchd-842.92.1.tar.gz
    • libplatform-177.200.16.tar.gz
      把他们下载好并解压之后放入同一个文件夹中,方便查找。
    1. 提示'sys/reason.h' file not found
      在当前项目下创建一个文件夹Common,里面用于存放所有缺失的头问题件:

      并且把它添加到项目的Header Search Paths中,依次选择objc->TARGETS->objc->Build Settings,搜索框中输入header search path,然后加入$(SRCROOT)/Common

    接下来需要去已下载好的开源项目中寻找reason.h头文件了,方式有两种:

    • 使用命令行:
      进入目录cd /Users/gcf/Desktop/OpenSource
      搜索文件名find . -name ‘reason.h’
      image.png
      可以看到搜索结果显示在./xnu-4903.221.2/bsd/sys/reason.h中,按照这个路径找到reason.h文件,根据编译错误提示知道,这个reason.h文件在路径sys下,那么在已创建的Common文件下创建一个新的sys文件夹,里面放入找到的reason.h文件:

    • 普通搜索
      直接在Opensource中搜索reason.h文件:

      接下来处理和上述一样。
    1. 再次编译,提示'mach-o/dyld_priv.h' file not found
      选择./dyld-551.3/include/mach-o/dyld_priv.h,和上述同样操作,不再重述。
    2. 提示'os/lock_private.h' file not found
      选择./libplatform-177.200.16/private/os/lock_private.h
    3. 提示'os/base_private.h' file not found
      选择./libplatform-177.200.16/private/os/base_private.h
    4. 提示'pthread/tsd_private.h' file not found
      选择./libpthread-330.220.2/private/tsd_private.h
    5. 提示'System/machine/cpu_capabilities.h' file not found
      选择./xnu-4903.221.2/osfmk/machine/cpu_capabilities.h
    6. 提示'os/tsd.h' file not found
      选择./xnu-4903.221.2/libsyscall/os/tsd.h
    7. 提示'pthread/spinlock_private.h' file not found
      选择./libpthread-330.220.2/private/spinlock_private.h
    8. 提示'System/pthread_machdep.h' file not found
      选择./Libc-825.40.1 2/pthreads/pthread_machdep.h
    9. 提示Typedef redefinition with different types ('int' vs 'volatile OSSpinLock' (aka 'volatile int’))
      这种redefinition错误时,在include文件夹下使用grep命令:
    // 如 重复定义 pthread_lock_t
    grep -rne "typedef.*pthread_lock_t” .
    // 输出
    ./pthread/spinlock_private.h:59:typedef volatile OSSpinLock pthread_lock_t __deprecated_msg("Use <os/lock.h> instead”);
    ./System/pthread_machdep.h:214:typedef int pthread_lock_t;
    

    可以看见有两处定义了pthread_lock_t,注释掉pthread_machdep.h文件中的定义即可。

    1. 提示Static declaration of '_pthread_getspecific_direct' follows non-static declaration
      这里有三个函数定义重复了:
    • _pthread_has_direct_tsd(void)
    • _pthread_getspecific_direct(unsigned long slot)
    • _pthread_setspecific_direct(unsigned long slot, void * val)
    grep -re "_pthread_has_direct_tsd(void)” .
    //输出
    ./pthread/tsd_private.h:_pthread_has_direct_tsd(void)
    ./System/pthread_machdep.h:_pthread_has_direct_tsd(void)
     grep -re "_pthread_getspecific_direct(unsigned long slot)” .
    //输出
    ./pthread/tsd_private.h:_pthread_getspecific_direct(unsigned long slot)
    ./System/pthread_machdep.h:_pthread_getspecific_direct(unsigned long slot)
    grep -re "_pthread_setspecific_direct(unsigned long slot, void \* val)” .
    //输出
    ./pthread/tsd_private.h:_pthread_setspecific_direct(unsigned long slot, void * val)
    ./System/pthread_machdep.h:_pthread_setspecific_direct(unsigned long slot, void * val)
    

    这里选择把pthread_machdep.h文件中的定义注释掉。

    1. 提示'CrashReporterClient.h' file not found
      选择./Libc-825.40.1 2/include/CrashReporterClient.h,放入Common文件夹下之后还是报错,需要在Build Settings->Preprocessor Macros中加入:LIBC_NO_LIBCRASHREPORTERCLIENT
    2. 提示'Block_private.h' file not found
      选择./libdispatch-1008.220.2/src/BlocksRuntime/Block_private.h
    3. 提示'objc-shared-cache.h' file not found
      选择./dyld-551.3/include/objc-shared-cache.h
    4. 提示Use of undeclared identifier ‘DYLD_MACOSX_VERSION_10_13
      在 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
    
    1. 提示'_simple.h' file not found
      选择./libplatform-177.200.16/private/_simple.h
    2. 提示'isa.h' file not found
      isa.h文件在项目的runtime文件夹中,新加入的一个头文件:

      我们把它引入Commone文件夹中去:
    3. 提示can't open order file: /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX10.14.sdk/AppleInternal/OrderFiles/libobjc.order
      修改工程配置,将Build Settings->Linking->Order File改为工程根目录下的libobjc.order,即:$(SRCROOT)/libobjc.order。
    4. 提示library not found for -lCrashReporterClient
      此时在 Build Settings -> Linking -> Other Linker Flags里删掉"-lCrashReporterClient"(Debug和Release都删了)
    5. 提示SDK "macosx.internal" cannot be located.unable to find utility "clang++", not a developer tool or in PATH
      把Target-objcBuild Phases->Run Script(markgc)里的内容macosx.internal改为macosx,这里我猜测macosx.internal为苹果内部的macosx,说的不对,大家指出来。
    6. 提示no such public header file: '/tmp/objc.dst/usr/include/objc/ObjectiveC.apinotes’
      这里需要把Target-objcBuild Settings->Other Text-Based InstallAPI Flags里的内容设为!

      并且一定记得要把Text-Based InstallAPI Verification Model里的值改为Errors Only

    相关警告

    1. 警告Traditional headermap style is no longer supported; please migrate to using separate headermaps and set 'ALWAYS_SEARCH_USER_PATHS' to NO. (in target 'objc-trampolines')Traditional headermap style is no longer supported; please migrate to using separate headermaps and set 'ALWAYS_SEARCH_USER_PATHS' to NO. (in target 'objc’)
      在项目Target->objc-trampolinesobjc中的Build Settings下设置ALWAYS_SEARCH_USER_PATHSNo
    2. 警告'_PTHREAD_TSD_SLOT_PTHREAD_SELF' macro redefined
      pthread_machdep.h头文件中共有四个宏定义重复了:
    • _PTHREAD_TSD_SLOT_PTHREAD_SELF
    • __PTK_LIBC_TTYNAME_KEY
    • LOCK_INIT
    • LOCK_INITIALIZER
      这里选择把pthread_machdep.h文件中的宏定义注释掉。
    1. 警告objc-exception.mm:584:5: Code will never be executed
      把不会执行到的代码__builtin_trap();注释掉
    2. 警告objc-class.mm:558:33: Possible misuse of comma operator here
      使用Xcode提示的Fix修复
    3. 还有一些Fixme...之类的警告,是苹果在自己代码里定义的一些警告提示,就不处理了。

    添加Debug Target

    1. 添加一个target 取名为 objc-test


    2. 为改target添加工程依赖


    3. 在objc-test中添加测试代码
    #import <Foundation/Foundation.h>
    #import <objc/runtime.h>
    #import <objc/message.h>
    int main(int argc, const char * argv[]) {
        @autoreleasepool {
            Class newClass = objc_allocateClassPair(objc_getClass("NSObject"), "newClass", 0);
                    objc_registerClassPair(newClass);
            id newObject = [[newClass alloc]init];
            NSLog(@"%@",newObject);
        }
        return 0;
    }
    

    参考文章:

    总结

    1. 所有头文件


    2. 所有其它开源项目


    3. 推荐给技巧,从别人博客中看到的:
      当缺少头文件时,不知道在哪个开源项目中,比如缺少CrashReporterClient.h,那么在谷歌中输入CrashReporterClient.h site:opensource.apple.com,搜索结果:
      看搜索结果的链接,很容易可知道它来自Libc项目。

    相关文章

      网友评论

        本文标题:最新Runtime源码objc4-750编译

        本文链接:https://www.haomeiwen.com/subject/ckmooftx.html