zhengxiaoyong

初识FFmpeg编译那些事

FFmpeg简介

FFMPEG是一套具有非常强大功能的多媒体处理工具,它几乎涵盖了目前所有主流的多媒体数据封装格式、多媒体传输协议以及音视频编解码器,并且支持多媒体后处理,视频色彩转换、滤镜和缩放等,也支持众多主流的协议:HTTP、RTP、RTSP、RTMP、HLS、UDP等
安装可以通过Homebrew

FFmpeg它主要含有以下几个核心库:

1、libavcodec-提供了更加全面的编解码实现的合集
2、libavformat-提供了更加全面的音视频容器格式的封装和解析以及所支持的协议
3、libavutil-提供了一些公共函数
4、libavfilter-提供音视频的过滤器,如视频加水印、音频变声等
5、libavdevice-提供支持众多设备数据的输入与输出,如读取摄像头数据、屏幕录制
6、libswresample,libavresample-提供音频的重采样工具
7、libswscale-提供对视频图像进行色彩转换、缩放以及像素格式转换,如图像的YUV转换
8、libpostproc-多媒体后处理器

以及包含以下几个工具:

1、ffmpeg-一个流媒体的编解码、格式转换以及多媒体流的内容处理工具
2、ffplay-一个使用FFmpeg编解码的播放器
3、ffprobe-一个多媒体分析工具
4、ffserver-一个流多媒体服务器

FFmpeg命令

一些命令:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
//获取帮助
ffmpeg -help
//支持的格式
ffmpeg -formats
//支持的解码
ffmpeg -decoders
//支持的编码
ffmpeg -encoders
//支持的协议
ffmpeg -protocols
//本地视频文件推流至指定rtmp流媒体服务器
ffmpeg -re -i movie.mp4 -c copy -f flv rtmp://localhost:1935/live/room
//媒体流保存为视频文件
ffmpeg -i rtmp://localhost:1935/live/room -c copy moive.flv

更多命令请看文档FFmpeg文档

FFmpeg编译

首先在官网下载FFmpeg

在root dir编辑configure文件,把以下内容

1
2
3
4
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)'

替换成:

1
2
3
4
SLIBNAME_WITH_MAJOR='$(SLIBPREF)$(FULLNAME)-$(LIBMAJOR)$(SLIBSUF)'
LIB_INSTALL_EXTRA_CMD='$$(RANLIB) "$(LIBDIR)/$(LIBNAME)"'
SLIB_INSTALL_NAME='$(SLIBNAME_WITH_MAJOR)'
SLIB_INSTALL_LINKS='$(SLIBNAME)'

目的是修改so的文件名,为了编译出来的so能够被android平台正常使用,否则将编译出libavcodec.so.57.64.100这种类型的so,替换后将编译成我们能正常使用的libavcodec-57.so动态链接库,后面的数字为版本号

一般对于so,我们只需兼容arm cpu架构的即可,在root dir下新建bash脚本文件,内容为:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
#!/bin/bash

NDK=$HOME/Library/Android/sdk/ndk-bundle
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
ADDI_CFLAGS=""
ADDI_LDFLAGS=""

function build_arm
{
./configure \
--prefix=$PREFIX \
--enable-shared \
--disable-static \
--disable-doc \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffprobe \
--disable-ffserver \
--disable-symver \
--enable-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

NDK目录改为自己的NDK目录即可,关于configure配置,可以根据自己需要进行配置,可配置的属性在FFmpeg root dir的configure文件中show_help方法中查看,我摘一段:

1
2
3
4
5
6
Program options:
--disable-programs do not build command line programs
--disable-ffmpeg disable ffmpeg build
--disable-ffplay disable ffplay build
--disable-ffprobe disable ffprobe build
--disable-ffserver disable ffserver build

--disable-ffmpeg意为禁用ffmpeg工具,编译时不编译出ffmpeg工具,--enable-ffmpeg为启用,但是configure文件配置有个特点,FFmpeg的默认的配置不是以show_help方法中的配置配置的,而是以前缀disableorenable取反配置的,也就是FFmpeg中各属性默认的配置把show_help中各个配置的前缀取反即可,disable变为enableenable变为disable,如:在上面那段配置中,--disable-ffmpeg意为启用ffmpeg工具。

但是我们编译的bash脚本中却不是这样,而是配置的实意配置,disable就是disable

运行上面的bash脚本:

1
2
cd ffmpeg-3.2
./build_ffmpeg_arm.sh

编译完成后可以发现在FFmpeg的root目录下多了一个android目录,该目录就是所生成对应cup架构的so目录


按以上配置的bash脚本,会发现libavcodec-57.so比较大,有7.6M,肯定不能在Android中使用,所以,在编译时,我们可以针对自己需要的功能来进行配置,更改bash脚本,加入配置:

1
--disable-everything

该属性会把下列所有的组件都不加入编译:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
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

而我们生成的libavcodec-57.so过大也是由于加入了过多我们不需要使用的组件,禁用了所有的,接下来就是把自己需要的加入到bash脚本中配置即可,比如加入下面两个配置确保h264和aac的解码功能的保留:

1
2
--enable-decoder=h264
--enable-decoder=aac

等等,当然还有其它的,可以参考上面禁用的配置进行选择性保留。

编译支持H.264编码的FFmpeg动态链接库

x264

首先下载x264源码
x264是一个开源的H.264编码器,据说是最好的视频有损编码器,和编译FFmpeg类似,要编译x264成动态so库,一样通过configure配置文件,需要先修改configure文件中的:

1
2
3
echo "SOSUFFIX=so" >> config.mak
echo "SONAME=libx264.so.$API" >> config.mak
echo "SOFLAGS=-shared -Wl,-soname,\$(SONAME) $SOFLAGS" >> config.mak

替换成:

1
2
3
echo "SOSUFFIX=so" >> config.mak
echo "SONAME=libx264-$API.so" >> config.mak
echo "SOFLAGS=-shared -Wl,-soname,\$(SONAME) $SOFLAGS" >> config.mak

目的一样,确保可以在Android平台下可用,然后使用下面的bash脚本进行编译,arm为例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
#!/bin/bash

NDK=$HOME/Library/Android/sdk/ndk-bundle
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
ADDI_CFLAGS=""
ADDI_LDFLAGS=""

function build_arm
{
./configure \
--prefix=$PREFIX \
--enable-shared \
--disable-asm \
--enable-pic \
--enable-strip \
--host=arm-linux-androideabi \
--cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \
--sysroot=$SYSROOT
--extra-cflags="-Os -fpic $ADDI_CFLAGS" \
--extra-ldflags="$ADDI_LDFLAGS" \
$ADDITIONAL_CONFIGURE_FLAG
make clean
make
make install
}

build_arm

NDK目录同样改为自己的
运行上面脚本:

1
2
cd x264
./build_x264_arm.sh

编译完成后可以android目录找到libx264-148.so

静态链接x264编译

FFmpeg默认是不支持H.264的编码,不过支持H.264解码,但是FFmpeg为我们提供了众多配置,可添加额外功能库的支持,方便我们自己链接所需要额外功能的库,在FFmpeg的configure文件配置中,可以看到这么一段配置:

1
2
3
4
5
6
7
8
9
10
11
12
External library support:
//...
--enable-libfdk-aac enable AAC de/encoding via libfdk-aac [no]
--enable-libopenh264 enable H.264 encoding via OpenH264 [no]
--enable-libopenjpeg enable JPEG 2000 de/encoding via OpenJPEG [no]
--enable-librtmp enable RTMP[E] support via librtmp [no]
--enable-libwebp enable WebP encoding via libwebp [no]
--enable-libx264 enable H.264 encoding via x264 [no]
--enable-opengl enable OpenGL rendering [no]
--enable-openssl enable openssl, needed for https support
if gnutls is not used [no]
//...

避免篇幅太长,做了一部分删减,留了一些比较眼熟的,FFmpeg可以让我们自己配置添加需要的功能库,如使用fdk-aac库来做aac的编解码,H.264编码可选x264或openh264以及可选其它功能库的支持

要编译支持x264编码的FFmpeg动态链接库,首先是编译出x264静态链接库libx264.a,然后修改FFmpeg的编译脚本,配置x264静态链接库和头文件路径,接下来即可编译了

1、编译x264静态链接库

因为FFmpeg是使用静态链接方式链接其它额外的外部功能库,所以需要把x264库编译成.a静态库,通过--enable-static配置参数来编译出.a静态库,而--enable-shared则是编译动态链接so库,所以只需在上面的x264脚本中增加个配置:

1
--enable-static

编译成功后可以看到如下目录结构:

2、在FFmpeg脚本中配置x264静态链接库和头文件路径及编译

在此之前,先安装yasm:

1
brew install yasm

然后在之前FFmpeg的脚本基础上添加以下配置:

1
2
3
4
5
--enable-gpl \
--enable-libx264 \
--enable-yasm \
--extra-cflags="-I../x264/android/arm/include" \
--extra-ldflags="-L../x264/android/arm/lib" \

FFmpeg默认的LICENSE是LGPL,而libx264需要GPL,所以加入--enable-gpl,最后面两项配置是配置编译x264出的头文件和静态链接库路径,在这里我把x264和FFmpeg放在了同一目录层级,故可这样配置。
如果你需要修改--extra-cflags--extra-ldflags路径为自己x264头文件和静态链接库路径,必须符合--extra-cflags以-I开头,--extra-ldflags以-L开头,这些参数含义为:

-D:用于在编译时定义宏
-I:编译阶段生效的参数,用于指定头文件的搜索路径
-L:链接阶段生效的参数,用于指定链接库的搜索路径,-l用于指定链接库的名称,一般两者一起使用的话,就可以指定动态链接库

比如x264在桌面上,路径为:

1
2
--extra-cflags="-I/Users/Sunzxyong/Desktop/x264/android/arm/include" \
--extra-ldflags="-L/Users/Sunzxyong/Desktop/x264/android/arm/lib" \

然后进行编译,编译中可以看到encode的支持选择中,多了libx264的支持:

其它外部添加库如fdk-aac、openh264等也类似方式链接编译

Github

FFmpeg编译脚本

坚持原创技术分享,您的支持将鼓励我继续创作!