G711编解码

时间:2021-10-08 07:54:31

http://blog.csdn.net/rightorwrong/article/details/4209467

搞语音对讲几天了,播放时声音干扰太大了。拖得时间久有两个原因:

1.每次采样的位数这个值设置的问题。本来是用的采样位数为16,但是服务端那边说用8。导致编解码时8位的始终有杂音。

1.G711编解码的问题:用了一个错误的编解码程序。目前用的编解码代码对于8为始终有杂音

下面把编解码的代码放在这里,16为采集效果很好

  1. //编码
  2. int CG711Decoder::G711_EnCode(unsigned char* pCodecBits, const char* pBuffer, int nBufferSize)
  3. {
  4. short* buffer = (short*)pBuffer;
  5. for(int i=0; i<nBufferSize/2; i++)
  6. {
  7. pCodecBits[i] = encode(buffer[i]);
  8. }
  9. return nBufferSize/2;
  10. }
  11. //解码
  12. int CG711Decoder::G711_Decode(char* pRawData, const unsigned char* pBuffer, int nBufferSize)
  13. {
  14. short *out_data = (short*)pRawData;
  15. for(int i=0; i<nBufferSize; i++)
  16. {
  17. out_data[i] = decode(pBuffer[i]);
  18. }
  19. return nBufferSize*2;
  20. }
  21. #define MAX (32635)
  22. unsigned char CG711Decoder::encode(short pcm)
  23. {
  24. int sign = (pcm & 0x8000) >> 8;
  25. if (sign != 0)
  26. pcm = -pcm;
  27. if (pcm > MAX) pcm = MAX;
  28. int exponent = 7;
  29. int expMask;
  30. for (expMask = 0x4000; (pcm & expMask) == 0
  31. && exponent>0; exponent--, expMask >>= 1) { }
  32. int mantissa = (pcm >> ((exponent == 0) ? 4 : (exponent + 3))) & 0x0f;
  33. unsigned char alaw = (unsigned char)(sign | exponent << 4 | mantissa);
  34. return (unsigned char)(alaw^0xD5);
  35. }
  36. short CG711Decoder::decode(unsigned char alaw)
  37. {
  38. alaw ^= 0xD5;
  39. int sign = alaw & 0x80;
  40. int exponent = (alaw & 0x70) >> 4;
  41. int data = alaw & 0x0f;
  42. data <<= 4;
  43. data += 8;
  44. if (exponent != 0)
  45. data += 0x100;
  46. if (exponent > 1)
  47. data <<= (exponent - 1);
  48. return (short)(sign == 0 ? data : -data);
  49. }