c – 使用Speex和DirectSound进行声学回声消除(AEC)

我正在尝试使用Speex编解码器库执行声学回声消除(AEC).根据Speex文档,我需要执行两个调用
speex_echo_playback(echo_state,echo_frame);

每次播放音频帧时,和

speex_echo_capture(echo_state,input_frame,output_frame);

对于每个捕获的帧.

由于我使用的是DirectSound,我以为我可以在调用speex_echo_playback时使用主DirectSound缓冲区作为echo_frame,例如,

DWORD offset = 0;
  DWORD length = 0;
  LPVOID block1,block2;
  DWORD length1,length2;
  DWORD flags = DSBLOCK_ENTIREBUFFER;

  HRESULT hr = primary_buffer->Lock(
        offset,length,&block1,&length1,&block2,&length2,flags
      );

  // Would like to convert the buffer into a form that
  // speex_echo_capture() can use.
  // Why does length1 == length2 == 0 always? 

  hr = primary_buffer->Unlock( block1,length1,block2,length2 );

文档确实说这些是只写指针,但是无论如何都不能自己使用缓冲区数据?

这基本上就是我创建缓冲区的方式:

CComPtr< IDirectSoundBuffer > primary_buffer;
  DSBUFFERDESC primarydesc = { sizeof( DSBUFFERDESC ),DSBCAPS_PRIMARYBUFFER | DSBCAPS_CTRL3D | DSBCAPS_LOCHARDWARE,NULL,DS3DALG_HRTF_LIGHT };
  HRESULT hr = directsound_->CreateSoundBuffer(
      &primarydesc,&primary_buffer,NULL );

看来,使用DirectSound缓冲区本身的替代方法是使用speex_decode()的输出并进行自己的软件混合.

让Speex和DirectSound一起工作的任何指示或建议?谢谢你的帮助.

解决方法

我做过一次.但我的方法如下:

我从未直接使用过主缓冲区.相反,我只使用一个辅助缓冲区.我有两个线程 – 回放线程和捕获线程.另外,我使用了另一个speex函数speex_echo_cancellation.

因此,在我的回放线程中,我将当前回放帧保存在全局缓冲区和捕获线程中,称为speex_echo_cancellation函数,具有当前捕获帧和先前存储的回放帧.

DMO不适合我,因为我也必须支持Windows XP.

您还可以通过0700或更好的方式查看subscribe here以获得更多有趣的信息.

祝好运,

安东尼

相关文章

对象的传值与返回说起函数,就不免要谈谈函数的参数和返回值...
从实现装饰者模式中思考C++指针和引用的选择最近在看...
关于vtordisp知多少?我相信不少人看到这篇文章,多半是来自...
那些陌生的C++关键字学过程序语言的人相信对关键字并...
命令行下的树形打印最近在处理代码分析问题时,需要将代码的...
虚函数与虚继承寻踪封装、继承、多态是面向对象语言的三大特...