pcm 转 g711a(输入13位编成8位)
1.将pcm二进制 转成 排列:x xx…x1 xxxx xxxx…
2.符号位取反
3.根据从高位起,第一个 1 的位数 查表
4.wxyz为 第一个 1 后四位
5.获取完 seeewxyz 之后,除符号位外,其他位 偶数位取反 奇数位保留。
例:
输入pcm数据为3210,二进制对应为(0000 1100 1000 1010)
二进制变换下排列组合方式(0 0001 1001 0001010)
(1) 获取符号位最高位为0,取反,s=1
(2) 获取强度位0001,查表,编码制应该是eee=100
(3) 获取高位样本wxyz=1001
(4) 组合为11001001,逢偶数为取反为10011100
编码完毕。
pcm 转 G711u (输入14位转成8位)
这个编码完全就是查表
1.根据输入的值 通过查表找到基础值,间隔值和区间基本值
2.区间基本值减去当前值 得到 差异值
3.差异值 除于 间隔值 取整 得偏移值
4.编码结果为 基础值 加 偏移值
代码实现:
static int ulaw2linear(unsigned char u_val)
{
int t;
u_val = ~u_val;
t = ((u_val & QUANT_MASK) << 3) + BIAS;
t <<= ((unsigned)u_val & SEG_MASK) >> SEG_SHIFT;
return ((u_val & SIGN_BIT) ? (BIAS - t) : (t - BIAS));
}
static int alaw2linear(unsigned char a_val)
{
int t;
int seg;
a_val ^= 0x55;
t = (a_val & QUANT_MASK) << 4;
seg = ((unsigned)a_val & SEG_MASK) >> SEG_SHIFT;
switch (seg)
{
case 0:
t += 8;
break;
case 1:
t += 0x108;
break;
default:
t += 0x108;
t <<= seg - 1;
}
return ((a_val & SIGN_BIT) ? t : -t);
}
补充理解:pcm 是 完整无损的音频采样方法,但为了传输,结合应用的场景不同,就有不同的编码方式。编码简单点来说,就是在保证音频不失真的情况下,把音频流按需压缩,利于传输。
而在程序使用时,步骤是 fopen fread 把各种编码音频文件读出来,在对其进行相应解码,得到最初的pcm数据,之后将得到的pcm数据 推到 音频驱动程序中。录音则反之。