add multi channel audio to raudio (#895)
* added multi channel sound replay to raudio added -fPIC to Makefile for Linux added simple lighting and audio multi channel to examples Makefile * not properly reporting audio buffer pool size...
This commit is contained in:
parent
a4e307ed96
commit
83a4eb0852
6 changed files with 259 additions and 14 deletions
|
@ -435,10 +435,12 @@ EXAMPLES = \
|
||||||
shaders/shaders_texture_waves \
|
shaders/shaders_texture_waves \
|
||||||
shaders/shaders_julia_set \
|
shaders/shaders_julia_set \
|
||||||
shaders/shaders_eratosthenes \
|
shaders/shaders_eratosthenes \
|
||||||
|
shaders/shaders_basic_lighting \
|
||||||
audio/audio_module_playing \
|
audio/audio_module_playing \
|
||||||
audio/audio_music_stream \
|
audio/audio_music_stream \
|
||||||
audio/audio_raw_stream \
|
audio/audio_raw_stream \
|
||||||
audio/audio_sound_loading \
|
audio/audio_sound_loading \
|
||||||
|
audio/audio_multichannel_sound \
|
||||||
physac/physics_demo \
|
physac/physics_demo \
|
||||||
physac/physics_friction \
|
physac/physics_friction \
|
||||||
physac/physics_movement \
|
physac/physics_movement \
|
||||||
|
|
116
examples/audio/audio_multichannel_sound.c
Normal file
116
examples/audio/audio_multichannel_sound.c
Normal file
|
@ -0,0 +1,116 @@
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This example was coded to demonstrate multi channel audio changes added to raylib
|
||||||
|
*
|
||||||
|
* This example has been created using raylib (www.raylib.com)
|
||||||
|
* raylib is licensed under an unmodified zlib/libpng license (View raylib.h for details)
|
||||||
|
*
|
||||||
|
* This example Copyright (c) 2018 Chris Camacho (codifies) http://bedroomcoders.co.uk/captcha/
|
||||||
|
*
|
||||||
|
* THIS EXAMPLE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
* IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
||||||
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||||||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||||||
|
* OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*
|
||||||
|
* This example may be freely redistributed.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "raylib.h"
|
||||||
|
#include <stdio.h> // sprintf
|
||||||
|
|
||||||
|
int main(void)
|
||||||
|
{
|
||||||
|
// Initialization
|
||||||
|
//--------------------------------------------------------------------------------------
|
||||||
|
const int screenWidth = 800;
|
||||||
|
const int screenHeight = 450;
|
||||||
|
|
||||||
|
InitWindow(screenWidth, screenHeight, "raylib [audio] abuse!");
|
||||||
|
|
||||||
|
InitAudioDevice(); // Initialize audio device
|
||||||
|
|
||||||
|
Sound fxWav = LoadSound("resources/sound.wav"); // Load WAV audio file
|
||||||
|
Sound fxOgg = LoadSound("resources/tanatana.ogg"); // Load OGG audio file
|
||||||
|
|
||||||
|
|
||||||
|
//InitPlayBufferPool();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
SetTargetFPS(60); // Set our game to run at 60 frames-per-second
|
||||||
|
//--------------------------------------------------------------------------------------
|
||||||
|
int frame = 0;
|
||||||
|
// Main game loop
|
||||||
|
|
||||||
|
// old system still works alongside
|
||||||
|
SetSoundVolume(fxWav, 0.2); // effects all mutltiplay replay (really annoying sound!!!)
|
||||||
|
PlaySound(fxOgg);
|
||||||
|
|
||||||
|
bool inhibitWav = false;
|
||||||
|
bool inhibitOgg = false;
|
||||||
|
int maxFrame = 60;
|
||||||
|
int numberPlaying = 0;
|
||||||
|
|
||||||
|
while (!WindowShouldClose()) // Detect window close button or ESC key
|
||||||
|
{
|
||||||
|
// Update
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
frame++;
|
||||||
|
|
||||||
|
if (IsKeyDown(KEY_ENTER)) { inhibitWav=true; } else { inhibitWav=false; }
|
||||||
|
if (IsKeyDown(KEY_SPACE)) { inhibitOgg=true; } else { inhibitOgg=false; }
|
||||||
|
|
||||||
|
// deliberatly hammer the play pool to see what dropping old
|
||||||
|
// pool entries sounds like....
|
||||||
|
if (frame % 5==0) {
|
||||||
|
if (!inhibitWav) PlaySoundEx(fxWav);
|
||||||
|
}
|
||||||
|
if (frame==maxFrame) {
|
||||||
|
if (!inhibitOgg) PlaySoundEx(fxOgg);
|
||||||
|
frame=0;
|
||||||
|
maxFrame = GetRandomValue(6,12);
|
||||||
|
}
|
||||||
|
|
||||||
|
numberPlaying = ConcurrentPlayChannels();
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
// Draw
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
BeginDrawing();
|
||||||
|
|
||||||
|
ClearBackground(RAYWHITE);
|
||||||
|
char msg[1024];
|
||||||
|
DrawText("multichannel sound abuse!", 200, 180, 20, LIGHTGRAY);
|
||||||
|
DrawText("Space to inhibit new ogg triggering", 200, 200, 20, LIGHTGRAY);
|
||||||
|
DrawText("Enter to inhibit new wav triggering", 200, 220, 20, LIGHTGRAY);
|
||||||
|
|
||||||
|
sprintf(msg,"concurrently playing %i", numberPlaying);
|
||||||
|
DrawText(msg, 200, 280, 20, LIGHTGRAY);
|
||||||
|
|
||||||
|
EndDrawing();
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
}
|
||||||
|
|
||||||
|
// De-Initialization
|
||||||
|
//--------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
// you MUST stop the buffer pool before unloading
|
||||||
|
// any Sounds it might have used...
|
||||||
|
StopPlayBufferPool();
|
||||||
|
|
||||||
|
UnloadSound(fxWav); // Unload sound data
|
||||||
|
UnloadSound(fxOgg); // Unload sound data
|
||||||
|
|
||||||
|
CloseAudioDevice(); // Close audio device
|
||||||
|
|
||||||
|
CloseWindow(); // Close window and OpenGL context
|
||||||
|
//--------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
|
@ -258,6 +258,10 @@ endif
|
||||||
# -fno-strict-aliasing jar_xm.h does shady stuff (breaks strict aliasing)
|
# -fno-strict-aliasing jar_xm.h does shady stuff (breaks strict aliasing)
|
||||||
CFLAGS += -Wall -std=c99 -D_DEFAULT_SOURCE -Wno-missing-braces -Werror=pointer-arith -fno-strict-aliasing
|
CFLAGS += -Wall -std=c99 -D_DEFAULT_SOURCE -Wno-missing-braces -Werror=pointer-arith -fno-strict-aliasing
|
||||||
|
|
||||||
|
ifeq ($(PLATFORM_OS),LINUX)
|
||||||
|
CFLAGS += -fPIC
|
||||||
|
endif
|
||||||
|
|
||||||
ifeq ($(RAYLIB_BUILD_MODE),DEBUG)
|
ifeq ($(RAYLIB_BUILD_MODE),DEBUG)
|
||||||
CFLAGS += -g
|
CFLAGS += -g
|
||||||
endif
|
endif
|
||||||
|
|
123
src/raudio.c
123
src/raudio.c
|
@ -178,10 +178,11 @@ typedef enum {
|
||||||
} TraceLogType;
|
} TraceLogType;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//----------------------------------------------------------------------------------
|
//----------------------------------------------------------------------------------
|
||||||
// Global Variables Definition
|
// Global Variables Definition
|
||||||
//----------------------------------------------------------------------------------
|
//----------------------------------------------------------------------------------
|
||||||
// ...
|
|
||||||
|
|
||||||
//----------------------------------------------------------------------------------
|
//----------------------------------------------------------------------------------
|
||||||
// Module specific Functions Declaration
|
// Module specific Functions Declaration
|
||||||
|
@ -230,7 +231,7 @@ struct rAudioBuffer {
|
||||||
unsigned int bufferSizeInFrames;
|
unsigned int bufferSizeInFrames;
|
||||||
rAudioBuffer *next;
|
rAudioBuffer *next;
|
||||||
rAudioBuffer *prev;
|
rAudioBuffer *prev;
|
||||||
unsigned char buffer[1];
|
unsigned char *buffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
// HACK: To avoid CoreAudio (macOS) symbol collision
|
// HACK: To avoid CoreAudio (macOS) symbol collision
|
||||||
|
@ -268,6 +269,24 @@ void SetAudioBufferPitch(AudioBuffer *audioBuffer, float pitch);
|
||||||
void TrackAudioBuffer(AudioBuffer *audioBuffer);
|
void TrackAudioBuffer(AudioBuffer *audioBuffer);
|
||||||
void UntrackAudioBuffer(AudioBuffer *audioBuffer);
|
void UntrackAudioBuffer(AudioBuffer *audioBuffer);
|
||||||
|
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
// multi channel playback globals
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// number of channels in the pool
|
||||||
|
#define PLAY_POOL_SIZE 16
|
||||||
|
|
||||||
|
// the buffer pool
|
||||||
|
AudioBuffer* PlayBufferPool[PLAY_POOL_SIZE];
|
||||||
|
|
||||||
|
// these are used to determine the oldest playing channel
|
||||||
|
unsigned long PlayPoolAge = 0;
|
||||||
|
unsigned long PlayPoolAges[PLAY_POOL_SIZE] = {0};
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
// Log callback function
|
// Log callback function
|
||||||
static void OnLog(ma_context *pContext, ma_device *pDevice, ma_uint32 logLevel, const char *message)
|
static void OnLog(ma_context *pContext, ma_device *pDevice, ma_uint32 logLevel, const char *message)
|
||||||
{
|
{
|
||||||
|
@ -462,6 +481,15 @@ static void MixAudioFrames(float *framesOut, const float *framesIn, ma_uint32 fr
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// initialise the multichannel buffer pool
|
||||||
|
static void InitPlayBufferPool()
|
||||||
|
{
|
||||||
|
// 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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------------------
|
//----------------------------------------------------------------------------------
|
||||||
// Module Functions Definition - Audio Device initialization and Closing
|
// Module Functions Definition - Audio Device initialization and Closing
|
||||||
//----------------------------------------------------------------------------------
|
//----------------------------------------------------------------------------------
|
||||||
|
@ -526,9 +554,21 @@ void InitAudioDevice(void)
|
||||||
TraceLog(LOG_INFO, "Audio sample rate: %d -> %d", device.sampleRate, device.playback.internalSampleRate);
|
TraceLog(LOG_INFO, "Audio sample rate: %d -> %d", device.sampleRate, device.playback.internalSampleRate);
|
||||||
TraceLog(LOG_INFO, "Audio buffer size: %d", device.playback.internalBufferSizeInFrames);
|
TraceLog(LOG_INFO, "Audio buffer size: %d", device.playback.internalBufferSizeInFrames);
|
||||||
|
|
||||||
|
InitPlayBufferPool();
|
||||||
|
TraceLog(LOG_INFO, "Audio multichannel pool size: %i", PLAY_POOL_SIZE);
|
||||||
|
|
||||||
isAudioInitialized = MA_TRUE;
|
isAudioInitialized = MA_TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// internal
|
||||||
|
static void FreePlayBufferPool() {
|
||||||
|
for (int i = 0; i < PLAY_POOL_SIZE; i++) {
|
||||||
|
// NB important free only the buffer struct not the attached data...!
|
||||||
|
RL_FREE(PlayBufferPool[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Close the audio device for all contexts
|
// Close the audio device for all contexts
|
||||||
void CloseAudioDevice(void)
|
void CloseAudioDevice(void)
|
||||||
{
|
{
|
||||||
|
@ -542,6 +582,8 @@ void CloseAudioDevice(void)
|
||||||
ma_device_uninit(&device);
|
ma_device_uninit(&device);
|
||||||
ma_context_uninit(&context);
|
ma_context_uninit(&context);
|
||||||
|
|
||||||
|
FreePlayBufferPool();
|
||||||
|
|
||||||
TraceLog(LOG_INFO, "Audio device closed successfully");
|
TraceLog(LOG_INFO, "Audio device closed successfully");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -567,7 +609,8 @@ void SetMasterVolume(float volume)
|
||||||
// Create a new audio buffer. Initially filled with silence
|
// Create a new audio buffer. Initially filled with silence
|
||||||
AudioBuffer *CreateAudioBuffer(ma_format format, ma_uint32 channels, ma_uint32 sampleRate, ma_uint32 bufferSizeInFrames, AudioBufferUsage usage)
|
AudioBuffer *CreateAudioBuffer(ma_format format, ma_uint32 channels, ma_uint32 sampleRate, ma_uint32 bufferSizeInFrames, AudioBufferUsage usage)
|
||||||
{
|
{
|
||||||
AudioBuffer *audioBuffer = (AudioBuffer *)RL_CALLOC(sizeof(*audioBuffer) + (bufferSizeInFrames*channels*ma_get_bytes_per_sample(format)), 1);
|
AudioBuffer *audioBuffer = (AudioBuffer *)RL_CALLOC(sizeof(*audioBuffer), 1);
|
||||||
|
audioBuffer->buffer = RL_CALLOC((bufferSizeInFrames*channels*ma_get_bytes_per_sample(format)), 1);
|
||||||
if (audioBuffer == NULL)
|
if (audioBuffer == NULL)
|
||||||
{
|
{
|
||||||
TraceLog(LOG_ERROR, "CreateAudioBuffer() : Failed to allocate memory for audio buffer");
|
TraceLog(LOG_ERROR, "CreateAudioBuffer() : Failed to allocate memory for audio buffer");
|
||||||
|
@ -623,6 +666,7 @@ void DeleteAudioBuffer(AudioBuffer *audioBuffer)
|
||||||
}
|
}
|
||||||
|
|
||||||
UntrackAudioBuffer(audioBuffer);
|
UntrackAudioBuffer(audioBuffer);
|
||||||
|
RL_FREE(audioBuffer->buffer);
|
||||||
RL_FREE(audioBuffer);
|
RL_FREE(audioBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -966,6 +1010,79 @@ void PlaySound(Sound sound)
|
||||||
PlayAudioBuffer((AudioBuffer *)sound.audioBuffer);
|
PlayAudioBuffer((AudioBuffer *)sound.audioBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// play a sound in the multichannel buffer pool
|
||||||
|
void PlaySoundEx(Sound s)
|
||||||
|
{
|
||||||
|
int found = -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];
|
||||||
|
oldIndex = i;
|
||||||
|
}
|
||||||
|
if (!IsAudioBufferPlaying(PlayBufferPool[i])) {
|
||||||
|
found = 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!
|
||||||
|
TraceLog(LOG_ERROR,"sound buffer pool couldn't determine oldest buffer not playing sound");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
found = oldIndex;
|
||||||
|
// just in case...
|
||||||
|
StopAudioBuffer(PlayBufferPool[found]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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
|
||||||
|
// 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;
|
||||||
|
|
||||||
|
PlayAudioBuffer(PlayBufferPool[found]);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// MUST be called before UnLoadSound is used on any sound played with PlaySoundEx
|
||||||
|
void StopPlayBufferPool()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < PLAY_POOL_SIZE; i++) {
|
||||||
|
StopAudioBuffer(PlayBufferPool[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// number of sounds playing in the multichannel buffer pool
|
||||||
|
int ConcurrentPlayChannels()
|
||||||
|
{
|
||||||
|
int n = 0;
|
||||||
|
for (int i=0; i<PLAY_POOL_SIZE; i++) {
|
||||||
|
if (IsAudioBufferPlaying(PlayBufferPool[i])) {
|
||||||
|
n++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Pause a sound
|
// Pause a sound
|
||||||
void PauseSound(Sound sound)
|
void PauseSound(Sound sound)
|
||||||
{
|
{
|
||||||
|
|
|
@ -139,6 +139,9 @@ void UpdateSound(Sound sound, const void *data, int samplesCount);// Update soun
|
||||||
void UnloadWave(Wave wave); // Unload wave data
|
void UnloadWave(Wave wave); // Unload wave data
|
||||||
void UnloadSound(Sound sound); // Unload sound
|
void UnloadSound(Sound sound); // Unload sound
|
||||||
void PlaySound(Sound sound); // Play a sound
|
void PlaySound(Sound sound); // Play a sound
|
||||||
|
void PlaySoundEx(Sound s); // Play a sound using the multi channel buffer pool
|
||||||
|
void StopPlayBufferPool(); // MUST be called before UnLoadSound is used on any sound played with PlaySoundEx
|
||||||
|
int ConcurrentPlayChannels(); // Number of sounds playing in the multichannel buffer pool
|
||||||
void PauseSound(Sound sound); // Pause a sound
|
void PauseSound(Sound sound); // Pause a sound
|
||||||
void ResumeSound(Sound sound); // Resume a paused sound
|
void ResumeSound(Sound sound); // Resume a paused sound
|
||||||
void StopSound(Sound sound); // Stop playing a sound
|
void StopSound(Sound sound); // Stop playing a sound
|
||||||
|
|
|
@ -1356,6 +1356,9 @@ RLAPI void ExportWaveAsCode(Wave wave, const char *fileName); // Export
|
||||||
|
|
||||||
// Wave/Sound management functions
|
// Wave/Sound management functions
|
||||||
RLAPI void PlaySound(Sound sound); // Play a sound
|
RLAPI void PlaySound(Sound sound); // Play a sound
|
||||||
|
RLAPI void PlaySoundEx(Sound s); // Play a sound using the multi channel buffer pool
|
||||||
|
RLAPI void StopPlayBufferPool(); // MUST be called before UnLoadSound is used on any sound played with PlaySoundEx
|
||||||
|
RLAPI int ConcurrentPlayChannels(); // Number of sounds playing in the multichannel buffer pool
|
||||||
RLAPI void PauseSound(Sound sound); // Pause a sound
|
RLAPI void PauseSound(Sound sound); // Pause a sound
|
||||||
RLAPI void ResumeSound(Sound sound); // Resume a paused sound
|
RLAPI void ResumeSound(Sound sound); // Resume a paused sound
|
||||||
RLAPI void StopSound(Sound sound); // Stop playing a sound
|
RLAPI void StopSound(Sound sound); // Stop playing a sound
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue