单个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
结果
编译结果.png3. 开启文件大小优化
$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
结果
编译结果.pngNDK version >14
表现
直接使用nkd大于14版本的进行编译时
问题表现.png
原因
https://android.googlesource.com/platform/ndk/+/ndk-r15-release/docs/UnifiedHeaders.md
解决方案
1. 使用小于ndk15的ndk来进行编译
Android NDK, Revision 14b (March 2017)
- Windows 32-bit android-ndk-r14b-windows-x86.zip
- Windows 64-bit android-ndk-r14b-windows-x86_64.zip
- Mac OS X android-ndk-r14b-darwin-x86_64.zip
- Linux 64-bit (x86) android-ndk-r14b-linux-x86_64.zip
2. 使用 Standalone Toolchains
- 使用脚本,预编译出需要使用的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
- 修改编译脚本对应路径,开始编译
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
网友评论