美文网首页iOS技术文章iOS进阶
编译Android FFmpeg3.2.2 ARM64动态库遇到

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

作者: 熊皮皮 | 来源:发表于2016-12-23 09:20 被阅读1868次

    这几天的开发任务是编写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。

    禁用nasm替换成yasm

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

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

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

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

    编译64位动态库出现文件格式错误 config.log的详细描述

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

    生成libutil报错

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

    编译64位动态库找不到pthread 编译64位动态库找不到pic

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

    不满足__UCLIBC__

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

    -llibc配置无效

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

    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
    
    arm-v7a编译得到armeabi一样的信息

    相关文章

      网友评论

      • 白藍鳮:我在mac上搭的ndk可以顺利编译arm64-v8a的ffmpeg库,调用也没问题,但在ubuntu上搭的ndk会出现问题
      • devzhaoyou:我也正在 编译 arm 64安卓端库,也是提示 s'y'mbols 找不到
      • difcareer:我也编译过,不过没有你这么多问题啊,你是想要提供native接口直接执行ffmpeg?如果是这样还有一些问题需要解决,比如参数解析
        熊皮皮:@difcareer v7a我也没遇到太大的意外,你编译64位试试
        difcareer: @熊皮皮 编译的是v7a的
        熊皮皮:@difcareer 你编译的是arm64-v8a架构的动态库吗?我之前负责编译iOS的,那边倒是很顺利。我通过JNI接口,FFmpeg接口都写在C/C++中。

      本文标题:编译Android FFmpeg3.2.2 ARM64动态库遇到

      本文链接:https://www.haomeiwen.com/subject/dxrfvttx.html