faad2交叉编译——aac解码为pcm,解决faad单通道转双通道问题

FAAD是比较成熟高效的开源AAC解码库,这里用于解码AAC生成PCM数据,用于音频播放。这里因为faad库,会将单通道转化为双通道踩了些坑,所以记录一下。
我使用的是2.11.0版本,貌似往前的版本没有使用CMake,需要configure配置编译

1.源码编译

使用git拉取

git clone https://github.com/knik0/faad2.git

因为是交叉编译,所以创建了一个cfg_file_path,其中设置一些参数,cfg_file_path内容如下:

##################################
# 配置 交叉编译
#################################set(CMAKE_SYSTEM_NAME Linux)    #设置目标系统名字
set(CMAKE_SYSTEM_PROCESSOR aarch64) #设置目标处理器架构
set(CMAKE_C_COMPILER aarch64-ca53-linux-gnu-gcc-10.4.0)#设置交叉编译器
add_compile_definitions(DISABLE_SBR) #禁止SBR和DECODER
add_compile_definitions(LC_ONLY_DECODER)

指定cfg_file_path进行编译,编译产物输出到output目录

mkdir output
/usr/bin/cmake  -DCMAKE_TOOLCHAIN_FILE=cfg_file_path -Boutput/
cd output
make
sudo make install

这样编译完成后在output下就会有libfaad.so,libfaad.so.2,libfaad.so.2.11.1,可以拷贝到开发板即可
image.png

2.测试代码

/** * faaddec.c * use faad library to decode AAC, only can decode frame with ADTS head  */  
#include <stdio.h>  
#include <memory.h>  
#include "faad.h"  #define FRAME_MAX_LEN 1024*5   
#define BUFFER_MAX_LEN 1024*1024  void show_usage()  
{  printf("usage\nfaaddec src_file dst_file");  
}  /** * fetch one ADTS frame */  
int get_one_ADTS_frame(unsigned char* buffer, size_t buf_size, unsigned char* data ,size_t* data_size)  
{  size_t size = 0;  if(!buffer || !data || !data_size )  {  return -1;  }  while(1)  {  if(buf_size  < 7 )  {  return -1;  }  if((buffer[0] == 0xff) && ((buffer[1] & 0xf0) == 0xf0) )  {  size |= ((buffer[3] & 0x03) <<11);     //high 2 bit  size |= buffer[4]<<3;                //middle 8 bit  size |= ((buffer[5] & 0xe0)>>5);        //low 3bit  break;  }  --buf_size;  ++buffer;  }  if(buf_size < size)  {  return -1;  }  memcpy(data, buffer, size);  *data_size = size;  return 0;  
}  int main(int argc, char* argv[])  
{  static unsigned char frame[FRAME_MAX_LEN];  static unsigned char buffer[BUFFER_MAX_LEN] = {0};  char src_file[128] = {0};  char dst_file[128] = {0};  FILE* ifile = NULL;  FILE* ofile = NULL;  unsigned long samplerate;  unsigned char channels;  NeAACDecHandle decoder = 0;  size_t data_size = 0;  size_t size = 0;  NeAACDecFrameInfo frame_info;  unsigned char* input_data = buffer;  unsigned char* pcm_data = NULL;  //analyse parameter  if(argc < 3)  {  show_usage();  return -1;  }  sscanf(argv[1], "%s", src_file);  sscanf(argv[2], "%s", dst_file);  ifile = fopen(src_file, "rb");  ofile = fopen(dst_file, "wb");  if(!ifile || !ofile)  {  printf("source or destination file");  return -1;  }  data_size = fread(buffer, 1, BUFFER_MAX_LEN, ifile);  //open decoder  decoder = NeAACDecOpen();      if(get_one_ADTS_frame(buffer, data_size, frame, &size) < 0)  {  return -1;  }  NeAACDecConfigurationPtr config = NeAACDecGetCurrentConfiguration(decoder);config->defObjectType = LC;config->defSampleRate = 8000;// config->defObjectType = HE_AAC;config->outputFormat = FAAD_FMT_16BIT;  //位深度设置成16bitconfig->downMatrix = 0;
//    源码部分如下:
//    if (*samplerate <= 24000 && (hDecoder->config.dontUpSampleImplicitSBR == 0))
//    {
//        *samplerate *= 2;
//        hDecoder->forceUpSampling = 1;
//    } else if (*samplerate > 24000 && (hDecoder->config.dontUpSampleImplicitSBR == 0)) {
//        hDecoder->downSampledSBR = 1;
//    }
//    dontUpSampleImplicitSBR 设置成1,可以禁止NeAACDecInit时,采样率(小于等于24000时)翻倍的问题config->dontUpSampleImplicitSBR = 1;NeAACDecSetConfiguration(decoder, config);//initialize decoder  NeAACDecInit(decoder, frame, size, &samplerate, &channels);  printf("samplerate %d, channels %d\n", samplerate, channels);  while(get_one_ADTS_frame(input_data, data_size, frame, &size) == 0)  {  // printf("frame size %d\n", size);  //decode ADTS frame  pcm_data = (unsigned char*)NeAACDecDecode(decoder, &frame_info, frame, size);   if(frame_info.error > 0)  {  printf("%s\n",NeAACDecGetErrorMessage(frame_info.error));              }  else if(pcm_data && frame_info.samples > 0)  {  printf("frame info: bytesconsumed %d, channels %d, header_type %d\  object_type %d, samples %d, samplerate %d\n",   frame_info.bytesconsumed,   frame_info.channels, frame_info.header_type,   frame_info.object_type, frame_info.samples,   frame_info.samplerate);  fwrite(pcm_data, 1, frame_info.samples * frame_info.channels * 2, ofile);      //2个通道  fflush(ofile);  }          data_size -= size;  input_data += size;  }      NeAACDecClose(decoder);  fclose(ifile);  fclose(ofile);  return 0;  
}  

注意这里很多博客给的都是错的,frame_info.samples * frame_info.channels * 2,这里需要乘以2,因为位宽是16bit,而一个字节是8bit。在使用单通道的情况下,这里如果不乘以2,会导致音频加速等问题

3.程序运行

使用下面的编译指令编译,拷贝可执行文件和动态库到板子里。

aarch64-ca53-linux-gnu-gcc-10.4.0 main.c -o main -lfaad -L/usr/local/lib/
./main_aarch high_temperature_shut_down.aac test.pcm

动态库拷贝到板子的/usr/lib下后,在板子上运行可执行文件。

4.关于单/双通道

faad这个在内部因为一些音频算法,开启一些编译选项之后,会将通道数乘以2,导致输出的音频通道数总是2,达不到预期效果,要解决这个问题,需要做以下两点
1)参考第二点测试代码,需要乘以2,否则音频加速
2)参考第一点中的编译配置选项,取消一些编译配置选项,防止通道乘以2
image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/9003.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++中如何自己封装一个Callable

需求背景 有时候静态编译太不灵活&#xff0c;我们需要更灵活的运行时操作。 又或真假设你在开发一个脚本&#xff0c;想注册本地的C函数到脚本语言的标准库中。例如gdscript的Callable。 下面是一个我的一个简单的实现。我们假设脚本语言中的变量类型是std::any。根据情况不…

《Fundamentals of Power Electronics》——一些常用变换器的正则电路参数值

对于理想的CCM PWM dc-dc转换器&#xff0c;其包含一个电感和电容&#xff0c;正则模型有效的低通滤波器需要包含一个电感和一个电容。正则模型简化为如下图所示。 假设电容与负载直接相连。基础的buck、boost和buck-boost转换器的参数值如下表所示。 该模型可以用传统的线性电…

使用网站内容进行多渠道品牌营销的主要优势

在选择服务提供商时&#xff0c;人们使用不同的方式来查找信息并与他们联系。有些人更喜欢网站&#xff0c;有些人则使用社交媒体或电子邮件。网站对于数字存在仍然至关重要&#xff0c;但跨多个渠道管理内容现在至关重要。多渠道营销以客户喜欢的方式与客户建立联系&#xff0…

记住这篇!论文查重降重aigc降低一定要看!

论文查重率降不下去真是受不了啊&#xff01;根本降不下去&#xff01;这些方法工具太适合我这样的论文裁缝了&#xff01; 一、论文降重/aigc降低工具 如果实在降不下去可以使用“蝌蚪论文”和“反向词典”这两个工具&#xff0c;也是我最常用的降重软件。 1.蝌蚪论文&#xf…

我从这些书籍中学来的财务以及税务知识

“你不能指望在开始工作的头两年攒下任何积蓄。” 这句话一直是我的座右铭&#xff0c;也是我给大学生的个人理财建议。这也就难怪我二十出头的时候&#xff0c;基本就是靠薪水过日子。 回想起来&#xff0c;我意识到其实这并不是最好的建议&#xff0c;甚至非常不好。 我现…

区块链的应用场景以及解释为什么能够保证安全提高信任度

区块链的不可篡改性和透明性是其最重要的特征之一。 不可篡改性&#xff1a;是指一旦数据被写入区块链&#xff0c;就无法被修改或删除。这是因为区块链中的每个区块都包含了前一个区块的哈希值&#xff0c;这个哈希值与当前区块的数据一起计算得出。如果对当前区块的数据进行…

BigInteger和BigDecimal类

BigInteger 和 BigDecimal 介绍 应用场景 BigInteger适合保存比较大的整型BigDecimal适合保存精度更高的浮点型&#xff08;小数&#xff09; BigInteger 和 BigDecimal 常见方法 1&#xff0c;add 加2&#xff0c;subtract 减3&#xff0c;multiply 乘4&#xff0c;divide…

mysql其它补充

exist和in的区别 exists 用于对外表记录做筛选。 exists 会遍历外表&#xff0c;将外查询表的每一行&#xff0c;代入内查询进行判断。 当 exists 里的条件语句能够返回记录行时&#xff0c;条件就为真&#xff0c;返回外表当前记录。反之如果 exists 里的条件语句不能返回记…

SpringBoot 使用Outlook邮箱发送邮件

目录 一、开启Outlook设置 二、依赖 三、配置文件 四、代码调用 一、开启Outlook设置 开启设置如图&#xff1a; 二、依赖 <!-- 邮箱依赖 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-mai…

常见的IP代理网站收集

收集的网站代理IP仅用于学习和测试&#xff0c;严禁用于非法用途&#xff0c;因非法使用导致的法律责任由用户承担。 免费代理网站&#xff1a;快代理&#xff0c; 芝麻&#xff0c;太阳 &#xff0c;豌豆 &#xff0c;云代理&#xff0c;开心代理 &#xff0c; 66ip&#xff0…

Nginx配置多个前端项目

1、修改nginx.conf配置文件&#xff1b; 2、必须包含默认的跟路径 location / { root D:/work/nginx-1.22.0/html; index index.html; } 3、添加要访问的前端项目信息&#xff0c;必须使用alias而不能使用root location /…

网页主题自动适配:网页跟随系统自动切换主题

主题切换是网站设计中一个非常有趣的功能&#xff0c;它允许用户在多种预先设计的样式之间轻松切换&#xff0c;以改变网站的视觉表现。最常见的就是白天和黑夜主题的切换&#xff0c;用户可以根据自己的喜好进行设置。 除了让用户手动去切换主题外&#xff0c;如果能够让用户第…

ChatGLM3大模型本地化部署、应用开发与微调

文章目录 写在前面ChatGLM3推荐图书作者简介推荐理由粉丝福利写在后面 写在前面 本期博主给大家推荐一本初学者学习并部署大模型的入门书籍&#xff0c;一起来看看吧&#xff01; ChatGLM3 ChatGLM3是继一系列先进语言模型之后的又一力作&#xff0c;专为追求高精度和广泛适…

Eel 项目中 Python端调用JS 使用一个括号和两个括号的区别

在使用 Python 第三方 GUI 库 EEL 的项目中&#xff0c;Python 文件中调用前端界面的 JS 函数时&#xff0c;使用一个括号 eel.my_function() 和使用两个括号 eel.my_function()() 存在以下区别&#xff1a; 一个括号eel.my_function()两个括号 eel.my_function()()第二个括号…

2023-2024年电力行业报告合集(精选69份)

电力行业报告/方案&#xff08;精选69份&#xff09; 2023-2024年 来源&#xff1a;2024年低空行业报告合集&#xff08;精选74份&#xff09; 【以下是资料目录】 2023中国发电企业和世界同类能源企业对标分析报告 2024电力部门碳减排技术经济管理报告 2024电力产业链分析…

计算机发展史故事【6】

电脑群英谱 本世纪三、四十年代&#xff0c;是计算机发展史里最重大的收获季节。群英荟萃&#xff0c;逐鹿中原&#xff0c;鹿究竟死于谁手&#xff0c;并不是没有争议的。除了马克1 号与埃历阿克&#xff0c;还有一大批科学家为计算机的诞生作出过巨大的贡献&#xff0c;他们…

智慧变电站守护者:TSINGSEE青犀AI视频智能管理系统引领行业革新

一、方案概述 随着科技的不断进步&#xff0c;人工智能&#xff08;AI&#xff09;技术已经深入到各个领域。在变电站安全监控领域&#xff0c;引入AI视频监控智能分析系统&#xff0c;可以实现对站内环境、设备状态的实时监控与智能分析&#xff0c;从而提高变电站的安全运行…

零基础入门学习Python第二阶02面向对象,迭代器生成器,并发编程

Python语言进阶 面向对象相关知识 三大支柱&#xff1a;封装、继承、多态 例子&#xff1a;工资结算系统。 """月薪结算系统 - 部门经理每月15000 程序员每小时200 销售员1800底薪加销售额5%提成"""from abc import ABCMeta, abstractmethodcl…

docker-compose集成elk(基于logstash+filebeat)采集java和nginx日志

1.准备compose.yml编排式文件 services: #日志信息同步logstash:container_name: logstashimage: docker.elastic.co/logstash/logstash:7.17.14 #logstash:command: logstash -f /usr/share/logstash/pipeline/logstash.confdepends_on:- elasticsearchrestart: on-failurepo…

解决$‘\r‘: command not found 或syntax error near unexpected token `$‘\r‘的四个方法

问题原因&#xff1a; 两个报错原因都是Linux和windows下的回车换行符不兼容 解决方法&#xff1a; 方法一&#xff1a;在windows系统可以用文本编辑器查看所有字符&#xff0c;例如notepad&#xff0c;编辑->档案格式转换->转换为UNIX格式 方法二&#xff1a;在Linux系…