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;  //位深度设置成16bit
    config->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.mfbz.cn/a/603667.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

自动化测试:Selenium入门指南!

Selenium是一个强大的自动化测试工具&#xff0c;特别适用于Web应用测试。本指南将介绍Selenium的安装、常用功能以及一些常见方法&#xff0c;帮助入门并能够更灵活地进行自动化测试。Selenium是一个用于自动化浏览器操作的工具&#xff0c;它广泛应用于Web应用程序的测试和网…

【前缀和】560. 和为 K 的子数组 974. 和可被 K 整除的子数组

题目链接 974. 和可被 K 整除的子数组 560. 和为 K 的子数组 今天刷题的时候&#xff0c;刷了这两题&#xff0c;感觉挺有意思的。代码写起来挺简单的&#xff0c;但是思路和其中的细节以及涉及到的知识点确实让我挺意外的。这里写个博客解析一波&#xff0c;也是巩固一下。 力…

分享《2024年中国企业级SaaS行业研究报告》

&#xff08;文章作者与来源&#xff1a;艾瑞咨询&#xff09; 大浪淘沙&#xff0c;SaaS行业进入关键转折点&#xff0c;企业级SaaS的总体市场规模达到888亿元&#xff0c;同比增长13.0%。内外部因素叠加之下&#xff0c;预计三年未来企业级SaaS市场规模的增速将稳定在15%-20…

请大数据把我推荐给正在申请小程序地理位置接口的人

小程序地理位置接口有什么功能&#xff1f; 若提审后被驳回&#xff0c;理由是“当前提审小程序代码包中地理位置相关接口( chooseAddress、getLocation )暂未开通&#xff0c;建议完成接口开通后或移除接口相关内容后再进行后续版本提审”&#xff0c;那么遇到这种情况&#x…

2024速通python之python基础

文章目录 一、你好&#xff0c;世界二、基本数据类型&#xff08;1&#xff09;数字型&#xff08;2&#xff09;字符串&#xff08;3&#xff09;列表&#xff08;4&#xff09;元组&#xff08;5&#xff09;集合&#xff08;6&#xff09;字典 二、注释&#xff08;1&#x…

【面试干货】http请求报文的组成与作用?

【面试干货】http请求报文的组成与作用&#xff1f; 一、http 的请求报文组成二、请求行&#xff08;Request Line&#xff09;三、请求头部&#xff08;Request Headers&#xff09;四、请求体&#xff08;Request Body&#xff09;五、响应头部 &#xff08;Response Headers…

LeetCode70:爬楼梯

题目描述 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢&#xff1f; 解题思想 1.确定dp数组以及下标的含义 dp[i]&#xff1a; 爬到第i层楼梯&#xff0c;有dp[i]种方法 2.确定递推公式 从dp[i]的定义可以…

Ansible任务剧本Playbook之变量、模板、角色介绍

前言 上篇介绍了 Ansible 单模块&#xff08;AD-Hoc&#xff09;的相关内容Ansible自动化运维工具单模块介绍-CSDN博客&#xff0c;Ad-Hoc 命令是一次性的、即时执行的命令&#xff0c;用于在远程主机上执行特定任务&#xff0c;这些命令通常用于快速执行简单的任务。当需要在…

【AI绘画】Midjourney 工笔画 水蓝色衣服的少女

using Midjourney 提示词&#xff1a; highly detailed,细节刻画细腻,超高清晰度,32k,HD,大师作品,高质量,动漫少女,水墨人像,20岁年轻身材很好的中国少女,惊人的美貌,五官精致,精致的妆容,华丽的水蓝色衣服,古风服饰,华丽的珠宝,飞扬的黑色长发,大风吹起头发,宝石发光,黄金装饰…

如何给正弦信号添加12V直流偏置

一个有趣问题的探究&#xff1a; 运放在单电源的情况下只能输出正电压&#xff08;单方向的&#xff09;&#xff0c;这就使得有正负值的信号电压只能输出一半&#xff1a; 【单电源供电的运放如何增加直流偏置】&#xff08;电阻分压法&#xff09;&#xff1a; 单电源供电的…

某云eHR PtFjk.mob 任意文件上传漏洞复现

0x01 产品简介 某云eHR是大中型企业广泛采用人力资源管理系统。某云是国内顶尖的HR软件供应商,是新一代eHR系统的领导者。 0x02 漏洞概述 某云EHR系统PtFjk.mob接口处存在未授权文件上传漏洞,攻击者可上传webshell来命令执行,获取服务器权限。 0x03 复现环境 FOFA:bod…

算法-并查集

目录 什么是并查集 并查集基础 &#xff08;1&#xff09;原理 &#xff08;2&#xff09;初始化 &#xff08;3&#xff09;查询 &#xff08;4&#xff09;合并 &#xff08;5&#xff09;判断是否同一集合 并查集优化 路径压缩 启发式合并 并查集模板 模板 例题…

线下订单平台操作步揍

收款管理 1微信收款查询 1. 获取微信数据 获取微信数据。通过时间范围 查找微信数据调用第三方接口如下&#xff1a; Map map HttpPost.doPost("https://qyapi.weixin.qq.com/cgi-bin/externalpay/get_bill_list?access_token"ApiUtils.getWxtoken(),args); 其中…

如何缩小图片尺寸不改变清晰度?几个方法教你解决

在平时对图片进行处理的时候&#xff0c;最害怕的就是修改过的图片质量下降&#xff0c;导致清晰度不够&#xff0c;尤其是缩小图片尺寸的时候&#xff0c;所以今天小编就来告诉大家几个关于修改图片尺寸又不改变清晰度的方法。 修改图片大小是非常普遍的图片编辑需求&#xf…

【SpringMVC 】什么是SpringMVC(三)?基于springmvc的文件上传、基于springmvc的拦截器、基于springmvc的邮件发送

文章目录 SpringMVC第五章1、SpringMVC文件上传1、基本步骤1-2345-82、邮件发送1、基本步骤1-234-5567-8 简单邮件带附件的邮件第六章1、拦截器的使用使用步骤232、调度的使用基本步骤1-56-8调度规则3、shiro安全框架核心概念基本语法1、基于ini文件的认证**测视类**2、基于rea…

计算机组成原理网课笔记

无符号整数的表示与运算 带符号整数的表示与运算 原反补码的特性对比 移码

基于 docker-compose 部署 LNMP 架构

目录 前言 1、任务要求 2、Nginx 2.1 建立工作目录并上传相关安装包 2.2 编写 Nginx Dockerfile 脚本 2.3 准备 nginx.conf 配置文件 3、Mysql 3.1 建立工作目录并上传相关安装包 3.2 编写 Mysql Dockerfile 脚本 3.3 编写 my.cnf 配置文件 4、PHP 4.1 建立工作目录…

Spring MVC(一)

1 Spring MVC概述 我们在之前学习Servlet的时候&#xff0c;认识了在WEB开发中MVC设计模式&#xff0c;其最为经典的设计就是&#xff0c;通过控制器&#xff08;Controller&#xff09;分离模型&#xff08;Model&#xff09;和视图&#xff08;View&#xff09;。在具体的WEB…

提高谷歌抓取成功率:代理IP的7个使用误区

在当今数字化时代&#xff0c;数据采集和网络爬取已成为许多企业和个人必不可少的业务活动。对于爬取搜索引擎数据&#xff0c;特别是Google&#xff0c;使用代理IP是常见的手段。然而&#xff0c;使用代理抓取Google并不是一件轻松的事情&#xff0c;有许多常见的误区可能会导…