|
@@ -207,7 +207,7 @@ void TraceLog(int msgType, const char *text, ...); // Show trace lo
|
|
|
#endif
|
|
|
|
|
|
//----------------------------------------------------------------------------------
|
|
|
-// miniaudio AudioBuffer Functionality
|
|
|
+// AudioBuffer Functionality
|
|
|
//----------------------------------------------------------------------------------
|
|
|
#define DEVICE_FORMAT ma_format_f32
|
|
|
#define DEVICE_CHANNELS 2
|
|
@@ -269,24 +269,21 @@ void SetAudioBufferPitch(AudioBuffer *audioBuffer, float pitch);
|
|
|
void TrackAudioBuffer(AudioBuffer *audioBuffer);
|
|
|
void UntrackAudioBuffer(AudioBuffer *audioBuffer);
|
|
|
|
|
|
-
|
|
|
//----------------------------------------------------------------------------------
|
|
|
-// multi channel playback globals
|
|
|
+// Multi channel playback globals
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
|
|
-// number of channels in the pool
|
|
|
-#define PLAY_POOL_SIZE 16
|
|
|
-
|
|
|
-// the buffer pool
|
|
|
-AudioBuffer* PlayBufferPool[PLAY_POOL_SIZE];
|
|
|
+// Number of channels in the audio pool
|
|
|
+#define MAX_AUDIO_BUFFER_POOL_CHANNELS 16
|
|
|
|
|
|
-// these are used to determine the oldest playing channel
|
|
|
-unsigned long PlayPoolAge = 0;
|
|
|
-unsigned long PlayPoolAges[PLAY_POOL_SIZE] = {0};
|
|
|
+// Audio buffer pool
|
|
|
+AudioBuffer *audioBufferPool[MAX_AUDIO_BUFFER_POOL_CHANNELS] = { 0 };
|
|
|
|
|
|
+// These are used to determine the oldest playing channel
|
|
|
+unsigned long audioBufferPoolCounter = 0;
|
|
|
+unsigned long audioBufferPoolChannels[MAX_AUDIO_BUFFER_POOL_CHANNELS] = { 0 };
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
|
|
-
|
|
|
// Log callback function
|
|
|
static void OnLog(ma_context *pContext, ma_device *pDevice, ma_uint32 logLevel, const char *message)
|
|
|
{
|
|
@@ -481,12 +478,13 @@ static void MixAudioFrames(float *framesOut, const float *framesIn, ma_uint32 fr
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-// initialise the multichannel buffer pool
|
|
|
-static void InitPlayBufferPool()
|
|
|
+// Initialise the multichannel buffer pool
|
|
|
+static void InitAudioBufferPool()
|
|
|
{
|
|
|
- // dummy buffers
|
|
|
- for (int i=0; i<PLAY_POOL_SIZE; i++) {
|
|
|
- PlayBufferPool[i] = CreateAudioBuffer(DEVICE_FORMAT, DEVICE_CHANNELS, DEVICE_SAMPLE_RATE, 0, AUDIO_BUFFER_USAGE_STATIC);
|
|
|
+ // Dummy buffers
|
|
|
+ for (int i = 0; i < MAX_AUDIO_BUFFER_POOL_CHANNELS; i++)
|
|
|
+ {
|
|
|
+ audioBufferPool[i] = CreateAudioBuffer(DEVICE_FORMAT, DEVICE_CHANNELS, DEVICE_SAMPLE_RATE, 0, AUDIO_BUFFER_USAGE_STATIC);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -554,17 +552,17 @@ void InitAudioDevice(void)
|
|
|
TraceLog(LOG_INFO, "Audio sample rate: %d -> %d", device.sampleRate, device.playback.internalSampleRate);
|
|
|
TraceLog(LOG_INFO, "Audio buffer size: %d", device.playback.internalBufferSizeInFrames);
|
|
|
|
|
|
- InitPlayBufferPool();
|
|
|
- TraceLog(LOG_INFO, "Audio multichannel pool size: %i", PLAY_POOL_SIZE);
|
|
|
+ InitAudioBufferPool();
|
|
|
+ TraceLog(LOG_INFO, "Audio multichannel pool size: %i", MAX_AUDIO_BUFFER_POOL_CHANNELS);
|
|
|
|
|
|
isAudioInitialized = MA_TRUE;
|
|
|
}
|
|
|
|
|
|
// internal
|
|
|
-static void FreePlayBufferPool() {
|
|
|
- for (int i = 0; i < PLAY_POOL_SIZE; i++) {
|
|
|
+static void FreeaudioBufferPool() {
|
|
|
+ for (int i = 0; i < MAX_AUDIO_BUFFER_POOL_CHANNELS; i++) {
|
|
|
// NB important free only the buffer struct not the attached data...!
|
|
|
- RL_FREE(PlayBufferPool[i]);
|
|
|
+ RL_FREE(audioBufferPool[i]);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -582,7 +580,7 @@ void CloseAudioDevice(void)
|
|
|
ma_device_uninit(&device);
|
|
|
ma_context_uninit(&context);
|
|
|
|
|
|
- FreePlayBufferPool();
|
|
|
+ FreeaudioBufferPool();
|
|
|
|
|
|
TraceLog(LOG_INFO, "Audio device closed successfully");
|
|
|
}
|
|
@@ -1010,79 +1008,87 @@ void PlaySound(Sound sound)
|
|
|
PlayAudioBuffer((AudioBuffer *)sound.audioBuffer);
|
|
|
}
|
|
|
|
|
|
-// play a sound in the multichannel buffer pool
|
|
|
-void PlaySoundEx(Sound s)
|
|
|
+// Play a sound in the multichannel buffer pool
|
|
|
+void PlaySoundMulti(Sound sound)
|
|
|
{
|
|
|
- int found = -1;
|
|
|
+ int index = -1;
|
|
|
unsigned long oldAge = 0;
|
|
|
int oldIndex = -1;
|
|
|
|
|
|
// find the first non playing pool entry
|
|
|
- for (int i=0; i<PLAY_POOL_SIZE; i++) {
|
|
|
- if (PlayPoolAges[i] > oldAge) {
|
|
|
- oldAge = PlayPoolAges[i];
|
|
|
+ for (int i = 0; i < MAX_AUDIO_BUFFER_POOL_CHANNELS; i++)
|
|
|
+ {
|
|
|
+ if (audioBufferPoolChannels[i] > oldAge)
|
|
|
+ {
|
|
|
+ oldAge = audioBufferPoolChannels[i];
|
|
|
oldIndex = i;
|
|
|
}
|
|
|
- if (!IsAudioBufferPlaying(PlayBufferPool[i])) {
|
|
|
- found = i;
|
|
|
+
|
|
|
+ if (!IsAudioBufferPlaying(audioBufferPool[i]))
|
|
|
+ {
|
|
|
+ index = i;
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- // if no none playing pool members can be found choose the oldest
|
|
|
- if (found == -1) {
|
|
|
- TraceLog(LOG_WARNING,"pool age %i ended a sound early no room in buffer pool",PlayPoolAge);
|
|
|
- if (oldIndex == -1) {
|
|
|
- // shouldn't be able to get here... but just in case something odd happens!
|
|
|
+ // If no none playing pool members can be index choose the oldest
|
|
|
+ if (index == -1)
|
|
|
+ {
|
|
|
+ TraceLog(LOG_WARNING,"pool age %i ended a sound early no room in buffer pool", audioBufferPoolCounter);
|
|
|
+
|
|
|
+ if (oldIndex == -1)
|
|
|
+ {
|
|
|
+ // Shouldn't be able to get here... but just in case something odd happens!
|
|
|
TraceLog(LOG_ERROR,"sound buffer pool couldn't determine oldest buffer not playing sound");
|
|
|
+
|
|
|
return;
|
|
|
}
|
|
|
- found = oldIndex;
|
|
|
- // just in case...
|
|
|
- StopAudioBuffer(PlayBufferPool[found]);
|
|
|
+
|
|
|
+ index = oldIndex;
|
|
|
+
|
|
|
+ // Just in case...
|
|
|
+ StopAudioBuffer(audioBufferPool[index]);
|
|
|
}
|
|
|
|
|
|
- // experimentally mutex lock doesn't seem to be needed this makes sense
|
|
|
- // as PlayBufferPool[found] isn't playing and the only stuff we're copying
|
|
|
+ // Experimentally mutex lock doesn't seem to be needed this makes sense
|
|
|
+ // as audioBufferPool[index] isn't playing and the only stuff we're copying
|
|
|
// shouldn't be changing...
|
|
|
|
|
|
- PlayPoolAges[found] = PlayPoolAge;
|
|
|
- PlayPoolAge++;
|
|
|
- PlayBufferPool[found]->volume = ((AudioBuffer*)s.audioBuffer)->volume;
|
|
|
- PlayBufferPool[found]->pitch = ((AudioBuffer*)s.audioBuffer)->pitch;
|
|
|
- PlayBufferPool[found]->looping = ((AudioBuffer*)s.audioBuffer)->looping;
|
|
|
- PlayBufferPool[found]->usage = ((AudioBuffer*)s.audioBuffer)->usage;
|
|
|
- PlayBufferPool[found]->isSubBufferProcessed[0] = false;
|
|
|
- PlayBufferPool[found]->isSubBufferProcessed[1] = false;
|
|
|
- PlayBufferPool[found]->bufferSizeInFrames = ((AudioBuffer*)s.audioBuffer)->bufferSizeInFrames;
|
|
|
- PlayBufferPool[found]->buffer = ((AudioBuffer*)s.audioBuffer)->buffer;
|
|
|
+ audioBufferPoolChannels[index] = audioBufferPoolCounter;
|
|
|
+ audioBufferPoolCounter++;
|
|
|
+
|
|
|
+ audioBufferPool[index]->volume = ((AudioBuffer*)sound.audioBuffer)->volume;
|
|
|
+ audioBufferPool[index]->pitch = ((AudioBuffer*)sound.audioBuffer)->pitch;
|
|
|
+ audioBufferPool[index]->looping = ((AudioBuffer*)sound.audioBuffer)->looping;
|
|
|
+ audioBufferPool[index]->usage = ((AudioBuffer*)sound.audioBuffer)->usage;
|
|
|
+ audioBufferPool[index]->isSubBufferProcessed[0] = false;
|
|
|
+ audioBufferPool[index]->isSubBufferProcessed[1] = false;
|
|
|
+ audioBufferPool[index]->bufferSizeInFrames = ((AudioBuffer*)sound.audioBuffer)->bufferSizeInFrames;
|
|
|
+ audioBufferPool[index]->buffer = ((AudioBuffer*)sound.audioBuffer)->buffer;
|
|
|
|
|
|
- PlayAudioBuffer(PlayBufferPool[found]);
|
|
|
+ PlayAudioBuffer(audioBufferPool[index]);
|
|
|
|
|
|
}
|
|
|
|
|
|
-// MUST be called before UnLoadSound is used on any sound played with PlaySoundEx
|
|
|
-void StopPlayBufferPool()
|
|
|
+// Stop any sound played with PlaySoundMulti()
|
|
|
+void StopSoundMulti(void)
|
|
|
{
|
|
|
- for (int i = 0; i < PLAY_POOL_SIZE; i++) {
|
|
|
- StopAudioBuffer(PlayBufferPool[i]);
|
|
|
- }
|
|
|
+ for (int i = 0; i < MAX_AUDIO_BUFFER_POOL_CHANNELS; i++) StopAudioBuffer(audioBufferPool[i]);
|
|
|
}
|
|
|
|
|
|
-// number of sounds playing in the multichannel buffer pool
|
|
|
-int ConcurrentPlayChannels()
|
|
|
+// Get number of sounds playing in the multichannel buffer pool
|
|
|
+int GetSoundsPlaying(void)
|
|
|
{
|
|
|
- int n = 0;
|
|
|
- for (int i=0; i<PLAY_POOL_SIZE; i++) {
|
|
|
- if (IsAudioBufferPlaying(PlayBufferPool[i])) {
|
|
|
- n++;
|
|
|
- }
|
|
|
+ int counter = 0;
|
|
|
+
|
|
|
+ for (int i = 0; i < MAX_AUDIO_BUFFER_POOL_CHANNELS; i++)
|
|
|
+ {
|
|
|
+ if (IsAudioBufferPlaying(audioBufferPool[i])) counter++;
|
|
|
}
|
|
|
- return n;
|
|
|
+
|
|
|
+ return counter;
|
|
|
}
|
|
|
|
|
|
-
|
|
|
-
|
|
|
// Pause a sound
|
|
|
void PauseSound(Sound sound)
|
|
|
{
|