项目里面用到了 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 命令代替。
最后
笔者写了一个 脚本,包含编译,提取头文件,合并架构来完成这整个流程。欢迎使用~