编译Android FFmpeg3.2.2 ARM64动态库遇到的问题

这几天的开发任务是编写Android 4.4以上的视频处理模块。由于MediaCodec及MediaExtrator的NDK接口只有Android 5.0以上才有,故使用FFmpeg解析MP4文件。
当然,也可以像ijkplayer一样,把Java层的接口在JNI_OnLoad函数中映射到C/C++运行环境,这反而维护起来更复杂,暂不考虑。
编译FFmpeg(最新版本为3.2.2)64及32位动态库时遇到不少问题,在此一一描述,相关工作环境:

  • 宿主机:Ubuntu 14
  • NDK 13
  • 调用设备:Nexus 6p(Android 6.0)
  • FFmpeg源码版本:3.2.2

结论:编译成armeabi动态库可正常使用,其余在CMakeList或gradle中都存在编译问题。具体原因留待进一步了解。

参考编译脚本:

#!/bin/bash
NDK=$HOME/Android/Sdk/ndk-bundle
SYSROOT=$NDK/platforms/android-21/arch-arm64/
TOOLCHAIN=$NDK/toolchains/aarch64-linux-android-4.9/prebuilt/linux-x86_64
ARCH=aarch64
function build_one
{
./configure \
    --prefix=$PREFIX \
    --disable-debug \
    --disable-shared \
    --enable-static \
    --enable-pthreads \
    --enable-pic \
    --enable-asm \
    --enable-version3 \
    --enable-gpl \
    --disable-muxers \
    --enable-muxer=mp4 \
    --disable-protocols \
    --enable-protocol=file \
    --disable-doc \
    --disable-ffmpeg \
    --disable-ffplay \
    --disable-ffprobe \
    --disable-ffserver \
    --disable-avdevice \
    --disable-programs \
    --disable-network \
    --disable-swresample \
    --disable-avresample \
    --disable-avfilter \
    --disable-swscale \
    --disable-postproc \
    --disable-doc \
    --disable-symver \
    --disable-encoders \
    --disable-decoders \
    --enable-decoder=h264 \
    --disable-muxers \
    --enable-muxer=mp4 \
    --disable-parsers \
    --enable-parser=h264 \
    --disable-devices \
    --disable-filters \
    --disable-iconv \
    --disable-audiotoolbox \
    --disable-videotoolbox \
    --enable-yasm \
    --disable-hwaccels \
    --cross-prefix=$TOOLCHAIN/bin/aarch64-linux-android- \
    --target-os=linux \
    --arch=$ARCH \
    --enable-cross-compile \
    --sysroot=$SYSROOT \
    --extra-cflags="-DANDROID -O3 -fpic"
make clean
make
make install
}
CPU=$ARCH
PREFIX=$(pwd)/android/$CPU 
build_one

从脚本上看,最终要编译的是64位ARM静态库。另外,有些编译脚本配置禁用nasm,在FFmpeg3.2.2的编译中存在如下错误提示,需改成yasm。

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第1张图片
禁用nasm替换成yasm

尝试一:可成功编译64位静态库,CMakeList中导入对应的静态库出现链接失败,相应静态库并没被拷贝到apk中,修改gradle配置也无效。

值得注意的是,平台需要21以上才有64位交叉编译链。感谢@大牙的提醒,开始我用android-19,找不到arch-arm64子目录。

SYSROOT=$NDK/platforms/android-21/arch-arm64/

尝试二:编译64位动态库失败。经多次修改编译脚本,禁用各种协议、编解码器,始终出现文件格式错误,如下所示。

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第2张图片
编译64位动态库出现文件格式错误
编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第3张图片
config.log的详细描述

有时能编译到生成libutil才报错。

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第4张图片
生成libutil报错

另外,编译动态库时提示pthread、启用pic等找不到,而编译静态库则无pthread找不到提示。

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第5张图片
编译64位动态库找不到pthread
编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第6张图片
编译64位动态库找不到pic

config.log中虽然提示UCLIBC等条件不满足错误,并不影响32位动态库的正常编译,而64位也能继续编译流程,可以认为也不受此影响。

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第7张图片
不满足__UCLIBC__

对此问题,部分博客提示配置上-llibc,在FFmpeg3.2.2的编译中,这是无效的

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第8张图片
-llibc配置无效

最后,虽然编译成功,但是CMake链接时还是提示文件格式错误,如下图所示。

编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第9张图片
clang++提示64位FFmpeg动态库文件格式错误

这里存在的疑问是CMake使用了linux-x84_64目录下的clang++,而不像我们在FFmpeg编译脚本中指定为aarch64-linux-android-4.9目录。因对CMake不了解,留待进一步尝试。

尝试三:编译32位armeabi-v7a动态库。虽然指定arch为arm-v7a或armeabi-v7a,然而,最终编译得到的静态库通过读取文件CPU架构信息都显示是v5TE,和armeabi一样。

readelf -A your.so
编译Android FFmpeg3.2.2 ARM64动态库遇到的问题_第10张图片
arm-v7a编译得到armeabi一样的信息

你可能感兴趣的:(编译Android FFmpeg3.2.2 ARM64动态库遇到的问题)