偶遇FFmpeg(番外)——FFmpeg花样编译入魔2之单个SO库和ndk15之后

96
deep_sadness
0.2 2018.10.18 09:42* 字数 677

单个SO文件

1. 编译o文件

 --disable-shared \
 --enable-static \

因为最后会将文件合成libffmpeg.so文件,所以就不需要去改configure文件了!!

2. 这些o文件link到一起,变成libffmpeg.so文件

$TOOLCHAIN/bin/arm-linux-androideabi-ld \
-rpath-link=$SYSROOT/usr/lib \
-L$SYSROOT/usr/lib \
-L$PREFIX/lib \
-soname libffmpeg.so -shared -nostdlib -Bsymbolic --whole-archive --no-undefined -o \
$PREFIX/libffmpeg.so \
libavcodec/libavcodec.a \
libavfilter/libavfilter.a \
libswresample/libswresample.a \
libavformat/libavformat.a \
libavutil/libavutil.a \
libswscale/libswscale.a \
libavdevice/libavdevice.a \
-lc -lm -lz -ldl -llog --dynamic-linker=/system/bin/linker \
$TOOLCHAIN/lib/gcc/arm-linux-androideabi/4.9.x/libgcc.a

结果

编译结果.png

3. 开启文件大小优化

$TOOLCHAIN/bin/arm-linux-androideabi-strip $PREFIX/libffmpeg.so

完整的脚本

#!/bin/bash

NDK=/Users/Cry/Library/Android/sdk/android-ndk-r14b
SYSROOT=$NDK/platforms/android-14/arch-arm/
TOOLCHAIN=$NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/darwin-x86_64

CPU=arm
# PREFIX=$(pwd)/android/$CPU
PREFIX=/Users/Cry/Documents/FFmpeg/1017/small_one_strip/$CPU
ADDI_CFLAGS=" -I$SYSROOT/usr/include"
ADDI_LDFLAGS=""

function build_arm
{
./configure \
    --prefix=$PREFIX \
    --disable-shared \
    --enable-static \
    --disable-encoders \
    --disable-muxers \
    --disable-decoders \
    --enable-decoder=h264 \
    --enable-decoder=aac \
    --disable-demuxers \
    --enable-demuxer=mov \
    --enable-demuxer=m4a \
    --disable-parsers \
    --enable-parser=aac \
    --enable-parser=h264 \
    --disable-protocols \
    --enable-protocol=file \
    --disable-filters \
    --enable-filter=scale \
    --disable-doc \
    --disable-ffmpeg \
    --disable-ffplay \
    --disable-ffprobe \
    --disable-ffserver \
    --disable-symver \
    --disable-avresample \
    --enable-small \
    --cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \
    --target-os=linux \
    --arch=arm \
    --enable-cross-compile \
    --sysroot=$SYSROOT \
    --extra-cflags="-Os -fpic $ADDI_CFLAGS" \
    --extra-ldflags="$ADDI_LDFLAGS" \
    $ADDITIONAL_CONFIGURE_FLAG
make clean
make
make install

$TOOLCHAIN/bin/arm-linux-androideabi-ld \
-rpath-link=$SYSROOT/usr/lib \
-L$SYSROOT/usr/lib \
-L$PREFIX/lib \
-soname libffmpeg.so -shared -nostdlib -Bsymbolic --whole-archive --no-undefined -o \
$PREFIX/libffmpeg.so \
libavcodec/libavcodec.a \
libavfilter/libavfilter.a \
libswresample/libswresample.a \
libavformat/libavformat.a \
libavutil/libavutil.a \
libswscale/libswscale.a \
libavdevice/libavdevice.a \
-lc -lm -lz -ldl -llog --dynamic-linker=/system/bin/linker \
$TOOLCHAIN/lib/gcc/arm-linux-androideabi/4.9.x/libgcc.a

$TOOLCHAIN/bin/arm-linux-androideabi-strip $PREFIX/libffmpeg.so

echo "compile complete!!"

}

build_arm

结果

编译结果.png

NDK version >14

表现

直接使用nkd大于14版本的进行编译时


问题表现.png

原因

https://android.googlesource.com/platform/ndk/+/ndk-r15-release/docs/UnifiedHeaders.md

解决方案

1. 使用小于ndk15的ndk来进行编译

NDK Archives

Android NDK, Revision 14b (March 2017)

2. 使用 Standalone Toolchains

官网-使用独立工具链

  1. 使用脚本,预编译出需要使用的toolchain
STANDALONE_DIR=/tmp/my-android-toolchain/
API=14
TARGET = arm
function standalone{
$NDK/build/tools/make-standalone-toolchain.sh \
--arch=$TARGET --platform=android-$API --install-dir=$STANDALONE_DIR
}
standalone
  1. 修改编译脚本对应路径,开始编译
SYSROOT=$STANDALONE_DIR/sysroot
TOOLCHAIN=$STANDALONE_DIR
完整的脚本

注意这里是编译生产多个so文件的脚本。同时需要去改configure文件!!

#!/bin/bash
NDK=/Users/Cry/Library/Android/sdk/ndk-bundle
STANDALONE_DIR=/private/tmp/my-android-toolchain/
SYSROOT=$STANDALONE_DIR/sysroot
TOOLCHAIN=$STANDALONE_DIR
TRIPLE=arm-linux-androideabi
API=14
TARGET = arm

CPU=arm
# PREFIX=$(pwd)/android/$CPU
PREFIX=/Users/Cry/Documents/FFmpeg/1017/small_new/$CPU
ADDI_CFLAGS=""

ADDI_LDFLAGS=""
ADDITIONAL_CONFIGURE_FLAG=""


function standalone
{
./$NDK/build/tools/make-standalone-toolchain.sh \
--arch=$TARGET --platform=android-$API --install-dir=$STANDALONE_DIR
}
standalone

function build_arm
{
./configure \
    --prefix=$PREFIX \
    --enable-shared \
    --disable-static \
    --disable-encoders \
    --disable-muxers \
    --disable-decoders \
    --enable-decoder=h264 \
    --enable-decoder=aac \
    --disable-demuxers \
    --enable-demuxer=mov \
    --enable-demuxer=m4a \
    --disable-parsers \
    --enable-parser=aac \
    --enable-parser=h264 \
    --disable-protocols \
    --enable-protocol=file \
    --disable-filters \
    --enable-filter=scale \
    --disable-doc \
    --disable-ffmpeg \
    --disable-ffplay \
    --disable-ffprobe \
    --disable-ffserver \
    --disable-symver \
    --disable-avresample \
    --enable-small \
    --cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \
    --target-os=linux \
    --arch=arm \
    --enable-cross-compile \
    --sysroot=$SYSROOT \
    --extra-cflags="-Os -fpic $ADDI_CFLAGS" \
    --extra-ldflags="$ADDI_LDFLAGS" \
    $ADDITIONAL_CONFIGURE_FLAG
make clean
make
make install
}

build_arm
缺陷

当编译成一个SO文件时,报错如下


编译成单个SO时报错.png

因为新版本的Ndk改了对应的方法,变成了内联函数。所以会找不到应用。
暂时没找到办法解决。

Just don't set _FILE_OFFSET_BITS=64 for pre-L targets. The difference between r14 and r15 here is that doing that used to be a no-op.

重要

在编译x86版本的单个SO文件的时候,遇到了很多问题。都无法解决。所以如果要兼容多个CPU版本,建议还是不要使用单个so版本。
ps: 泪目,花了好多时间。

使用ndk 14时
image.png

合成单个SO文件时,会出现 undefined reference to '__page_size' .
找了一圈,找不到答案。答案应该是如下

__page_size was only used for android-12 and below.. I have fixed getpagesize for these versions to use PAGE_SIZE instead
https://stackoverflow.com/questions/36002823/ndk-build-fails-to-build-superpowered-audio-library-after-recent-ndk-update

找了一下,但这确实是使用了PAGE_SIZE。这。。。。


image.png

但是具体没找到,要去哪儿改源码。。。所以就失败了。

使用ndk 15以上时

会出现上面的mmap64的错误
可以在这边文章中找到描述

https://android.googlesource.com/platform/ndk/+/master/docs/user/common_problems.md
https://github.com/android-ndk/ndk/issues/536

arm 编译时

指定Android 的API 版本及同时设置 file_offset就可以解决这个问题了
“ -D__ANDROID_API__=19 -D_FILE_OFFSET_BITS=32”

x86 时

但是在x86时,编译出来的单个so还是有问题。甚至
使用Clang进行编译

    # --cc=$TOOLCHAIN/bin/clang \

编译能够成功,但是在运行是,同样会提示这个错误。。

  • 提示 has text relocations


    image.png
FFmpeg