How to build FFmpeg(ffmpeg-4.2.2 最新) for Android

一、简介

        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源码会遇到对应的不同的坑,譬如像如下两个问题:

udp.c
C compiler test failed.

(3) Linux编译环境使用(Ubuntu 14.04 ,x86_64)
Ubuntu 14.04

三、编译流程

(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
版本有关,在使用r15cr9b低版本的时候都没有遇到这个问题;
解决方法: 修改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改成b0xB0改成xb0yB0改成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应该就能解决问题,但是不知道又会不会出现其它的问题…还需后续研究。

你可能感兴趣的:(How to build FFmpeg(ffmpeg-4.2.2 最新) for Android)