使用 clang 编译 FFmpeg
一、 准备工作
- 下载 FFmpeg、 最新版 ndk(上篇文章已经提及)
- 整理下文件(当然用你喜欢的就行, 只要配置的路径对就没问题)
二、 明确使用 target-os=android, 使用 clang 进行编译
2.1 明确 clang 编译环境的位置
# 在相应 ndk 的这个目录下
ndk/toolchains/llvm/prebuilt/darwin-x86_64
让我们看看 clang 等编译工具的内容(bin目录)
aarch64-linux-android-addr2line i686-linux-android-gprof
aarch64-linux-android-ar i686-linux-android-ld
aarch64-linux-android-as i686-linux-android-ld.bfd
aarch64-linux-android-c++filt i686-linux-android-ld.gold
aarch64-linux-android-dwp i686-linux-android-nm
aarch64-linux-android-elfedit i686-linux-android-objcopy
aarch64-linux-android-gprof i686-linux-android-objdump
aarch64-linux-android-ld i686-linux-android-ranlib
aarch64-linux-android-ld.bfd i686-linux-android-readelf
aarch64-linux-android-ld.gold i686-linux-android-size
aarch64-linux-android-nm i686-linux-android-strings
aarch64-linux-android-objcopy i686-linux-android-strip
aarch64-linux-android-objdump i686-linux-android16-clang
aarch64-linux-android-ranlib i686-linux-android16-clang++
aarch64-linux-android-readelf i686-linux-android17-clang
aarch64-linux-android-size i686-linux-android17-clang++
aarch64-linux-android-strings i686-linux-android18-clang
aarch64-linux-android-strip i686-linux-android18-clang++
aarch64-linux-android21-clang i686-linux-android19-clang
aarch64-linux-android21-clang++ i686-linux-android19-clang++
aarch64-linux-android22-clang i686-linux-android21-clang
aarch64-linux-android22-clang++ i686-linux-android21-clang++
aarch64-linux-android23-clang i686-linux-android22-clang
aarch64-linux-android23-clang++ i686-linux-android22-clang++
aarch64-linux-android24-clang i686-linux-android23-clang
aarch64-linux-android24-clang++ i686-linux-android23-clang++
aarch64-linux-android26-clang i686-linux-android24-clang
aarch64-linux-android26-clang++ i686-linux-android24-clang++
aarch64-linux-android27-clang i686-linux-android26-clang
aarch64-linux-android27-clang++ i686-linux-android26-clang++
aarch64-linux-android28-clang i686-linux-android27-clang
aarch64-linux-android28-clang++ i686-linux-android27-clang++
aarch64-linux-android29-clang i686-linux-android28-clang
aarch64-linux-android29-clang++ i686-linux-android28-clang++
arm-linux-androideabi-addr2line i686-linux-android29-clang
arm-linux-androideabi-ar i686-linux-android29-clang++
arm-linux-androideabi-as ld.lld
arm-linux-androideabi-c++filt llvm-ar
arm-linux-androideabi-dwp llvm-as
arm-linux-androideabi-elfedit llvm-config
arm-linux-androideabi-gprof llvm-cov
arm-linux-androideabi-ld llvm-dis
arm-linux-androideabi-ld.bfd llvm-link
arm-linux-androideabi-ld.gold llvm-modextract
arm-linux-androideabi-nm llvm-nm
arm-linux-androideabi-objcopy llvm-objcopy
arm-linux-androideabi-objdump llvm-profdata
arm-linux-androideabi-ranlib llvm-readobj
arm-linux-androideabi-readelf llvm-strip
arm-linux-androideabi-size llvm-symbolizer
arm-linux-androideabi-strings pkg-config
arm-linux-androideabi-strip sancov
armv7a-linux-androideabi16-clang sanstats
armv7a-linux-androideabi16-clang++ scan-build
armv7a-linux-androideabi17-clang scan-view
armv7a-linux-androideabi17-clang++ x86_64-linux-android-addr2line
armv7a-linux-androideabi18-clang x86_64-linux-android-ar
armv7a-linux-androideabi18-clang++ x86_64-linux-android-as
armv7a-linux-androideabi19-clang x86_64-linux-android-c++filt
armv7a-linux-androideabi19-clang++ x86_64-linux-android-dwp
armv7a-linux-androideabi21-clang x86_64-linux-android-elfedit
armv7a-linux-androideabi21-clang++ x86_64-linux-android-gprof
armv7a-linux-androideabi22-clang x86_64-linux-android-ld
armv7a-linux-androideabi22-clang++ x86_64-linux-android-ld.bfd
armv7a-linux-androideabi23-clang x86_64-linux-android-ld.gold
armv7a-linux-androideabi23-clang++ x86_64-linux-android-nm
armv7a-linux-androideabi24-clang x86_64-linux-android-objcopy
armv7a-linux-androideabi24-clang++ x86_64-linux-android-objdump
armv7a-linux-androideabi26-clang x86_64-linux-android-ranlib
armv7a-linux-androideabi26-clang++ x86_64-linux-android-readelf
armv7a-linux-androideabi27-clang x86_64-linux-android-size
armv7a-linux-androideabi27-clang++ x86_64-linux-android-strings
armv7a-linux-androideabi28-clang x86_64-linux-android-strip
armv7a-linux-androideabi28-clang++ x86_64-linux-android21-clang
armv7a-linux-androideabi29-clang x86_64-linux-android21-clang++
armv7a-linux-androideabi29-clang++ x86_64-linux-android22-clang
bisect_driver.py x86_64-linux-android22-clang++
clang x86_64-linux-android23-clang
clang++ x86_64-linux-android23-clang++
clang-check x86_64-linux-android24-clang
clang-format x86_64-linux-android24-clang++
clang-tidy x86_64-linux-android26-clang
clang-tidy.real x86_64-linux-android26-clang++
git-clang-format x86_64-linux-android27-clang
i686-linux-android-addr2line x86_64-linux-android27-clang++
i686-linux-android-ar x86_64-linux-android28-clang
i686-linux-android-as x86_64-linux-android28-clang++
i686-linux-android-c++filt x86_64-linux-android29-clang
i686-linux-android-dwp x86_64-linux-android29-clang++
i686-linux-android-elfedit yasm
我们可以在这里发现不同 Android 版本和架构的编译工具。 通过查看里面的编译工具我们发现, clang、 clang++ 是不但分架构还分 Android 版本的, 而其他工具只分架构。 比如我选几个哈
armv7a-linux-androideabi16-clang
armv7a-linux-androideabi16-clang++
armv7a-linux-androideabi29-clang
armv7a-linux-androideabi29-clang++
arm-linux-androideabi-addr2line
arm-linux-androideabi-ar
arm-linux-androideabi-as
arm-linux-androideabi-c++filt
arm-linux-androideabi-dwp
arm-linux-androideabi-elfedit
arm-linux-androideabi-gprof
arm-linux-androideabi-ld
arm-linux-androideabi-ld.bfd
arm-linux-androideabi-ld.gold
arm-linux-androideabi-nm
arm-linux-androideabi-objcopy
arm-linux-androideabi-objdump
arm-linux-androideabi-ranlib
arm-linux-androideabi-readelf
arm-linux-androideabi-size
arm-linux-androideabi-strings
arm-linux-androideabi-strip
发现问题了吗, 前缀不一样耶。 这个问题在配置 configure 时需要特别注意。(这里知道前缀不一样就好了)
2.2 对比 configure 源码明确问题所在
在查看 configure 时 发现如下代码:
set_default target_os
if test "$target_os" = android; then
cc_default="clang"
fi
ar_default="${cross_prefix}${ar_default}"
cc_default="${cross_prefix}${cc_default}"
cxx_default="${cross_prefix}${cxx_default}"
nm_default="${cross_prefix}${nm_default}"
pkg_config_default="${cross_prefix}${pkg_config_default}"
发现问题了吗?
如果 target-os 设置的是 Android 的话, 那么 使用的 clang 去编译呢。
我们先不管 ar_default、cc_default、cxx_default、....等等的默认值,我们先看最后几行各个工具的拼接, 发现都是 cross_prefix 去拼接的。 cross_prefix 就是我们在设置 configure 里配置的 --cross-prefix 前缀。 发现问题没, 发现问题没, 发现问题没, 我们在 2.1 中刚看到, 很明显 clang、 clang++ 的前缀和 其他工具的不一样。 所以这个地方很矛盾呀。 为了通用,我们可以定义一个参数在外部去单独设置 clang 和 clang++ 的前缀(这里我随便命名成--cross-prefix-clang)。跟设置 --cross-prefix 一样。
现在看下 ar_default、 cc_default、 cxx_default、 nm_default、 pkg_config_default等还有其他工具的默认名字, 可以在configure中全局搜索这个名字看默认值, 发现 ar_default 默认值是ar, 如果拼上前缀的话(--cross-prefix + ar),在编译工具中是存在这个工具的, 其他的也一样。 但是 clang 和 clang++ 也不行, 不存在的, 我们看到在上面已经把 cc_default="clang"了, 但是 clang++ 还没有, 所以在上面需要手动添加一个 cxx_default="clang++", 这样的话(--cross-prefix-clang + clang 或者 + clang++) 才存在哦!
2.3 明确所需要的头文件和库
使用了 clang 编译就需要指定 clang 的头文件和库
使用该目录下的头文件和库, 在编写脚本时, 引入即可
/ndk/toolchains/llvm/prebuilt/darwin-x86_64/sysroot
三、 修改 configure
最新版已经支持 target-os 设置成 android, 所以不用再像以前那样去修改那四个值了。
通过二中我们知道要增加一个变量 --cross-prefix-clang(随便命名)
--cross-prefix=PREFIX use PREFIX for compilation tools [$cross_prefix]
# 这里添加
--cross-prefix-clang=PREFIX use PREFIX for compilation clang tools [$cross_prefix]
先设置一个帮助信息呢, 规范嘛要
- 在 CMDLINE_SET 中把新增的变量加入
CMDLINE_SET="
$PATHS_LIST
ar
arch
as
assert_level
build_suffix
cc
objcc
cpu
cross_prefix
# 这里添加
cross_prefix_clang
custom_allocator
cxx
- 修改 clang++ 编译的默认值
set_default target_os
if test "$target_os" = android; then
cc_default="clang"
# 这里修改 默认值
cxx_default="clang++"
fi
ar_default="${cross_prefix}${ar_default}"
# 这里修改成我们新定义的前缀
cc_default="${cross_prefix_clang}${cc_default}"
# 这里修改成我们新定义的前缀
cxx_default="${cross_prefix_clang}${cxx_default}"
nm_default="${cross_prefix}${nm_default}"
pkg_config_default="${cross_prefix}${pkg_config_default}"
- 修改完后重新 ./configure
四、 编写脚本
#!/bin/bash
export TMPDIR=../temp
# 定义变量(可以不定义, 直接在下面写死也行, 这样不是更加清晰复用和简易嘛)
SYSROOT=/Users/liushuai/ffmpeg_2/ndk/toolchains/llvm/prebuilt/darwin-x86_64/sysroot
# 定义变量(可以不定义, 直接在下面写死也行, 这样不是更加清晰复用和简易嘛)
PLATFORM=/Users/liushuai/ffmpeg_2/ndk/toolchains/llvm/prebuilt/darwin-x86_64
function build
{
./configure \
--prefix=$PREFIX \
--target-os=android \
--arch=$CPU \
--enable-shared \
--disable-static \
--disable-doc \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffprobe \
--disable-avdevice \
--disable-symver \
--enable-cross-compile \
--sysroot=$SYSROOT \
--cross-prefix=$PLATFORM/bin/arm-linux-androideabi- \
--cross-prefix-clang=$PLATFORM/bin/armv7a-linux-androideabi16- \
--extra-cflags="-I$SYSROOT/usr/include" \
--extra-ldflags="-L$SYSROOT/usr/lib"
$ADDITIONAL_CONFIGURE_FLAG
make clean
make -j4
make install
}
CPU=armv7-a
PREFIX=../os
build
你会发现直接编译完,动态库生成了!!!, 没有发现中间有任何问题