ref: 17911753a6d21d11a47c813c3505d3016010ff6f
dir: /src/Backends/Audio/SoftwareMixer/SDL2.cpp/
// Released under the MIT licence. // See LICENCE.txt for details. #include "Backend.h" #include <stddef.h> #include <string.h> #include <string> #include "SDL.h" #include "../../Misc.h" #define MIN(a, b) ((a) < (b) ? (a) : (b)) static void (*parent_callback)(long *stream, size_t frames_total); static SDL_AudioDeviceID device_id; static void Callback(void *user_data, Uint8 *stream_uint8, int len) { (void)user_data; short *stream = (short*)stream_uint8; const size_t frames_total = len / sizeof(short) / 2; size_t frames_done = 0; while (frames_done != frames_total) { long mix_buffer[0x800 * 2]; // 2 because stereo size_t subframes = MIN(0x800, frames_total - frames_done); memset(mix_buffer, 0, subframes * sizeof(long) * 2); parent_callback(mix_buffer, subframes); for (size_t i = 0; i < subframes * 2; ++i) { if (mix_buffer[i] > 0x7FFF) *stream++ = 0x7FFF; else if (mix_buffer[i] < -0x7FFF) *stream++ = -0x7FFF; else *stream++ = mix_buffer[i]; } frames_done += subframes; } } unsigned long SoftwareMixerBackend_Init(void (*callback)(long *stream, size_t frames_total)) { if (SDL_InitSubSystem(SDL_INIT_AUDIO) < 0) { std::string errorMessage = std::string("'SDL_InitSubSystem(SDL_INIT_AUDIO)' failed: ") + SDL_GetError(); Backend_ShowMessageBox("Fatal error (SDL2 audio backend)", errorMessage.c_str()); return 0; } Backend_PrintInfo("Available SDL audio drivers:"); for (int i = 0; i < SDL_GetNumAudioDrivers(); ++i) Backend_PrintInfo("%s", SDL_GetAudioDriver(i)); SDL_AudioSpec specification; specification.freq = 48000; specification.format = AUDIO_S16; specification.channels = 2; specification.samples = 0x400; // Roughly 10 milliseconds for 48000Hz specification.callback = Callback; specification.userdata = NULL; SDL_AudioSpec obtained_specification; device_id = SDL_OpenAudioDevice(NULL, 0, &specification, &obtained_specification, SDL_AUDIO_ALLOW_FREQUENCY_CHANGE); if (device_id == 0) { std::string error_message = std::string("'SDL_OpenAudioDevice' failed: ") + SDL_GetError(); Backend_ShowMessageBox("Fatal error (SDL2 audio backend)", error_message.c_str()); return 0; } Backend_PrintInfo("Selected SDL audio driver: %s", SDL_GetCurrentAudioDriver()); parent_callback = callback; return obtained_specification.freq; } void SoftwareMixerBackend_Deinit(void) { SDL_CloseAudioDevice(device_id); SDL_QuitSubSystem(SDL_INIT_AUDIO); } bool SoftwareMixerBackend_Start(void) { SDL_PauseAudioDevice(device_id, 0); return true; } void SoftwareMixerBackend_LockMixerMutex(void) { SDL_LockAudioDevice(device_id); } void SoftwareMixerBackend_UnlockMixerMutex(void) { SDL_UnlockAudioDevice(device_id); } void SoftwareMixerBackend_LockOrganyaMutex(void) { SDL_LockAudioDevice(device_id); } void SoftwareMixerBackend_UnlockOrganyaMutex(void) { SDL_UnlockAudioDevice(device_id); }