前言
这篇文件简单介绍下移动端Android系统下利用FFmpeg的Filter进行音频数据预处理的方法。
按照惯例先上一份源码 AndroidFFmpegFilter。
项目实现了:
- 音量调节功能 volume_filter.cpp
- 混音 amix_filter.cpp
FFmepg编译须知
由于需要用到Filter模块,
所以在FFmpeg编译脚本中需要开启相关编译选项。
否则会出现 avfilter_get_by_name(filter_name)
找不到对应的处理器。
--enable-filters
#or
--enable-filter=name #name 指定需要使用到的filter name
如果不想自己编译,可以使用项目编译好的 动态库。
使用FFmpeg相关动态库
接下来要将ffmpeg的动态库链接到我们的工程上面。
Filter相关只需要使用到libavfilter.so、libavformat.so、libavutil.so这三个动态库。
参考代码如下(提供CMake实现,Android.mk请自己转换):
set(LIB_DIR ${PROJECT_SOURCE_DIR}/libs)
#设置ffmpeg的头文件目录位置
include_directories(${LIB_DIR}/include/ffmpeg)
#导入avfilter动态库
add_library( avfilter
SHARED
IMPORTED )
set_target_properties( avfilter
PROPERTIES
IMPORTED_LOCATION
${LIB_DIR}/${ANDROID_ABI}/libavfilter-6.so )
#导入avformat动态库
add_library( avformat
SHARED
IMPORTED )
set_target_properties( avformat
PROPERTIES
IMPORTED_LOCATION
${LIB_DIR}/${ANDROID_ABI}/libavformat-57.so )
#导入avutil动态库
add_library( avutil
SHARED
IMPORTED )
set_target_properties( avutil
PROPERTIES
IMPORTED_LOCATION
${LIB_DIR}/${ANDROID_ABI}/libavutil-55.so )
#连接动态库
target_link_libraries(
your-lib
avfilter
avutil
avformat
)
FFmpeg Filter初始化流程
导入头文件
extern "C" {
#include <libavfilter/buffersink.h>
#include <libavfilter/buffersrc.h>
#include <libavutil/channel_layout.h>
};
注册相关filter
avfilter_register_all();
获取一个AVFilterGraph
利用这个Graph可以对后续的AVFilter进行管理。
AVFilterGraph *graph = avfilter_graph_alloc();
对于AVFilter的处理
一般步骤都是:
1、通过filter_name获取到需要使用的AVFilter。
AVFilter filter = avfilter_get_by_name(filter_name);
2、利用AVFilter从AVFilterGraph获取到相应的上下文环境。
AVFilterContext filter_ctx = avfilter_graph_alloc_filter(graph, filter, NULL);
3、构造初始化参数配置(多种方式)
- 方式一
char options_str[1024];
snprintf(options_str, sizeof(options_str),
"sample_fmt=%s:sample_rate=%d:channel_layout=0x%" PRIx64 ,
av_get_sample_fmt_name(sample_format),
sample_rate,
sample_channel);
avfilter_init_str(filter_ctx, options_str);
- 方式二
char ch_layout[64];
av_get_channel_layout_string(ch_layout, sizeof(ch_layout), 0, sample_channel);
av_opt_set(filter_ctx, "channel_layout", ch_layout, AV_OPT_SEARCH_CHILDREN);
av_opt_set(filter_ctx, "sample_fmt", av_get_sample_fmt_name(sample_format), AV_OPT_SEARCH_CHILDREN);
av_opt_set_int(filter_ctx, "sample_rate", sample_rate, AV_OPT_SEARCH_CHILDREN);
avfilter_init_str(filter_ctx, NULL);
- 方式三
AVDictionary *options_dict = NULL;
char ch_layout[64];
av_get_channel_layout_string(ch_layout, sizeof(ch_layout), 0, sample_channel);
av_dict_set(&options_dict, "channel_layout", ch_layout, AV_OPT_SEARCH_CHILDREN);
av_dict_set(&options_dict, "sample_fmt", av_get_sample_fmt_name(sample_format), AV_OPT_SEARCH_CHILDREN);
av_dict_set(&options_dict, "sample_rate", sample_rate, AV_OPT_SEARCH_CHILDREN);
avfilter_init_dict(volume_ctx, &options_dict);
PS:以上三种方式的实现效果是一致的。
对各个Filter进行链接
- 连接情况一(例如音量调节):
//abuffersrc_ctx -> volume_ctx -> abuffersink_ctx
avfilter_link(abuffersrc_ctx, 0, volume_ctx, 0);
avfilter_link(volume_ctx, 0, abuffersink_ctx, 0);
- 连接情况二 (例如混音):
//abuffersrc1_ctx
// -> amix_ctx -> abuffersink_ctx
//abuffersrc2_ctx
avfilter_link(abuffersrc1_ctx, 0, amix_ctx, 0);
avfilter_link(abuffersrc2_ctx, 0, amix_ctx, 1);
avfilter_link(amix_ctx, 0, abuffersink_ctx, 0);
初始化整个filters链
avfilter_graph_config(graph, NULL);
以上的流程就是整个FFmpeg Filter的初始化过程。
FFmpeg Filter使用流程
源音频数据输入
1、构造一个AVFrame:
//获取一个AVFrame实例
AVFrame *avframe = av_frame_alloc();
//配置输入音频的格式、采样率、声道和采样数
avframe->sample_rate = sample_rate;
avframe->format = sample_format;
avframe->channel_layout = sample_channel;
avframe->nb_samples = nb_sample;
//根据上面设置的情况,申请音频数据缓冲区
av_frame_get_buffer(avframe, 1);
2、将源音频输入送入Filter链中:
av_buffersrc_add_frame(abuffersrc_ctx, avframe);
3、销毁AVFrame相关资源
av_frame_free(&avframe);
处理后音频数据输出
1、申请一个AVFrame实例,值得提醒的是我们不需要对这个AVFrame做任何配置
AVFrame *avframe = av_frame_alloc();
2、从Filters链中获取处理后的数据包
av_buffersink_get_frame(abuffersink_ctx, avframe);
3、提取完毕AVFrame的数据后,我们需要将其销毁
av_frame_free(&avframe);
最后说几句
对于FFmpeg Filter的使用,基本都是遵循上述流程。
- 注册Filters
- 获取一个AVFilterGraph
- 获取多个AVFilter和AVFilterContext并进行参数配置
- 连接各个AVFilterContext
- 初始化整个Filters链
- 将源数据AVFrame输入Filters链接收端
- 从Filters链输出端获取处理后数据AVFrame
对于音量调节,我们需要获取如下几个filter:
- abuffer:提供了音频数据的输入端。
- volume:提供了音频数据音量调节的模块。
- aformat:提供了转换成我们期望输出格式的模块,是因为Graph会在abuffer和volume之间自动做了格式转换。
- abuffersink:提供了音频数据的输出端。
对于混音,我们需要获取如下几个filter:
- abuffer:提供了音频数据的输入端,我们需要获取两个,因为有两路输入。
- amix:提供了多路音频数据混合的模块。
- aformat:提供了转换成我们期望输出格式的模块,是因为Graph会在abuffer和amix之间自动做了格式转换。
- abuffersink:提供了音频数据的输出端。
播放PCM文件可以利用Audacity这个工具可以导入pcm原始文件,并且提供了波形图查看和播放功能。
End!