WaveOutXX 播放PCM流
2009-09-18 21:24
218 查看
最新在WM上开发音频想关的东西, 遇到的麻烦是播放流文件的时候会一顿一顿的,网上找了找答案,是由于没有用到双缓冲。网上遇到了一段代码,给了我很大帮助。在此分享。。。。
//waveout.h
#ifndef waveout_h
#define waveout_h
#pragma once
#include <mmsystem.h>
#include <list>
using namespace std;
#define BUFFER_SIZE 16
#define BLOCK_SIZE 1280 + sizeof(WAVEHDR)
class waveout
{
WAVEFORMATEX wave_format;
HWAVEOUT h_waveout;
public:
char m_buf[BUFFER_SIZE][BLOCK_SIZE];
list<char*>m_buf_free;
waveout();
~waveout();
bool start();
bool stop();
int input(unsigned char* buf);
bool inlineis_start(){return (h_waveout != NULL);}
};
#endif
//waveout.cpp
#include "stdafx.h"
#include "waveout.h"
waveout::waveout(){
wave_format.wFormatTag= WAVE_FORMAT_PCM;
wave_format.nChannels= 1;
wave_format.nSamplesPerSec= 16000;
wave_format.wBitsPerSample= 16;
wave_format.nAvgBytesPerSec= 32000;
wave_format.nBlockAlign= 2;
wave_format.cbSize= 0;
h_waveout = NULL;
}
waveout::~waveout(){
stop();
}
void __stdcall waveOutProc(HWAVEOUT hwo, UINT uMsg, DWORD dwInstance,DWORD dwParam1, DWORD dwParam2){
if(uMsg == WOM_DONE){
WAVEHDR* p_wavehdr = (WAVEHDR *) dwParam1;
waveout* p_wvo = (waveout *) dwInstance;
p_wvo->m_buf_free.push_back((char *) p_wavehdr);
}
}
int waveout::input(unsigned char* buf){
size_t nSize = m_buf_free.size();
if(nSize == 0)
return 0;
WAVEHDR *p_wavehdr = (WAVEHDR *) m_buf_free.front();
m_buf_free.pop_front();
memcpy(p_wavehdr->lpData, buf, 1280);
waveOutWrite(h_waveout, p_wavehdr, sizeof(WAVEHDR));
return 0;
}
bool waveout::start(){
if(h_waveout != NULL)
return true;
MMRESULT nRet = waveOutOpen(&h_waveout, WAVE_MAPPER, &wave_format, (DWORD) waveOutProc, (DWORD) this, CALLBACK_FUNCTION);
if(nRet != MMSYSERR_NOERROR)
return false;
for(int i = 0; i < BUFFER_SIZE; ++ i)
{
WAVEHDR *p_wavehdr = (WAVEHDR *)m_buf[i];
p_wavehdr->dwBufferLength = 1280;
p_wavehdr->lpData = m_buf[i] + sizeof(WAVEHDR);
p_wavehdr->dwFlags = 0;
waveOutPrepareHeader(h_waveout, p_wavehdr, sizeof(WAVEHDR));
m_buf_free.push_back((char *) p_wavehdr);
}
return true;
}
bool waveout::stop(){
if(h_waveout != NULL){
while(m_buf_free.size() != BUFFER_SIZE)
Sleep(80);
m_buf_free.clear();
waveOutReset(h_waveout);
for(int i = 0; i < BUFFER_SIZE; ++ i){
WAVEHDR *p_wavehdr = (WAVEHDR *)m_buf[i];
waveOutUnprepareHeader(h_waveout, p_wavehdr, sizeof(WAVEHDR));
}
waveOutClose(h_waveout);
h_waveout = NULL;
}
return true;
}
//waveout.h
#ifndef waveout_h
#define waveout_h
#pragma once
#include <mmsystem.h>
#include <list>
using namespace std;
#define BUFFER_SIZE 16
#define BLOCK_SIZE 1280 + sizeof(WAVEHDR)
class waveout
{
WAVEFORMATEX wave_format;
HWAVEOUT h_waveout;
public:
char m_buf[BUFFER_SIZE][BLOCK_SIZE];
list<char*>m_buf_free;
waveout();
~waveout();
bool start();
bool stop();
int input(unsigned char* buf);
bool inlineis_start(){return (h_waveout != NULL);}
};
#endif
//waveout.cpp
#include "stdafx.h"
#include "waveout.h"
waveout::waveout(){
wave_format.wFormatTag= WAVE_FORMAT_PCM;
wave_format.nChannels= 1;
wave_format.nSamplesPerSec= 16000;
wave_format.wBitsPerSample= 16;
wave_format.nAvgBytesPerSec= 32000;
wave_format.nBlockAlign= 2;
wave_format.cbSize= 0;
h_waveout = NULL;
}
waveout::~waveout(){
stop();
}
void __stdcall waveOutProc(HWAVEOUT hwo, UINT uMsg, DWORD dwInstance,DWORD dwParam1, DWORD dwParam2){
if(uMsg == WOM_DONE){
WAVEHDR* p_wavehdr = (WAVEHDR *) dwParam1;
waveout* p_wvo = (waveout *) dwInstance;
p_wvo->m_buf_free.push_back((char *) p_wavehdr);
}
}
int waveout::input(unsigned char* buf){
size_t nSize = m_buf_free.size();
if(nSize == 0)
return 0;
WAVEHDR *p_wavehdr = (WAVEHDR *) m_buf_free.front();
m_buf_free.pop_front();
memcpy(p_wavehdr->lpData, buf, 1280);
waveOutWrite(h_waveout, p_wavehdr, sizeof(WAVEHDR));
return 0;
}
bool waveout::start(){
if(h_waveout != NULL)
return true;
MMRESULT nRet = waveOutOpen(&h_waveout, WAVE_MAPPER, &wave_format, (DWORD) waveOutProc, (DWORD) this, CALLBACK_FUNCTION);
if(nRet != MMSYSERR_NOERROR)
return false;
for(int i = 0; i < BUFFER_SIZE; ++ i)
{
WAVEHDR *p_wavehdr = (WAVEHDR *)m_buf[i];
p_wavehdr->dwBufferLength = 1280;
p_wavehdr->lpData = m_buf[i] + sizeof(WAVEHDR);
p_wavehdr->dwFlags = 0;
waveOutPrepareHeader(h_waveout, p_wavehdr, sizeof(WAVEHDR));
m_buf_free.push_back((char *) p_wavehdr);
}
return true;
}
bool waveout::stop(){
if(h_waveout != NULL){
while(m_buf_free.size() != BUFFER_SIZE)
Sleep(80);
m_buf_free.clear();
waveOutReset(h_waveout);
for(int i = 0; i < BUFFER_SIZE; ++ i){
WAVEHDR *p_wavehdr = (WAVEHDR *)m_buf[i];
waveOutUnprepareHeader(h_waveout, p_wavehdr, sizeof(WAVEHDR));
}
waveOutClose(h_waveout);
h_waveout = NULL;
}
return true;
}
相关文章推荐
- waveout 播放流媒体
- WinAPI: waveOutGetPosition - 获取输出设备当前的播放位置
- WinAPI: waveOutGetPlaybackRate - 设置输出设备的播放速度(默认速度值的倍数)
- WaveOut播放声音死锁问题原因
- WaveOut播放音频
- Wave 文件(9): 使用 waveOut... 函数播放 wav 文件
- c++实现waveOutOpen音频播放功能
- 操作 Wave 文件(12): 使用 waveOut...重复播放 wav 文件
- waveOutOpen、waveOutWrite系统函数用法编程实现声音播放
- 操作 Wave 文件(9): 使用 waveOut... 函数播放 wav 文件
- Wave 文件(12): 使用 waveOut...重复播放 wav 文件
- waveout**实现音频播放
- 使用WaveOut API播放WAV音频文件(解决卡顿)
- waveout 播放声音
- 首战用DirectSound改造waveout,贴出成功播放声音的代码
- WinAPI: waveOutGetPlaybackRate - 获取输出设备当前的播放速度设置(默认速度值的倍数)
- 首战用DirectSound改造waveout,贴出成功播放声音的代码。
- c# 利用WaveOut播放音频流
- -XX:HeapDumpOnOutofMemoryError
- DShow之DSound与WaveOut的学习笔记