mirror of
https://github.com/RaySollium99/picodrive.git
synced 2025-09-05 15:27:46 -04:00

git-svn-id: file:///home/notaz/opt/svn/PicoDrive@341 be3aeb3a-fb24-0410-a615-afba39da0efa
136 lines
3.2 KiB
C++
136 lines
3.2 KiB
C++
|
|
#include "app.h"
|
|
|
|
#ifndef _XBOX
|
|
#pragma warning (disable:4201)
|
|
#include <mmsystem.h>
|
|
#include <dsound.h>
|
|
#endif
|
|
|
|
static IDirectSound *DSound=NULL;
|
|
static IDirectSoundBuffer *LoopBuffer=NULL;
|
|
static int LoopLen=0,LoopWrite=0; // Next position in loop to write
|
|
|
|
short *DSoundNext=NULL; // Buffer for next sound data to put in loop
|
|
|
|
static int LoopBlank()
|
|
{
|
|
void *mema=NULL,*memb=NULL;
|
|
DWORD sizea=0,sizeb=0;
|
|
|
|
LoopBuffer->Lock(0,LoopLen<<2, &mema,&sizea, &memb,&sizeb, 0);
|
|
|
|
if (mema) memset(mema,0,sizea);
|
|
|
|
LoopBuffer->Unlock(mema,sizea, memb,sizeb);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DSoundInit()
|
|
{
|
|
DSBUFFERDESC dsbd;
|
|
WAVEFORMATEX wfx;
|
|
|
|
memset(&dsbd,0,sizeof(dsbd));
|
|
memset(&wfx,0,sizeof(wfx));
|
|
|
|
// Make wave format:
|
|
wfx.wFormatTag=WAVE_FORMAT_PCM;
|
|
wfx.nChannels=(unsigned short)((PicoOpt&8) ? 2 : 1); // Stereo/mono
|
|
wfx.nSamplesPerSec=PsndRate;
|
|
wfx.wBitsPerSample=16;
|
|
|
|
wfx.nBlockAlign=(WORD)((wfx.nChannels*wfx.wBitsPerSample)>>3);
|
|
wfx.nAvgBytesPerSec=wfx.nBlockAlign*wfx.nSamplesPerSec;
|
|
|
|
// Make buffer for the next seg to put into the loop:
|
|
DSoundNext=(short *)malloc((PsndLen<<2)+64); if (DSoundNext==NULL) return 1;
|
|
memset(DSoundNext,0,PsndLen<<2);
|
|
|
|
// Create the DirectSound interface:
|
|
DirectSoundCreate(NULL,&DSound,NULL);
|
|
if (DSound==NULL) return 1;
|
|
|
|
LoopLen=PsndLen<<1; // 2 segs
|
|
|
|
#ifndef _XBOX
|
|
LoopLen<<=1; // 4 segs
|
|
DSound->SetCooperativeLevel(FrameWnd,DSSCL_PRIORITY);
|
|
dsbd.dwFlags=DSBCAPS_GLOBALFOCUS; // Play in background
|
|
#endif
|
|
|
|
// Create the looping buffer:
|
|
dsbd.dwSize=sizeof(dsbd);
|
|
dsbd.dwBufferBytes=LoopLen<<wfx.nChannels; // 16bit stereo?
|
|
dsbd.lpwfxFormat=&wfx;
|
|
|
|
DSound->CreateSoundBuffer(&dsbd,&LoopBuffer,NULL);
|
|
if (LoopBuffer==NULL) return 1;
|
|
|
|
LoopBlank();
|
|
LoopBuffer->Play(0,0,DSBPLAY_LOOPING);
|
|
return 0;
|
|
}
|
|
|
|
void DSoundExit()
|
|
{
|
|
if (LoopBuffer) LoopBuffer->Stop();
|
|
RELEASE(LoopBuffer)
|
|
RELEASE(DSound)
|
|
free(DSoundNext); DSoundNext=NULL;
|
|
}
|
|
|
|
static int WriteSeg()
|
|
{
|
|
void *mema=NULL,*memb=NULL;
|
|
DWORD sizea=0,sizeb=0;
|
|
|
|
// Lock the segment at 'LoopWrite' and copy the next segment in
|
|
LoopBuffer->Lock(LoopWrite<<((PicoOpt&8) ? 2 : 1),PsndLen<<((PicoOpt&8) ? 2 : 1), &mema,&sizea, &memb,&sizeb, 0);
|
|
|
|
//dprintf2("lock %p, cpy %x\n", mema, sizea);
|
|
|
|
if (mema) memcpy(mema,DSoundNext,sizea);
|
|
// if (memb) memcpy(memb,DSoundNext+sizea,sizeb);
|
|
|
|
LoopBuffer->Unlock(mema,sizea, memb,0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DSoundUpdate()
|
|
{
|
|
DWORD play=0;
|
|
int pos=0;
|
|
|
|
if (LoopBuffer==NULL) return -1;
|
|
|
|
LoopBuffer->GetCurrentPosition(&play,NULL);
|
|
pos=play>>((PicoOpt&8) ? 2 : 1);
|
|
|
|
//dprintf2("loop %i pos %i\n", LoopWrite, pos);
|
|
|
|
// 'LoopWrite' is the next seg in the loop that we want to write
|
|
// First check that the sound 'play' pointer has moved out of it:
|
|
if (pos>=LoopWrite && pos<LoopWrite+PsndLen) return 1; // No, it hasn't
|
|
|
|
WriteSeg();
|
|
|
|
// Advance LoopWrite to next seg:
|
|
LoopWrite+=PsndLen; if (LoopWrite+PsndLen>LoopLen) LoopWrite=0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void DSoundMute()
|
|
{
|
|
if (LoopBuffer==NULL) return;
|
|
LoopBuffer->Stop();
|
|
}
|
|
|
|
void DSoundUnMute()
|
|
{
|
|
if (LoopBuffer==NULL) return;
|
|
LoopBuffer->Play(0,0,DSBPLAY_LOOPING);
|
|
}
|