一、简介
FFmpeg既是一款音视频编解码工具,同时也是一组音视频编解码开发套件,作为编解码开发套件,它为开发者提供了丰富的音视频处理的调用接口。
FFmpeg提供了多种媒体格式的封装和解封装,包括多种音视频编码、多种协议的流媒体、多种色彩格式转换、多种采样率转换、多种码率转换等。FFmpeg框架提供了多种丰富的插件模块,包含封装与解封装的插件、编码与解码的插件等。
二、编译环境
(1) FFmpeg 版本(ffmpeg-4.2.2) 最新
ffmpeg-4.2.2源码可去FFmpeg官网下载:https://ffmpeg.org/download.html#releases
也可以是用git 直接clone:
git clone "https://git.ffmpeg.org/ffmpeg.git" ffmpeg
(2) NDK版本使用 android-ndk-r17c
去官网下载对应的Android NDK:https://developer.android.com/ndk/downloads/older_releases
也可以使用其它版本,但是使用不同版本的NDK来编译FFmpeg源码会遇到对应的不同的坑,譬如像如下两个问题:
(3) Linux编译环境使用(Ubuntu 14.04 ,x86_64)
三、编译流程
(1) 修改FFmpeg的configure
下载FFmpeg源代码之后,首先需要对源代码中的configure
文件进行修改。由于编译出来的动态库文件名的版本号在.so之后(例如“libavcodec.so.5.100.1
”),而android平台不能识别这样文件名,所以需要修改这种文件名。在configure文件中找到下面几行代码:
SLIBNAME_WITH_MAJOR='$(SLIBNAME).$(LIBMAJOR)'
LIB_INSTALL_EXTRA_CMD='$$(RANLIB) "$(LIBDIR)/$(LIBNAME)"'
SLIB_INSTALL_NAME='$(SLIBNAME_WITH_VERSION)'
SLIB_INSTALL_LINKS='$(SLIBNAME_WITH_MAJOR)$(SLIBNAME)'
替换为下面内容就可以了:
SLIBNAME_WITH_MAJOR='$(SLIBPREF)$(FULLNAME)-$(LIBMAJOR)$(SLIBSUF)'
LIB_INSTALL_EXTRA_CMD='$$(RANLIB) "$(LIBDIR)/$(LIBNAME)"'
SLIB_INSTALL_NAME='$(SLIBNAME_WITH_MAJOR)'
SLIB_INSTALL_LINKS='$(SLIBNAME)'
(2) 编写脚本生成类库
在ffmpeg中创建一个build_android.sh
的脚本,并赋予可执行的权限,脚本内容如下:
#!/bin/bash
export NDK=/home/startimes/disk6/Android/android-ndk-r17c
export SYSROOT=${NDK}/platforms/android-23/arch-arm
export PREBUILT=${NDK}/toolchains/arm-linux-androideabi-4.9/prebuilt
export PREFIX=./android/arm
function build_so
{
./configure \
--prefix=$PREFIX \
--cross-prefix=${PREBUILT}/linux-x86_64/bin/arm-linux-androideabi- \
--cc=${PREBUILT}/linux-x86_64/bin/arm-linux-androideabi-gcc \
--nm=${PREBUILT}/linux-x86_64/bin/arm-linux-androideabi-nm \
--sysroot=$SYSROOT \
--target-os=android \
--arch=arm \
--enable-shared \
--disable-static \
--enable-cross-compile \
--extra-cflags="-I$NDK/sysroot/usr/include/arm-linux-androideabi -isysroot $NDK/sysroot -fPIC -DANDROID -D__thumb__ -mthumb -Wfatal-errors -Wno-deprecated -mfloat-abi=softfp -marm -march=armv7-a"
}
build_so
make clean
make -j4
make install
其中:
- SYSROOT 为so文件支持的最低Android版本的平台目录
- PREFIX 为生成的so文件存放目录
- cross-prefix 为编译所使用的工具链文件
- target-os 为目标操作系统
- enable和disable 指定了需要编译的项
四、编译FFmpeg
在ffmpeg目录下,直接执行命令:
./build_android.sh
理想状态下,经过上面的步骤后,会成功在android/arm/
目录下生成lib
,include
目录,lib文件夹里面就是需要的编译好的、适用于Android平台的so库文件。如下图:
五、编译问题修改
1)
上面演示的build_android.sh
脚本中使用的ndk版本是r17c
,编译FFmpeg 4.2.2
版本你应该会遇到如下error:
CC libavdevice/alldevices.o
In file included from ./libavformat/internal.h:24:0,
from libavdevice/alldevices.c:23:
/opt/andorid_sdk/adt-bundle-linux-x86-20131030/android-ndk-r17c/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64/lib/gcc/arm-linux-androideabi/4.9.x/include/stdint.h:9:26: fatal error: stdint.h: No such file or directory
出现这个错误是因为NDK r17c
版本将头文件和库文件进行了分离,我们指定的sysroot
文件夹下只有库文件,而头文件放在了NDK目录下的sysroot
内;
解决方法: 需在--extra-cflags
中添加 “-isysroot $NDK/sysroot”
还有有关汇编的头文件也进行了分离;
解决方法: 根据目标平台进行指定 “-I$NDK/sysroot/usr/include/arm-linux-androideabi”
,将 “arm-linux-androideabi”
改为需要的平台就可以,此问题上述脚本中已经添加完成。
2)
libavcodec/aaccoder.c: In function 'search_for_ms':
libavcodec/aaccoder.c:803:25: error: expected identifier or '(' before numeric constant
int B0 = 0, B1 = 0;
^
这是由于定义冲突导致的一个error,和NDK
版本有关,在使用r15c
和r9b
低版本的时候都没有遇到这个问题;
解决方法: 修改libavcodec/aaccoder.c
文件 B0
改成b0
(ps:就是把int型变量名改一下,避免冲突,名字随便起)。
3)
libavcodec/hevc_mvs.c: In function 'derive_spatial_merge_candidates':
libavcodec/hevc_mvs.c:208:15: error: 'y0000000' undeclared (first use in this function)
((y ## v) >> s->ps.sps->log2_min_pu_size))
^
解决方法:将libavcodec/hevc_mvs.c
文件的变量B0
改成b0
,xB0
改成xb0
,yB0
改成yb0
。
4)
libavcodec/opus_pvq.c: In function 'quant_band_template':
libavcodec/opus_pvq.c:498:9: error: expected identifier or '(' before numeric constant
int B0 = blocks;
^
解决方法:将libavcodec/opus_pvq.c
文件的变量B0
改成b0
。
结语
除了上面的问题外,中间也遇到过configure.sh脚本没执行成功的问题,原因大都是空格键或一些老的参数过时了等问题。比如 -ffserver
这个参数已经不用了。
上面提到说推荐NDK版本是r17c
以下的,是因为我们的build_android.sh
,编译脚本用用的编译工具是gcc,但是NDK r17c
以后的版本把gcc给移除了。所以如果现在用最新的版本r21
的话会出现关于gcc的error。考虑更换编译工具不用gcc应该就能解决问题,但是不知道又会不会出现其它的问题…还需后续研究。