一.初始化音频滤镜

  初始化音频滤镜的方法基本上和初始化视频滤镜的方法相同,不懂的可以看上篇博客,这里直接给出代码:

//audio_filter_core.cpp
#define INPUT_SAMPLERATE 44100
#define INPUT_FORMAT AV_SAMPLE_FMT_FLTP
#define INPUT_CHANNEL_LAYOUT AV_CH_LAYOUT_STEREO
static AVFilterGraph *filter_graph;
static AVFilterContext *abuffersrc_ctx;
static AVFilterContext *volume_ctx;
static AVFilterContext *aformat_ctx;
static AVFilterContext *abuffersink_ctx;
static AVFrame *input_frame= nullptr,*output_frame= nullptr;
int32_t init_audio_filter(const char *volume_factor){
int32_t result=0;
char ch_layout[64];
char options_str[1024];
AVDictionary *options_dict= nullptr;
//创建滤镜图
filter_graph=avfilter_graph_alloc();
if(!filter_graph){
cerr<<"Error:Unable to create filter graph."<<endl;
return -1;
}
//创建abuffer滤镜
const AVFilter *abuffer= avfilter_get_by_name("abuffer");
if(!abuffer){
cerr<<"Error:Could not find abuffer filter."<<endl;
return -1;
}
abuffersrc_ctx= avfilter_graph_alloc_filter(filter_graph,abuffer,"src");
if(!abuffersrc_ctx){
cerr<<"Error:could not allocate the abuffer instance."<<endl;
return -1;
}
av_get_channel_layout_string(ch_layout,sizeof(ch_layout),0,INPUT_CHANNEL_LAYOUT);
av_opt_set(abuffersrc_ctx,"channel_layout",ch_layout,AV_OPT_SEARCH_CHILDREN);
av_opt_set(abuffersrc_ctx,"sample_fmt",av_get_sample_fmt_name(INPUT_FORMAT),AV_OPT_SEARCH_CHILDREN);
av_opt_set_q(abuffersrc_ctx,"time_base",(AVRational){1,INPUT_SAMPLERATE},AV_OPT_SEARCH_CHILDREN);
av_opt_set_int(abuffersrc_ctx,"sample_rate",INPUT_SAMPLERATE,AV_OPT_SEARCH_CHILDREN);
result= avfilter_init_str(abuffersrc_ctx, nullptr);
if(result<0){
cerr<<"Error:could not initialize the abuffer filter."<<endl;
return -1;
}
//创建volume滤镜
const AVFilter *volume= avfilter_get_by_name("volume");
if(!volume){
cerr<<"Error:could not find volume filter."<<endl;
return -1;
}
volume_ctx= avfilter_graph_alloc_filter(filter_graph,volume,"volume");
if(!volume_ctx){
cerr<<"Error:could not allocate volume filter instance."<<endl;
return -1;
}
av_dict_set(&options_dict,"volume",volume_factor,0);
result= avfilter_init_dict(volume_ctx,&options_dict);
av_dict_free(&options_dict);
if(result<0){
cerr<<"Error:could not initialize volume filter instance."<<endl;
return -1;
}
//创建aformat滤镜
const AVFilter *aformat=avfilter_get_by_name("aformat");
if(!aformat){
cerr<<"Error:could not find aformat filter."<<endl;
return -1;
}
aformat_ctx= avfilter_graph_alloc_filter(filter_graph,aformat,"aformat");
if(!aformat_ctx){
cerr<<"Error:could not allocate aformat filter instance."<<endl;
return -1;
}
snprintf(options_str,sizeof(options_str),"sample_fmts=%s:sample_rates=%d:channel_layouts=stereo",av_get_sample_fmt_name(AV_SAMPLE_FMT_S16),44100);
result= avfilter_init_str(aformat_ctx,options_str);
if(result<0){
cerr<<"Error:could not initialize aformat filter."<<endl;
return -1;
}
//创建abuffersink滤镜
const AVFilter *abuffersink= avfilter_get_by_name("abuffersink");
if(!abuffersink){
cerr<<"Error:could not find abuffersink filter."<<endl;
return -1;
}
abuffersink_ctx= avfilter_graph_alloc_filter(filter_graph,abuffersink,"sink");
if(!abuffersink_ctx){
cerr<<"Error:could not allocate abuffersink filter instance."<<endl;
return -1;
}
result= avfilter_init_str(abuffersink_ctx, nullptr);
if(result<0){
cerr<<"Error:could not initialize abuffersink filter."<<endl;
return -1;
}
//连接创建好的滤镜
result=avfilter_link(abuffersrc_ctx,0,volume_ctx,0);
if(result>=0){
result=avfilter_link(volume_ctx,0,aformat_ctx,0);
}
if(result>=0){
result=avfilter_link(aformat_ctx,0,abuffersink_ctx,0);
}
if(result<0){
fprintf(stderr,"Error connecting filters\n");
return -1;
}
//配置滤镜图
result=avfilter_graph_config(filter_graph, nullptr);
if(result<0){
cerr<<"Error:Error configuring the filter graph."<<endl;
return -1;
}
//创建输入输出帧
input_frame=av_frame_alloc();
output_frame=av_frame_alloc();
if(!input_frame||!output_frame){
cerr<<"Error:frame allocation failed."<<endl;
return -1;
}
return 0;
}

二.初始化输入音频帧

  在这一步需要给输入音频帧设置一些参数,包括采样率,采样点个数,声道布局,音频帧格式等,然后就可以给音频帧分配内存空间了。代码如下:

//audio_filter_core.cpp
static int32_t init_frames(){
int result=0;
input_frame->sample_rate=44100;
input_frame->format=AV_SAMPLE_FMT_FLTP;
input_frame->channel_layout=AV_CH_LAYOUT_STEREO;
input_frame->nb_samples=1024;
result= av_frame_get_buffer(input_frame,0);
if(result<0){
cerr<<"Error:av_frame_get_buffer failed."<<endl;
return -1;
}
result= av_frame_make_writable(input_frame);
if(result<0){
cerr<<"Error:av_frame_make_writable failed."<<endl;
return -1;
}
return 0;
}

三.循环编辑音频帧

  在这一步需要注意的是,每次将输入音频帧放入滤镜图前,都要做一次初始化音频帧操作,否则会报错:filter context - fmt: fltp r: 44100 layout: 3 ch: 2, incoming frame - fmt: (null) r: 0 layout: 3 ch: 2 pts_time: NOPTS【Changing audio frame properties on the fly is not supported.】。注意一定是每次,不要只初始化一次,这样只有第一帧初始化了,后面的帧还是会报错,因为输入帧的格式要和滤镜上下文保持一致,如果没有每次都初始化,后面的帧的格式和采样率就识别不到,为null了。下面给出代码:

//audio_filter_core.cpp
static int32_t filter_frame(){
int32_t result= av_buffersrc_add_frame(abuffersrc_ctx,input_frame);
if(result<0){
cerr<<"Error:add frame to buffersrc failed."<<endl;
return -1;
}
while(1){
result=av_buffersink_get_frame(abuffersink_ctx,output_frame);
if(result==AVERROR(EAGAIN)||result==AVERROR_EOF){
return 1;
}
else if(result<0){
cerr<<"Error:av_buffersink_get_frame failed."<<endl;
return -1;
}
cout<<"Output channels:"<<output_frame->channels<<",nb_samples:"<<output_frame->nb_samples<<",sample_fmt:"<<output_frame->format<<endl;
write_samples_to_pcm2(output_frame);
av_frame_unref(output_frame);
}
return 0;
}
int32_t audio_filtering(){
int32_t result=0;
while(!end_of_input_file()){
init_frames();//每次都要执行
result=read_pcm_to_frame2(input_frame,INPUT_FORMAT,2);
if(result<0){
cerr<<"Error:read_pcm_to_frame2 failed."<<endl;
return -1;
}
result=filter_frame();
if(result<0){
cerr<<"Error:filter_frame failed."<<endl;
return -1;
}
}
return 0;
}

四.将编辑后的数据写入输出文件

  在这一步需要注意的是,由于在滤镜图中有一个滤镜实例将音频帧的采样格式设置为了AV_SAMPLE_FMT_S16,这是packed格式的帧,左右声道的数据交错存储在frame->data[0]指向的内存单元中,所以在写入的时候,需要注意这一点。下面给出代码:

//io_data.cpp
int32_t write_samples_to_pcm2(AVFrame* frame){
int16_t* samples = reinterpret_cast<int16_t*>(frame->data[0]);
int dataSize = frame->nb_samples * frame->channels * sizeof(int16_t);
fwrite(samples, 1, dataSize, output_file);
return 0;
}

  数据读入代码:

//io_data.cpp
static FILE* input_file= nullptr;
static FILE* output_file= nullptr;
int32_t read_pcm_to_frame2(AVFrame *frame,enum AVSampleFormat sample_fmt,int channels){
int data_size= av_get_bytes_per_sample(sample_fmt);
if(data_size<0){
cerr<<"Error:Failed to calculate data size."<<endl;
return -1;
}
for(int i=0;i<frame->nb_samples;i++){
for(int ch=0;ch<channels;ch++){
fread(frame->data[ch]+i*data_size,1,data_size,input_file);
}
}
return 0;
}
int32_t open_input_output_files(const char* input_name,const char* output_name){
if(strlen(input_name)==0||strlen(output_name)==0){
cout<<"Error:empty input or output file name."<<endl;
return -1;
}
close_input_output_files();
input_file=fopen(input_name,"rb");//rb:读取一个二进制文件,该文件必须存在
if(input_file==nullptr){
cerr<<"Error:failed to open input file."<<endl;
return -1;
}
output_file=fopen(output_name,"wb");//wb:打开或新建一个二进制文件,只允许写
if(output_file== nullptr){
cout<<"Error:failed to open output file."<<endl;
return -1;
}
return 0;
}
void close_input_output_files(){
if(input_file!= nullptr){
fclose(input_file);
input_file= nullptr;
}
if(output_file!= nullptr){
fclose(output_file);
output_file= nullptr;
}
}
int32_t end_of_input_file(){
return feof(input_file);
}

五.销毁资源

audio_filter_core.cpp
static void free_frames(){
av_frame_free(&input_frame);
av_frame_free(&output_frame);
}
void destroy_audio_filter(){
free_frames();
avfilter_graph_free(&filter_graph);
}

六.main函数实现

int main(){
const char *input_file_name="../input.pcm";
const char *output_file_name="../output.pcm";
const char *volume_factor="0.9";
int32_t result=0;
result= open_input_output_files(input_file_name,output_file_name);
if(result<0){
return -1;
}
result= init_audio_filter(volume_factor);
if(result<0){
return -1;
}
result=audio_filtering();
if(result<0){
return -1;
}
destroy_audio_filter();
close_input_output_files();
return 0;
}

  最后,可以使用下面的指令测试输出的pcm文件:

  ffplay -ac 2 -ar 44100 -f s16le -i output.pcm

 

如何使用libavfilter库给pcm音频采样数据添加音频滤镜?的更多相关文章

  1. 视音频数据处理入门:PCM音频采样数据处理

    ===================================================== 视音频数据处理入门系列文章: 视音频数据处理入门:RGB.YUV像素数据处理 视音频数据处理 ...

  2. 音频采样中left-or right-justified(左对齐,右对齐), I2S时钟关系

    音频采样中left-or right-justified(左对齐,右对齐), I2S时钟关系 原创 2014年02月11日 13:56:51 4951 0 0 刚刚过完春节,受假期综合症影响脑袋有点发 ...

  3. Java中解析wav音频文件信息:音频声道数,采样频率,采样位数、声音尺寸

    前言:请各大网友尊重本人原创知识分享,谨记本人博客:南国以南i 音频解析方法: public static int toInt(byte[] b) { return ((b[3] << 2 ...

  4. WebRTC 音频采样算法 附完整C++示例代码

    之前有大概介绍了音频采样相关的思路,详情见<简洁明了的插值音频重采样算法例子 (附完整C代码)>. 音频方面的开源项目很多很多. 最知名的莫过于谷歌开源的WebRTC, 其中的音频模块就包 ...

  5. 常用音频协议介绍&&有关音频编码的知识与技术参数

    (转载)常用音频协议介绍 会议电视常用音频协议介绍及对比白皮书 一.数字化音频原理:声音其实是一种能量波,因此也有频率和振幅的特征,频率对应于时间轴线,振幅对应于电平轴线.通常人耳可以听到的频率在20 ...

  6. iOS从零开始学习直播之音频1.播放本地音频文件

      现在直播越来越火,俨然已经成为了下一个红海.作为一个资深码农(我只喜欢这样称呼自己,不喜欢别人这样称呼我),我必须赶上时代的潮流,开始研究视频直播.发现视屏直播类的文章上来就讲拉流.推流.采集.美 ...

  7. sqlserver 多库查询 sp_addlinkedserver使用方法(添加链接服务器)

    sqlserver 多库查询 sp_addlinkedserver使用方法(添加链接服务器) 我们日常使用SQL Server数据库时,经常遇到需要在实例Instance01中跨实例访问Instanc ...

  8. iOS 直播-获取音频(视频)数据

    iOS 直播-获取音频(视频)数据 // // ViewController.m // capture-test // // Created by caoxu on 16/6/3. // Copyri ...

  9. HI3516EV100 RTMP添加音频

    1.先试试只发音频 失败 2.保存音频,发送视频,成功

  10. Android Volley 库通过网络获取 JSON 数据

    本文内容 什么是 Volley 库 Volley 能做什么 Volley 架构 环境 演示 Volley 库通过网络获取 JSON 数据 参考资料 Android 关于网络操作一般都会介绍 HttpC ...

随机推荐

  1. 四月十二号java基础知识

    1.面向对象的编程思想是力图使在计算机语言中对事物的描述与现实世界中该事物的本来面目尽可能地一致.2.类(class)和对象(object)是面向对象程序设计方法中最核心的概念3.类是对某一事物的描述 ...

  2. Thread面试题

    面试题目录https://www.cnblogs.com/Kaelthas/p/15005844.html 1.一个Thread对象代表一个线程,同一个线程能否多次启动? 不能,在Thread类中变量 ...

  3. Java学习笔记04

    1. 循环进阶 1.1 无限循环 概念 ​ 循环一直停不下来,又叫死循环. for格式 for (;;) { 循环语句; } while格式 while (true) { 循环语句; } do...w ...

  4. Go语言基础: goroutine和通道

    并发编程表现为程序由若干个自主的活动单元组成. goroutine 在Go语言里,每一个并发执行的活动称为goroutine.当一个程序启动时,只有一个goroutine来调用main函数,称之为主g ...

  5. Python ArcPy批量计算多时相遥感影像的各项元平均值

      本文介绍基于Python中ArcPy模块,对大量长时间序列栅格遥感影像文件的每一个像元进行多时序平均值的求取.   在遥感应用中,我们经常需要对某一景遥感影像中的全部像元的像素值进行平均值求取-- ...

  6. 【ACM算法竞赛日常训练】DAY16【奇♂妙拆分】【区区区间间间】【小AA的数列】数学 | 位运算 | 前缀和

    DAY16共3题: 奇♂妙拆分(简单数学) 区区区间间间(单调栈) 小AA的数列(位运算dp) 作者:Eriktse 简介:19岁,211计算机在读,现役ACM银牌选手力争以通俗易懂的方式讲解算法!️ ...

  7. Appweb+ESP学习笔记

    Appweb+ESP学习笔记 1.Appweb简介        Appweb HTTP Web服务器是最快的小型Web服务器.这是一个高性能,紧凑的嵌入式网络服务器,具有模块化,安全的核心.它支持广 ...

  8. 绝对强大的三大linux指令:ar, nm, objdump

    前言 如果普通编程不需要了解这些东西,如果想精确控制你的对象文件的格式或者你想查看一下文件对象里的内容以便作出某种判断,刚你可以看一下下面的工具:objdump, nm, ar.当然,本文不可能非常详 ...

  9. Django笔记三十五之admin后台界面介绍

    本文首发于公众号:Hunter后端 原文链接:Django笔记三十五之admin后台界面介绍 这一篇介绍一下 Django 的后台界面使用. Django 自带了一套后台管理界面,可用于我们直接操作数 ...

  10. AutoGPT:有手就会的安装教程

    AutoGPT 是什么 Auto-GPT 是一个实验性开源应用程序,展示了 GPT-4 语言模型的功能.该程序由 GPT-4 驱动,将 LLM 的"思想"链接在一起,以自主实现您设 ...