/* RetroArch - A frontend for libretro. * Copyright (C) 2010-2013 - Hans-Kristian Arntzen * * RetroArch is free software: you can redistribute it and/or modify it under the terms * of the GNU General Public License as published by the Free Software Found- * ation, either version 3 of the License, or (at your option) any later version. * * RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with RetroArch. * If not, see . */ #include "../driver.h" #ifdef __APPLE__ #include #include #else #include #include #endif #include #include #ifdef _WIN32 #include #endif #define BUFSIZE 128 typedef struct al { ALuint source; ALuint *buffers; ALuint *res_buf; int res_ptr; ALenum format; int num_buffers; int rate; int queue; uint8_t tmpbuf[BUFSIZE]; int tmpbuf_ptr; ALCdevice *handle; ALCcontext *ctx; bool nonblock; } al_t; static void *al_init(const char *device, unsigned rate, unsigned latency) { (void)device; al_t *al = (al_t*)calloc(1, sizeof(al_t)); if (!al) return NULL; al->handle = alcOpenDevice(NULL); if (al->handle == NULL) goto error; al->ctx = alcCreateContext(al->handle, NULL); if (al->ctx == NULL) goto error; alcMakeContextCurrent(al->ctx); al->rate = rate; al->num_buffers = (latency * rate * 2 * 2) / (1000 * BUFSIZE); al->buffers = (ALuint*)calloc(al->num_buffers, sizeof(ALuint)); al->res_buf = (ALuint*)calloc(al->num_buffers, sizeof(ALuint)); if (al->buffers == NULL || al->res_buf == NULL) goto error; alGenSources(1, &al->source); alGenBuffers(al->num_buffers, al->buffers); memcpy(al->res_buf, al->buffers, al->num_buffers * sizeof(ALuint)); al->res_ptr = al->num_buffers; return al; error: if (al) { alcMakeContextCurrent(NULL); if (al->ctx) alcDestroyContext(al->ctx); if (al->handle) alcCloseDevice(al->handle); if (al->buffers) free(al->buffers); if (al->res_buf) free(al->res_buf); free(al); } return NULL; } static bool al_unqueue_buffers(al_t *al) { ALint val; alGetSourcei(al->source, AL_BUFFERS_PROCESSED, &val); if (val > 0) { alSourceUnqueueBuffers(al->source, val, &al->res_buf[al->res_ptr]); al->res_ptr += val; return true; } return false; } static bool al_get_buffer(al_t *al, ALuint *buffer) { if (al->res_ptr == 0) { #ifndef _WIN32 struct timespec tv = {0}; tv.tv_sec = 0; tv.tv_nsec = 1000000; #endif for (;;) { if (al_unqueue_buffers(al)) break; if (al->nonblock) return false; #ifdef _WIN32 Sleep(1); #else nanosleep(&tv, NULL); #endif } } *buffer = al->res_buf[--al->res_ptr]; return true; } static size_t al_fill_internal_buf(al_t *al, const void *buf, size_t size) { size_t read_size = (BUFSIZE - al->tmpbuf_ptr > (ssize_t)size) ? size : (BUFSIZE - al->tmpbuf_ptr); memcpy(al->tmpbuf + al->tmpbuf_ptr, buf, read_size); al->tmpbuf_ptr += read_size; return read_size; } static ssize_t al_write(void *data, const void *buf, size_t size) { al_t *al = (al_t*)data; size_t written = 0; while (written < size) { size_t rc = al_fill_internal_buf(al, (const char*)buf + written, size - written); written += rc; if (al->tmpbuf_ptr != BUFSIZE) break; ALuint buffer; if (!al_get_buffer(al, &buffer)) return 0; alBufferData(buffer, AL_FORMAT_STEREO16, al->tmpbuf, BUFSIZE, al->rate); al->tmpbuf_ptr = 0; alSourceQueueBuffers(al->source, 1, &buffer); if (alGetError() != AL_NO_ERROR) return -1; ALint val; alGetSourcei(al->source, AL_SOURCE_STATE, &val); if (val != AL_PLAYING) alSourcePlay(al->source); if (alGetError() != AL_NO_ERROR) return -1; } return size; } static bool al_stop(void *data) { (void)data; return true; } static void al_set_nonblock_state(void *data, bool state) { al_t *al = (al_t*)data; al->nonblock = state; } static bool al_start(void *data) { (void)data; return true; } static void al_free(void *data) { al_t *al = (al_t*)data; if (al) { alSourceStop(al->source); alDeleteSources(1, &al->source); if (al->buffers) { alDeleteBuffers(al->num_buffers, al->buffers); free(al->buffers); free(al->res_buf); } } alcMakeContextCurrent(NULL); alcDestroyContext(al->ctx); alcCloseDevice(al->handle); free(al); } const audio_driver_t audio_openal = { al_init, al_write, al_stop, al_start, al_set_nonblock_state, al_free, NULL, "openal" };