这几天的开发任务是编写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位动态库找不到picconfig.log中虽然提示UCLIBC等条件不满足错误,并不影响32位动态库的正常编译,而64位也能继续编译流程,可以认为也不受此影响。
不满足__UCLIBC__对此问题,部分博客提示配置上-llibc
,在FFmpeg3.2.2的编译中,这是无效的。
最后,虽然编译成功,但是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一样的信息
网友评论