美文网首页
WebRTC Native 模块单独编译静态库(iOS)

WebRTC Native 模块单独编译静态库(iOS)

作者: Nemocdz | 来源:发表于2019-12-31 16:37 被阅读0次

    项目里面用到了 WebRTC 某些模块,以前是源码拷贝集成的。这样不仅需要处理平台的宏(WebRTC Native 跨平台),而且升级版本也会涉及依赖等问题,所以探索了一下怎么用编码编译出可用的静态库。

    网上相关的资料很少,且大部分都是 Android 平台编译单独的 apm(Audio Processing Module)进行使用,但我想找到更通用的方法。

    WebRTC 编译流程

    要编译单独的模块,首先要了解完整的 WebRTC 源码是如何编译的。根据 官网教程,WebRTC 项目出自于 Chromium,所以使用的工具和流程基本都一致。

    • 安装 Chromium depot tools,这是 Chromium 特有的工具集,包含 git 管理,编译工具等
    • 使用该工具下载源码(Android 依赖 Linux 环境)
    • 使用 GN 命令生成编译配置
    • 使用 ninja 进行编译

    Ninja

    Ninja 是一个小型轻量的 C++ 构建系统,使用 build.ninja (类似 Makefile)来进行编译。而一般需要用工具来生成这个描述文件,而不是手动编写。而官方采用的就是 GN(Generate Ninja)工具。

    GN

    GN 是专门用于创建 Ninja 描述文件的工具。可以对项目多个模块分别编写 BUILD.gn 文件,对模块描述依赖,编译文件,头文件,可见性,命名空间等。算是一种 DSL,用于描述模块,和 Cocoapods 里的 Podspec 文件作用差不多。

    单独编译

    虽然可以进行整体编译了,但是单独编译有没有办法呢?先查看 Ninja 官方文档 发现 ninja 命令是可以带上编译目标的。编译目标的名字,是用 GN 生成的。查看 BUILD.gn 文件,发现基本都有 rtc_static_library。这个模版被定义在 webrtc.gni 文件里。

    template("rtc_static_library") {
      static_library(target_name) {
        forward_variables_from(invoker,
                               "*",
                               [
                                 "configs",
                                 "public_configs",
                                 "suppressed_configs",
                                 "visibility",
                               ])
        forward_variables_from(invoker, [ "visibility" ])
            // ...
      }
    }
    

    看来这个就是要找的 target_name。比如需要单独编译 modules/audio_processing 模块,查看 modules/audio_processing/BUILD.gn,发现里面定义了

    rtc_static_library("audio_processing")
    

    尝试使用在 gn 执行后使用

    ninja -C out/Default audio_processing
    

    可以看到,在 out/Default/obj/modules/audio_processing 目录下生成了 libaudio_processing.a。

    但这样编译出来的静态库大小不大,怀疑是没有包含依赖的编译产物的。使用命令 nm 查看

    nm -u libaudio_processing.a
    

    发现有未定义的符号,联想 ninja 是个轻量构建系统,感觉很合理。只能再翻翻文档,最后在 GN 文档 中发现,BUILD.gn 定义了 complete_static_lib 参数,可以用来设置在编译时是否链接依赖产物。在 BUILD.gn 中 rtc_static_library 中加上 omplete_static_lib = true,再次尝试,发现正常了。

    提取头文件

    WebRTC 的头文件有文件层次结构,C++ 依赖头文件的路径,所以按照原始文件目录层次提出头文件就是必要的。Linux 系统可以是用 cp --parents 来进行保留文件目录的拷贝,但 Mac 上的 cp 命令并不支持 --parents 参数,只有 ditto 命令代替。

    最后

    笔者写了一个 脚本,包含编译,提取头文件,合并架构来完成这整个流程。欢迎使用~

    参考链接

    相关文章

      网友评论

          本文标题:WebRTC Native 模块单独编译静态库(iOS)

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