本篇文章主要来源自https://blog.csdn.net/qqqq245425070/article/details/90041475,修改了文章中一些错误,并附上自己遇到的一些问题的解决方案
编译流程
step1:下载FFmpeg库和NDK库
(1)我在usr目录下建立了一个ndk文件夹
(2)然后进入ndk文件夹,FFmpeg官网找到下载地址,再在命令行输入 “wget https://www.ffmpeg.org/releases/ffmpeg-4.0.2.tar.gz”
(3)解压文件夹 “tar -xzvf ffmpeg-4.0.2.tar.gz”
(4)下载NDK,输入命
https://blog.csdn.net/momo0853/article/details/73898066
(5)解压ndk包 “unzip android-ndk-r14b-linux-x86_64.zip”,如果没有安装unzip,系统会提示,按照系统提示安装就好了,最后会解压生成文件夹android-ndk-r14b
step2:NDK安装
将android-ndk-r14b加入环境变量
step3:配置解压后的文件夹ffmpeg-4.0.2中的configure,更改下方内容:
修改前:
SLIBNAME_WITH_MAJOR='$(SLIBNAME).$(LIBMAJOR)'
LIB_INSTALL_EXTRA_CMD='$$(RANLIB) "$(LIBDIR)/$(LIBNAME)"'
SLIB_INSTALL_NAME='$(SLIBNAME_WITH_VERSION)'
SLIB_INSTALL_LINKS='$(SLIBNAME_WITH_MAJOR) $(SLIBNAME)'
修改后:
SLIBNAME_WITH_MAJOR='$(SLIBPREF)$(FULLNAME)-$(LIBMAJOR)$(SLIBSUF)'
LIB_INSTALL_EXTRA_CMD='$$(RANLIB) "$(LIBDIR)/$(LIBNAME)"'
SLIB_INSTALL_NAME='$(SLIBNAME_WITH_MAJOR)'
SLIB_INSTALL_LINKS='$(SLIBNAME)'
这一步的主要目的是生成Android能够使用的 名称-版本.so文件的格式,不然的话生成的是Linux上使用库,Android不能用。
step3:编写Android编译的脚本
在解压后的文件夹ffmpeg-4.0.2中,新建build_android.sh文件(使用的shell命令“touch build_android.sh”)
输入命令“vim build_android.sh”,点击Enter打开文件,在文件中粘贴下面内容,(注意修改NDK的地址):
#!/bin/bash
export NDK_HOME=/Users/zamp/Documents/ffmpeg/android-ndk-r14b
export PLATFORM_VERSION=android-9
function build
{
echo "start build ffmpeg for $ARCH"
./configure --target-os=linux \
--prefix=$PREFIX --arch=$ARCH \
--disable-doc \
--enable-shared \
--disable-static \
--disable-yasm \
--disable-asm \
--disable-symver \
--enable-gpl \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffprobe \
--disable-ffserver \
--cross-prefix=$CROSS_COMPILE \
--enable-cross-compile \
--sysroot=$SYSROOT \
--enable-small \
--disable-linux-perf \
--extra-cflags="-Os -fpic $ADDI_CFLAGS" \
--extra-ldflags="$ADDI_LDFLAGS" \
$ADDITIONAL_CONFIGURE_FLAG
make clean
make
make install
echo "build ffmpeg for $ARCH finished"
}
#arm
ARCH=arm
CPU=arm
PREFIX=$(pwd)/android/$ARCH
TOOLCHAIN=$NDK_HOME/toolchains/arm-linux-androideabi-4.9/prebuilt/darwin-x86_64
CROSS_COMPILE=$TOOLCHAIN/bin/arm-linux-androideabi-
ADDI_CFLAGS="-marm"
SYSROOT=$NDK_HOME/platforms/$PLATFORM_VERSION/arch-$ARCH/
build
# #x86
# ARCH=x86
# CPU=x86
# PREFIX=$(pwd)/android/$ARCH
# TOOLCHAIN=$NDK_HOME/toolchains/x86-4.9/prebuilt/darwin-x86_64
# CROSS_COMPILE=$TOOLCHAIN/bin/i686-linux-android-
# ADDI_CFLAGS="-march=i686 -mtune=intel -mssse3 -mfpmath=sse -m32"
# SYSROOT=$NDK_HOME/platforms/$PLATFORM_VERSION/arch-$ARCH/
# build
–disable-ffmpeg意为禁用ffmpeg工具,编译时不编译出ffmpeg工具,–enable-ffmpeg为启用,但是configure文件配置有个特点,FFmpeg的默认的配置不是以show_help方法中的配置配置的,而是以前缀disable or enable取反配置的,也就是FFmpeg中各属性默认的配置把show_help中各个配置的前缀取反即可。
step4:编译ffmpeg
1)给ndk里的所有文件设置权限 chmod -R 777 ndk
2)执行 ./build_android.sh
编译成功后文件会在/usr/ndk/ffmpeg-4.0.2/android路径目录下。实际上这个脚本执行完,会编译出现多个.so文件,我们使用的取/usr/ndk/ffmpeg-4.0.2/android/arm/lib中的大版本号的,就是名字里面有数字的。
减少包体积
如果你的APP比较在意包的大小,在编译时,我们可以针对自己需要的功能来进行配置,更改bash脚本(也就是我们创建的build_android.sh文件),加入配置:
–disable-everything
会把下面的组件不加入编译:
Individual component options:--disable-everything
disable all components listed below--disable-encoder=NAME
disable encoder NAME
--enable-encoder=NAME
enable encoder NAME
--disable-encoders
disable all encoders--disable-decoder=NAME
disable decoder NAME
--enable-decoder=NAME
enable decoder NAME--disable-decoders
disable all decoders--disable-hwaccel=NAME
disable hwaccel NAME--enable-hwaccel=NAME
enable hwaccel NAME--disable-hwaccels
disable all hwaccels--disable-muxer=NAME
disable muxer NAME--enable-muxer=NAME
enable muxer NAME--disable-muxers
disable all muxers--disable-demuxer=NAME
disable demuxer NAME--enable-demuxer=NAME
enable demuxer NAME--disable-demuxers
disable all demuxers--enable-parser=NAME
enable parser NAME--disable-parser=NAME
disable parser NAME--disable-parsers
disable all parsers--enable-bsf=NAME
enable bitstream filter NAME--disable-bsf=NAME
disable bitstream filter NAME--disable-bsfs
disable all bitstream filters--enable-protocol=NAME
enable protocol NAME--disable-protocol=NAME
disable protocol NAME--disable-protocols
disable all protocols--enable-indev=NAME
enable input device NAME--disable-indev=NAME
disable input device NAME--disable-indevs
disable input devices--enable-outdev=NAME
enable output device NAME--disable-outdev=NAME
disable output device NAME--disable-outdevs
disable output devices--disable-devices
disable all devices--enable-filter=NAME
enable filter NAME--disable-filter=NAME
disable filter NAME--disable-filters
disable all filters
如果只加入–disable-everything那你编译出来的东西,里面几乎什么都没有了,所以你要在build_android.sh加入要编译的组件,比如:
#!/bin/bash
export TMPDIR=$(pwd)/ffmpegtemp #这句很重要,不然会报错 unable to create temporary file in
# NDK的路径,根据自己的安装位置进行设置
NDK=/softdata/android-sdk-macosx/android-ndk-r16b
# 编译针对的平台,可以根据自己的需求进行设置
# 这里选择最低支持android-14, arm架构,生成的so库是放在
# libs/armeabi文件夹下的,若针对x86架构,要选择arch-x86
PLATFORM=$NDK/platforms/android-14/arch-arm
# 工具链的路径,根据编译的平台不同而不同
# arm-linux-androideabi-4.9与上面设置的PLATFORM对应,4.9为工具的版本号,
# 根据自己安装的NDK版本来确定,一般使用最新的版本
TOOLCHAIN=$NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/darwin-x86_64
function build_one
{
./configure \
--prefix=$PREFIX \
--target-os=linux \
--cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \
--arch=arm \
--sysroot=$PLATFORM \
--extra-cflags="-I$PLATFORM/usr/include" \
--cc=$TOOLCHAIN/bin/arm-linux-androideabi-gcc \
--nm=$TOOLCHAIN/bin/arm-linux-androideabi-nm \
--enable-shared \
--enable-runtime-cpudetect \
--enable-gpl \
--enable-small \
--enable-cross-compile \
--disable-debug \
--disable-static \
--disable-doc \
--disable-asm \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffprobe \
--disable-ffserver \
--disable-postproc \
--disable-avdevice \
--disable-symver \
--disable-stripping \
$ADDITIONAL_CONFIGURE_FLAG
sed -i '' 's/HAVE_LRINT 0/HAVE_LRINT 1/g' config.h
sed -i '' 's/HAVE_LRINTF 0/HAVE_LRINTF 1/g' config.h
sed -i '' 's/HAVE_ROUND 0/HAVE_ROUND 1/g' config.h
sed -i '' 's/HAVE_ROUNDF 0/HAVE_ROUNDF 1/g' config.h
sed -i '' 's/HAVE_TRUNC 0/HAVE_TRUNC 1/g' config.h
sed -i '' 's/HAVE_TRUNCF 0/HAVE_TRUNCF 1/g' config.h
sed -i '' 's/HAVE_CBRT 0/HAVE_CBRT 1/g' config.h
sed -i '' 's/HAVE_RINT 0/HAVE_RINT 1/g' config.h
make clean
make -j4
make install
}
# arm v7vfp
CPU=armv7-a
OPTIMIZE_CFLAGS="-mfloat-abi=softfp -mfpu=vfp -marm -march=$CPU "
PREFIX=./android/$CPU-vfp
ADDITIONAL_CONFIGURE_FLAG=
build_one
遇到的坑
如果在执行build_android.sh时,如果提示没有权限“Permission Denied”,这是因为这个文件还没有被标记为可执行文件,先执行chmod a+x build_android.sh,再执行./build_android.sh就可以了。
(一)
上面演示的build_android.sh 脚本中使用的ndk版本是r15c,那在FFmpeg 4.1版本中,你应该会遇到这个error:
libavformat/udp.c: In function ‘udp_set_multicast_sources’:
libavformat/udp.c:290:28: error: request for member ‘s_addr’ in something not a structure or union
网上很少关于这个错误的描述,官方的回复也没看出来啥子有用的价值.
https://trac.ffmpeg.org/ticket/7741
解决方法:
有两种解决方案
1.ndk版本升到r17c
2.如果不想升ndk版本的,那就修改libavformat/udp.c 文件,把报错的相关代码注释掉就好.前提是你的项目中用不到这块功能.
在低版本的ndk应该都会有这个问题,我有试过r9b版本的,也有这个问题.
(二)
如果使用ndk版本是r17c的话,那上面的问题就不会遇到,但是会遇到新的问题.
CC libavdevice/alldevices.o
In file included from ./libavformat/internal.h:24:0,
from libavdevice/alldevices.c:23:
/opt/andorid_sdk/adt-bundle-linux-x86-20131030/android-ndk-r17c/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64/lib/gcc/arm-linux-androideabi/4.9.x/include/stdint.h:9:26: fatal error: stdint.h: No such file or directory
出现这个错误是因为NDK r17c版本将头文件和库文件进行了分离,我们指定的sysroot文件夹下只有库文件,而头文件放在了NDK目录下的sysroot内.
解决方法: 需在–extra-cflags中添加 “-isysroot $NDK/sysroot”
还有有关汇编的头文件也进行了分离.
解决方法:根据目标平台进行指定 “-I$NDK/sysroot/usr/include/arm-linux-androideabi”,将 “arm-linux-androideabi” 改为需要的平台就可以
修改之后的build_android.sh :
!/bin/bash
set -x
API=14
NDK=/opt/andorid_sdk/adt-bundle-linux-x86-20131030/android-ndk-r17c
SYSROOT=N D K / p l a t f o r m s / a n d r o i d − NDK/platforms/android-NDK/platforms/android−API/arch-arm/
TOOLCHAIN=KaTeX parse error: Expected '}', got '\ ' at position 100: … { ./configure \̲ ̲ --prefix=PREFIX
–disable-shared
–enable-static
–disable-doc
–disable-ffplay
–disable-ffprobe
–disable-symver
–disable-ffmpeg
–cc=KaTeX parse error: Expected 'EOF', got '\ ' at position 41: …ndroideabi-gcc \̲ ̲ --cross-prefi…TOOLCHAIN/bin/arm-linux-androideabi-
–target-os=linux
–arch=arm
–enable-cross-compile
–sysroot=KaTeX parse error: Expected 'EOF', got '\ ' at position 9: SYSROOT \̲ ̲ --extra-cflag…NDK/sysroot/usr/include/arm-linux-androideabi -isysroot $NDK/sysroot -fPIC -DANDROID -D__thumb__ -mthumb -Wfatal-errors -Wno-deprecated -mfloat-abi=softfp -marm -march=armv7-a"
–enable-neon
make clean all
make
make install
}
CPU=armv7-a
PREFIX=( p w d ) / a n d r o i d / (pwd)/android/(pwd)/android/CPU
function_one
(三)
libavcodec/aaccoder.c: In function ‘search_for_ms’:
libavcodec/aaccoder.c:803:25: error: expected identifier or ‘(’ before numeric constant
int B0 = 0, B1 = 0;
^
这是由于定义冲突导致的一个error,和ndk
版本有关,我在使用r15c和r9b版本的时候都没有遇到这个问题.
解决方法:修改libavcodec/aaccoder.c 文件 B0改成b0(ps:就是把int型变量名改一下,避免冲突,名字随便起)
(四)
libavcodec/hevc_mvs.c: In function ‘derive_spatial_merge_candidates’:
libavcodec/hevc_mvs.c:208:15: error: ‘y0000000’ undeclared (first use in this function)
((y ## v) >> s->ps.sps->log2_min_pu_size))
^
解决方法:将libavcodec/hevc_mvs.c文件的变量B0改成b0,xB0改成xb0,yB0改成yb0
(五)
libavcodec/opus_pvq.c: In function ‘quant_band_template’:
libavcodec/opus_pvq.c:498:9: error: expected identifier or ‘(’ before numeric constant
int B0 = blocks;
^
解决方法:将libavcodec/opus_pvq.c文件的变量B0改成b0
(六)
chmod: -R: No such file or directory 各种文件无法创建
先执行 ./configure
如果报nasm/yasm not found or too old. ,则安装yasm
方法:
1)下载:yasm的下载链接
2)解压:把下载下来的压缩包进行解压
3)切换路径: cd yasm-1.3.0
4)执行配置: ./configure
5)编译:make
6)安装:make install(提示:Permission denied,就执行sudo make install)
(七)fatal error: linux/perf_event.h: No such file or directory
ffmpeg 的android交叉编译 ,报这个错误最简单的方式是
--disable-linux-perf
编译后的so库请自行取用:
https://gitee.com/zhangxingzx0/ff-so