2010-11-20 10:37:14 +00:00
|
|
|
/* (c) Magnus Auvinen. See licence.txt in the root of the distribution for more information. */
|
|
|
|
/* If you are missing that file, acquire a complete release at teeworlds.com. */
|
2012-01-06 18:27:18 +00:00
|
|
|
#include <base/math.h>
|
2010-05-29 07:25:38 +00:00
|
|
|
#include <base/system.h>
|
2012-01-06 18:27:18 +00:00
|
|
|
|
2011-02-27 14:03:57 +00:00
|
|
|
#include <engine/graphics.h>
|
|
|
|
#include <engine/storage.h>
|
2012-01-06 18:27:18 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
#include <engine/shared/config.h>
|
|
|
|
|
|
|
|
#include "SDL.h"
|
|
|
|
|
|
|
|
#include "sound.h"
|
|
|
|
|
|
|
|
extern "C" { // wavpack
|
|
|
|
#include <engine/external/wavpack/wavpack.h>
|
|
|
|
}
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
NUM_SAMPLES = 512,
|
|
|
|
NUM_VOICES = 64,
|
|
|
|
NUM_CHANNELS = 16,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct CSample
|
|
|
|
{
|
|
|
|
short *m_pData;
|
|
|
|
int m_NumFrames;
|
|
|
|
int m_Rate;
|
|
|
|
int m_Channels;
|
|
|
|
int m_LoopStart;
|
|
|
|
int m_LoopEnd;
|
2011-07-02 11:11:32 +00:00
|
|
|
int m_PausedAt;
|
2010-05-29 07:25:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct CChannel
|
|
|
|
{
|
|
|
|
int m_Vol;
|
|
|
|
int m_Pan;
|
|
|
|
} ;
|
|
|
|
|
|
|
|
struct CVoice
|
|
|
|
{
|
|
|
|
CSample *m_pSample;
|
|
|
|
CChannel *m_pChannel;
|
|
|
|
int m_Tick;
|
|
|
|
int m_Vol; // 0 - 255
|
|
|
|
int m_Flags;
|
|
|
|
int m_X, m_Y;
|
|
|
|
} ;
|
|
|
|
|
|
|
|
static CSample m_aSamples[NUM_SAMPLES] = { {0} };
|
|
|
|
static CVoice m_aVoices[NUM_VOICES] = { {0} };
|
|
|
|
static CChannel m_aChannels[NUM_CHANNELS] = { {255, 0} };
|
|
|
|
|
|
|
|
static LOCK m_SoundLock = 0;
|
|
|
|
|
|
|
|
static int m_CenterX = 0;
|
|
|
|
static int m_CenterY = 0;
|
|
|
|
|
|
|
|
static int m_MixingRate = 48000;
|
|
|
|
static volatile int m_SoundVolume = 100;
|
|
|
|
|
|
|
|
static int m_NextVoice = 0;
|
2012-01-06 18:27:18 +00:00
|
|
|
static int *m_pMixBuffer = 0; // buffer only used by the thread callback function
|
|
|
|
static unsigned m_MaxFrames = 0;
|
2010-05-29 07:25:38 +00:00
|
|
|
|
|
|
|
// TODO: there should be a faster way todo this
|
|
|
|
static short Int2Short(int i)
|
|
|
|
{
|
|
|
|
if(i > 0x7fff)
|
|
|
|
return 0x7fff;
|
|
|
|
else if(i < -0x7fff)
|
|
|
|
return -0x7fff;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int IntAbs(int i)
|
|
|
|
{
|
|
|
|
if(i<0)
|
|
|
|
return -i;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void Mix(short *pFinalOut, unsigned Frames)
|
|
|
|
{
|
|
|
|
int MasterVol;
|
2012-01-06 18:27:18 +00:00
|
|
|
mem_zero(m_pMixBuffer, m_MaxFrames*2*sizeof(int));
|
|
|
|
Frames = min(Frames, m_MaxFrames);
|
2010-05-29 07:25:38 +00:00
|
|
|
|
|
|
|
// aquire lock while we are mixing
|
|
|
|
lock_wait(m_SoundLock);
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
MasterVol = m_SoundVolume;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
for(unsigned i = 0; i < NUM_VOICES; i++)
|
|
|
|
{
|
|
|
|
if(m_aVoices[i].m_pSample)
|
|
|
|
{
|
|
|
|
// mix voice
|
|
|
|
CVoice *v = &m_aVoices[i];
|
2012-01-06 18:27:18 +00:00
|
|
|
int *pOut = m_pMixBuffer;
|
2010-05-29 07:25:38 +00:00
|
|
|
|
|
|
|
int Step = v->m_pSample->m_Channels; // setup input sources
|
|
|
|
short *pInL = &v->m_pSample->m_pData[v->m_Tick*Step];
|
|
|
|
short *pInR = &v->m_pSample->m_pData[v->m_Tick*Step+1];
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
unsigned End = v->m_pSample->m_NumFrames-v->m_Tick;
|
|
|
|
|
|
|
|
int Rvol = v->m_pChannel->m_Vol;
|
|
|
|
int Lvol = v->m_pChannel->m_Vol;
|
|
|
|
|
|
|
|
// make sure that we don't go outside the sound data
|
|
|
|
if(Frames < End)
|
|
|
|
End = Frames;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// check if we have a mono sound
|
|
|
|
if(v->m_pSample->m_Channels == 1)
|
|
|
|
pInR = pInL;
|
|
|
|
|
|
|
|
// volume calculation
|
|
|
|
if(v->m_Flags&ISound::FLAG_POS && v->m_pChannel->m_Pan)
|
|
|
|
{
|
|
|
|
// TODO: we should respect the channel panning value
|
|
|
|
const int Range = 1500; // magic value, remove
|
|
|
|
int dx = v->m_X - m_CenterX;
|
|
|
|
int dy = v->m_Y - m_CenterY;
|
2010-08-16 00:21:18 +00:00
|
|
|
int Dist = (int)sqrtf((float)dx*dx+dy*dy); // float here. nasty
|
2010-05-29 07:25:38 +00:00
|
|
|
int p = IntAbs(dx);
|
2011-08-03 21:13:35 +00:00
|
|
|
if(Dist >= 0 && Dist < Range)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
|
|
|
// panning
|
|
|
|
if(dx > 0)
|
|
|
|
Lvol = ((Range-p)*Lvol)/Range;
|
|
|
|
else
|
|
|
|
Rvol = ((Range-p)*Rvol)/Range;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// falloff
|
|
|
|
Lvol = (Lvol*(Range-Dist))/Range;
|
|
|
|
Rvol = (Rvol*(Range-Dist))/Range;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Lvol = 0;
|
|
|
|
Rvol = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// process all frames
|
|
|
|
for(unsigned s = 0; s < End; s++)
|
|
|
|
{
|
|
|
|
*pOut++ += (*pInL)*Lvol;
|
|
|
|
*pOut++ += (*pInR)*Rvol;
|
|
|
|
pInL += Step;
|
|
|
|
pInR += Step;
|
|
|
|
v->m_Tick++;
|
|
|
|
}
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// free voice if not used any more
|
|
|
|
if(v->m_Tick == v->m_pSample->m_NumFrames)
|
2011-04-13 18:00:54 +00:00
|
|
|
{
|
|
|
|
if(v->m_Flags&ISound::FLAG_LOOP)
|
|
|
|
v->m_Tick = 0;
|
|
|
|
else
|
|
|
|
v->m_pSample = 0;
|
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-08-11 08:59:14 +00:00
|
|
|
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// release the lock
|
|
|
|
lock_release(m_SoundLock);
|
|
|
|
|
|
|
|
{
|
|
|
|
// clamp accumulated values
|
|
|
|
// TODO: this seams slow
|
|
|
|
for(unsigned i = 0; i < Frames; i++)
|
|
|
|
{
|
|
|
|
int j = i<<1;
|
2012-01-06 18:27:18 +00:00
|
|
|
int vl = ((m_pMixBuffer[j]*MasterVol)/101)>>8;
|
|
|
|
int vr = ((m_pMixBuffer[j+1]*MasterVol)/101)>>8;
|
2010-05-29 07:25:38 +00:00
|
|
|
|
|
|
|
pFinalOut[j] = Int2Short(vl);
|
|
|
|
pFinalOut[j+1] = Int2Short(vr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(CONF_ARCH_ENDIAN_BIG)
|
|
|
|
swap_endian(pFinalOut, sizeof(short), Frames * 2);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SdlCallback(void *pUnused, Uint8 *pStream, int Len)
|
|
|
|
{
|
|
|
|
(void)pUnused;
|
|
|
|
Mix((short *)pStream, Len/2/2);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int CSound::Init()
|
|
|
|
{
|
2011-06-27 20:57:06 +00:00
|
|
|
m_SoundEnabled = 0;
|
2010-05-29 07:25:38 +00:00
|
|
|
m_pGraphics = Kernel()->RequestInterface<IEngineGraphics>();
|
|
|
|
m_pStorage = Kernel()->RequestInterface<IStorage>();
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
SDL_AudioSpec Format;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_SoundLock = lock_create();
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(!g_Config.m_SndEnable)
|
|
|
|
return 0;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2012-01-03 20:39:10 +00:00
|
|
|
if(SDL_InitSubSystem(SDL_INIT_AUDIO) < 0)
|
|
|
|
{
|
|
|
|
dbg_msg("gfx", "unable to init SDL audio: %s", SDL_GetError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_MixingRate = g_Config.m_SndRate;
|
|
|
|
|
|
|
|
// Set 16-bit stereo audio at 22Khz
|
|
|
|
Format.freq = g_Config.m_SndRate; // ignore_convention
|
|
|
|
Format.format = AUDIO_S16; // ignore_convention
|
|
|
|
Format.channels = 2; // ignore_convention
|
|
|
|
Format.samples = g_Config.m_SndBufferSize; // ignore_convention
|
|
|
|
Format.callback = SdlCallback; // ignore_convention
|
|
|
|
Format.userdata = NULL; // ignore_convention
|
|
|
|
|
|
|
|
// Open the audio device and start playing sound!
|
|
|
|
if(SDL_OpenAudio(&Format, NULL) < 0)
|
|
|
|
{
|
|
|
|
dbg_msg("client/sound", "unable to open audio: %s", SDL_GetError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dbg_msg("client/sound", "sound init successful");
|
|
|
|
|
2012-01-06 18:27:18 +00:00
|
|
|
m_MaxFrames = g_Config.m_SndBufferSize*2;
|
|
|
|
m_pMixBuffer = (int *)mem_alloc(m_MaxFrames*2*sizeof(int), 1);
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
SDL_PauseAudio(0);
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_SoundEnabled = 1;
|
|
|
|
Update(); // update the volume
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSound::Update()
|
|
|
|
{
|
|
|
|
// update volume
|
|
|
|
int WantedVolume = g_Config.m_SndVolume;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(!m_pGraphics->WindowActive() && g_Config.m_SndNonactiveMute)
|
|
|
|
WantedVolume = 0;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(WantedVolume != m_SoundVolume)
|
|
|
|
{
|
|
|
|
lock_wait(m_SoundLock);
|
|
|
|
m_SoundVolume = WantedVolume;
|
|
|
|
lock_release(m_SoundLock);
|
|
|
|
}
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSound::Shutdown()
|
|
|
|
{
|
|
|
|
SDL_CloseAudio();
|
2012-01-03 20:39:10 +00:00
|
|
|
SDL_QuitSubSystem(SDL_INIT_AUDIO);
|
2010-05-29 07:25:38 +00:00
|
|
|
lock_destroy(m_SoundLock);
|
2012-01-06 18:27:18 +00:00
|
|
|
if(m_pMixBuffer)
|
|
|
|
{
|
|
|
|
mem_free(m_pMixBuffer);
|
|
|
|
m_pMixBuffer = 0;
|
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
int CSound::AllocID()
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
|
|
|
// TODO: linear search, get rid of it
|
2011-02-12 10:40:36 +00:00
|
|
|
for(unsigned SampleID = 0; SampleID < NUM_SAMPLES; SampleID++)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
if(m_aSamples[SampleID].m_pData == 0x0)
|
|
|
|
return SampleID;
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
void CSound::RateConvert(int SampleID)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
CSample *pSample = &m_aSamples[SampleID];
|
2010-05-29 07:25:38 +00:00
|
|
|
int NumFrames = 0;
|
|
|
|
short *pNewData = 0;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// make sure that we need to convert this sound
|
|
|
|
if(!pSample->m_pData || pSample->m_Rate == m_MixingRate)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// allocate new data
|
|
|
|
NumFrames = (int)((pSample->m_NumFrames/(float)pSample->m_Rate)*m_MixingRate);
|
|
|
|
pNewData = (short *)mem_alloc(NumFrames*pSample->m_Channels*sizeof(short), 1);
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
for(int i = 0; i < NumFrames; i++)
|
|
|
|
{
|
|
|
|
// resample TODO: this should be done better, like linear atleast
|
|
|
|
float a = i/(float)NumFrames;
|
|
|
|
int f = (int)(a*pSample->m_NumFrames);
|
|
|
|
if(f >= pSample->m_NumFrames)
|
|
|
|
f = pSample->m_NumFrames-1;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// set new data
|
|
|
|
if(pSample->m_Channels == 1)
|
|
|
|
pNewData[i] = pSample->m_pData[f];
|
|
|
|
else if(pSample->m_Channels == 2)
|
|
|
|
{
|
|
|
|
pNewData[i*2] = pSample->m_pData[f*2];
|
|
|
|
pNewData[i*2+1] = pSample->m_pData[f*2+1];
|
|
|
|
}
|
|
|
|
}
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// free old data and apply new
|
|
|
|
mem_free(pSample->m_pData);
|
|
|
|
pSample->m_pData = pNewData;
|
|
|
|
pSample->m_NumFrames = NumFrames;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSound::ReadData(void *pBuffer, int Size)
|
|
|
|
{
|
|
|
|
return io_read(ms_File, pBuffer, Size);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSound::LoadWV(const char *pFilename)
|
|
|
|
{
|
|
|
|
CSample *pSample;
|
2011-02-12 10:40:36 +00:00
|
|
|
int SampleID = -1;
|
2010-05-29 07:25:38 +00:00
|
|
|
char aError[100];
|
|
|
|
WavpackContext *pContext;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// don't waste memory on sound when we are stress testing
|
|
|
|
if(g_Config.m_DbgStress)
|
|
|
|
return -1;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// no need to load sound when we are running with no sound
|
|
|
|
if(!m_SoundEnabled)
|
|
|
|
return 1;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(!m_pStorage)
|
|
|
|
return -1;
|
|
|
|
|
2010-10-06 21:07:35 +00:00
|
|
|
ms_File = m_pStorage->OpenFile(pFilename, IOFLAG_READ, IStorage::TYPE_ALL);
|
2010-05-29 07:25:38 +00:00
|
|
|
if(!ms_File)
|
|
|
|
{
|
2011-01-06 21:18:19 +00:00
|
|
|
dbg_msg("sound/wv", "failed to open file. filename='%s'", pFilename);
|
2010-05-29 07:25:38 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
SampleID = AllocID();
|
|
|
|
if(SampleID < 0)
|
2010-05-29 07:25:38 +00:00
|
|
|
return -1;
|
2011-02-12 10:40:36 +00:00
|
|
|
pSample = &m_aSamples[SampleID];
|
2010-05-29 07:25:38 +00:00
|
|
|
|
|
|
|
pContext = WavpackOpenFileInput(ReadData, aError);
|
|
|
|
if (pContext)
|
|
|
|
{
|
|
|
|
int m_aSamples = WavpackGetNumSamples(pContext);
|
|
|
|
int BitsPerSample = WavpackGetBitsPerSample(pContext);
|
|
|
|
unsigned int SampleRate = WavpackGetSampleRate(pContext);
|
|
|
|
int m_aChannels = WavpackGetNumChannels(pContext);
|
|
|
|
int *pData;
|
|
|
|
int *pSrc;
|
|
|
|
short *pDst;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
pSample->m_Channels = m_aChannels;
|
|
|
|
pSample->m_Rate = SampleRate;
|
|
|
|
|
|
|
|
if(pSample->m_Channels > 2)
|
|
|
|
{
|
|
|
|
dbg_msg("sound/wv", "file is not mono or stereo. filename='%s'", pFilename);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
if(snd->rate != 44100)
|
|
|
|
{
|
|
|
|
dbg_msg("sound/wv", "file is %d Hz, not 44100 Hz. filename='%s'", snd->rate, filename);
|
|
|
|
return -1;
|
|
|
|
}*/
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(BitsPerSample != 16)
|
|
|
|
{
|
|
|
|
dbg_msg("sound/wv", "bps is %d, not 16, filname='%s'", BitsPerSample, pFilename);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
pData = (int *)mem_alloc(4*m_aSamples*m_aChannels, 1);
|
|
|
|
WavpackUnpackSamples(pContext, pData, m_aSamples); // TODO: check return value
|
|
|
|
pSrc = pData;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
pSample->m_pData = (short *)mem_alloc(2*m_aSamples*m_aChannels, 1);
|
|
|
|
pDst = pSample->m_pData;
|
|
|
|
|
|
|
|
for (i = 0; i < m_aSamples*m_aChannels; i++)
|
|
|
|
*pDst++ = (short)*pSrc++;
|
|
|
|
|
|
|
|
mem_free(pData);
|
|
|
|
|
|
|
|
pSample->m_NumFrames = m_aSamples;
|
|
|
|
pSample->m_LoopStart = -1;
|
|
|
|
pSample->m_LoopEnd = -1;
|
2011-07-02 11:11:32 +00:00
|
|
|
pSample->m_PausedAt = 0;
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dbg_msg("sound/wv", "failed to open %s: %s", pFilename, aError);
|
|
|
|
}
|
|
|
|
|
|
|
|
io_close(ms_File);
|
|
|
|
ms_File = NULL;
|
|
|
|
|
|
|
|
if(g_Config.m_Debug)
|
|
|
|
dbg_msg("sound/wv", "loaded %s", pFilename);
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
RateConvert(SampleID);
|
|
|
|
return SampleID;
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSound::SetListenerPos(float x, float y)
|
|
|
|
{
|
|
|
|
m_CenterX = (int)x;
|
|
|
|
m_CenterY = (int)y;
|
|
|
|
}
|
2011-04-13 18:00:54 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
void CSound::SetChannel(int ChannelID, float Vol, float Pan)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
m_aChannels[ChannelID].m_Vol = (int)(Vol*255.0f);
|
|
|
|
m_aChannels[ChannelID].m_Pan = (int)(Pan*255.0f); // TODO: this is only on and off right now
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
int CSound::Play(int ChannelID, int SampleID, int Flags, float x, float y)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
int VoiceID = -1;
|
2010-05-29 07:25:38 +00:00
|
|
|
int i;
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2014-03-22 09:24:10 +00:00
|
|
|
if(SampleID == 107) // GetSampleID(SOUND_CHAT_SERVER)
|
|
|
|
{
|
|
|
|
if(!g_Config.m_SndServerMessage)
|
|
|
|
return VoiceID;
|
|
|
|
}
|
|
|
|
else if(SampleID == 108) // GetSampleID(SOUND_CHAT_CLIENT)
|
2014-05-04 16:35:37 +00:00
|
|
|
{}
|
2014-03-22 09:24:10 +00:00
|
|
|
else if(SampleID == 109) // GetSampleID(SOUND_CHAT_HIGHLIGHT)
|
|
|
|
{
|
|
|
|
if(!g_Config.m_SndHighlight)
|
|
|
|
return VoiceID;
|
|
|
|
}
|
|
|
|
else if(!g_Config.m_SndGame)
|
2013-10-21 00:16:45 +00:00
|
|
|
return VoiceID;
|
|
|
|
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
lock_wait(m_SoundLock);
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// search for voice
|
|
|
|
for(i = 0; i < NUM_VOICES; i++)
|
|
|
|
{
|
|
|
|
int id = (m_NextVoice + i) % NUM_VOICES;
|
|
|
|
if(!m_aVoices[id].m_pSample)
|
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
VoiceID = id;
|
2010-05-29 07:25:38 +00:00
|
|
|
m_NextVoice = id+1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
// voice found, use it
|
2011-02-12 10:40:36 +00:00
|
|
|
if(VoiceID != -1)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
m_aVoices[VoiceID].m_pSample = &m_aSamples[SampleID];
|
|
|
|
m_aVoices[VoiceID].m_pChannel = &m_aChannels[ChannelID];
|
2011-07-02 11:11:32 +00:00
|
|
|
if(Flags & FLAG_LOOP)
|
|
|
|
m_aVoices[VoiceID].m_Tick = m_aSamples[SampleID].m_PausedAt;
|
|
|
|
else
|
|
|
|
m_aVoices[VoiceID].m_Tick = 0;
|
2011-02-12 10:40:36 +00:00
|
|
|
m_aVoices[VoiceID].m_Vol = 255;
|
|
|
|
m_aVoices[VoiceID].m_Flags = Flags;
|
|
|
|
m_aVoices[VoiceID].m_X = (int)x;
|
|
|
|
m_aVoices[VoiceID].m_Y = (int)y;
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
2011-08-11 08:59:14 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
lock_release(m_SoundLock);
|
2011-02-12 10:40:36 +00:00
|
|
|
return VoiceID;
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
int CSound::PlayAt(int ChannelID, int SampleID, int Flags, float x, float y)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
return Play(ChannelID, SampleID, Flags|ISound::FLAG_POS, x, y);
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
int CSound::Play(int ChannelID, int SampleID, int Flags)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
return Play(ChannelID, SampleID, Flags, 0, 0);
|
2010-05-29 07:25:38 +00:00
|
|
|
}
|
|
|
|
|
2011-04-13 18:00:54 +00:00
|
|
|
void CSound::Stop(int SampleID)
|
2010-05-29 07:25:38 +00:00
|
|
|
{
|
|
|
|
// TODO: a nice fade out
|
|
|
|
lock_wait(m_SoundLock);
|
2011-04-13 18:00:54 +00:00
|
|
|
CSample *pSample = &m_aSamples[SampleID];
|
|
|
|
for(int i = 0; i < NUM_VOICES; i++)
|
|
|
|
{
|
|
|
|
if(m_aVoices[i].m_pSample == pSample)
|
2011-07-02 11:11:32 +00:00
|
|
|
{
|
|
|
|
if(m_aVoices[i].m_Flags & FLAG_LOOP)
|
|
|
|
m_aVoices[i].m_pSample->m_PausedAt = m_aVoices[i].m_Tick;
|
|
|
|
else
|
|
|
|
m_aVoices[i].m_pSample->m_PausedAt = 0;
|
2011-04-13 18:00:54 +00:00
|
|
|
m_aVoices[i].m_pSample = 0;
|
2011-07-02 11:11:32 +00:00
|
|
|
}
|
2011-04-13 18:00:54 +00:00
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
lock_release(m_SoundLock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSound::StopAll()
|
|
|
|
{
|
|
|
|
// TODO: a nice fade out
|
|
|
|
lock_wait(m_SoundLock);
|
|
|
|
for(int i = 0; i < NUM_VOICES; i++)
|
|
|
|
{
|
2011-07-02 11:11:32 +00:00
|
|
|
if(m_aVoices[i].m_pSample)
|
|
|
|
{
|
|
|
|
if(m_aVoices[i].m_Flags & FLAG_LOOP)
|
|
|
|
m_aVoices[i].m_pSample->m_PausedAt = m_aVoices[i].m_Tick;
|
|
|
|
else
|
|
|
|
m_aVoices[i].m_pSample->m_PausedAt = 0;
|
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
m_aVoices[i].m_pSample = 0;
|
|
|
|
}
|
|
|
|
lock_release(m_SoundLock);
|
|
|
|
}
|
|
|
|
|
|
|
|
IOHANDLE CSound::ms_File = 0;
|
|
|
|
|
|
|
|
IEngineSound *CreateEngineSound() { return new CSound; }
|
|
|
|
|