2017-04-27 04:31:01 +00:00
|
|
|
#include "b3SoundEngine.h"
|
|
|
|
|
|
|
|
#include "RtAudio.h"
|
|
|
|
|
|
|
|
#include "b3AudioListener.h"
|
|
|
|
#include "b3SoundSource.h"
|
|
|
|
#include "Bullet3Common/b3AlignedObjectArray.h"
|
2017-04-28 18:35:00 +00:00
|
|
|
#include "b3ReadWavFile.h"
|
|
|
|
#include "../Utils/b3ResourcePath.h"
|
2017-05-01 18:14:09 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
#include "Bullet3Common/b3HashMap.h"
|
2017-04-27 04:31:01 +00:00
|
|
|
|
|
|
|
// The default real-time audio input and output buffer size. If
|
|
|
|
// clicks are occuring in the input and/or output sound stream, a
|
|
|
|
// larger buffer size may help. Larger buffer sizes, however, produce
|
|
|
|
// more latency.
|
|
|
|
//const unsigned int RT_BUFFER_SIZE = 1024;
|
|
|
|
const unsigned int RT_BUFFER_SIZE = 256;
|
|
|
|
|
|
|
|
struct b3SoundEngineInternalData
|
|
|
|
{
|
|
|
|
b3AudioListener m_listener;
|
|
|
|
RtAudio m_dac;
|
2017-04-28 18:35:00 +00:00
|
|
|
bool m_useRealTimeDac;
|
2017-04-27 04:31:01 +00:00
|
|
|
|
|
|
|
b3AlignedObjectArray<b3SoundSource*> m_soundSources;
|
2017-04-28 18:35:00 +00:00
|
|
|
b3HashMap<b3HashInt, b3ReadWavFile*> m_wavFiles;
|
2017-05-01 18:14:09 +00:00
|
|
|
b3HashMap<b3HashString, int> m_name2wav;
|
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
int m_wavFileUidGenerator;
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
b3SoundEngineInternalData()
|
|
|
|
: m_useRealTimeDac(false),
|
|
|
|
m_wavFileUidGenerator(123)
|
|
|
|
{
|
|
|
|
}
|
2017-04-27 04:31:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
b3SoundEngine::b3SoundEngine()
|
|
|
|
{
|
|
|
|
m_data = new b3SoundEngineInternalData();
|
|
|
|
}
|
|
|
|
|
|
|
|
b3SoundEngine::~b3SoundEngine()
|
|
|
|
{
|
|
|
|
exit();
|
|
|
|
delete m_data;
|
|
|
|
}
|
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
void b3SoundEngine::init(int maxNumSoundSources, bool useRealTimeDac)
|
2017-04-27 04:31:01 +00:00
|
|
|
{
|
2017-04-28 18:35:00 +00:00
|
|
|
for (int i = 0; i < maxNumSoundSources; i++)
|
|
|
|
{
|
|
|
|
b3SoundSource* source = new b3SoundSource();
|
|
|
|
m_data->m_soundSources.push_back(source);
|
|
|
|
m_data->m_listener.addSoundSource(source);
|
|
|
|
}
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
this->m_data->m_useRealTimeDac = useRealTimeDac;
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
if (useRealTimeDac)
|
|
|
|
{
|
|
|
|
RtAudioFormat format = (sizeof(double) == 8) ? RTAUDIO_FLOAT64 : RTAUDIO_FLOAT32;
|
|
|
|
RtAudio::StreamParameters parameters;
|
|
|
|
parameters.deviceId = m_data->m_dac.getDefaultOutputDevice();
|
|
|
|
parameters.nChannels = 2;
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
unsigned int bufferFrames = RT_BUFFER_SIZE;
|
|
|
|
double sampleRate = m_data->m_listener.getSampleRate();
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
m_data->m_dac.openStream(¶meters, NULL, format, (unsigned int)sampleRate, &bufferFrames, &b3AudioListener::tick,
|
|
|
|
(void*)m_data->m_listener.getTickData());
|
|
|
|
|
|
|
|
m_data->m_dac.startStream();
|
|
|
|
}
|
2017-04-27 04:31:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void b3SoundEngine::exit()
|
|
|
|
{
|
|
|
|
m_data->m_dac.closeStream();
|
2017-04-28 18:35:00 +00:00
|
|
|
m_data->m_useRealTimeDac = false;
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
for (int i = 0; i < m_data->m_soundSources.size(); i++)
|
2017-04-27 04:31:01 +00:00
|
|
|
{
|
|
|
|
m_data->m_listener.removeSoundSource(m_data->m_soundSources[i]);
|
2017-04-28 18:35:00 +00:00
|
|
|
m_data->m_soundSources[i]->stopSound();
|
2017-04-27 04:31:01 +00:00
|
|
|
delete m_data->m_soundSources[i];
|
|
|
|
}
|
|
|
|
m_data->m_soundSources.clear();
|
2017-05-01 18:14:09 +00:00
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
for (int i = 0; i < m_data->m_wavFiles.size(); i++)
|
2017-05-01 18:14:09 +00:00
|
|
|
{
|
|
|
|
b3ReadWavFile** wavPtr = m_data->m_wavFiles.getAtIndex(i);
|
|
|
|
if (wavPtr && *wavPtr)
|
|
|
|
{
|
|
|
|
b3ReadWavFile* wav = *wavPtr;
|
|
|
|
delete wav;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_data->m_wavFiles.clear();
|
|
|
|
m_data->m_name2wav.clear();
|
2017-04-27 04:31:01 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
int b3SoundEngine::getAvailableSoundSource()
|
|
|
|
{
|
|
|
|
for (int i = 0; i < m_data->m_soundSources.size(); i++)
|
|
|
|
{
|
|
|
|
if (m_data->m_soundSources[i]->isAvailable())
|
|
|
|
{
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
2017-04-27 04:31:01 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
void b3SoundEngine::startSound(int soundSourceIndex, b3SoundMessage msg)
|
2017-04-27 04:31:01 +00:00
|
|
|
{
|
2017-04-28 18:35:00 +00:00
|
|
|
b3SoundSource* soundSource = m_data->m_soundSources[soundSourceIndex];
|
2018-09-23 21:17:31 +00:00
|
|
|
soundSource->setOscillatorAmplitude(0, msg.m_amplitude);
|
|
|
|
soundSource->setOscillatorAmplitude(1, msg.m_amplitude);
|
2017-04-28 19:39:51 +00:00
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
soundSource->setADSR(msg.m_attackRate, msg.m_decayRate, msg.m_sustainLevel, msg.m_releaseRate);
|
2017-04-28 19:39:51 +00:00
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
switch (msg.m_type)
|
|
|
|
{
|
|
|
|
case B3_SOUND_SOURCE_SINE_OSCILLATOR:
|
|
|
|
{
|
|
|
|
soundSource->setOscillatorFrequency(0, msg.m_frequency);
|
|
|
|
soundSource->setOscillatorFrequency(1, msg.m_frequency);
|
2018-09-23 21:17:31 +00:00
|
|
|
|
2017-05-01 18:14:09 +00:00
|
|
|
soundSource->startSound(msg.m_autoKeyOff);
|
2017-04-28 18:35:00 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case B3_SOUND_SOURCE_WAV_FILE:
|
|
|
|
{
|
|
|
|
b3ReadWavFile** wavFilePtr = m_data->m_wavFiles[msg.m_wavId];
|
|
|
|
if (wavFilePtr)
|
|
|
|
{
|
|
|
|
b3ReadWavFile* wavFile = *wavFilePtr;
|
2018-09-23 21:17:31 +00:00
|
|
|
soundSource->setWavFile(0, wavFile, getSampleRate());
|
|
|
|
soundSource->setWavFile(1, wavFile, getSampleRate());
|
2017-05-01 18:14:09 +00:00
|
|
|
soundSource->startSound(msg.m_autoKeyOff);
|
2017-04-28 18:35:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
2017-04-27 04:31:01 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 18:35:00 +00:00
|
|
|
void b3SoundEngine::releaseSound(int soundSourceIndex)
|
2017-04-27 04:31:01 +00:00
|
|
|
{
|
2017-04-28 18:35:00 +00:00
|
|
|
b3SoundSource* soundSource = m_data->m_soundSources[soundSourceIndex];
|
|
|
|
soundSource->stopSound();
|
2017-04-27 04:31:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int b3SoundEngine::loadWavFile(const char* fileName)
|
|
|
|
{
|
2017-05-01 18:14:09 +00:00
|
|
|
int* wavUidPtr = m_data->m_name2wav[fileName];
|
|
|
|
if (wavUidPtr)
|
|
|
|
{
|
|
|
|
return *wavUidPtr;
|
|
|
|
}
|
2017-04-28 18:35:00 +00:00
|
|
|
char resourcePath[1024];
|
|
|
|
|
|
|
|
if (b3ResourcePath::findResourcePath(fileName, resourcePath, 1024))
|
|
|
|
{
|
|
|
|
b3ReadWavFile* wavFile = new b3ReadWavFile();
|
|
|
|
wavFile->getWavInfo(resourcePath);
|
|
|
|
wavFile->resize();
|
|
|
|
wavFile->read(0, true);
|
|
|
|
wavFile->normalize(1);
|
|
|
|
int wavUID = m_data->m_wavFileUidGenerator++;
|
|
|
|
m_data->m_wavFiles.insert(wavUID, wavFile);
|
2018-09-23 21:17:31 +00:00
|
|
|
m_data->m_name2wav.insert(fileName, wavUID);
|
2017-04-28 18:35:00 +00:00
|
|
|
return wavUID;
|
|
|
|
}
|
2017-04-27 04:31:01 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
double b3SoundEngine::getSampleRate() const
|
|
|
|
{
|
|
|
|
return m_data->m_listener.getSampleRate();
|
|
|
|
}
|