You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
dr_libs/old/dr_audio_ancient.h

4311 lines
138 KiB
C

// Public domain. See "unlicense" statement at the end of this file.
//
// QUICK NOTES
//
// If you've stumbled across this library, be aware that this is very, very early in development. A lot of APIs
// are very temporary, in particular the effects API which at the moment are tied very closely to DirectSound.
//
// Currently, the only backend available is DirectSound while I figure out the API.
//
// General
// - This library is NOT thread safe. Functions can be called from any thread, but it is up to the host application
// to do synchronization.
// - This library is only concerned with playback and recording of raw audio data. It does not load audio files
// such as WAV, OGG and MP3.
// - Before you can create an output (playback) or input (recording) device you need to first create a context.
// - Each backend (DirectSound, ALSA, etc.) has it's own context. Using draudio_create_context() will find
// a backend implementation based on the platform dr_audio has been compiled for.
// - A context for a specific backend can be created as well. For example, draudio_create_context_dsound() will
// create a context which uses DirectSound as it's backend.
// - Currently, devices are enumerated once when the context is created. Thus, when a device is plugged in or
// unplugged it will not be detected by dr_audio and the context will need to be deleted and re-created.
// - Currently, dr_audio will only consider the first DRAUDIO_MAX_DEVICE_COUNT output and input devices, which
// is currently set to 16 and should be plenty for the vast majority of cases. Feel free to increase (or decrease)
// this number to suit your own requirements.
//
// Events
// - Events are handled via callbacks. The different types of events include stop, pause, play and markers.
// - The Stop event is fired when an output buffer is stopped, either from finishing it's playback or if it was
// stopped manually.
// - The Pause event is fired when the output buffer is paused.
// - The Play event is fired when the output buffer begins being played from either a stopped or paused state.
// - A Marker event is fired when the playback position of an output buffer reaches a certain point within the
// buffer. This is useful for streaming audio data because it can tell you when a particular section of the
// buffer can be filled with new data.
// - Due to the inherent multi-threaded nature of audio playback, events can be fired from any thread. It is up
// to the application to ensure events are handled safely.
// - Currently, the maximum number of markers is set by DRAUDIO_MAX_MARKER_COUNT which is set to 4 by default. This
// can be increased, however doing so increases memory usage for each sound buffer.
//
// Performance Considerations
// - Creating and deleting buffers should be considered an expensive operation because there is quite a bit of thread
// management being performed under the hood. Prefer caching sound buffers.
//
//
// OPTIONS
//
// #define DRAUDIO_NO_DIRECTSOUND
// Disables support for the DirectSound backend. Note that at the moment this is the only backend available for
// Windows platforms, so you will likely not want to set this. DirectSound will only be compiled on Win32 builds.
//
//
// TODO
//
// - DirectSound: Consider using Win32 critical sections instead of events where possible.
// - DirectSound: Remove the semaphore and replace with an auto-reset event.
// - Implement a better error handling API.
// - Implement effects
// - Implement velocity
// - Implement cones
// - Implement attenuation min/max distances
//
#ifndef dr_audio_h
#define dr_audio_h
#ifdef __cplusplus
extern "C" {
#endif
#include <stdbool.h>
#include <stdint.h>
#define DRAUDIO_MAX_DEVICE_COUNT 16
#define DRAUDIO_MAX_MARKER_COUNT 4
#define DRAUDIO_MAX_MESSAGE_QUEUE_SIZE 1024 // The maximum number of messages that can be cached in the internal message queues.
#if defined(_WIN32) && !defined(DRAUDIO_NO_DIRECTSOUND)
#define DRAUDIO_BUILD_DSOUND
#endif
#define DRAUDIO_EVENT_ID_STOP 0xFFFFFFFF
#define DRAUDIO_EVENT_ID_PAUSE 0xFFFFFFFE
#define DRAUDIO_EVENT_ID_PLAY 0xFFFFFFFD
#define DRAUDIO_EVENT_ID_MARKER 0
#define DRAUDIO_ENABLE_3D (1 << 0)
#define DRAUDIO_RELATIVE_3D (1 << 1) // <-- Uses relative 3D positioning by default instead of absolute. Only used if DRAUDIO_ENABLE_3D is also specified.
// Data formats.
typedef enum
{
draudio_format_pcm,
draudio_format_float
} draudio_format;
// Playback states.
typedef enum
{
draudio_stopped,
draudio_paused,
draudio_playing
} draudio_playback_state;
// Effect types.
typedef enum
{
draudio_effect_type_none,
draudio_effect_type_i3dl2reverb,
draudio_effect_type_chorus,
draudio_effect_type_compressor,
draudio_effect_type_distortion,
draudio_effect_type_echo,
draudio_effect_type_flanger
} draudio_effect_type;
// 3D processing modes.
typedef enum
{
draudio_3d_mode_absolute,
draudio_3d_mode_relative,
draudio_3d_mode_disabled
} draudio_3d_mode;
typedef struct draudio_context draudio_context;
typedef struct draudio_device draudio_device;
typedef struct draudio_buffer draudio_buffer;
typedef void (* draudio_event_callback_proc)(draudio_buffer* pBuffer, unsigned int eventID, void *pUserData);
typedef struct
{
/// The callback function.
draudio_event_callback_proc callback;
/// The user data.
void* pUserData;
} draudio_event_callback;
typedef struct
{
/// The description of the device.
char description[256];
} draudio_device_info;
typedef struct
{
/// Boolean flags.
/// DRAUDIO_ENABLE_3D: Enable 3D positioning
unsigned int flags;
/// The data format.
draudio_format format;
/// The number of channels. This should be 1 for mono, 2 for stereo.
unsigned int channels;
/// The sample rate.
unsigned int sampleRate;
/// The number of bits per sample.
unsigned int bitsPerSample;
/// The size in bytes of the data.
size_t sizeInBytes;
/// A pointer to the initial data.
void* pData;
} draudio_buffer_desc;
typedef struct
{
/// The effect type.
draudio_effect_type type;
struct
{
float room;
float roomHF;
float roomRolloffFactor;
float decayTime;
float reflections;
float reflectionsDelay;
float reverb;
float reverbDelay;
float diffusion;
float density;
float hfReference;
} i3dl2reverb;
struct
{
int waveform;
int phase;
float depth;
float feedback;
float frequency;
float delay;
} chorus;
struct
{
float gain;
float attack;
float release;
float threshold;
float ratio;
float predelay;
} compressor;
struct
{
float gain;
float edge;
float postEQCenterFrequency;
float postEQBandwidth;
float preLowpassCutoff;
} distortion;
struct
{
float wetDryMix;
float feedback;
float leftDelay;
float rightDelay;
float panDelay;
} echo;
struct
{
float wetDryMix;
float depth;
float feedback;
float frequency;
float waveform;
float delay;
float phase;
} flanger;
} draudio_effect;
/// Creates a context which chooses an appropriate backend based on the given platform.
draudio_context* draudio_create_context();
#ifdef DRAUDIO_BUILD_DSOUND
/// Creates a context that uses DirectSound for it's backend.
draudio_context* draudio_create_context_dsound();
#endif
/// Deletes the given context.
void draudio_delete_context(draudio_context* pContext);
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// OUTPUT / PLAYBACK
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
/// Retrieves the number of output devices that were enumerated when the context was created.
unsigned int draudio_get_output_device_count(draudio_context* pContext);
/// Retrieves information about the device at the given index.
bool draudio_get_output_device_info(draudio_context* pContext, unsigned int deviceIndex, draudio_device_info* pInfoOut);
/// Creates a output device.
///
/// @param pContext [in] A pointer to the main context.
/// @param deviceIndex [in] The index of the device.
///
/// @remarks
/// Use a device index of 0 to use the default output device.
draudio_device* draudio_create_output_device(draudio_context* pContext, unsigned int deviceIndex);
/// Deletes the given output device.
void draudio_delete_output_device(draudio_device* pDevice);
/// Create a buffer.
///
/// @remarks
/// This will fail if 3D positioning is requested when the sound has more than 1 channel.
draudio_buffer* draudio_create_buffer(draudio_device* pDevice, draudio_buffer_desc* pBufferDesc, size_t extraDataSize);
/// Deletes the given buffer.
void draudio_delete_buffer(draudio_buffer* pBuffer);
/// Retrieves the size in bytes of the given buffer's extra data.
unsigned int draudio_get_buffer_extra_data_size(draudio_buffer* pBuffer);
/// Retrieves a pointer to the given buffer's extra data.
void* draudio_get_buffer_extra_data(draudio_buffer* pBuffer);
/// Sets the audio data of the given buffer.
void draudio_set_buffer_data(draudio_buffer* pBuffer, size_t offset, const void* pData, size_t dataSizeInBytes);
/// Begins or resumes playing the given buffer.
///
/// @remarks
/// If the sound is already playing, it will continue to play, but the \c loop setting will be replaced with that specified
/// by the most recent call.
void draudio_play(draudio_buffer* pBuffer, bool loop);
/// Pauses playback of the given buffer.
void draudio_pause(draudio_buffer* pBuffer);
/// Stops playback of the given buffer.
void draudio_stop(draudio_buffer* pBuffer);
/// Retrieves the playback state of the given buffer.
draudio_playback_state draudio_get_playback_state(draudio_buffer* pBuffer);
/// Determines whether or not the given audio buffer is looping.
bool draudio_is_looping(draudio_buffer* pBuffer);
/// Sets the playback position for the given buffer.
void draudio_set_playback_position(draudio_buffer* pBuffer, unsigned int position);
/// Retrieves hte playback position of the given buffer.
unsigned int draudio_get_playback_position(draudio_buffer* pBuffer);
/// Sets the pan of the given buffer.
///
/// @remarks
/// This does nothing for 3D sounds.
void draudio_set_pan(draudio_buffer* pBuffer, float pan);
/// Retrieves the pan of the given buffer.
float draudio_get_pan(draudio_buffer* pBuffer);
/// Sets the volume of the given buffer.
///
/// @param volume [in] The new volume.
///
/// @remarks
/// Amplificiation is not currently supported, so the maximum value is 1. A value of 1 represents the volume of the original
/// data.
void draudio_set_volume(draudio_buffer* pBuffer, float volume);
/// Retrieves the volume of the sound.
float draudio_get_volume(draudio_buffer* pBuffer);
/// Removes every marker.
void draudio_remove_markers(draudio_buffer* pBuffer);
/// Registers the callback to fire when the playback position hits a certain position in the given buffer.
///
/// @param eventID [in] The event ID that will be passed to the callback and can be used to identify a specific marker.
///
/// @remarks
/// This will fail if the buffer is not in a stopped state.
/// @par
/// Set the event ID to DRAUDIO_EVENT_ID_MARKER + n, where "n" is your own application-specific identifier.
bool draudio_register_marker_callback(draudio_buffer* pBuffer, size_t offsetInBytes, draudio_event_callback_proc callback, unsigned int eventID, void* pUserData);
/// Registers the callback to fire when the buffer stops playing.
///
/// @remarks
/// This will fail if the buffer is not in a stopped state and the callback is non-null. It is fine to call this
/// with a null callback while the buffer is in the middle of playback in which case the callback will be cleared.
/// @par
/// The will replace any previous callback.
bool draudio_register_stop_callback(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData);
/// Registers the callback to fire when the buffer is paused.
///
/// @remarks
/// This will fail if the buffer is not in a stopped state and the callback is non-null. It is fine to call this
/// with a null callback while the buffer is in the middle of playback in which case the callback will be cleared.
/// @par
/// The will replace any previous callback.
bool draudio_register_pause_callback(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData);
/// Registers the callback to fire when the buffer begins playing from either a stopped or paused state.
///
/// @remarks
/// This will fail if the buffer is not in a stopped state and the callback is non-null. It is fine to call this
/// with a null callback while the buffer is in the middle of playback in which case the callback will be cleared.
/// @par
/// The will replace any previous callback.
bool draudio_register_play_callback(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData);
/// Retrieves the callback that is currently set for the stop event.
draudio_event_callback draudio_get_stop_callback(draudio_buffer* pBuffer);
/// Retrieves the callback that is currently set for the pause event.
draudio_event_callback draudio_get_pause_callback(draudio_buffer* pBuffer);
/// Retrieves the callback that is currently set for the play event.
draudio_event_callback draudio_get_play_callback(draudio_buffer* pBuffer);
/// Sets the position of the given buffer in 3D space.
///
/// @remarks
/// This does nothing for buffers that do not support 3D positioning.
void draudio_set_position(draudio_buffer* pBuffer, float x, float y, float z);
/// Retrieves the position of the given buffer in 3D space.
void draudio_get_position(draudio_buffer* pBuffer, float* pPosOut);
/// Sets the position of the listener for the given output device.
void draudio_set_listener_position(draudio_device* pDevice, float x, float y, float z);
/// Retrieves the position of the listner for the given output device.
void draudio_get_listener_position(draudio_device* pDevice, float* pPosOut);
/// Sets the orientation of the listener for the given output device.
void draudio_set_listener_orientation(draudio_device* pDevice, float forwardX, float forwardY, float forwardZ, float upX, float upY, float upZ);
/// Retrieves the orientation of the listener for the given output device.
void draudio_get_listener_orientation(draudio_device* pDevice, float* pForwardOut, float* pUpOut);
/// Sets the 3D processing mode (absolute, relative or disabled).
///
/// @remarks
/// This applies to position, orientation and velocity.
void draudio_set_3d_mode(draudio_buffer* pBuffer, draudio_3d_mode mode);
/// Retrieves the 3D processing mode (absolute, relative or disabled).
draudio_3d_mode draudio_get_3d_mode(draudio_buffer* pBuffer);
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// INPUT / RECORDING
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// HIGH-LEVEL API
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//// SYNCHRONIZATION ////
typedef void* draudio_mutex;
/// Creates a mutex object.
draudio_mutex draudio_create_mutex();
/// Deletes a mutex object.
void draudio_delete_mutex(draudio_mutex mutex);
/// Locks the given mutex.
void draudio_lock_mutex(draudio_mutex mutex);
/// Unlocks the given mutex.
void draudio_unlock_mutex(draudio_mutex mutex);
//// STREAMING ////
typedef int draudio_bool;
typedef draudio_bool (* draudio_stream_read_proc)(void* pUserData, void* pDataOut, size_t bytesToRead, size_t* bytesReadOut);
typedef draudio_bool (* draudio_stream_seek_proc)(void* pUserData, size_t offsetInBytesFromStart);
typedef struct
{
/// A pointer to the user data to pass to each callback.
void* pUserData;
/// A pointer to the function to call when more data needs to be read.
draudio_stream_read_proc read;
/// Seeks source data from the beginning of the file.
draudio_stream_seek_proc seek;
} draudio_streaming_callbacks;
/// Creates a buffer that's pre-configured for use for streaming audio data.
///
/// @remarks
/// This function is just a high-level convenience wrapper. The returned buffer is just a regular buffer with pre-configured
/// markers attached to the buffer. This will attach 3 markers in total which means there is only DRAUDIO_MAX_MARKER_COUNT - 3
/// marker slots available to the application.
/// @par
/// You must play the buffer with draudio_play_streaming_buffer() because the underlying buffer management is slightly different
/// to a regular buffer.
/// @par
/// Looping and stop callbacks may be inaccurate by up to half a second.
/// @par
/// Callback functions use bytes to determine how much data to process. This is always a multiple of samples * channels, however.
/// @par
/// The first chunk of data is not loaded until the buffer is played with draudio_play_streaming_buffer().
draudio_buffer* draudio_create_streaming_buffer(draudio_device* pDevice, draudio_buffer_desc* pBufferDesc, draudio_streaming_callbacks callbacks, unsigned int extraDataSize);
/// Retrieves the size of the extra data of the given streaming buffer..
size_t draudio_get_streaming_buffer_extra_data_size(draudio_buffer* pBuffer);
/// Retrieves a pointer to the extra data of the given streaming buffer.
void* draudio_get_streaming_buffer_extra_data(draudio_buffer* pBuffer);
/// Begins playing the given streaming buffer.
bool draudio_play_streaming_buffer(draudio_buffer* pBuffer, bool loop);
/// Determines whether or not the given streaming buffer is looping.
bool draudio_is_streaming_buffer_looping(draudio_buffer* pBuffer);
///////////////////////////////////////////////////////////////////////////////
//
// Sound World
//
// When a sound is created, whether or not it will be streamed is determined by
// whether or not a pointer to some initial data is specified when creating the
// sound. When initial data is specified, the sound data will be loaded into
// buffer once at creation time. If no data is specified, sound data will be
// loaded dynamically at playback time.
//
///////////////////////////////////////////////////////////////////////////////
typedef struct draudio_sound draudio_sound;
typedef struct draudio_world draudio_world;
typedef void (* draudio_on_sound_delete_proc) (draudio_sound* pSound);
typedef bool (* draudio_on_sound_read_data_proc)(draudio_sound* pSound, void* pDataOut, size_t bytesToRead, size_t* bytesReadOut);
typedef bool (* draudio_on_sound_seek_data_proc)(draudio_sound* pSound, size_t offsetInBytesFromStart);
/// The structure that is used for creating a sound object.
typedef struct
{
/// Boolean flags.
/// DRAUDIO_ENABLE_3D: Enable 3D positioning
unsigned int flags;
/// The data format.
draudio_format format;
/// The number of channels. This should be 1 for mono, 2 for stereo.
unsigned int channels;
/// The sample rate.
unsigned int sampleRate;
/// The number of bits per sample.
unsigned int bitsPerSample;
/// The size in bytes of the data. When this is non-zero, and pInitialData is non-null, the onRead and onSeek streaming
/// callbacks are not used, and instead the sound's audio data is made up exclusively with this data.
size_t sizeInBytes;
/// A pointer to the initial data. Can be null, in which case the audio data is streamed with the onRead and onSeek
/// callbacks below. It is an error for this to be null in addition to onRead and onSeek.
void* pData;
/// A pointer to the function to call when the sound is being deleted. This gives the application the opportunity
/// to delete internal objects that are used for streaming or whatnot.
draudio_on_sound_delete_proc onDelete;
/// A pointer to the function to call when dr_audio needs to request a chunk of audio data. This is only used when
/// streaming data.
draudio_on_sound_read_data_proc onRead;
/// A pointer to the function to call when dr_audio needs to seek the audio data. This is only used when streaming
/// data.
draudio_on_sound_seek_data_proc onSeek;
/// The size of the extra data to associate with the sound. Extra data is how an application can link custom data to the
/// sound object.
unsigned int extraDataSize;
/// A pointer to a buffer containing the initial extra data. This buffer is copied when the sound is initially created,
/// and can be null.
const void* pExtraData;
} draudio_sound_desc;
struct draudio_sound
{
/// A pointer to the world that owns the sound.
draudio_world* pWorld;
/// A pointer to the audio buffer for playback.
draudio_buffer* pBuffer;
/// [Internal Use Only] The state of the buffer's playback at the time the associated world overwrote it.
draudio_playback_state prevPlaybackState;
/// [Internal Use Only] A pointer to the next sound in the local list.
draudio_sound* pNextSound;
/// [Internal Use Only] A pointer ot the previous sound in the local list.
draudio_sound* pPrevSound;
/// [Internal Use Only] Keeps track of whether or not a streaming buffer is being used.
bool isUsingStreamingBuffer;
/// [Internal Use Only] Keeps track of whether or not the sound has been marked for deletion. This is used to
/// ensure onRead and onSeek are never called after the sound has been deleted. This scenario is possible because
/// these functions are called in response to the sound buffer hitting markers which can be slightly delayed
/// due to multi-threading synchronization.
bool markedForDeletion;
/// [Internal Use Only] the onDelete function. Can be null.
draudio_on_sound_delete_proc onDelete;
/// [Internal Use Only] The onRead streaming function. Can be null, in which case streaming will not be used.
draudio_on_sound_read_data_proc onRead;
/// [Internal Use Only] The onSeek streaming function. Can be null, in which case streaming will not be used.
draudio_on_sound_seek_data_proc onSeek;
};
struct draudio_world
{
/// A pointer to the dr_audio device to output audio to.
draudio_device* pDevice;
/// The global playback state of the world.
draudio_playback_state playbackState;
/// A pointer to the first sound in the local list of sounds.
draudio_sound* pFirstSound;
/// Mutex for thread-safety.
draudio_mutex lock;
};
/// Creates a new sound world which will output audio from the given device.
draudio_world* draudio_create_world(draudio_device* pDevice);
/// Deletes a sound world that was previously created with draudio_create_world().
void draudio_delete_world(draudio_world* pWorld);
/// Creates a sound in 3D space.
draudio_sound* draudio_create_sound(draudio_world* pWorld, draudio_sound_desc desc);
/// Deletes a sound that was previously created with draudio_create_sound().
void draudio_delete_sound(draudio_sound* pSound);
/// Deletes every sound from the given world.
void draudio_delete_all_sounds(draudio_world* pWorld);
/// Retrieves the size in bytes of the given sound's extra data.
size_t draudio_get_sound_extra_data_size(draudio_sound* pSound);
/// Retrieves a pointer to the buffer containing the given sound's extra data.
void* draudio_get_sound_extra_data(draudio_sound* pSound);
/// Plays or resumes the given sound.
void draudio_play_sound(draudio_sound* pSound, bool loop);
/// Pauses playback the given sound.
void draudio_pause_sound(draudio_sound* pSound);
/// Stops playback of the given sound.
void draudio_stop_sound(draudio_sound* pSound);
/// Retrieves the playback state of the given sound.
draudio_playback_state draudio_get_sound_playback_state(draudio_sound* pSound);
/// Determines if the given sound is looping.
bool draudio_is_sound_looping(draudio_sound* pSound);
/// Begins playing a sound using the given streaming callbacks.
void draudio_play_inline_sound(draudio_world* pWorld, draudio_sound_desc desc);
/// Begins playing the given sound at the given position.
void draudio_play_inline_sound_3f(draudio_world* pWorld, draudio_sound_desc desc, float posX, float posY, float posZ);
/// Stops playback of all sounds in the given world.
void draudio_stop_all_sounds(draudio_world* pWorld);
/// Pauses playback of all sounds in the given world.
void draudio_pause_all_sounds(draudio_world* pWorld);
/// Resumes playback of all sounds in the given world.
void draudio_resume_all_sounds(draudio_world* pWorld);
/// Sets the callback for the stop event for the given sound.
void draudio_set_sound_stop_callback(draudio_sound* pSound, draudio_event_callback_proc callback, void* pUserData);
/// Sets the callback for the pause event for the given sound.
void draudio_set_sound_pause_callback(draudio_sound* pSound, draudio_event_callback_proc callback, void* pUserData);
/// Sets the callback for the play event for the given sound.
void draudio_set_sound_play_callback(draudio_sound* pSound, draudio_event_callback_proc callback, void* pUserData);
/// Sets the position of the given sound.
void draudio_set_sound_position(draudio_sound* pSound, float posX, float posY, float posZ);
/// Sets the 3D mode of the given sound (absolute positioning, relative positioning, no positioning).
void draudio_set_sound_3d_mode(draudio_sound* pSound, draudio_3d_mode mode);
/// Retrieves the 3D mode of the given sound.
draudio_3d_mode draudio_get_sound_3d_mode(draudio_sound* pSound);
#ifdef __cplusplus
}
#endif
#endif //dr_audio_h
///////////////////////////////////////////////////////////////////////////////
//
// IMPLEMENTATION
//
///////////////////////////////////////////////////////////////////////////////
#ifdef DR_AUDIO_IMPLEMENTATION
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include <errno.h>
// Annotations
#ifndef DRAUDIO_PRIVATE
#define DRAUDIO_PRIVATE
#endif
////////////////////////////////////////////////////////
// Utilities
// strcpy()
int draudio_strcpy(char* dst, size_t dstSizeInBytes, const char* src)
{
#if defined(_MSC_VER)
return strcpy_s(dst, dstSizeInBytes, src);
#else
if (dst == 0) {
return EINVAL;
}
if (dstSizeInBytes == 0) {
return ERANGE;
}
if (src == 0) {
dst[0] = '\0';
return EINVAL;
}
char* iDst = dst;
const char* iSrc = src;
size_t remainingSizeInBytes = dstSizeInBytes;
while (remainingSizeInBytes > 0 && iSrc[0] != '\0')
{
iDst[0] = iSrc[0];
iDst += 1;
iSrc += 1;
remainingSizeInBytes -= 1;
}
if (remainingSizeInBytes > 0) {
iDst[0] = '\0';
} else {
dst[0] = '\0';
return ERANGE;
}
return 0;
#endif
}
typedef void (* draudio_delete_context_proc)(draudio_context* pContext);
typedef draudio_device* (* draudio_create_output_device_proc)(draudio_context* pContext, unsigned int deviceIndex);
typedef void (* draudio_delete_output_device_proc)(draudio_device* pDevice);
typedef unsigned int (* draudio_get_output_device_count_proc)(draudio_context* pContext);
typedef bool (* draudio_get_output_device_info_proc)(draudio_context* pContext, unsigned int deviceIndex, draudio_device_info* pInfoOut);
typedef draudio_buffer* (* draudio_create_buffer_proc)(draudio_device* pDevice, draudio_buffer_desc* pBufferDesc, size_t extraDataSize);
typedef void (* draudio_delete_buffer_proc)(draudio_buffer* pBuffer);
typedef unsigned int (* draudio_get_buffer_extra_data_size_proc)(draudio_buffer* pBuffer);
typedef void* (* draudio_get_buffer_extra_data_proc)(draudio_buffer* pBuffer);
typedef void (* draudio_set_buffer_data_proc)(draudio_buffer* pBuffer, size_t offset, const void* pData, size_t dataSizeInBytes);
typedef void (* draudio_play_proc)(draudio_buffer* pBuffer, bool loop);
typedef void (* draudio_pause_proc)(draudio_buffer* pBuffer);
typedef void (* draudio_stop_proc)(draudio_buffer* pBuffer);
typedef draudio_playback_state (* draudio_get_playback_state_proc)(draudio_buffer* pBuffer);
typedef void (* draudio_set_playback_position_proc)(draudio_buffer* pBuffer, unsigned int position);
typedef unsigned int (* draudio_get_playback_position_proc)(draudio_buffer* pBuffer);
typedef void (* draudio_set_pan_proc)(draudio_buffer* pBuffer, float pan);
typedef float (* draudio_get_pan_proc)(draudio_buffer* pBuffer);
typedef void (* draudio_set_volume_proc)(draudio_buffer* pBuffer, float volume);
typedef float (* draudio_get_volume_proc)(draudio_buffer* pBuffer);
typedef void (* draudio_remove_markers_proc)(draudio_buffer* pBuffer);
typedef bool (* draudio_register_marker_callback_proc)(draudio_buffer* pBuffer, size_t offsetInBytes, draudio_event_callback_proc callback, unsigned int eventID, void* pUserData);
typedef bool (* draudio_register_stop_callback_proc)(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData);
typedef bool (* draudio_register_pause_callback_proc)(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData);
typedef bool (* draudio_register_play_callback_proc)(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData);
typedef void (* draudio_set_position_proc)(draudio_buffer* pBuffer, float x, float y, float z);
typedef void (* draudio_get_position_proc)(draudio_buffer* pBuffer, float* pPosOut);
typedef void (* draudio_set_listener_position_proc)(draudio_device* pDevice, float x, float y, float z);
typedef void (* draudio_get_listener_position_proc)(draudio_device* pDevice, float* pPosOut);
typedef void (* draudio_set_listener_orientation_proc)(draudio_device* pDevice, float forwardX, float forwardY, float forwardZ, float upX, float upY, float upZ);
typedef void (* draudio_get_listener_orientation_proc)(draudio_device* pDevice, float* pForwardOut, float* pUpOut);
typedef void (* draudio_set_3d_mode_proc)(draudio_buffer* pBuffer, draudio_3d_mode mode);
typedef draudio_3d_mode (* draudio_get_3d_mode_proc)(draudio_buffer* pBuffer);
struct draudio_context
{
// Callbacks.
draudio_delete_context_proc delete_context;
draudio_create_output_device_proc create_output_device;
draudio_delete_output_device_proc delete_output_device;
draudio_get_output_device_count_proc get_output_device_count;
draudio_get_output_device_info_proc get_output_device_info;
draudio_create_buffer_proc create_buffer;
draudio_delete_buffer_proc delete_buffer;
draudio_get_buffer_extra_data_size_proc get_buffer_extra_data_size;
draudio_get_buffer_extra_data_proc get_buffer_extra_data;
draudio_set_buffer_data_proc set_buffer_data;
draudio_play_proc play;
draudio_pause_proc pause;
draudio_stop_proc stop;
draudio_get_playback_state_proc get_playback_state;
draudio_set_playback_position_proc set_playback_position;
draudio_get_playback_position_proc get_playback_position;
draudio_set_pan_proc set_pan;
draudio_get_pan_proc get_pan;
draudio_set_volume_proc set_volume;
draudio_get_volume_proc get_volume;
draudio_remove_markers_proc remove_markers;
draudio_register_marker_callback_proc register_marker_callback;
draudio_register_stop_callback_proc register_stop_callback;
draudio_register_pause_callback_proc register_pause_callback;
draudio_register_play_callback_proc register_play_callback;
draudio_set_position_proc set_position;
draudio_get_position_proc get_position;
draudio_set_listener_position_proc set_listener_position;
draudio_get_listener_position_proc get_listener_position;
draudio_set_listener_orientation_proc set_listener_orientation;
draudio_get_listener_orientation_proc get_listener_orientation;
draudio_set_3d_mode_proc set_3d_mode;
draudio_get_3d_mode_proc get_3d_mode;
};
struct draudio_device
{
/// The context that owns this device.
draudio_context* pContext;
/// Whether or not the device is marked for deletion. A device is not always deleted straight away, so we use this
/// to determine whether or not it has been marked for deletion.
bool markedForDeletion;
};
struct draudio_buffer
{
/// The device that owns this buffer.
draudio_device* pDevice;
/// The stop callback.
draudio_event_callback stopCallback;
/// The pause callback.
draudio_event_callback pauseCallback;
/// The play callback.
draudio_event_callback playCallback;
/// Whether or not playback is looping.
bool isLooping;
/// Whether or not the buffer is marked for deletion. A buffer is not always deleted straight away, so we use this
/// to determine whether or not it has been marked for deletion.
bool markedForDeletion;
};
draudio_context* draudio_create_context()
{
draudio_context* pContext = NULL;
#ifdef DRAUDIO_BUILD_DSOUND
pContext = draudio_create_context_dsound();
if (pContext != NULL) {
return pContext;
}
#endif
// If we get here it means we weren't able to create a context, so return NULL. Later on we're going to have a null implementation so that
// we don't ever need to return NULL here.
assert(pContext == NULL);
return pContext;
}
void draudio_delete_context(draudio_context* pContext)
{
if (pContext == NULL) {
return;
}
pContext->delete_context(pContext);
}
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// OUTPUT
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
unsigned int draudio_get_output_device_count(draudio_context* pContext)
{
if (pContext == NULL) {
return 0;
}
return pContext->get_output_device_count(pContext);
}
bool draudio_get_output_device_info(draudio_context* pContext, unsigned int deviceIndex, draudio_device_info* pInfoOut)
{
if (pContext == NULL) {
return false;
}
if (pInfoOut == NULL) {
return false;
}
return pContext->get_output_device_info(pContext, deviceIndex, pInfoOut);
}
draudio_device* draudio_create_output_device(draudio_context* pContext, unsigned int deviceIndex)
{
if (pContext == NULL) {
return NULL;
}
draudio_device* pDevice = pContext->create_output_device(pContext, deviceIndex);
if (pDevice != NULL)
{
pDevice->markedForDeletion = false;
}
return pDevice;
}
void draudio_delete_output_device(draudio_device* pDevice)
{
if (pDevice == NULL) {
return;
}
// If the device is already marked for deletion we just return straight away. However, this is an erroneous case so we trigger a failed assertion in this case.
if (pDevice->markedForDeletion) {
assert(false);
return;
}
pDevice->markedForDeletion = true;
assert(pDevice->pContext != NULL);
pDevice->pContext->delete_output_device(pDevice);
}
draudio_buffer* draudio_create_buffer(draudio_device* pDevice, draudio_buffer_desc* pBufferDesc, size_t extraDataSize)
{
if (pDevice == NULL) {
return NULL;
}
if (pBufferDesc == NULL) {
return NULL;
}
assert(pDevice->pContext != NULL);
draudio_buffer* pBuffer = pDevice->pContext->create_buffer(pDevice, pBufferDesc, extraDataSize);
if (pBuffer != NULL)
{
draudio_event_callback nullcb = {NULL, NULL};
pBuffer->pDevice = pDevice;
pBuffer->stopCallback = nullcb;
pBuffer->pauseCallback = nullcb;
pBuffer->playCallback = nullcb;
pBuffer->isLooping = false;
pBuffer->markedForDeletion = false;
}
return pBuffer;
}
void draudio_delete_buffer(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return;
}
// We don't want to do anything if the buffer is marked for deletion.
if (pBuffer->markedForDeletion) {
assert(false);
return;
}
pBuffer->markedForDeletion = true;
// The sound needs to be stopped first.
draudio_stop(pBuffer);
// Now we need to remove every event.
draudio_remove_markers(pBuffer);
draudio_register_stop_callback(pBuffer, NULL, NULL);
draudio_register_pause_callback(pBuffer, NULL, NULL);
draudio_register_play_callback(pBuffer, NULL, NULL);
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->delete_buffer(pBuffer);
}
unsigned int draudio_get_buffer_extra_data_size(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return 0;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_buffer_extra_data_size(pBuffer);
}
void* draudio_get_buffer_extra_data(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return NULL;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_buffer_extra_data(pBuffer);
}
void draudio_set_buffer_data(draudio_buffer* pBuffer, size_t offset, const void* pData, size_t dataSizeInBytes)
{
if (pBuffer == NULL) {
return;
}
if (pData == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->set_buffer_data(pBuffer, offset, pData, dataSizeInBytes);
}
void draudio_play(draudio_buffer* pBuffer, bool loop)
{
if (pBuffer == NULL) {
return;
}
pBuffer->isLooping = loop;
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->play(pBuffer, loop);
}
void draudio_pause(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->pause(pBuffer);
}
void draudio_stop(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->stop(pBuffer);
}
draudio_playback_state draudio_get_playback_state(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return draudio_stopped;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_playback_state(pBuffer);
}
bool draudio_is_looping(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return false;
}
return pBuffer->isLooping;
}
void draudio_set_playback_position(draudio_buffer* pBuffer, unsigned int position)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->set_playback_position(pBuffer, position);
}
unsigned int draudio_get_playback_position(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return 0;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_playback_position(pBuffer);
}
void draudio_set_pan(draudio_buffer* pBuffer, float pan)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->set_pan(pBuffer, pan);
}
float draudio_get_pan(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return 0.0f;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_pan(pBuffer);
}
void draudio_set_volume(draudio_buffer* pBuffer, float volume)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->set_volume(pBuffer, volume);
}
float draudio_get_volume(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return 1.0f;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_volume(pBuffer);
}
void draudio_remove_markers(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->remove_markers(pBuffer);
}
bool draudio_register_marker_callback(draudio_buffer* pBuffer, size_t offsetInBytes, draudio_event_callback_proc callback, unsigned int eventID, void* pUserData)
{
if (pBuffer == NULL) {
return false;
}
if (eventID == DRAUDIO_EVENT_ID_STOP ||
eventID == DRAUDIO_EVENT_ID_PAUSE ||
eventID == DRAUDIO_EVENT_ID_PLAY)
{
return false;
}
if (draudio_get_playback_state(pBuffer) != draudio_stopped) {
return false;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->register_marker_callback(pBuffer, offsetInBytes, callback, eventID, pUserData);
}
bool draudio_register_stop_callback(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData)
{
if (pBuffer == NULL) {
return false;
}
if (callback != NULL && draudio_get_playback_state(pBuffer) != draudio_stopped) {
return false;
}
pBuffer->stopCallback.callback = callback;
pBuffer->stopCallback.pUserData = pUserData;
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->register_stop_callback(pBuffer, callback, pUserData);
}
bool draudio_register_pause_callback(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData)
{
if (pBuffer == NULL) {
return false;
}
if (callback != NULL && draudio_get_playback_state(pBuffer) != draudio_stopped) {
return false;
}
pBuffer->pauseCallback.callback = callback;
pBuffer->pauseCallback.pUserData = pUserData;
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->register_pause_callback(pBuffer, callback, pUserData);
}
bool draudio_register_play_callback(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData)
{
if (pBuffer == NULL) {
return false;
}
if (callback != NULL && draudio_get_playback_state(pBuffer) != draudio_stopped) {
return false;
}
pBuffer->playCallback.callback = callback;
pBuffer->playCallback.pUserData = pUserData;
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->register_play_callback(pBuffer, callback, pUserData);
}
draudio_event_callback draudio_get_stop_callback(draudio_buffer* pBuffer)
{
if (pBuffer != NULL) {
return pBuffer->stopCallback;
} else {
draudio_event_callback result = {NULL, NULL};
return result;
}
}
draudio_event_callback draudio_get_pause_callback(draudio_buffer* pBuffer)
{
if (pBuffer != NULL) {
return pBuffer->pauseCallback;
} else {
draudio_event_callback result = {NULL, NULL};
return result;
}
}
draudio_event_callback draudio_get_play_callback(draudio_buffer* pBuffer)
{
if (pBuffer != NULL) {
return pBuffer->playCallback;
} else {
draudio_event_callback result = {NULL, NULL};
return result;
}
}
void draudio_set_position(draudio_buffer* pBuffer, float x, float y, float z)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->set_position(pBuffer, x, y, z);
}
void draudio_get_position(draudio_buffer* pBuffer, float* pPosOut)
{
if (pBuffer == NULL) {
return;
}
if (pPosOut == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->get_position(pBuffer, pPosOut);
}
void draudio_set_listener_position(draudio_device* pDevice, float x, float y, float z)
{
if (pDevice == NULL) {
return;
}
pDevice->pContext->set_listener_position(pDevice, x, y, z);
}
void draudio_get_listener_position(draudio_device* pDevice, float* pPosOut)
{
if (pDevice == NULL || pPosOut == NULL) {
return;
}
pDevice->pContext->get_listener_position(pDevice, pPosOut);
}
void draudio_set_listener_orientation(draudio_device* pDevice, float forwardX, float forwardY, float forwardZ, float upX, float upY, float upZ)
{
if (pDevice == NULL) {
return;
}
pDevice->pContext->set_listener_orientation(pDevice, forwardX, forwardY, forwardZ, upX, upY, upZ);
}
void draudio_get_listener_orientation(draudio_device* pDevice, float* pForwardOut, float* pUpOut)
{
if (pDevice == NULL) {
return;
}
pDevice->pContext->get_listener_orientation(pDevice, pForwardOut, pUpOut);
}
void draudio_set_3d_mode(draudio_buffer* pBuffer, draudio_3d_mode mode)
{
if (pBuffer == NULL) {
return;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
pBuffer->pDevice->pContext->set_3d_mode(pBuffer, mode);
}
draudio_3d_mode draudio_get_3d_mode(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return draudio_3d_mode_disabled;
}
assert(pBuffer->pDevice != NULL);
assert(pBuffer->pDevice->pContext != NULL);
return pBuffer->pDevice->pContext->get_3d_mode(pBuffer);
}
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// INPUT
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// HIGH-LEVEL API
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//// SYNCHRONIZATION ////
#if defined(_WIN32)
#include <windows.h>
draudio_mutex draudio_create_mutex()
{
draudio_mutex mutex = malloc(sizeof(CRITICAL_SECTION));
if (mutex != NULL)
{
InitializeCriticalSection((LPCRITICAL_SECTION)mutex);
}
return mutex;
}
void draudio_delete_mutex(draudio_mutex mutex)
{
DeleteCriticalSection((LPCRITICAL_SECTION)mutex);
free(mutex);
}
void draudio_lock_mutex(draudio_mutex mutex)
{
EnterCriticalSection((LPCRITICAL_SECTION)mutex);
}
void draudio_unlock_mutex(draudio_mutex mutex)
{
LeaveCriticalSection((LPCRITICAL_SECTION)mutex);
}
#else
#include <pthread.h>
draudio_mutex draudio_create_mutex()
{
pthread_mutex_t* mutex = malloc(sizeof(pthread_mutex_t));
if (pthread_mutex_init(mutex, NULL) != 0) {
free(mutex);
mutex = NULL;
}
return mutex;
}
void draudio_delete_mutex(draudio_mutex mutex)
{
pthread_mutex_destroy(mutex);
}
void draudio_lock_mutex(draudio_mutex mutex)
{
pthread_mutex_lock(mutex);
}
void draudio_unlock_mutex(draudio_mutex mutex)
{
pthread_mutex_unlock(mutex);
}
#endif
//// STREAMING ////
#define DRAUDIO_STREAMING_MARKER_0 DRAUDIO_EVENT_ID_MARKER + 0
#define DRAUDIO_STREAMING_MARKER_1 DRAUDIO_EVENT_ID_MARKER + 1
#define DRAUDIO_STREAMING_CHUNK_INVALID 0
typedef struct
{
/// The steaming buffer callbacks.
draudio_streaming_callbacks callbacks;
/// Keeps track of whether or not we are at the start of the playback.
bool atStart;
/// Keeps track of whether or not we should stop at the end of the next chunk.
bool stopAtEndOfCurrentChunk;
/// Keeps track of whether or not the sound should loop.
bool isLoopingEnabled;
/// The size of the extra data.
size_t extraDataSize;
/// The size of an individual chunk. A chunk is half the size of the buffer.
size_t chunkSize;
/// A pointer to the temporary buffer for loading chunk data.
unsigned char pTempChunkData[1];
} ea_streaming_buffer_data;
bool ea_streaming_buffer_load_next_chunk(draudio_buffer* pBuffer, ea_streaming_buffer_data* pStreamingData, size_t offset, size_t chunkSize)
{
assert(pStreamingData != NULL);
assert(pStreamingData->callbacks.read != NULL);
assert(pStreamingData->callbacks.seek != NULL);
assert(pStreamingData->chunkSize >= chunkSize);
// A chunk size of 0 is valid, but we just return immediately.
if (chunkSize == 0) {
return true;
}
size_t bytesRead;
if (!pStreamingData->callbacks.read(pStreamingData->callbacks.pUserData, pStreamingData->pTempChunkData, chunkSize, &bytesRead))
{
// There was an error reading the data. We might have run out of data.
return false;
}
pStreamingData->stopAtEndOfCurrentChunk = false;
draudio_set_buffer_data(pBuffer, offset, pStreamingData->pTempChunkData, bytesRead);
if (chunkSize > bytesRead)
{
// The number of bytes read is less than our chunk size. This is our cue that we've reached the end of the steam. If we're looping, we
// just seek back to the start and read more data. There is a chance the data total size of the streaming data is smaller than our
// chunk, so we actually want to do this recursively.
//
// If we're not looping, we fill the remaining data with silence.
if (pStreamingData->isLoopingEnabled)
{
pStreamingData->callbacks.seek(pStreamingData->callbacks.pUserData, 0);
return ea_streaming_buffer_load_next_chunk(pBuffer, pStreamingData, offset + bytesRead, chunkSize - bytesRead);
}
else
{
memset(pStreamingData->pTempChunkData + bytesRead, 0, chunkSize - bytesRead);
draudio_set_buffer_data(pBuffer, offset + bytesRead, pStreamingData->pTempChunkData + bytesRead, chunkSize - bytesRead);
pStreamingData->stopAtEndOfCurrentChunk = true;
}
}
return true;
}
void ea_steaming_buffer_marker_callback(draudio_buffer* pBuffer, unsigned int eventID, void *pUserData)
{
ea_streaming_buffer_data* pStreamingData = (ea_streaming_buffer_data*)pUserData;
assert(pStreamingData != NULL);
size_t offset = 0;
if (eventID == DRAUDIO_STREAMING_MARKER_0) {
offset = pStreamingData->chunkSize;
}
if (pStreamingData->stopAtEndOfCurrentChunk)
{
if (!pStreamingData->atStart) {
draudio_stop(pBuffer);
}
}
else
{
ea_streaming_buffer_load_next_chunk(pBuffer, pStreamingData, offset, pStreamingData->chunkSize);
}
pStreamingData->atStart = false;
}
draudio_buffer* draudio_create_streaming_buffer(draudio_device* pDevice, draudio_buffer_desc* pBufferDesc, draudio_streaming_callbacks callbacks, unsigned int extraDataSize)
{
if (callbacks.read == NULL) {
return NULL;
}
if (pBufferDesc == NULL) {
return NULL;
}
// We are determining for ourselves what the size of the buffer should be. We need to create our own copy rather than modify the input descriptor.
draudio_buffer_desc bufferDesc = *pBufferDesc;
bufferDesc.sizeInBytes = pBufferDesc->sampleRate * pBufferDesc->channels * (pBufferDesc->bitsPerSample / 8);
bufferDesc.pData = NULL;
size_t chunkSize = bufferDesc.sizeInBytes / 2;
draudio_buffer* pBuffer = draudio_create_buffer(pDevice, &bufferDesc, sizeof(ea_streaming_buffer_data) - sizeof(unsigned char) + chunkSize + extraDataSize);
if (pBuffer == NULL) {
return NULL;
}
ea_streaming_buffer_data* pStreamingData = (ea_streaming_buffer_data*)draudio_get_buffer_extra_data(pBuffer);
assert(pStreamingData != NULL);
pStreamingData->callbacks = callbacks;
pStreamingData->atStart = true;
pStreamingData->stopAtEndOfCurrentChunk = false;
pStreamingData->isLoopingEnabled = false;
pStreamingData->chunkSize = chunkSize;
// Register two markers - one for the first half and another for the second half. When a half is finished playing we need to
// replace it with new data.
draudio_register_marker_callback(pBuffer, 0, ea_steaming_buffer_marker_callback, DRAUDIO_STREAMING_MARKER_0, pStreamingData);
draudio_register_marker_callback(pBuffer, chunkSize, ea_steaming_buffer_marker_callback, DRAUDIO_STREAMING_MARKER_1, pStreamingData);
return pBuffer;
}
size_t draudio_get_streaming_buffer_extra_data_size(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return 0;
}
ea_streaming_buffer_data* pStreamingData = (ea_streaming_buffer_data*)draudio_get_buffer_extra_data(pBuffer);
assert(pStreamingData != NULL);
return pStreamingData->extraDataSize;
}
void* draudio_get_streaming_buffer_extra_data(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return NULL;
}
ea_streaming_buffer_data* pStreamingData = (ea_streaming_buffer_data*)draudio_get_buffer_extra_data(pBuffer);
assert(pStreamingData != NULL);
return ((char*)pStreamingData->pTempChunkData) + pStreamingData->chunkSize;
}
bool draudio_play_streaming_buffer(draudio_buffer* pBuffer, bool loop)
{
if (pBuffer == NULL) {
return false;
}
ea_streaming_buffer_data* pStreamingData = (ea_streaming_buffer_data*)draudio_get_buffer_extra_data(pBuffer);
assert(pStreamingData != NULL);
// If the buffer was previously in a paused state, we just play like normal. If it was in a stopped state we need to start from the beginning.
if (draudio_get_playback_state(pBuffer) == draudio_stopped)
{
// We need to load some initial data into the first chunk.
pStreamingData->atStart = true;
pStreamingData->callbacks.seek(pStreamingData->callbacks.pUserData, 0);
if (!ea_streaming_buffer_load_next_chunk(pBuffer, pStreamingData, 0, pStreamingData->chunkSize))
{
// There was an error loading the initial data.
return false;
}
}
pStreamingData->isLoopingEnabled = loop;
draudio_play(pBuffer, true); // <-- Always loop on a streaming buffer. Actual looping is done a bit differently for streaming buffers.
return true;
}
bool draudio_is_streaming_buffer_looping(draudio_buffer* pBuffer)
{
if (pBuffer == NULL) {
return false;
}
ea_streaming_buffer_data* pStreamingData = (ea_streaming_buffer_data*)draudio_get_buffer_extra_data(pBuffer);
assert(pStreamingData != NULL);
return pStreamingData->isLoopingEnabled;
}
///////////////////////////////////////////////////////////////////////////////
//
// Sound World
//
///////////////////////////////////////////////////////////////////////////////
DRAUDIO_PRIVATE draudio_bool draudio_on_sound_read_callback(void* pUserData, void* pDataOut, size_t bytesToRead, size_t* bytesReadOut)
{
draudio_sound* pSound = (draudio_sound*)pUserData;
assert(pSound != NULL);
assert(pSound->onRead != NULL);
bool result = false;
draudio_lock_mutex(pSound->pWorld->lock);
{
if (!pSound->markedForDeletion) {
result = pSound->onRead(pSound, pDataOut, bytesToRead, bytesReadOut);
}
}
draudio_unlock_mutex(pSound->pWorld->lock);
return result;
}
DRAUDIO_PRIVATE static draudio_bool draudio_on_sound_seek_callback(void* pUserData, size_t offsetInBytesFromStart)
{
draudio_sound* pSound = (draudio_sound*)pUserData;
assert(pSound != NULL);
assert(pSound->onRead != NULL);
bool result = false;
draudio_lock_mutex(pSound->pWorld->lock);
{
if (!pSound->markedForDeletion) {
result = pSound->onSeek(pSound, offsetInBytesFromStart);
}
}
draudio_unlock_mutex(pSound->pWorld->lock);
return result;
}
DRAUDIO_PRIVATE static void draudio_inline_sound_stop_callback(draudio_buffer* pBuffer, unsigned int eventID, void *pUserData)
{
(void)pBuffer;
(void)eventID;
assert(pBuffer != NULL);
assert(eventID == DRAUDIO_EVENT_ID_STOP);
assert(pUserData != NULL);
draudio_sound* pSound = (draudio_sound*)pUserData;
draudio_delete_sound(pSound);
}
DRAUDIO_PRIVATE void draudio_prepend_sound(draudio_sound* pSound)
{
assert(pSound != NULL);
assert(pSound->pWorld != NULL);
assert(pSound->pPrevSound == NULL);
draudio_lock_mutex(pSound->pWorld->lock);
{
pSound->pNextSound = pSound->pWorld->pFirstSound;
if (pSound->pNextSound != NULL) {
pSound->pNextSound->pPrevSound = pSound;
}
pSound->pWorld->pFirstSound = pSound;
}
draudio_unlock_mutex(pSound->pWorld->lock);
}
DRAUDIO_PRIVATE void draudio_remove_sound_nolock(draudio_sound* pSound)
{
assert(pSound != NULL);
assert(pSound->pWorld != NULL);
if (pSound == pSound->pWorld->pFirstSound) {
pSound->pWorld->pFirstSound = pSound->pNextSound;
}
if (pSound->pNextSound != NULL) {
pSound->pNextSound->pPrevSound = pSound->pPrevSound;
}
if (pSound->pPrevSound != NULL) {
pSound->pPrevSound->pNextSound = pSound->pNextSound;
}
}
DRAUDIO_PRIVATE bool draudio_is_inline_sound(draudio_sound* pSound)
{
assert(pSound != NULL);
return draudio_get_stop_callback(pSound->pBuffer).callback == draudio_inline_sound_stop_callback;
}
draudio_world* draudio_create_world(draudio_device* pDevice)
{
draudio_world* pWorld = (draudio_world*)malloc(sizeof(*pWorld));
if (pWorld != NULL)
{
pWorld->pDevice = pDevice;
pWorld->playbackState = draudio_playing;
pWorld->pFirstSound = NULL;
pWorld->lock = draudio_create_mutex();
}
return pWorld;
}
void draudio_delete_world(draudio_world* pWorld)
{
if (pWorld == NULL) {
return;
}
// Delete every sound first.
draudio_delete_all_sounds(pWorld);
// Delete the lock after deleting every sound because we still need thread-safety at this point.
draudio_delete_mutex(pWorld->lock);
// Free the world last.
free(pWorld);
}
draudio_sound* draudio_create_sound(draudio_world* pWorld, draudio_sound_desc desc)
{
if (pWorld == NULL) {
return NULL;
}
if ((desc.pData == NULL || desc.sizeInBytes == 0) && (desc.onRead == NULL || desc.onSeek == NULL)) {
// When streaming is not being used, the initial data must be valid at creation time.
return NULL;
}
draudio_sound* pSound = (draudio_sound*)malloc(sizeof(*pSound));
if (pSound == NULL) {
return NULL;
}
pSound->pWorld = pWorld;
pSound->prevPlaybackState = draudio_stopped;
pSound->pNextSound = NULL;
pSound->pPrevSound = NULL;
pSound->isUsingStreamingBuffer = desc.sizeInBytes == 0 || desc.pData == NULL;
pSound->markedForDeletion = false;
pSound->onDelete = desc.onDelete;
pSound->onRead = desc.onRead;
pSound->onSeek = desc.onSeek;
draudio_buffer_desc bufferDesc;
bufferDesc.flags = desc.flags;
bufferDesc.format = desc.format;
bufferDesc.channels = desc.channels;
bufferDesc.sampleRate = desc.sampleRate;
bufferDesc.bitsPerSample = desc.bitsPerSample;
bufferDesc.sizeInBytes = desc.sizeInBytes;
bufferDesc.pData = desc.pData;
if (pSound->isUsingStreamingBuffer)
{
draudio_streaming_callbacks streamingCallbacks;
streamingCallbacks.pUserData = pSound;
streamingCallbacks.read = draudio_on_sound_read_callback;
streamingCallbacks.seek = draudio_on_sound_seek_callback;
pSound->pBuffer = draudio_create_streaming_buffer(pWorld->pDevice, &bufferDesc, streamingCallbacks, desc.extraDataSize);
if (pSound->pBuffer != NULL && desc.pExtraData != NULL)
{
memcpy(draudio_get_streaming_buffer_extra_data(pSound->pBuffer), desc.pExtraData, desc.extraDataSize);
}
}
else
{
pSound->pBuffer = draudio_create_buffer(pWorld->pDevice, &bufferDesc, desc.extraDataSize);
if (pSound->pBuffer != NULL && desc.pExtraData != NULL)
{
memcpy(draudio_get_buffer_extra_data(pSound->pBuffer), desc.pExtraData, desc.extraDataSize);
}
}
// Return NULL if we failed to create the internal audio buffer.
if (pSound->pBuffer == NULL) {
free(pSound);
return NULL;
}
// Only attach the sound to the internal list at the end when we know everything has worked.
draudio_prepend_sound(pSound);
return pSound;
}
void draudio_delete_sound(draudio_sound* pSound)
{
if (pSound == NULL) {
return;
}
draudio_lock_mutex(pSound->pWorld->lock);
{
if (pSound->markedForDeletion) {
assert(false);
return;
}
pSound->markedForDeletion = true;
// Remove the sound from the internal list first.
draudio_remove_sound_nolock(pSound);
// If we're deleting an inline sound, we want to remove the stop event callback. If we don't do this, we'll end up trying to delete
// the sound twice.
if (draudio_is_inline_sound(pSound)) {
draudio_register_stop_callback(pSound->pBuffer, NULL, NULL);
}
// Let the application know that the sound is being deleted. We want to do this after removing the stop event just to be sure the
// application doesn't try to explicitly stop the sound in this callback - that would be a problem for inlined sounds because they
// are configured to delete themselves upon stopping which we are already in the process of doing.
if (pSound->onDelete != NULL) {
pSound->onDelete(pSound);
}
// Delete the internal audio buffer before letting the host application know about the deletion.
draudio_delete_buffer(pSound->pBuffer);
}
draudio_unlock_mutex(pSound->pWorld->lock);
// Only free the sound after the application has been made aware the sound is being deleted.
free(pSound);
}
void draudio_delete_all_sounds(draudio_world* pWorld)
{
if (pWorld == NULL) {
return;
}
while (pWorld->pFirstSound != NULL) {
draudio_delete_sound(pWorld->pFirstSound);
}
}
size_t draudio_get_sound_extra_data_size(draudio_sound* pSound)
{
if (pSound == NULL) {
return 0;
}
if (pSound->isUsingStreamingBuffer) {
return draudio_get_streaming_buffer_extra_data_size(pSound->pBuffer);
} else {
return draudio_get_buffer_extra_data_size(pSound->pBuffer);
}
}
void* draudio_get_sound_extra_data(draudio_sound* pSound)
{
if (pSound == NULL) {
return NULL;
}
if (pSound->isUsingStreamingBuffer) {
return draudio_get_streaming_buffer_extra_data(pSound->pBuffer);
} else {
return draudio_get_buffer_extra_data(pSound->pBuffer);
}
}
void draudio_play_sound(draudio_sound* pSound, bool loop)
{
if (pSound != NULL) {
if (pSound->isUsingStreamingBuffer) {
draudio_play_streaming_buffer(pSound->pBuffer, loop);
} else {
draudio_play(pSound->pBuffer, loop);
}
}
}
void draudio_pause_sound(draudio_sound* pSound)
{
if (pSound != NULL) {
draudio_pause(pSound->pBuffer);
}
}
void draudio_stop_sound(draudio_sound* pSound)
{
if (pSound != NULL) {
draudio_stop(pSound->pBuffer);
}
}
draudio_playback_state draudio_get_sound_playback_state(draudio_sound* pSound)
{
if (pSound == NULL) {
return draudio_stopped;
}
return draudio_get_playback_state(pSound->pBuffer);
}
bool draudio_is_sound_looping(draudio_sound* pSound)
{
if (pSound == NULL) {
return false;
}
if (pSound->isUsingStreamingBuffer) {
return draudio_is_streaming_buffer_looping(pSound->pBuffer);
} else {
return draudio_is_looping(pSound->pBuffer);
}
}
void draudio_play_inline_sound(draudio_world* pWorld, draudio_sound_desc desc)
{
if (pWorld == NULL) {
return;
}
// We need to explicitly ensure 3D positioning is disabled.
desc.flags &= ~DRAUDIO_ENABLE_3D;
draudio_sound* pSound = draudio_create_sound(pWorld, desc);
if (pSound != NULL)
{
// For inline sounds we set a callback for when the sound is stopped. When this callback is fired, the sound is deleted.
draudio_set_sound_stop_callback(pSound, draudio_inline_sound_stop_callback, pSound);
// Start playing the sound once everything else has been set up.
draudio_play_sound(pSound, false);
}
}
void draudio_play_inline_sound_3f(draudio_world* pWorld, draudio_sound_desc desc, float posX, float posY, float posZ)
{
if (pWorld == NULL) {
return;
}
draudio_sound* pSound = draudio_create_sound(pWorld, desc);
if (pSound != NULL)
{
// For inline sounds we set a callback for when the sound is stopped. When this callback is fired, the sound is deleted.
draudio_set_sound_stop_callback(pSound, draudio_inline_sound_stop_callback, pSound);
// Set the position before playing anything.
draudio_set_sound_position(pSound, posX, posY, posZ);
// Start playing the sound once everything else has been set up.
draudio_play_sound(pSound, false);
}
}
void draudio_stop_all_sounds(draudio_world* pWorld)
{
if (pWorld == NULL) {
return;
}
bool wasPlaying = pWorld->playbackState == draudio_playing;
if (pWorld->playbackState != draudio_stopped)
{
// We need to loop over every sound and stop them. We also need to keep track of their previous playback state
// so that when resume_all_sounds() is called, it can be restored correctly.
for (draudio_sound* pSound = pWorld->pFirstSound; pSound != NULL; pSound = pSound->pNextSound)
{
if (wasPlaying) {
pSound->prevPlaybackState = draudio_get_sound_playback_state(pSound);
}
draudio_stop_sound(pSound);
}
}
}
void draudio_pause_all_sounds(draudio_world* pWorld)
{
if (pWorld == NULL) {
return;
}
if (pWorld->playbackState == draudio_playing)
{
// We need to loop over every sound and stop them. We also need to keep track of their previous playback state
// so that when resume_all_sounds() is called, it can be restored correctly.
for (draudio_sound* pSound = pWorld->pFirstSound; pSound != NULL; pSound = pSound->pNextSound)
{
pSound->prevPlaybackState = draudio_get_sound_playback_state(pSound);
draudio_pause_sound(pSound);
}
}
}
void draudio_resume_all_sounds(draudio_world* pWorld)
{
if (pWorld == NULL) {
return;
}
if (pWorld->playbackState != draudio_playing)
{
// When resuming playback, we use the previous playback state to determine how to resume.
for (draudio_sound* pSound = pWorld->pFirstSound; pSound != NULL; pSound = pSound->pNextSound)
{
if (pSound->prevPlaybackState == draudio_playing) {
draudio_play_sound(pSound, draudio_is_sound_looping(pSound));
}
}
}
}
void draudio_set_sound_stop_callback(draudio_sound* pSound, draudio_event_callback_proc callback, void* pUserData)
{
if (pSound != NULL) {
draudio_register_stop_callback(pSound->pBuffer, callback, pUserData);
}
}
void draudio_set_sound_pause_callback(draudio_sound* pSound, draudio_event_callback_proc callback, void* pUserData)
{
if (pSound != NULL) {
draudio_register_pause_callback(pSound->pBuffer, callback, pUserData);
}
}
void draudio_set_sound_play_callback(draudio_sound* pSound, draudio_event_callback_proc callback, void* pUserData)
{
if (pSound != NULL) {
draudio_register_play_callback(pSound->pBuffer, callback, pUserData);
}
}
void draudio_set_sound_position(draudio_sound* pSound, float posX, float posY, float posZ)
{
if (pSound != NULL) {
draudio_set_position(pSound->pBuffer, posX, posY, posZ);
}
}
void draudio_set_sound_3d_mode(draudio_sound* pSound, draudio_3d_mode mode)
{
if (pSound != NULL) {
draudio_set_3d_mode(pSound->pBuffer, mode);
}
}
draudio_3d_mode draudio_get_sound_3d_mode(draudio_sound* pSound)
{
if (pSound == NULL) {
return draudio_3d_mode_disabled;
}
return draudio_get_3d_mode(pSound->pBuffer);
}
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// BACKENDS
//
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//
// DirectSound
//
// The DirectSound backend is mostly simple, except for event handling. Events
// are achieved through the use of Win32 event objects and waiting on them to
// be put into a signaled state by DirectSound. Due to this mechanic we need to
// create a worker thread that waits on each event.
//
// The worker thread waits on three general types of events. The first is an
// event that is signaled when the thread needs to be terminated. The second
// is an event that is signaled when a new set of events need to be waited on.
// The third is a set of events that correspond to an output buffer event (such
// as stop, pause, play and marker events.)
//
///////////////////////////////////////////////////////////////////////////////
#ifdef DRAUDIO_BUILD_DSOUND
#include <windows.h>
#include <dsound.h>
#include <mmreg.h>
#include <stdio.h> // For testing and debugging with printf(). Delete this later.
// Define a NULL GUID for use later on. If we don't do this and use GUID_NULL we'll end
// up with a link error.
GUID DRAUDIO_GUID_NULL = {0x00000000, 0x0000, 0x0000, {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}};
//// Message Queue ////
#define DRAUDIO_MESSAGE_ID_UNKNOWN 0
#define DRAUDIO_MESSAGE_ID_EVENT 1
#define DRAUDIO_MESSAGE_ID_DELETE_BUFFER 2
#define DRAUDIO_MESSAGE_ID_DELETE_DEVICE 3
#define DRAUDIO_MESSAGE_ID_TERMINATE_THREAD 4
/// Structure representing an individual message
typedef struct
{
/// The message ID.
int id;
/// A pointer to the relevant buffer.
draudio_buffer* pBuffer;
// The message-specific data.
union
{
struct
{
/// A pointer to the callback function.
draudio_event_callback_proc callback;
/// The event ID.
unsigned int eventID;
/// The callback user data.
void* pUserData;
} callback_event;
struct
{
/// A pointer to the DirectSound buffer object.
LPDIRECTSOUNDBUFFER8 pDSBuffer;
/// A pointer to the 3D DirectSound buffer object. This will be NULL if 3D positioning is disabled for the buffer.
LPDIRECTSOUND3DBUFFER pDSBuffer3D;
/// A pointer to the object for handling notification events.
LPDIRECTSOUNDNOTIFY pDSNotify;
} delete_buffer;
struct
{
/// A pointer to the DIRECTSOUND object that was created with DirectSoundCreate8().
LPDIRECTSOUND8 pDS;
/// A pointer to the DIRECTSOUNDBUFFER object for the primary buffer.
LPDIRECTSOUNDBUFFER pDSPrimaryBuffer;
/// A pointer to the DIRECTSOUND3DLISTENER8 object associated with the device.
LPDIRECTSOUND3DLISTENER pDSListener;
/// A pointer to the device object being deleted.
draudio_device* pDevice;
} delete_device;
} data;
} draudio_message_dsound;
/// Structure representing the main message queue.
///
/// The message queue is implemented as a fixed-sized cyclic array which means there should be no significant data
/// movement and fast pushing and popping.
typedef struct
{
/// The buffer containing the list of events.
draudio_message_dsound messages[DRAUDIO_MAX_MESSAGE_QUEUE_SIZE];
/// The number of active messages in the queue.
unsigned int messageCount;
/// The index of the first message in the queue.
unsigned int iFirstMessage;
/// The mutex for synchronizing access to message pushing and popping.
draudio_mutex queueLock;
/// The semaphore that's used for blocking draudio_next_message_dsound(). The maximum value is set to DRAUDIO_MAX_MESSAGE_QUEUE_SIZE.
HANDLE hMessageSemaphore;
/// The message handling thread.
HANDLE hMessageHandlingThread;
/// Keeps track of whether or not the queue is deleted. We use this to ensure a thread does not try to post an event.
bool isDeleted;
} draudio_message_queue_dsound;
/// The function to run on the message handling thread. This is implemented down the bottom.
DWORD WINAPI MessageHandlingThread_DSound(draudio_message_queue_dsound* pQueue);
/// Posts a new message. This is thread safe.
void draudio_post_message_dsound(draudio_message_queue_dsound* pQueue, draudio_message_dsound msg);
/// Initializes the given mesasge queue.
bool draudio_init_message_queue_dsound(draudio_message_queue_dsound* pQueue)
{
if (pQueue == NULL) {
return false;
}
pQueue->messageCount = 0;
pQueue->iFirstMessage = 0;
pQueue->queueLock = draudio_create_mutex();
if (pQueue->queueLock == NULL) {
return false;
}
pQueue->hMessageSemaphore = CreateSemaphoreA(NULL, 0, DRAUDIO_MAX_MESSAGE_QUEUE_SIZE, NULL);
if (pQueue->hMessageSemaphore == NULL)
{
draudio_delete_mutex(pQueue->queueLock);
return false;
}
pQueue->hMessageHandlingThread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)MessageHandlingThread_DSound, pQueue, 0, NULL);
if (pQueue->hMessageHandlingThread == NULL)
{
CloseHandle(pQueue->hMessageSemaphore);
draudio_delete_mutex(pQueue->queueLock);
return false;
}
pQueue->isDeleted = false;
return true;
}
/// Uninitializes the given message queue.
void draudio_uninit_message_queue_dsound(draudio_message_queue_dsound* pQueue)
{
// We need to make sure the thread is closed properly before returning from here. To do this we just post an DRAUDIO_MESSAGE_ID_TERMINATE_THREAD
// event to the message queue and wait for the thread to finish.
draudio_message_dsound msg;
msg.id = DRAUDIO_MESSAGE_ID_TERMINATE_THREAD;
draudio_post_message_dsound(pQueue, msg);
// Once we posted the event we just wait for the thread to process it and terminate.
WaitForSingleObject(pQueue->hMessageHandlingThread, INFINITE);
// At this point the thread has been terminated and we can clear everything.
CloseHandle(pQueue->hMessageHandlingThread);
pQueue->hMessageHandlingThread = NULL;
CloseHandle(pQueue->hMessageSemaphore);
pQueue->hMessageSemaphore = NULL;
pQueue->isDeleted = true;
draudio_lock_mutex(pQueue->queueLock);
{
pQueue->messageCount = 0;
pQueue->iFirstMessage = 0;
}
draudio_unlock_mutex(pQueue->queueLock);
draudio_delete_mutex(pQueue->queueLock);
pQueue->queueLock = NULL;
}
void draudio_post_message_dsound(draudio_message_queue_dsound* pQueue, draudio_message_dsound msg)
{
assert(pQueue != NULL);
if (pQueue->isDeleted) {
return;
}
draudio_lock_mutex(pQueue->queueLock);
{
if (pQueue->messageCount < DRAUDIO_MAX_MESSAGE_QUEUE_SIZE)
{
pQueue->messages[(pQueue->iFirstMessage + pQueue->messageCount) % DRAUDIO_MAX_MESSAGE_QUEUE_SIZE] = msg;
pQueue->messageCount += 1;
ReleaseSemaphore(pQueue->hMessageSemaphore, 1, NULL);
}
}
draudio_unlock_mutex(pQueue->queueLock);
}
/// Retrieves the next message in the queue.
///
/// @remarks
/// This blocks until a message is available. This will return false when it receives the DRAUDIO_MESSAGE_ID_TERMINATE_THREAD message.
bool draudio_next_message_dsound(draudio_message_queue_dsound* pQueue, draudio_message_dsound* pMsgOut)
{
if (WaitForSingleObject(pQueue->hMessageSemaphore, INFINITE) == WAIT_OBJECT_0) // Wait for a message to become available.
{
draudio_message_dsound msg;
msg.id = DRAUDIO_MESSAGE_ID_UNKNOWN;
draudio_lock_mutex(pQueue->queueLock);
{
assert(pQueue->messageCount > 0);
msg = pQueue->messages[pQueue->iFirstMessage];
pQueue->iFirstMessage = (pQueue->iFirstMessage + 1) % DRAUDIO_MAX_MESSAGE_QUEUE_SIZE;
pQueue->messageCount -= 1;
}
draudio_unlock_mutex(pQueue->queueLock);
if (pMsgOut != NULL) {
pMsgOut[0] = msg;
}
return msg.id != DRAUDIO_MESSAGE_ID_TERMINATE_THREAD;
}
return false;
}
DWORD WINAPI MessageHandlingThread_DSound(draudio_message_queue_dsound* pQueue)
{
assert(pQueue != NULL);
draudio_message_dsound msg;
while (draudio_next_message_dsound(pQueue, &msg))
{
assert(msg.id != DRAUDIO_MESSAGE_ID_TERMINATE_THREAD); // <-- draudio_next_message_dsound() will return false when it receives DRAUDIO_MESSAGE_ID_TERMINATE_THREAD.
switch (msg.id)
{
case DRAUDIO_MESSAGE_ID_EVENT:
{
assert(msg.data.callback_event.callback != NULL);
msg.data.callback_event.callback(msg.pBuffer, msg.data.callback_event.eventID, msg.data.callback_event.pUserData);
break;
}
case DRAUDIO_MESSAGE_ID_DELETE_BUFFER:
{
if (msg.data.delete_buffer.pDSNotify != NULL) {
IDirectSoundNotify_Release(msg.data.delete_buffer.pDSNotify);
}
if (msg.data.delete_buffer.pDSBuffer3D != NULL) {
IDirectSound3DBuffer_Release(msg.data.delete_buffer.pDSBuffer3D);
}
if (msg.data.delete_buffer.pDSBuffer != NULL) {
IDirectSoundBuffer8_Release(msg.data.delete_buffer.pDSBuffer);
}
free(msg.pBuffer);
break;
}
case DRAUDIO_MESSAGE_ID_DELETE_DEVICE:
{
if (msg.data.delete_device.pDSListener != NULL) {
IDirectSound3DListener_Release(msg.data.delete_device.pDSListener);
}
if (msg.data.delete_device.pDSPrimaryBuffer != NULL) {
IDirectSoundBuffer_Release(msg.data.delete_device.pDSPrimaryBuffer);
}
if (msg.data.delete_device.pDS != NULL) {
IDirectSound_Release(msg.data.delete_device.pDS);
}
free(msg.data.delete_device.pDevice);
break;
}
default:
{
// Should never hit this.
assert(false);
break;
}
}
}
return 0;
}
/// Deactivates (but does not delete) every event associated with the given buffer.
void draudio_deactivate_buffer_events_dsound(draudio_buffer* pBuffer);
//// Event Management ////
typedef struct draudio_event_manager_dsound draudio_event_manager_dsound;
typedef struct draudio_event_dsound draudio_event_dsound;
struct draudio_event_dsound
{
/// A pointer to the event manager that owns this event.
draudio_event_manager_dsound* pEventManager;
/// The event.
HANDLE hEvent;
/// The callback.
draudio_event_callback_proc callback;
/// A pointer to the applicable buffer.
draudio_buffer* pBuffer;
/// The event ID. For on_stop events, this will be set to DRAUDIO_EVENT_STOP
unsigned int eventID;
/// A pointer to the user data.
void* pUserData;
/// The marker offset. Only used for marker events. Should be set to 0 for non-markers.
DWORD markerOffset;
/// Events are stored in a linked list. This is a pointer to the next event in the list.
draudio_event_dsound* pNextEvent;
/// A pointer to the previous event.
draudio_event_dsound* pPrevEvent;
};
struct draudio_event_manager_dsound
{
/// A pointer to the message queue where messages will be posted for event processing.
draudio_message_queue_dsound* pMessageQueue;
/// A handle to the event worker thread.
HANDLE hThread;
/// A handle to the terminator event object.
HANDLE hTerminateEvent;
/// A handle to the refresher event object.
HANDLE hRefreshEvent;
/// The mutex to use when refreshing the worker thread. This is used to ensure only one refresh is done at a time.
draudio_mutex refreshMutex;
/// The synchronization lock.
draudio_mutex mainLock;
/// The event object for notifying dr_audio when an event has finished being handled by the event handling thread.
HANDLE hEventCompletionLock;
/// The first event in a list.
draudio_event_dsound* pFirstEvent;
/// The last event in the list of events.
draudio_event_dsound* pLastEvent;
};
/// Locks the event manager.
void draudio_lock_events_dsound(draudio_event_manager_dsound* pEventManager)
{
draudio_lock_mutex(pEventManager->mainLock);
}
/// Unlocks the event manager.
void draudio_unlock_events_dsound(draudio_event_manager_dsound* pEventManager)
{
draudio_unlock_mutex(pEventManager->mainLock);
}
/// Removes the given event from the event lists.
///
/// @remarks
/// This will be used when moving the event to a new location in the list or when it is being deleted.
void draudio_remove_event_dsound_nolock(draudio_event_dsound* pEvent)
{
assert(pEvent != NULL);
draudio_event_manager_dsound* pEventManager = pEvent->pEventManager;
assert(pEventManager != NULL);
if (pEventManager->pFirstEvent == pEvent) {
pEventManager->pFirstEvent = pEvent->pNextEvent;
}
if (pEventManager->pLastEvent == pEvent) {
pEventManager->pLastEvent = pEvent->pPrevEvent;
}
if (pEvent->pPrevEvent != NULL) {
pEvent->pPrevEvent->pNextEvent = pEvent->pNextEvent;
}
if (pEvent->pNextEvent != NULL) {
pEvent->pNextEvent->pPrevEvent = pEvent->pPrevEvent;
}
pEvent->pNextEvent = NULL;
pEvent->pPrevEvent = NULL;
}
/// @copydoc draudio_remove_event_dsound_nolock()
void draudio_remove_event_dsound(draudio_event_dsound* pEvent)
{
assert(pEvent != NULL);
draudio_event_manager_dsound* pEventManager = pEvent->pEventManager;
draudio_lock_events_dsound(pEventManager);
{
draudio_remove_event_dsound_nolock(pEvent);
}
draudio_unlock_events_dsound(pEventManager);
}
/// Adds the given event to the end of the internal list.
void draudio_append_event_dsound(draudio_event_dsound* pEvent)
{
assert(pEvent != NULL);
draudio_event_manager_dsound* pEventManager = pEvent->pEventManager;
draudio_lock_events_dsound(pEventManager);
{
draudio_remove_event_dsound_nolock(pEvent);
assert(pEvent->pNextEvent == NULL);
if (pEventManager->pLastEvent != NULL) {
pEvent->pPrevEvent = pEventManager->pLastEvent;
pEvent->pPrevEvent->pNextEvent = pEvent;
}
if (pEventManager->pFirstEvent == NULL) {
pEventManager->pFirstEvent = pEvent;
}
pEventManager->pLastEvent = pEvent;
}
draudio_unlock_events_dsound(pEventManager);
}
void draudio_refresh_worker_thread_event_queue(draudio_event_manager_dsound* pEventManager)
{
assert(pEventManager != NULL);
// To refresh the worker thread we just need to signal the refresh event. We then just need to wait for
// processing to finish which we can do by waiting on another event to become signaled.
draudio_lock_mutex(pEventManager->refreshMutex);
{
// Signal a refresh.
SetEvent(pEventManager->hRefreshEvent);
// Wait for refreshing to complete.
WaitForSingleObject(pEventManager->hEventCompletionLock, INFINITE);
}
draudio_unlock_mutex(pEventManager->refreshMutex);
}
/// Closes the Win32 event handle of the given event.
void draudio_close_win32_event_handle_dsound(draudio_event_dsound* pEvent)
{
assert(pEvent != NULL);
assert(pEvent->pEventManager != NULL);
// At the time of calling this function, pEvent should have been removed from the internal list. The issue is that
// the event notification thread is waiting on it. Thus, we need to refresh the worker thread to ensure the event
// have been flushed from that queue. To do this we just signal a special event that's used to trigger a refresh.
draudio_refresh_worker_thread_event_queue(pEvent->pEventManager);
// The worker thread should not be waiting on the event so we can go ahead and close the handle now.
CloseHandle(pEvent->hEvent);
pEvent->hEvent = NULL;
}
/// Updates the given event to use the given callback and user data.
void draudio_update_event_dsound(draudio_event_dsound* pEvent, draudio_event_callback_proc callback, void* pUserData)
{
assert(pEvent != NULL);
pEvent->callback = callback;
pEvent->pUserData = pUserData;
draudio_refresh_worker_thread_event_queue(pEvent->pEventManager);
}
/// Creates a new event, but does not activate it.
///
/// @remarks
/// When an event is created, it just sits dormant and will never be triggered until it has been
/// activated with draudio_activate_event_dsound().
draudio_event_dsound* draudio_create_event_dsound(draudio_event_manager_dsound* pEventManager, draudio_event_callback_proc callback, draudio_buffer* pBuffer, unsigned int eventID, void* pUserData)
{
draudio_event_dsound* pEvent = (draudio_event_dsound*)malloc(sizeof(draudio_event_dsound));
if (pEvent != NULL)
{
pEvent->pEventManager = pEventManager;
pEvent->hEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
pEvent->callback = NULL;
pEvent->pBuffer = pBuffer;
pEvent->eventID = eventID;
pEvent->pUserData = NULL;
pEvent->markerOffset = 0;
pEvent->pNextEvent = NULL;
pEvent->pPrevEvent = NULL;
// Append the event to the internal list.
draudio_append_event_dsound(pEvent);
// This roundabout way of setting the callback and user data is to ensure the worker thread is made aware that it needs
// to refresh it's local event data.
draudio_update_event_dsound(pEvent, callback, pUserData);
}
return pEvent;
}
/// Deletes an event, and deactivates it.
///
/// @remarks
/// This will not return until the event has been deleted completely.
void draudio_delete_event_dsound(draudio_event_dsound* pEvent)
{
assert(pEvent != NULL);
// Set everything to NULL so the worker thread is aware that the event is about to get deleted.
pEvent->pBuffer = NULL;
pEvent->callback = NULL;
pEvent->eventID = 0;
pEvent->pUserData = NULL;
pEvent->markerOffset = 0;
// Remove the event from the list.
draudio_remove_event_dsound(pEvent);
// Close the Win32 event handle.
if (pEvent->hEvent != NULL) {
draudio_close_win32_event_handle_dsound(pEvent);
}
// At this point everything has been closed so we can safely free the memory and return.
free(pEvent);
}
/// Gathers the event handles and callback data for all of the relevant buffer events.
unsigned int draudio_gather_events_dsound(draudio_event_manager_dsound *pEventManager, HANDLE* pHandlesOut, draudio_event_dsound** ppEventsOut, unsigned int outputBufferSize)
{
assert(pEventManager != NULL);
assert(pHandlesOut != NULL);
assert(ppEventsOut != NULL);
assert(outputBufferSize >= 2);
unsigned int i = 2;
draudio_lock_events_dsound(pEventManager);
{
pHandlesOut[0] = pEventManager->hTerminateEvent;
ppEventsOut[0] = NULL;
pHandlesOut[1] = pEventManager->hRefreshEvent;
ppEventsOut[1] = NULL;
draudio_event_dsound* pEvent = pEventManager->pFirstEvent;
while (i < outputBufferSize && pEvent != NULL)
{
if (pEvent->hEvent != NULL)
{
pHandlesOut[i] = pEvent->hEvent;
ppEventsOut[i] = pEvent;
i += 1;
}
pEvent = pEvent->pNextEvent;
}
}
draudio_unlock_events_dsound(pEventManager);
return i;
}
/// The entry point to the event worker thread.
DWORD WINAPI DSound_EventWorkerThreadProc(draudio_event_manager_dsound *pEventManager)
{
if (pEventManager != NULL)
{
HANDLE hTerminateEvent = pEventManager->hTerminateEvent;
HANDLE hRefreshEvent = pEventManager->hRefreshEvent;
HANDLE eventHandles[1024];
draudio_event_dsound* events[1024];
unsigned int eventCount = draudio_gather_events_dsound(pEventManager, eventHandles, events, 1024); // <-- Initial gather.
bool requestedRefresh = false;
for (;;)
{
if (requestedRefresh)
{
eventCount = draudio_gather_events_dsound(pEventManager, eventHandles, events, 1024);
// Refreshing is done, so now we need to let other threads know about it.
SetEvent(pEventManager->hEventCompletionLock);
requestedRefresh = false;
}
DWORD rc = WaitForMultipleObjects(eventCount, eventHandles, FALSE, INFINITE);
if (rc >= WAIT_OBJECT_0 && rc < eventCount)
{
const unsigned int eventIndex = rc - WAIT_OBJECT_0;
HANDLE hEvent = eventHandles[eventIndex];
if (hEvent == hTerminateEvent)
{
// The terminator event was signaled. We just return from the thread immediately.
return 0;
}
if (hEvent == hRefreshEvent)
{
assert(hRefreshEvent == pEventManager->hRefreshEvent);
// This event will get signaled when a new set of events need to be waited on, such as when a new event has been registered on a buffer.
requestedRefresh = true;
continue;
}
// If we get here if means we have hit a callback event.
draudio_event_dsound* pEvent = events[eventIndex];
if (pEvent->callback != NULL)
{
assert(pEvent->hEvent == hEvent);
// The stop event will be signaled by DirectSound when IDirectSoundBuffer::Stop() is called. The problem is that we need to call that when the
// sound is paused as well. Thus, we need to check if we got the stop event, and if so DON'T call the callback function if it is in a non-stopped
// state.
bool isStopEventButNotStopped = pEvent->eventID == DRAUDIO_EVENT_ID_STOP && draudio_get_playback_state(pEvent->pBuffer) != draudio_stopped;
if (!isStopEventButNotStopped)
{
// We don't call the callback directly. Instead we post a message to the message handling thread for processing later.
draudio_message_dsound msg;
msg.id = DRAUDIO_MESSAGE_ID_EVENT;
msg.pBuffer = pEvent->pBuffer;
msg.data.callback_event.callback = pEvent->callback;
msg.data.callback_event.eventID = pEvent->eventID;
msg.data.callback_event.pUserData = pEvent->pUserData;
draudio_post_message_dsound(pEventManager->pMessageQueue, msg);
}
}
}
}
}
return 0;
}
/// Initializes the event manager by creating the thread and event objects.
bool draudio_init_event_manager_dsound(draudio_event_manager_dsound* pEventManager, draudio_message_queue_dsound* pMessageQueue)
{
assert(pEventManager != NULL);
assert(pMessageQueue != NULL);
pEventManager->pMessageQueue = pMessageQueue;
HANDLE hTerminateEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
if (hTerminateEvent == NULL) {
return false;
}
HANDLE hRefreshEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
if (hRefreshEvent == NULL)
{
CloseHandle(hTerminateEvent);
return false;
}
draudio_mutex refreshMutex = draudio_create_mutex();
if (refreshMutex == NULL)
{
CloseHandle(hTerminateEvent);
CloseHandle(hRefreshEvent);
return false;
}
draudio_mutex mainLock = draudio_create_mutex();
if (mainLock == NULL)
{
CloseHandle(hTerminateEvent);
CloseHandle(hRefreshEvent);
draudio_delete_mutex(refreshMutex);
return false;
}
HANDLE hEventCompletionLock = CreateEventA(NULL, FALSE, FALSE, NULL);
if (hEventCompletionLock == NULL)
{
CloseHandle(hTerminateEvent);
CloseHandle(hRefreshEvent);
draudio_delete_mutex(refreshMutex);
draudio_delete_mutex(mainLock);
return false;
}
HANDLE hThread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)DSound_EventWorkerThreadProc, pEventManager, 0, NULL);
if (hThread == NULL)
{
CloseHandle(hTerminateEvent);
CloseHandle(hRefreshEvent);
draudio_delete_mutex(refreshMutex);
draudio_delete_mutex(mainLock);
CloseHandle(hEventCompletionLock);
return false;
}
pEventManager->hTerminateEvent = hTerminateEvent;
pEventManager->hRefreshEvent = hRefreshEvent;
pEventManager->refreshMutex = refreshMutex;
pEventManager->mainLock = mainLock;
pEventManager->hEventCompletionLock = hEventCompletionLock;
pEventManager->hThread = hThread;
pEventManager->pFirstEvent = NULL;
pEventManager->pLastEvent = NULL;
return true;
}
/// Shuts down the event manager by closing the thread and event objects.
///
/// @remarks
/// This does not return until the worker thread has been terminated completely.
/// @par
/// This will delete every event, so any pointers to events will be made invalid upon calling this function.
void draudio_uninit_event_manager_dsound(draudio_event_manager_dsound* pEventManager)
{
assert(pEventManager != NULL);
// Cleanly delete every event first.
while (pEventManager->pFirstEvent != NULL) {
draudio_delete_event_dsound(pEventManager->pFirstEvent);
}
// Terminate the thread and wait for the thread to finish executing before freeing the context for real.
SignalObjectAndWait(pEventManager->hTerminateEvent, pEventManager->hThread, INFINITE, FALSE);
// Only delete the thread after it has returned naturally.
CloseHandle(pEventManager->hThread);
pEventManager->hThread = NULL;
// Once the thread has been terminated we can delete the terminator and refresher events.
CloseHandle(pEventManager->hTerminateEvent);
pEventManager->hTerminateEvent = NULL;
CloseHandle(pEventManager->hRefreshEvent);
pEventManager->hRefreshEvent = NULL;
draudio_delete_mutex(pEventManager->refreshMutex);
pEventManager->refreshMutex = NULL;
draudio_delete_mutex(pEventManager->mainLock);
pEventManager->mainLock = NULL;
CloseHandle(pEventManager->hEventCompletionLock);
pEventManager->hEventCompletionLock = NULL;
}
//// End Event Management ////
static GUID _g_DSListenerGUID = {0x279AFA84, 0x4981, 0x11CE, {0xA5, 0x21, 0x00, 0x20, 0xAF, 0x0B, 0xE5, 0x60}};
static GUID _g_DirectSoundBuffer8GUID = {0x6825a449, 0x7524, 0x4d82, {0x92, 0x0f, 0x50, 0xe3, 0x6a, 0xb3, 0xab, 0x1e}};
static GUID _g_DirectSound3DBuffer8GUID = {0x279AFA86, 0x4981, 0x11CE, {0xA5, 0x21, 0x00, 0x20, 0xAF, 0x0B, 0xE5, 0x60}};
static GUID _g_DirectSoundNotifyGUID = {0xb0210783, 0x89cd, 0x11d0, {0xaf, 0x08, 0x00, 0xa0, 0xc9, 0x25, 0xcd, 0x16}};
static GUID _g_KSDATAFORMAT_SUBTYPE_PCM_GUID = {0x00000001, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}};
static GUID _g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID = {0x00000003, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}};
#ifdef __cplusplus
static GUID g_DSListenerGUID = _g_DSListenerGUID;
static GUID g_DirectSoundBuffer8GUID = _g_DirectSoundBuffer8GUID;
static GUID g_DirectSound3DBuffer8GUID = _g_DirectSound3DBuffer8GUID;
static GUID g_DirectSoundNotifyGUID = _g_DirectSoundNotifyGUID;
static GUID g_KSDATAFORMAT_SUBTYPE_PCM_GUID = _g_KSDATAFORMAT_SUBTYPE_PCM_GUID;
static GUID g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID = _g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID;
#else
static GUID* g_DSListenerGUID = &_g_DSListenerGUID;
static GUID* g_DirectSoundBuffer8GUID = &_g_DirectSoundBuffer8GUID;
static GUID* g_DirectSound3DBuffer8GUID = &_g_DirectSound3DBuffer8GUID;
static GUID* g_DirectSoundNotifyGUID = &_g_DirectSoundNotifyGUID;
static GUID* g_KSDATAFORMAT_SUBTYPE_PCM_GUID = &_g_KSDATAFORMAT_SUBTYPE_PCM_GUID;
static GUID* g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID = &_g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID;
#endif
typedef HRESULT (WINAPI * pDirectSoundCreate8Proc)(LPCGUID pcGuidDevice, LPDIRECTSOUND8 *ppDS8, LPUNKNOWN pUnkOuter);
typedef HRESULT (WINAPI * pDirectSoundEnumerateAProc)(LPDSENUMCALLBACKA pDSEnumCallback, LPVOID pContext);
typedef HRESULT (WINAPI * pDirectSoundCaptureCreate8Proc)(LPCGUID pcGuidDevice, LPDIRECTSOUNDCAPTURE8 *ppDSC8, LPUNKNOWN pUnkOuter);
typedef HRESULT (WINAPI * pDirectSoundCaptureEnumerateAProc)(LPDSENUMCALLBACKA pDSEnumCallback, LPVOID pContext);
typedef struct
{
/// A pointer to the GUID of the device. This will be set to all zeros for the default device.
GUID guid;
/// The description of the device.
char description[256];
/// The module name of the DirectSound driver corresponding to this device.
char moduleName[256];
} draudio_device_info_dsound;
typedef struct
{
/// The base context data. This must always be the first item in the struct.
draudio_context base;
/// A handle to the dsound.dll file that was loaded by LoadLibrary().
HMODULE hDSoundDLL;
// DirectSound APIs.
pDirectSoundCreate8Proc pDirectSoundCreate8;
pDirectSoundEnumerateAProc pDirectSoundEnumerateA;
pDirectSoundCaptureCreate8Proc pDirectSoundCaptureCreate8;
pDirectSoundCaptureEnumerateAProc pDirectSoundCaptureEnumerateA;
/// The number of output devices that were iterated when the context was created. This is static, so if the user was to unplug
/// a device one would need to re-create the context.
unsigned int outputDeviceCount;
/// The buffer containing the list of enumerated output devices.
draudio_device_info_dsound outputDeviceInfo[DRAUDIO_MAX_DEVICE_COUNT];
/// The number of capture devices that were iterated when the context was created. This is static, so if the user was to unplug
/// a device one would need to re-create the context.
unsigned int inputDeviceCount;
/// The buffer containing the list of enumerated input devices.
draudio_device_info_dsound inputDeviceInfo[DRAUDIO_MAX_DEVICE_COUNT];
/// The event manager.
draudio_event_manager_dsound eventManager;
/// The message queue.
draudio_message_queue_dsound messageQueue;
} draudio_context_dsound;
typedef struct
{
/// The base device data. This must always be the first item in the struct.
draudio_device base;
/// A pointer to the DIRECTSOUND object that was created with DirectSoundCreate8().
LPDIRECTSOUND8 pDS;
/// A pointer to the DIRECTSOUNDBUFFER object for the primary buffer.
LPDIRECTSOUNDBUFFER pDSPrimaryBuffer;
/// A pointer to the DIRECTSOUND3DLISTENER8 object associated with the device.
LPDIRECTSOUND3DLISTENER pDSListener;
} draudio_device_dsound;
typedef struct
{
/// The base buffer data. This must always be the first item in the struct.
draudio_buffer base;
/// A pointer to the DirectSound buffer object.
LPDIRECTSOUNDBUFFER8 pDSBuffer;
/// A pointer to the 3D DirectSound buffer object. This will be NULL if 3D positioning is disabled for the buffer.
LPDIRECTSOUND3DBUFFER pDSBuffer3D;
/// A pointer to the object for handling notification events.
LPDIRECTSOUNDNOTIFY pDSNotify;
/// The current playback state.
draudio_playback_state playbackState;
/// The number of marker events that have been registered. This will never be more than DRAUDIO_MAX_MARKER_COUNT.
unsigned int markerEventCount;
/// The marker events.
draudio_event_dsound* pMarkerEvents[DRAUDIO_MAX_MARKER_COUNT];
/// The event to trigger when the sound is stopped.
draudio_event_dsound* pStopEvent;
/// The event to trigger when the sound is paused.
draudio_event_dsound* pPauseEvent;
/// The event to trigger when the sound is played or resumed.
draudio_event_dsound* pPlayEvent;
/// The size in bytes of the buffer's extra data.
unsigned int extraDataSize;
/// The buffer's extra data.
unsigned char pExtraData[1];
} draudio_buffer_dsound;
void draudio_activate_buffer_events_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
unsigned int dwPositionNotifies = 0;
DSBPOSITIONNOTIFY n[DRAUDIO_MAX_MARKER_COUNT + 1]; // +1 because we use this array for the markers + stop event.
// Stop
if (pBufferDS->pStopEvent != NULL)
{
LPDSBPOSITIONNOTIFY pN = n + dwPositionNotifies;
pN->dwOffset = DSBPN_OFFSETSTOP;
pN->hEventNotify = pBufferDS->pStopEvent->hEvent;
dwPositionNotifies += 1;
}
// Markers
for (unsigned int iMarker = 0; iMarker < pBufferDS->markerEventCount; ++iMarker)
{
LPDSBPOSITIONNOTIFY pN = n + dwPositionNotifies;
pN->dwOffset = pBufferDS->pMarkerEvents[iMarker]->markerOffset;
pN->hEventNotify = pBufferDS->pMarkerEvents[iMarker]->hEvent;
dwPositionNotifies += 1;
}
HRESULT hr = IDirectSoundNotify_SetNotificationPositions(pBufferDS->pDSNotify, dwPositionNotifies, n);
#if 0
if (FAILED(hr)) {
printf("WARNING: FAILED TO CREATE DIRECTSOUND NOTIFIERS\n");
}
#else
(void)hr;
#endif
}
void draudio_deactivate_buffer_events_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
HRESULT hr = IDirectSoundNotify_SetNotificationPositions(pBufferDS->pDSNotify, 0, NULL);
#if 0
if (FAILED(hr)) {
printf("WARNING: FAILED TO CLEAR DIRECTSOUND NOTIFIERS\n");
}
#else
(void)hr;
#endif
}
void draudio_delete_context_dsound(draudio_context* pContext)
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pContext;
assert(pContextDS != NULL);
draudio_uninit_event_manager_dsound(&pContextDS->eventManager);
// The message queue needs to uninitialized after the DirectSound marker notification thread.
draudio_uninit_message_queue_dsound(&pContextDS->messageQueue);
FreeLibrary(pContextDS->hDSoundDLL);
free(pContextDS);
}
unsigned int draudio_get_output_device_count_dsound(draudio_context* pContext)
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pContext;
assert(pContextDS != NULL);
return pContextDS->outputDeviceCount;
}
bool draudio_get_output_device_info_dsound(draudio_context* pContext, unsigned int deviceIndex, draudio_device_info* pInfoOut)
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pContext;
assert(pContextDS != NULL);
assert(pInfoOut != NULL);
if (deviceIndex >= pContextDS->outputDeviceCount) {
return false;
}
draudio_strcpy(pInfoOut->description, sizeof(pInfoOut->description), pContextDS->outputDeviceInfo[deviceIndex].description);
return true;
}
draudio_device* draudio_create_output_device_dsound(draudio_context* pContext, unsigned int deviceIndex)
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pContext;
assert(pContextDS != NULL);
if (deviceIndex >= pContextDS->outputDeviceCount) {
return NULL;
}
LPDIRECTSOUND8 pDS;
// Create the device.
HRESULT hr;
if (deviceIndex == 0) {
hr = pContextDS->pDirectSoundCreate8(NULL, &pDS, NULL);
} else {
hr = pContextDS->pDirectSoundCreate8(&pContextDS->outputDeviceInfo[deviceIndex].guid, &pDS, NULL);
}
if (FAILED(hr)) {
return NULL;
}
// Set the cooperative level. Must be done before anything else.
hr = IDirectSound_SetCooperativeLevel(pDS, GetForegroundWindow(), DSSCL_EXCLUSIVE);
if (FAILED(hr)) {
IDirectSound_Release(pDS);
return NULL;
}
// Primary buffer.
DSBUFFERDESC descDSPrimary;
memset(&descDSPrimary, 0, sizeof(DSBUFFERDESC));
descDSPrimary.dwSize = sizeof(DSBUFFERDESC);
descDSPrimary.dwFlags = DSBCAPS_PRIMARYBUFFER | DSBCAPS_CTRLVOLUME | DSBCAPS_CTRL3D;
descDSPrimary.guid3DAlgorithm = DRAUDIO_GUID_NULL;
LPDIRECTSOUNDBUFFER pDSPrimaryBuffer;
hr = IDirectSound_CreateSoundBuffer(pDS, &descDSPrimary, &pDSPrimaryBuffer, NULL);
if (FAILED(hr)) {
IDirectSound_Release(pDS);
return NULL;
}
WAVEFORMATIEEEFLOATEX wf = {0};
wf.Format.cbSize = sizeof(wf);
wf.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE;
wf.Format.nChannels = 2;
wf.Format.nSamplesPerSec = 48000;
wf.Format.wBitsPerSample = 32;
wf.Format.nBlockAlign = (wf.Format.nChannels * wf.Format.wBitsPerSample) / 8;
wf.Format.nAvgBytesPerSec = wf.Format.nBlockAlign * wf.Format.nSamplesPerSec;
wf.Samples.wValidBitsPerSample = wf.Format.wBitsPerSample;
wf.dwChannelMask = 0;
wf.SubFormat = _g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID;
hr = IDirectSoundBuffer_SetFormat(pDSPrimaryBuffer, (WAVEFORMATEX*)&wf);
if (FAILED(hr)) {
IDirectSoundBuffer_Release(pDSPrimaryBuffer);
IDirectSound_Release(pDS);
return NULL;
}
// Listener.
LPDIRECTSOUND3DLISTENER pDSListener = NULL;
hr = IDirectSound3DListener_QueryInterface(pDSPrimaryBuffer, g_DSListenerGUID, (LPVOID*)&pDSListener);
if (FAILED(hr)) {
IDirectSoundBuffer_Release(pDSPrimaryBuffer);
IDirectSound_Release(pDS);
return NULL;
}
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)malloc(sizeof(draudio_device_dsound));
if (pDeviceDS != NULL)
{
pDeviceDS->base.pContext = pContext;
pDeviceDS->pDS = pDS;
pDeviceDS->pDSPrimaryBuffer = pDSPrimaryBuffer;
pDeviceDS->pDSListener = pDSListener;
return (draudio_device*)pDeviceDS;
}
else
{
IDirectSound3DListener_Release(pDSListener);
IDirectSoundBuffer_Release(pDeviceDS->pDSPrimaryBuffer);
IDirectSound_Release(pDS);
return NULL;
}
}
void draudio_delete_output_device_dsound(draudio_device* pDevice)
{
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)pDevice;
assert(pDeviceDS != NULL);
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pDevice->pContext;
assert(pContextDS != NULL);
// The device id not deleted straight away. Instead we post a message to the message for delayed processing. The reason for this is that buffer
// deletion is also delayed which means we want to ensure any delayed processing of buffers is handled before deleting the device.
draudio_message_dsound msg;
msg.id = DRAUDIO_MESSAGE_ID_DELETE_DEVICE;
msg.pBuffer = NULL;
msg.data.delete_device.pDSListener = pDeviceDS->pDSListener;
msg.data.delete_device.pDSPrimaryBuffer = pDeviceDS->pDSPrimaryBuffer;
msg.data.delete_device.pDS = pDeviceDS->pDS;
msg.data.delete_device.pDevice = pDevice;
draudio_post_message_dsound(&pContextDS->messageQueue, msg);
#if 0
IDirectSound3DListener_Release(pDeviceDS->pDSListener);
IDirectSoundBuffer_Release(pDeviceDS->pDSPrimaryBuffer);
IDirectSound_Release(pDeviceDS->pDS);
free(pDeviceDS);
#endif
}
draudio_buffer* draudio_create_buffer_dsound(draudio_device* pDevice, draudio_buffer_desc* pBufferDesc, size_t extraDataSize)
{
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)pDevice;
assert(pDeviceDS != NULL);
assert(pBufferDesc != NULL);
// 3D is only valid for mono sounds.
if (pBufferDesc->channels > 1 && (pBufferDesc->flags & DRAUDIO_ENABLE_3D) != 0) {
return NULL;
}
WAVEFORMATIEEEFLOATEX wf = {0};
wf.Format.cbSize = sizeof(wf);
wf.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE;
wf.Format.nChannels = (WORD)pBufferDesc->channels;
wf.Format.nSamplesPerSec = pBufferDesc->sampleRate;
wf.Format.wBitsPerSample = (WORD)pBufferDesc->bitsPerSample;
wf.Format.nBlockAlign = (wf.Format.nChannels * wf.Format.wBitsPerSample) / 8;
wf.Format.nAvgBytesPerSec = wf.Format.nBlockAlign * wf.Format.nSamplesPerSec;
wf.Samples.wValidBitsPerSample = wf.Format.wBitsPerSample;
wf.dwChannelMask = 0;
if (pBufferDesc->format == draudio_format_pcm) {
wf.SubFormat = _g_KSDATAFORMAT_SUBTYPE_PCM_GUID;
} else if (pBufferDesc->format == draudio_format_float) {
wf.SubFormat = _g_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT_GUID;
} else {
return NULL;
}
// We want to try and create a 3D enabled buffer, however this will fail whenever the number of channels is > 1. In this case
// we do not want to attempt to create a 3D enabled buffer because it will just fail anyway. Instead we'll just create a normal
// buffer with panning enabled.
DSBUFFERDESC descDS;
memset(&descDS, 0, sizeof(DSBUFFERDESC));
descDS.dwSize = sizeof(DSBUFFERDESC);
descDS.dwFlags = DSBCAPS_CTRLVOLUME | DSBCAPS_CTRLPOSITIONNOTIFY | DSBCAPS_GETCURRENTPOSITION2 | DSBCAPS_GLOBALFOCUS;
descDS.dwBufferBytes = (DWORD)pBufferDesc->sizeInBytes;
descDS.lpwfxFormat = (WAVEFORMATEX*)&wf;
LPDIRECTSOUNDBUFFER8 pDSBuffer = NULL;
LPDIRECTSOUND3DBUFFER pDSBuffer3D = NULL;
if ((pBufferDesc->flags & DRAUDIO_ENABLE_3D) == 0)
{
// 3D Disabled.
descDS.dwFlags |= DSBCAPS_CTRLPAN;
LPDIRECTSOUNDBUFFER pDSBufferTemp;
HRESULT hr = IDirectSound_CreateSoundBuffer(pDeviceDS->pDS, &descDS, &pDSBufferTemp, NULL);
if (FAILED(hr)) {
return NULL;
}
hr = IDirectSoundBuffer_QueryInterface(pDSBufferTemp, g_DirectSoundBuffer8GUID, (void**)&pDSBuffer);
if (FAILED(hr)) {
IDirectSoundBuffer_Release(pDSBufferTemp);
return NULL;
}
IDirectSoundBuffer_Release(pDSBufferTemp);
}
else
{
// 3D Enabled.
descDS.dwFlags |= DSBCAPS_CTRL3D;
descDS.guid3DAlgorithm = DS3DALG_DEFAULT;
LPDIRECTSOUNDBUFFER pDSBufferTemp;
HRESULT hr = IDirectSound_CreateSoundBuffer(pDeviceDS->pDS, &descDS, &pDSBufferTemp, NULL);
if (FAILED(hr)) {
return NULL;
}
hr = IDirectSoundBuffer_QueryInterface(pDSBufferTemp, g_DirectSoundBuffer8GUID, (void**)&pDSBuffer);
if (FAILED(hr)) {
IDirectSoundBuffer_Release(pDSBufferTemp);
return NULL;
}
IDirectSoundBuffer_Release(pDSBufferTemp);
hr = IDirectSoundBuffer_QueryInterface(pDSBuffer, g_DirectSound3DBuffer8GUID, (void**)&pDSBuffer3D);
if (FAILED(hr)) {
return NULL;
}
IDirectSound3DBuffer_SetPosition(pDSBuffer3D, 0, 0, 0, DS3D_IMMEDIATE);
if ((pBufferDesc->flags & DRAUDIO_RELATIVE_3D) != 0) {
IDirectSound3DBuffer_SetMode(pDSBuffer3D, DS3DMODE_HEADRELATIVE, DS3D_IMMEDIATE);
}
}
// We need to create a notification object so we can notify the host application when the playback buffer hits a certain point.
LPDIRECTSOUNDNOTIFY pDSNotify;
HRESULT hr = IDirectSoundBuffer8_QueryInterface(pDSBuffer, g_DirectSoundNotifyGUID, (void**)&pDSNotify);
if (FAILED(hr)) {
IDirectSound3DBuffer_Release(pDSBuffer3D);
IDirectSoundBuffer8_Release(pDSBuffer);
return NULL;
}
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)malloc(sizeof(draudio_buffer_dsound) - sizeof(pBufferDS->pExtraData) + extraDataSize);
if (pBufferDS == NULL) {
IDirectSound3DBuffer_Release(pDSBuffer3D);
IDirectSoundBuffer8_Release(pDSBuffer);
return NULL;
}
pBufferDS->base.pDevice = pDevice;
pBufferDS->pDSBuffer = pDSBuffer;
pBufferDS->pDSBuffer3D = pDSBuffer3D;
pBufferDS->pDSNotify = pDSNotify;
pBufferDS->playbackState = draudio_stopped;
pBufferDS->markerEventCount = 0;
memset(pBufferDS->pMarkerEvents, 0, sizeof(pBufferDS->pMarkerEvents));
pBufferDS->pStopEvent = NULL;
pBufferDS->pPauseEvent = NULL;
pBufferDS->pPlayEvent = NULL;
// Fill with initial data, if applicable.
if (pBufferDesc->pData != NULL) {
draudio_set_buffer_data((draudio_buffer*)pBufferDS, 0, pBufferDesc->pData, pBufferDesc->sizeInBytes);
}
return (draudio_buffer*)pBufferDS;
}
void draudio_delete_buffer_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
assert(pBuffer->pDevice != NULL);
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pBuffer->pDevice->pContext;
assert(pContextDS != NULL);
// Deactivate the DirectSound notify events for sanity.
draudio_deactivate_buffer_events_dsound(pBuffer);
draudio_message_dsound msg;
msg.id = DRAUDIO_MESSAGE_ID_DELETE_BUFFER;
msg.pBuffer = pBuffer;
msg.data.delete_buffer.pDSNotify = pBufferDS->pDSNotify;
msg.data.delete_buffer.pDSBuffer3D = pBufferDS->pDSBuffer3D;
msg.data.delete_buffer.pDSBuffer = pBufferDS->pDSBuffer;
draudio_post_message_dsound(&pContextDS->messageQueue, msg);
#if 0
if (pBufferDS->pDSNotify != NULL) {
IDirectSoundNotify_Release(pBufferDS->pDSNotify);
}
if (pBufferDS->pDSBuffer3D != NULL) {
IDirectSound3DBuffer_Release(pBufferDS->pDSBuffer3D);
}
if (pBufferDS->pDSBuffer != NULL) {
IDirectSoundBuffer8_Release(pBufferDS->pDSBuffer);
}
free(pBufferDS);
#endif
}
unsigned int draudio_get_buffer_extra_data_size_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
return pBufferDS->extraDataSize;
}
void* draudio_get_buffer_extra_data_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
return pBufferDS->pExtraData;
}
void draudio_set_buffer_data_dsound(draudio_buffer* pBuffer, size_t offset, const void* pData, size_t dataSizeInBytes)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
assert(pData != NULL);
LPVOID lpvWrite;
DWORD dwLength;
HRESULT hr = IDirectSoundBuffer8_Lock(pBufferDS->pDSBuffer, (DWORD)offset, (DWORD)dataSizeInBytes, &lpvWrite, &dwLength, NULL, NULL, 0);
if (FAILED(hr)) {
return;
}
assert(dataSizeInBytes <= dwLength);
memcpy(lpvWrite, pData, dataSizeInBytes);
hr = IDirectSoundBuffer8_Unlock(pBufferDS->pDSBuffer, lpvWrite, dwLength, NULL, 0);
if (FAILED(hr)) {
return;
}
}
void draudio_play_dsound(draudio_buffer* pBuffer, bool loop)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
bool postEvent = true;
if (pBufferDS->playbackState == draudio_playing) {
postEvent = false;
}
// Events need to be activated.
if (pBufferDS->playbackState == draudio_stopped) {
draudio_activate_buffer_events_dsound(pBuffer);
}
DWORD dwFlags = 0;
if (loop) {
dwFlags |= DSBPLAY_LOOPING;
}
pBufferDS->playbackState = draudio_playing;
IDirectSoundBuffer8_Play(pBufferDS->pDSBuffer, 0, 0, dwFlags);
// If we have a play event we need to signal the event which will cause the worker thread to call the callback function.
if (pBufferDS->pPlayEvent != NULL && postEvent) {
SetEvent(pBufferDS->pPlayEvent->hEvent);
}
}
void draudio_pause_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (pBufferDS->playbackState == draudio_playing)
{
pBufferDS->playbackState = draudio_paused;
IDirectSoundBuffer8_Stop(pBufferDS->pDSBuffer);
// If we have a pause event we need to signal the event which will cause the worker thread to call the callback function.
if (pBufferDS->pPlayEvent != NULL) {
SetEvent(pBufferDS->pPauseEvent->hEvent);
}
}
}
void draudio_stop_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (pBufferDS->playbackState == draudio_playing)
{
pBufferDS->playbackState = draudio_stopped;
IDirectSoundBuffer8_Stop(pBufferDS->pDSBuffer);
IDirectSoundBuffer8_SetCurrentPosition(pBufferDS->pDSBuffer, 0);
}
else if (pBufferDS->playbackState == draudio_paused)
{
pBufferDS->playbackState = draudio_stopped;
IDirectSoundBuffer8_SetCurrentPosition(pBufferDS->pDSBuffer, 0);
if (pBufferDS->pStopEvent != NULL) {
SetEvent(pBufferDS->pStopEvent->hEvent);
}
}
}
draudio_playback_state draudio_get_playback_state_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
return pBufferDS->playbackState;
}
void draudio_set_playback_position_dsound(draudio_buffer* pBuffer, unsigned int position)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
IDirectSoundBuffer8_SetCurrentPosition(pBufferDS->pDSBuffer, position);
}
unsigned int draudio_get_playback_position_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
DWORD position;
HRESULT hr = IDirectSoundBuffer8_GetCurrentPosition(pBufferDS->pDSBuffer, &position, NULL);
if (FAILED(hr)) {
return 0;
}
return position;
}
void draudio_set_pan_dsound(draudio_buffer* pBuffer, float pan)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
LONG panDB;
if (pan == 0) {
panDB = DSBPAN_CENTER;
} else {
if (pan > 1) {
panDB = DSBPAN_RIGHT;
} else if (pan < -1) {
panDB = DSBPAN_LEFT;
} else {
if (pan < 0) {
panDB = (LONG)((20*log10f(1 + pan)) * 100);
} else {
panDB = -(LONG)((20*log10f(1 - pan)) * 100);
}
}
}
IDirectSoundBuffer_SetPan(pBufferDS->pDSBuffer, panDB);
}
float draudio_get_pan_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
LONG panDB;
HRESULT hr = IDirectSoundBuffer_GetPan(pBufferDS->pDSBuffer, &panDB);
if (FAILED(hr)) {
return 0;
}
if (panDB < 0) {
return -(1 - (float)(1.0f / powf(10.0f, -panDB / (20.0f*100.0f))));
}
if (panDB > 0) {
return (1 - (float)(1.0f / powf(10.0f, panDB / (20.0f*100.0f))));
}
return 0;
}
void draudio_set_volume_dsound(draudio_buffer* pBuffer, float volume)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
LONG volumeDB;
if (volume > 0) {
if (volume < 1) {
volumeDB = (LONG)((20*log10f(volume)) * 100);
} else {
volumeDB = DSBVOLUME_MAX;
}
} else {
volumeDB = DSBVOLUME_MIN;
}
IDirectSoundBuffer_SetVolume(pBufferDS->pDSBuffer, volumeDB);
}
float draudio_get_volume_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
LONG volumeDB;
HRESULT hr = IDirectSoundBuffer_GetVolume(pBufferDS->pDSBuffer, &volumeDB);
if (FAILED(hr)) {
return 1;
}
return (float)(1.0f / powf(10.0f, -volumeDB / (20.0f*100.0f)));
}
void draudio_remove_markers_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
for (unsigned int iMarker = 0; iMarker < pBufferDS->markerEventCount; ++iMarker)
{
if (pBufferDS->pMarkerEvents[iMarker] != NULL) {
draudio_delete_event_dsound(pBufferDS->pMarkerEvents[iMarker]);
pBufferDS->pMarkerEvents[iMarker] = NULL;
}
}
pBufferDS->markerEventCount = 0;
}
bool draudio_register_marker_callback_dsound(draudio_buffer* pBuffer, size_t offsetInBytes, draudio_event_callback_proc callback, unsigned int eventID, void* pUserData)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
assert(pBufferDS->markerEventCount <= DRAUDIO_MAX_MARKER_COUNT);
if (pBufferDS->markerEventCount == DRAUDIO_MAX_MARKER_COUNT) {
// Too many markers.
return false;
}
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pBuffer->pDevice->pContext;
assert(pContextDS != NULL);
draudio_event_dsound* pEvent = draudio_create_event_dsound(&pContextDS->eventManager, callback, pBuffer, eventID, pUserData);
if (pEvent == NULL) {
return false;
}
// draudio_create_event_dsound() will initialize the marker offset to 0, so we'll need to set it manually here.
pEvent->markerOffset = (DWORD)offsetInBytes;
pBufferDS->pMarkerEvents[pBufferDS->markerEventCount] = pEvent;
pBufferDS->markerEventCount += 1;
return true;
}
bool draudio_register_stop_callback_dsound(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (callback == NULL)
{
if (pBufferDS->pStopEvent != NULL) {
draudio_delete_event_dsound(pBufferDS->pStopEvent);
pBufferDS->pStopEvent = NULL;
}
return true;
}
else
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pBuffer->pDevice->pContext;
// If we already have a stop event, just replace the existing one.
if (pBufferDS->pStopEvent != NULL) {
draudio_update_event_dsound(pBufferDS->pStopEvent, callback, pUserData);
} else {
pBufferDS->pStopEvent = draudio_create_event_dsound(&pContextDS->eventManager, callback, pBuffer, DRAUDIO_EVENT_ID_STOP, pUserData);
}
return pBufferDS->pStopEvent != NULL;
}
}
bool draudio_register_pause_callback_dsound(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (callback == NULL)
{
if (pBufferDS->pPauseEvent != NULL) {
draudio_delete_event_dsound(pBufferDS->pPauseEvent);
pBufferDS->pPauseEvent = NULL;
}
return true;
}
else
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pBuffer->pDevice->pContext;
// If we already have a stop event, just replace the existing one.
if (pBufferDS->pPauseEvent != NULL) {
draudio_update_event_dsound(pBufferDS->pPauseEvent, callback, pUserData);
} else {
pBufferDS->pPauseEvent = draudio_create_event_dsound(&pContextDS->eventManager, callback, pBuffer, DRAUDIO_EVENT_ID_PAUSE, pUserData);
}
return pBufferDS->pPauseEvent != NULL;
}
}
bool draudio_register_play_callback_dsound(draudio_buffer* pBuffer, draudio_event_callback_proc callback, void* pUserData)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (callback == NULL)
{
if (pBufferDS->pPlayEvent != NULL) {
draudio_delete_event_dsound(pBufferDS->pPlayEvent);
pBufferDS->pPlayEvent = NULL;
}
return true;
}
else
{
draudio_context_dsound* pContextDS = (draudio_context_dsound*)pBuffer->pDevice->pContext;
// If we already have a stop event, just replace the existing one.
if (pBufferDS->pPlayEvent != NULL) {
draudio_update_event_dsound(pBufferDS->pPlayEvent, callback, pUserData);
} else {
pBufferDS->pPlayEvent = draudio_create_event_dsound(&pContextDS->eventManager, callback, pBuffer, DRAUDIO_EVENT_ID_PLAY, pUserData);
}
return pBufferDS->pPlayEvent != NULL;
}
}
void draudio_set_position_dsound(draudio_buffer* pBuffer, float x, float y, float z)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (pBufferDS->pDSBuffer3D != NULL) {
IDirectSound3DBuffer_SetPosition(pBufferDS->pDSBuffer3D, x, y, z, DS3D_IMMEDIATE);
}
}
void draudio_get_position_dsound(draudio_buffer* pBuffer, float* pPosOut)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
assert(pPosOut != NULL);
if (pBufferDS->pDSBuffer3D != NULL)
{
D3DVECTOR pos;
IDirectSound3DBuffer_GetPosition(pBufferDS->pDSBuffer3D, &pos);
pPosOut[0] = pos.x;
pPosOut[1] = pos.y;
pPosOut[2] = pos.z;
}
else
{
pPosOut[0] = 0;
pPosOut[1] = 1;
pPosOut[2] = 2;
}
}
void draudio_set_listener_position_dsound(draudio_device* pDevice, float x, float y, float z)
{
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)pDevice;
assert(pDeviceDS != NULL);
IDirectSound3DListener_SetPosition(pDeviceDS->pDSListener, x, y, z, DS3D_IMMEDIATE);
}
void draudio_get_listener_position_dsound(draudio_device* pDevice, float* pPosOut)
{
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)pDevice;
assert(pDeviceDS != NULL);
assert(pPosOut != NULL);
D3DVECTOR pos;
IDirectSound3DListener_GetPosition(pDeviceDS->pDSListener, &pos);
pPosOut[0] = pos.x;
pPosOut[1] = pos.y;
pPosOut[2] = pos.z;
}
void draudio_set_listener_orientation_dsound(draudio_device* pDevice, float forwardX, float forwardY, float forwardZ, float upX, float upY, float upZ)
{
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)pDevice;
assert(pDeviceDS != NULL);
IDirectSound3DListener_SetOrientation(pDeviceDS->pDSListener, forwardX, forwardY, forwardZ, upX, upY, upZ, DS3D_IMMEDIATE);
}
void draudio_get_listener_orientation_dsound(draudio_device* pDevice, float* pForwardOut, float* pUpOut)
{
draudio_device_dsound* pDeviceDS = (draudio_device_dsound*)pDevice;
assert(pDeviceDS != NULL);
assert(pForwardOut != NULL);
assert(pUpOut != NULL);
D3DVECTOR forward;
D3DVECTOR up;
IDirectSound3DListener_GetOrientation(pDeviceDS->pDSListener, &forward, &up);
pForwardOut[0] = forward.x;
pForwardOut[1] = forward.y;
pForwardOut[2] = forward.z;
pUpOut[0] = up.x;
pUpOut[1] = up.y;
pUpOut[2] = up.z;
}
void draudio_set_3d_mode_dsound(draudio_buffer* pBuffer, draudio_3d_mode mode)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (pBufferDS->pDSBuffer3D == NULL) {
return;
}
DWORD dwMode = DS3DMODE_NORMAL;
if (mode == draudio_3d_mode_relative) {
dwMode = DS3DMODE_HEADRELATIVE;
} else if (mode == draudio_3d_mode_disabled) {
dwMode = DS3DMODE_DISABLE;
}
IDirectSound3DBuffer_SetMode(pBufferDS->pDSBuffer3D, dwMode, DS3D_IMMEDIATE);
}
draudio_3d_mode draudio_get_3d_mode_dsound(draudio_buffer* pBuffer)
{
draudio_buffer_dsound* pBufferDS = (draudio_buffer_dsound*)pBuffer;
assert(pBufferDS != NULL);
if (pBufferDS->pDSBuffer3D == NULL) {
return draudio_3d_mode_disabled;
}
DWORD dwMode;
if (FAILED(IDirectSound3DBuffer_GetMode(pBufferDS->pDSBuffer3D, &dwMode))) {
return draudio_3d_mode_disabled;
}
if (dwMode == DS3DMODE_NORMAL) {
return draudio_3d_mode_absolute;
}
if (dwMode == DS3DMODE_HEADRELATIVE) {
return draudio_3d_mode_relative;
}
return draudio_3d_mode_disabled;
}
static BOOL CALLBACK DSEnumCallback_OutputDevices(LPGUID lpGuid, LPCSTR lpcstrDescription, LPCSTR lpcstrModule, LPVOID lpContext)
{
// From MSDN:
//
// The first device enumerated is always called the Primary Sound Driver, and the lpGUID parameter of the callback is
// NULL. This device represents the preferred output device set by the user in Control Panel.
draudio_context_dsound* pContextDS = (draudio_context_dsound*)lpContext;
assert(pContextDS != NULL);
if (pContextDS->outputDeviceCount < DRAUDIO_MAX_DEVICE_COUNT)
{
if (lpGuid != NULL) {
memcpy(&pContextDS->outputDeviceInfo[pContextDS->outputDeviceCount].guid, lpGuid, sizeof(GUID));
} else {
memset(&pContextDS->outputDeviceInfo[pContextDS->outputDeviceCount].guid, 0, sizeof(GUID));
}
draudio_strcpy(pContextDS->outputDeviceInfo[pContextDS->outputDeviceCount].description, 256, lpcstrDescription);
draudio_strcpy(pContextDS->outputDeviceInfo[pContextDS->outputDeviceCount].moduleName, 256, lpcstrModule);
pContextDS->outputDeviceCount += 1;
return TRUE;
}
else
{
// Ran out of device slots.
return FALSE;
}
}
static BOOL CALLBACK DSEnumCallback_InputDevices(LPGUID lpGuid, LPCSTR lpcstrDescription, LPCSTR lpcstrModule, LPVOID lpContext)
{
// From MSDN:
//
// The first device enumerated is always called the Primary Sound Driver, and the lpGUID parameter of the callback is
// NULL. This device represents the preferred output device set by the user in Control Panel.
draudio_context_dsound* pContextDS = (draudio_context_dsound*)lpContext;
assert(pContextDS != NULL);
if (pContextDS->inputDeviceCount < DRAUDIO_MAX_DEVICE_COUNT)
{
if (lpGuid != NULL) {
memcpy(&pContextDS->inputDeviceInfo[pContextDS->inputDeviceCount].guid, lpGuid, sizeof(GUID));
} else {
memset(&pContextDS->inputDeviceInfo[pContextDS->inputDeviceCount].guid, 0, sizeof(GUID));
}
draudio_strcpy(pContextDS->inputDeviceInfo[pContextDS->inputDeviceCount].description, 256, lpcstrDescription);
draudio_strcpy(pContextDS->inputDeviceInfo[pContextDS->inputDeviceCount].moduleName, 256, lpcstrModule);
pContextDS->inputDeviceCount += 1;
return TRUE;
}
else
{
// Ran out of device slots.
return FALSE;
}
}
draudio_context* draudio_create_context_dsound()
{
// Load the DLL.
HMODULE hDSoundDLL = LoadLibraryW(L"dsound.dll");
if (hDSoundDLL == NULL) {
return NULL;
}
// Retrieve the APIs.
pDirectSoundCreate8Proc pDirectSoundCreate8 = (pDirectSoundCreate8Proc)GetProcAddress(hDSoundDLL, "DirectSoundCreate8");
if (pDirectSoundCreate8 == NULL){
FreeLibrary(hDSoundDLL);
return NULL;
}
pDirectSoundEnumerateAProc pDirectSoundEnumerateA = (pDirectSoundEnumerateAProc)GetProcAddress(hDSoundDLL, "DirectSoundEnumerateA");
if (pDirectSoundEnumerateA == NULL){
FreeLibrary(hDSoundDLL);
return NULL;
}
pDirectSoundCaptureCreate8Proc pDirectSoundCaptureCreate8 = (pDirectSoundCaptureCreate8Proc)GetProcAddress(hDSoundDLL, "DirectSoundCaptureCreate8");
if (pDirectSoundCaptureCreate8 == NULL) {
FreeLibrary(hDSoundDLL);
return NULL;
}
pDirectSoundCaptureEnumerateAProc pDirectSoundCaptureEnumerateA = (pDirectSoundCaptureEnumerateAProc)GetProcAddress(hDSoundDLL, "DirectSoundCaptureEnumerateA");
if (pDirectSoundCaptureEnumerateA == NULL ){
FreeLibrary(hDSoundDLL);
return NULL;
}
// At this point we can almost certainly assume DirectSound is usable so we'll now go ahead and create the context.
draudio_context_dsound* pContext = (draudio_context_dsound*)malloc(sizeof(draudio_context_dsound));
if (pContext != NULL)
{
pContext->base.delete_context = draudio_delete_context_dsound;
pContext->base.create_output_device = draudio_create_output_device_dsound;
pContext->base.delete_output_device = draudio_delete_output_device_dsound;
pContext->base.get_output_device_count = draudio_get_output_device_count_dsound;
pContext->base.get_output_device_info = draudio_get_output_device_info_dsound;
pContext->base.create_buffer = draudio_create_buffer_dsound;
pContext->base.delete_buffer = draudio_delete_buffer_dsound;
pContext->base.get_buffer_extra_data_size = draudio_get_buffer_extra_data_size_dsound;
pContext->base.get_buffer_extra_data = draudio_get_buffer_extra_data_dsound;
pContext->base.set_buffer_data = draudio_set_buffer_data_dsound;
pContext->base.play = draudio_play_dsound;
pContext->base.pause = draudio_pause_dsound;
pContext->base.stop = draudio_stop_dsound;
pContext->base.get_playback_state = draudio_get_playback_state_dsound;
pContext->base.set_playback_position = draudio_set_playback_position_dsound;
pContext->base.get_playback_position = draudio_get_playback_position_dsound;
pContext->base.set_pan = draudio_set_pan_dsound;
pContext->base.get_pan = draudio_get_pan_dsound;
pContext->base.set_volume = draudio_set_volume_dsound;
pContext->base.get_volume = draudio_get_volume_dsound;
pContext->base.remove_markers = draudio_remove_markers_dsound;
pContext->base.register_marker_callback = draudio_register_marker_callback_dsound;
pContext->base.register_stop_callback = draudio_register_stop_callback_dsound;
pContext->base.register_pause_callback = draudio_register_pause_callback_dsound;
pContext->base.register_play_callback = draudio_register_play_callback_dsound;
pContext->base.set_position = draudio_set_position_dsound;
pContext->base.get_position = draudio_get_position_dsound;
pContext->base.set_listener_position = draudio_set_listener_position_dsound;
pContext->base.get_listener_position = draudio_get_listener_position_dsound;
pContext->base.set_listener_orientation = draudio_set_listener_orientation_dsound;
pContext->base.get_listener_orientation = draudio_get_listener_orientation_dsound;
pContext->base.set_3d_mode = draudio_set_3d_mode_dsound;
pContext->base.get_3d_mode = draudio_get_3d_mode_dsound;
pContext->hDSoundDLL = hDSoundDLL;
pContext->pDirectSoundCreate8 = pDirectSoundCreate8;
pContext->pDirectSoundEnumerateA = pDirectSoundEnumerateA;
pContext->pDirectSoundCaptureCreate8 = pDirectSoundCaptureCreate8;
pContext->pDirectSoundCaptureEnumerateA = pDirectSoundCaptureEnumerateA;
// Enumerate output devices.
pContext->outputDeviceCount = 0;
pContext->pDirectSoundEnumerateA(DSEnumCallback_OutputDevices, pContext);
// Enumerate input devices.
pContext->inputDeviceCount = 0;
pContext->pDirectSoundCaptureEnumerateA(DSEnumCallback_InputDevices, pContext);
// The message queue and marker notification thread.
if (!draudio_init_message_queue_dsound(&pContext->messageQueue) || !draudio_init_event_manager_dsound(&pContext->eventManager, &pContext->messageQueue))
{
// Failed to initialize the event manager.
FreeLibrary(hDSoundDLL);
free(pContext);
return NULL;
}
}
return (draudio_context*)pContext;
}
#endif // !DRAUDIO_BUILD_DSOUND
///////////////////////////////////////////////////////////////////////////////
//
// XAudio2
//
///////////////////////////////////////////////////////////////////////////////
#if 0
#define uuid(x)
#define DX_BUILD
#define INITGUID 1
#include <xaudio2.h>
#endif
#endif
/*
This is free and unencumbered software released into the public domain.
Anyone is free to copy, modify, publish, use, compile, sell, or
distribute this software, either in source code form or as a compiled
binary, for any purpose, commercial or non-commercial, and by any
means.
In jurisdictions that recognize copyright laws, the author or authors
of this software dedicate any and all copyright interest in the
software to the public domain. We make this dedication for the benefit
of the public at large and to the detriment of our heirs and
successors. We intend this dedication to be an overt act of
relinquishment in perpetuity of all present and future rights to this
software under copyright law.
THE SOFTWARE 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.
For more information, please refer to <http://unlicense.org/>
*/