结合实例分析A2DP SBC帧结构和编解码算法

SBC也就是Sub band codec,俗称子带编码,是蓝牙A2DP必须支持的唯一编码。下面结合实例看看SBC编码数据在蓝牙传输中帧结构以及SBC的编解码算法流程:

AVDTP Frame

在这里插入图片描述
首先AVDTP Frame是基于L2CAP协议,所以会包含有4个字节的L2CAP头部,如上图看到2个字节的length = 650 bytes,两个字节的Destination CID = 0xA040 (AVDTP)。
接下来看看12个字节的AVDTP的头部:
在这里插入图片描述
这12个字节分成了三部分:

  • Padding ,Extension, Maker 和 Payload Type组成了2个字节,也就是0x8060;
  • Sequence Number 也就是0x0019, 占用2个字节,就是给AVDTP Frame一个序号,方便接收方能检查发送的数据包是不是连续的,有没有丢包。
  • Time Stamp 也就是0x00005400,占用4个字节,这个就是计算数据包播放的时间戳,接收方Audio DSP会根据时间戳来做音频播放同步以及丢包错包检查。
  • SSRC 也就是0x00000000,占用4个字节。

SBC Frame

在这里插入图片描述
从上图可以看到一个2-DH5数据包的Media Payload包含一个字节SBC头部(其中4个bit用来表示包含几个子frame),和7个frame。
根据A2DP_SPEC_V1.3,立体声和联合立体声的Frame_length计算方法:

frame_length = 4 + (4 * nrof_subbands * nrof_channels ) / 8 + (join * nrof_subbands + nrof_blocks * bitpool ) / 8.;联合立体声时,join == 1;

MONO和DUAL CHANNEL mode 的frame_length计算方法:

frame_length = 4 + (4 * nrof_subbands * nrof_channels ) / 8 + nrof_blocks * nrof_channels * bitpool / 8 .

本实例我们得到的nrof_subbands 等于8, nrof_channels 等于2,join 等于1,bitpool等于39,nrof_blocks等于16,所以:

frame_length = 4 +(4* 8 * 2) /8 +1 * 8 + 16 * 39/8 = 4 + 8 + 1 +78 = 91 byte

这样我们挑一个frame 1来看看:
在这里插入图片描述
从上图可以看出一个SBC子帧是由13个字节的Frame Header + 78个字节的audio samples组成:

第n字节参数描述
1syncword0x9C占用一个字节,这是个固定的值,标志着一个子帧的开始
2Sampling Frequency10(44.1 kHz)占用两个bit:
00表示16kHZ;
01表示32KHZ;
10表示44.1KHZ;
11表示48KHZ
2Blocks(11)16占用2个bit:
00表示4个block;
01表示8个block;
10表示12个block;
11表示16个block
2Channel ModeJoint Stereo占用2个bit:
00表示MONO;
01表示DUAL_CHANNEL;
10表示STEREO;
11表示JOINT STEREO
2Allocation MethodLoudness占用1个bit:
0表示LOUDNESS;
1表示SNR;
2Subbands1(8)占用1个bit:
0表示4个subbands;
1表示8个subbands;
3Bitpool39占用一个字节,表示bitpool大小
4crc_check0x5E占用一个字节,用作CRC校验
5join0xFE占用一个字节,一个bit为1表示一个subband编码为联合立体声,否则为立体声编码。
6-13Scale Factors比例因子,占用8个字节,4个bit表示一个通道,一个子带;此实例有2个通道,8个子带,所以需要4x2x8/8 = 8字节表示;
14-91Audio Samples78 bytes占用78个字节,量化编码后的音频数据,16x39/8 = 78,大小也就是nrf_blocks x bitpool / 8

SBC解码

在这里插入图片描述

  1. 音频接收端收到SBC帧数据会检查frame_header,然后做CRC的校验:G(X) = X8 + X4 + X3 + X2 + 1 (CRC-8).
  2. 计算真实的比例因子scalefactors : scalefactor[ch][sb] = pow(2.0, (scale _ factor[ch][sb] + 1)). 这里的scale_factor[ch][sb]就是上一节我们在帧头部的4个bit,而所谓的scale factor就是每个子段的最大幅值。
  3. Bit Allocation:这个是跟scalefactors关联的,每个子频段的幅值不一样,需要用到的bit数也不一样,但同一个子频段的幅值所用的bit数是一样的。
  4. Reconstruction of the Subband Samples:就是把每个子频段的78个字节的audio_sample和真实的比例因子scalefactor做一个乘法运算,进行采样数据的还原:sb_sample[blk][ch][sb] = scalefactor[ch][sb] * ((audio_sample[blk][ch][sb]*2.0+1.0) /levels[ch][sb]-1.0);
  5. Joint Processing :
for (blk=0;blk< nrof_blocks;blk++)
{
for (sb=0;sb<nrof_subbands;sb++)
{
if ((channel_mode==JOINT_STEREO) && (join[sb]==1))
{
sb_sample[blk][0][sb] = sb_sample[blk][0][sb] + sb_sample[blk][1][sb];
sb_sample[blk][1][sb] = sb_sample[blk][0][sb]2 * sb_sample[blk][1][sb];
}
}
}
  1. Synthesis Filter:多项滤波器组把每个子频带采样数据进行滤波器的合成为PCM数据输出给音频codec。

SBC编码

在这里插入图片描述
SBC编码过程跟解码是反过来的:

  1. Analysis Filter :把输入的PCM数据通过多项滤波器组转化为4或者8个子频带的频域数据;
  2. Scale Factors :计算每个子频段的比例因子,用4个比特表示,实际值是2的n次方,表示这个子频段的最高幅值 ;
  3. Bit Allocation:跟解码过程是一样的,跟scalefactors关联 , 每个子频段的幅值不一样,需要用到的bit数也不一样,为每个子频段分配幅值bit数的过程就是bit allocation;
  4. Quantization:量化,通过如下公式进行: quantized_sb_sample[blk][ch][sb] = ((sb_sample[blk][ch][sb] / scalefactor[ch][sb] + 1.0) * levels[ch][sb]) / 2.0
  5. BitStream packing,把每个子频段的frame_header和audio_sample打包到AVDTP数据包中。

SBC的压缩比

首先看一下SBC的比特率的计算:

bit_rate = 8 * frame_length* fs / nrof_subbands / nrof_blocks

如果不进行SBC编码,我们的bit_rate = 44100 x 16 = 705600 bit/s.
进行了SBC编码的bit_rate = 8 * 91 *44100/16/8 = 250818 bit/s
那么压缩比为:705600/250818 = 2.81

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

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

相关文章

接口(3)

大家好&#xff0c;今天我们接着上一篇的内容继续往下来看看&#xff0c;了解一下如何实现多个接口&#xff0c;与继承父类不同&#xff0c;一个类可以实现接口&#xff0c;那么话不多说&#xff0c;来看。 2.5实现多个接口. 在Java中,类和类之间是单继承的,一个类只能有一个…

MATLAB提供的颜色映射表colormap——伪彩色

图像处理领域的一个习惯&#xff1a;不是真实的颜色&#xff0c;一般用伪彩色。一是说明不是物体本身的颜色&#xff0c;二是彩色更容易分辨。 MATLAB陆续提供了16种颜色映射表colormap。 之前的都很丑&#xff0c;近5年新增的4种还可以。总的说来还是丑。 这是一种鸟的名字。…

网络安全 | 入侵检测系统(IDS)与入侵防御系统(IPS):如何识别并阻止威胁

网络安全 | 入侵检测系统&#xff08;IDS&#xff09;与入侵防御系统&#xff08;IPS&#xff09;&#xff1a;如何识别并阻止威胁 一、前言二、入侵检测系统&#xff08;IDS&#xff09;2.1 IDS 的工作原理2.2 IDS 的技术类型2.3 IDS 的部署方式 三、入侵防御系统&#xff08;…

工业“MCU+AI”

随着工业4.0的推进&#xff0c;传统工业设备正向智能化和自动化方向转型。这要求设备具备更高的算力、更强的实时处理能力以及支持AI算法的能力&#xff0c;以应对工业机器人、电机控制、预测性维护等复杂应用场景。 近年来越来越多的芯片厂商纷纷推出工业“MCUAI”产品&#…

【统计的思想】假设检验(二)

假设检验是根据人为设定的显著水平&#xff0c;对被测对象的总体质量特性进行统计推断的方法。 如果我们通过假设检验否定了零假设&#xff0c;只是说明在设定的显著水平下&#xff0c;零假设成立的概率比较小&#xff0c;并不是说零假设就肯定不成立。如果零假设事实上是成立…

[云讷科技]Kerloud Falcon四旋翼飞车虚拟仿真空间发布

虚拟仿真环境作为一个独立的专有软件包提供给我们的客户&#xff0c;用于帮助用户在实际测试之前验证自身的代码&#xff0c;并通过在仿真引擎中添加新的场景来探索新的飞行驾驶功能。 环境要求 由于环境依赖关系&#xff0c;虚拟仿真只能运行在装有Ubuntu 18.04的Intel-64位…

【数据结构】深入解析:构建父子节点树形数据结构并返回前端

树形数据结构列表 一、前言二、测试数据生成三、树形代码3.1、获取根节点3.2、遍历根节点&#xff0c;递归获取所有子节点3.3、排序3.4、完整代码 一、前言 返回前端VO对象中&#xff0c;有列情况列表展示需要带树形结构&#xff0c;例如基于RBAC权限模型中的菜单返回&#xf…

Versal - ChipScoPy + XSDB + Python CLI

目录 1. 简介 1.1 ChipScoPy 1.2 XSDB 1.3 Python CLI 2. 安装 ChipScoPy 2.1 Python 环境 2.2 设置虚拟环境 2.3 安装 ChipScoPy 2.4 安装依赖 2.5 安装示例 2.6 打开 JupyterLab 2.7 启动 hw/cs 服务 2.8 更新 ChipScoPy 2.9 参考资料 3. ChipScoPy 示例 3.1…

【优选算法】8----四数之和

有看过我上篇算法博客并且去做过的铁子们&#xff0c;对这道题的话应该就不会那么陌生了&#xff0c;因为这两道题 的解题思路有着异曲同工之妙~ -----------------------------------------begin------------------------------------- 题目解析&#xff1a; 跟三数之和就多了…

安卓日常问题杂谈(一)

背景 关于安卓开发中&#xff0c;有很多奇奇怪怪的问题&#xff0c;有时候这个控件闪一下&#xff0c;有时候这个页面移动一下&#xff0c;这些对于快速开发中&#xff0c;去查询&#xff0c;都是很耗费时间的&#xff0c;因此&#xff0c;本系列文章&#xff0c;旨在记录安卓…

Python3 OS模块中的文件/目录方法说明九

一. 简介 前面文章简单学习了 Python3 中 OS模块中的文件/目录的部分函数。 本文继续来学习 OS 模块中文件、目录的操作方法&#xff1a;os.pipe() 方法、os.popen() 方法。 二. Python3 OS模块中的文件/目录方法 1. os.pipe() 方法 os.pipe() 方法用于创建一个管道, 返回…

2025.1.20——四、[强网杯 2019]Upload1 文件上传|反序列化

题目来源&#xff1a;buuctf [强网杯 2019]Upload 1 目录 一、打开靶机&#xff0c;查看信息 二、解题思路 step 1&#xff1a;登陆进去看情况 step 2&#xff1a;大佬来支援——问题在cookie step 3&#xff1a;测试两个思路 1.目录穿越 2.目录扫描 step 4&#xff…

Go学习:iota枚举

iota注意事项&#xff1a; iota&#xff1a;常量自动生成器&#xff0c;每隔一行&#xff0c;自动累加iota给常量赋值使用iota 遇到 const&#xff0c;重置为 0可以只写一个iotaiota如果是同一行&#xff0c;值都一样 简单代码&#xff1a; package mainimport "fmt&qu…

2025 OWASP十大智能合约漏洞

随着去中心化金融&#xff08;DeFi&#xff09;和区块链技术的不断发展&#xff0c;智能合约安全的重要性愈发凸显。在此背景下&#xff0c;开放网络应用安全项目&#xff08;OWASP&#xff09;发布了备受期待的《2025年智能合约十大漏洞》报告。 这份最新报告反映了不断演变的…

双指针+前缀和习题(一步步讲解)

前言&#xff1a;如果解决下面这几道题有些问题&#xff0c;或者即使看了我画的过程图也不理解的可以去看看我的上一篇文章&#xff0c;有可能会对你有帮助。 一、《数值元素的目标和》---来自AcWing 数组元素的目标和 给定两个升序排序的有序数组 A和 B&#xff0c;以及一个…

单路由及双路由端口映射指南

远程登录总会遇到登陆不上的情况&#xff0c;可能是访问的大门没有打开哦&#xff0c;下面我们来看看具体是怎么回事&#xff1f; 当软件远程访问时&#xff0c;主机需要两个条件&#xff0c;一是有一个唯一的公网IP地址&#xff08;运营商提供&#xff09;&#xff0c;二是开…

Addressable学习

AssetsBundle是Unity的资源管理机制,将资源打包到AssetsBundle资源包并提供接口能从ab包里面加载资源出来。有了这个机制以后&#xff0c;我们要做资源管理&#xff0c;还需要做: a: 根据项目需求,编写编辑器扩展,提供指定资源打入对应bundle包工具策略; b: 根据项目的需求,资源…

大华相机DH-IPC-HFW3237M支持的ONVIF协议

使用libONVIF C库。 先发现相机。 配置 lib目录 包含 编译提示缺的文件&#xff0c;到libonvif里面拷贝过来。 改UDP端口 代码 使用msvc 2022的向导生成空项目&#xff0c;从项目的main示例拷贝过来。 CameraOnvif.h #pragma once#include <QObject> #include &l…

leetcode刷题记录(八十六)——84. 柱状图中最大的矩形

&#xff08;一&#xff09;问题描述 84. 柱状图中最大的矩形 - 力扣&#xff08;LeetCode&#xff09;84. 柱状图中最大的矩形 - 给定 n 个非负整数&#xff0c;用来表示柱状图中各个柱子的高度。每个柱子彼此相邻&#xff0c;且宽度为 1 。求在该柱状图中&#xff0c;能够勾…

centos9编译安装opensips 二【进阶篇-定制目录+模块】推荐

环境&#xff1a;centos9 last opensips -V version: opensips 3.6.0-dev (x86_64/linux) flags: STATS: On, DISABLE_NAGLE, USE_MCAST, SHM_MMAP, PKG_MALLOC, Q_MALLOC, F_MALLOC, HP_MALLOC, DBG_MALLOC, CC_O0, FAST_LOCK-ADAPTIVE_WAIT ADAPTIVE_WAIT_LOOPS1024, MAX_RE…