最近用libmad做了些mp3解碼的工作,順便也研究了一下windows下播放PCM音頻資料的雙緩衝用法。
libmad的調用在此暫略過不表。
libmad解碼出來的是16bit的PCM資料,調用windows API可對其實現播放。不過如果解碼一段播放一段,聽起來會有一頓一頓的感覺,不流暢,究其原因,是沒有使用雙緩衝。
吭哧吭哧研究了半天,終於編碼實現,播放出來的效果倒也很流暢。
流程如下:
1)聲明兩個WAVEHDR結構waveHeader1,waveHeader2,並分別對其lpData參數分配緩衝buf1,buf2;
2)聲明WAVEFORMATEX結構waveFormat,以及HWAVEOUT結構hWaveOut。調用函數
waveOutOpen( &hWaveOut, WAVE_MAPPER, &waveFormat, (DWORD)waveOutProc, NULL, CALLBACK_FUNCTION );
waveOutProc為回呼函數,在後面會提到。
也說一下waveFormat的各參數。根據MSDN解釋,nChannels為通道數,nSamplesPerSec為採樣率,wFormatTag的值為WAVE_FORMAT_PCM,wBitsPerSample為16,nBlockAlign為nChannels*wBitsPerSample/8,nAvgBytesPerSec為nSamplesPerSec*nBlockAlign;
3)讀入buf1,buf2,並設定好相應長度;
4)將waveHeader1,waveHeader2寫入wave裝置:
waveOutPrepareHeader( hWaveOut, &waveHeader1, sizeof(WAVEHDR));
waveOutPrepareHeader( hWaveOut, &waveHeader2, sizeof(WAVEHDR));
waveOutWrite( hWaveOut, &waveHeader1, sizeof(WAVEHDR) );
waveOutWrite( hWaveOut, &waveHeader2, sizeof(WAVEHDR) );
5)關於回調
void CALLBACK waveOutProc( HWAVEOUT hwo,
UINT uMsg,
DWORD dwInstance,
DWORD dwParam1,
DWORD dwParam2 )
{
if(uMsg == WOM_DONE)
{
LPWAVEHDR pWaveHeader = (LPWAVEHDR)dwParam1;//系統自動識別是哪一個WAVEHDR播放完畢
waveOutUnprepareHeader( hwo, pWaveHeader, sizeof(WAVEHDR) );//播放完後須調用此函數
//此處填充WAVEHDR的lpdate緩衝
waveOutPrepareHeader( hwo, pWaveHeader, sizeof(WAVEHDR));
waveOutWrite( hwo, pWaveHeader, sizeof(WAVEHDR) );
//...
}
return ;
}
6)播放完畢後,調用waveOutClose,釋放緩衝。其他的一些waveOut函數,如waveOutPause、waveOutReset等等,在做播放器的時候會用得到。若播放過程中終止,須先調用waveOutReset,再調用waveOutClose。