2011-08-08 17:27:52 +02:00
|
|
|
/* SSNES - A Super Nintendo Entertainment System (SNES) Emulator frontend for libsnes.
|
|
|
|
* Copyright (C) 2010-2011 - Hans-Kristian Arntzen
|
2011-08-09 12:58:13 +02:00
|
|
|
* Copyright (C) 2011 - Chris Moeller
|
2011-08-08 17:27:52 +02:00
|
|
|
*
|
|
|
|
* Some code herein may be based on code found in BSNES.
|
|
|
|
*
|
|
|
|
* SSNES 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.
|
|
|
|
*
|
|
|
|
* SSNES 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 SSNES.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "driver.h"
|
|
|
|
#include "general.h"
|
|
|
|
#include "fifo_buffer.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
|
|
|
|
#include <CoreAudio/CoreAudio.h>
|
|
|
|
#include <CoreServices/CoreServices.h>
|
|
|
|
#include <AudioUnit/AudioUnit.h>
|
|
|
|
#include <AudioUnit/AUComponent.h>
|
|
|
|
|
|
|
|
typedef struct coreaudio
|
|
|
|
{
|
|
|
|
pthread_mutex_t lock;
|
|
|
|
pthread_cond_t cond;
|
|
|
|
|
|
|
|
ComponentInstance dev;
|
|
|
|
bool dev_alive;
|
|
|
|
|
|
|
|
fifo_buffer_t *buffer;
|
|
|
|
bool nonblock;
|
|
|
|
} coreaudio_t;
|
|
|
|
|
|
|
|
static void coreaudio_free(void *data)
|
|
|
|
{
|
|
|
|
coreaudio_t *dev = data;
|
|
|
|
if (!dev)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (dev->dev_alive)
|
|
|
|
{
|
|
|
|
AudioOutputUnitStop(dev->dev);
|
|
|
|
CloseComponent(dev->dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dev->buffer)
|
|
|
|
fifo_free(dev->buffer);
|
|
|
|
|
|
|
|
pthread_mutex_destroy(&dev->lock);
|
|
|
|
pthread_cond_destroy(&dev->cond);
|
|
|
|
|
|
|
|
free(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
static OSStatus audio_cb(void *userdata, AudioUnitRenderActionFlags *action_flags,
|
|
|
|
const AudioTimeStamp *time_stamp, UInt32 bus_number,
|
|
|
|
UInt32 number_frames, AudioBufferList *io_data)
|
|
|
|
{
|
|
|
|
coreaudio_t *dev = userdata;
|
|
|
|
(void)time_stamp;
|
|
|
|
(void)bus_number;
|
|
|
|
(void)number_frames;
|
|
|
|
|
|
|
|
if (!io_data)
|
|
|
|
return noErr;
|
|
|
|
if (io_data->mNumberBuffers != 1)
|
|
|
|
return noErr;
|
|
|
|
|
|
|
|
unsigned write_avail = io_data->mBuffers[0].mDataByteSize;
|
2011-08-09 22:07:11 +02:00
|
|
|
void *outbuf = io_data->mBuffers[0].mData;
|
2011-08-08 17:27:52 +02:00
|
|
|
|
|
|
|
pthread_mutex_lock(&dev->lock);
|
|
|
|
if (fifo_read_avail(dev->buffer) < write_avail)
|
|
|
|
{
|
|
|
|
*action_flags = kAudioUnitRenderAction_OutputIsSilence;
|
2011-08-09 22:07:11 +02:00
|
|
|
memset(outbuf, 0, write_avail); // Seems to be needed.
|
2011-08-08 17:27:52 +02:00
|
|
|
pthread_mutex_unlock(&dev->lock);
|
|
|
|
pthread_cond_signal(&dev->cond); // Technically possible to deadlock without.
|
|
|
|
return noErr;
|
|
|
|
}
|
|
|
|
|
2011-08-08 17:31:03 +02:00
|
|
|
fifo_read(dev->buffer, outbuf, write_avail);
|
2011-08-08 17:27:52 +02:00
|
|
|
pthread_mutex_unlock(&dev->lock);
|
|
|
|
pthread_cond_signal(&dev->cond);
|
|
|
|
return noErr;
|
|
|
|
}
|
|
|
|
|
2011-11-02 19:31:36 +01:00
|
|
|
static void *coreaudio_init(const char *device, unsigned rate, unsigned latency)
|
2011-08-08 17:27:52 +02:00
|
|
|
{
|
|
|
|
(void)device;
|
|
|
|
|
|
|
|
coreaudio_t *dev = calloc(1, sizeof(*dev));
|
|
|
|
if (!dev)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
CFRunLoopRef run_loop = NULL;
|
|
|
|
AudioObjectPropertyAddress addr = {
|
|
|
|
kAudioHardwarePropertyRunLoop,
|
|
|
|
kAudioObjectPropertyScopeGlobal,
|
2011-08-08 17:31:03 +02:00
|
|
|
kAudioObjectPropertyElementMaster
|
2011-08-08 17:27:52 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
pthread_mutex_init(&dev->lock, NULL);
|
|
|
|
pthread_cond_init(&dev->cond, NULL);
|
|
|
|
AudioObjectSetPropertyData(kAudioObjectSystemObject, &addr, 0, NULL,
|
|
|
|
sizeof(CFRunLoopRef), &run_loop);
|
|
|
|
|
|
|
|
ComponentDescription desc = {
|
|
|
|
.componentType = kAudioUnitType_Output,
|
|
|
|
.componentSubType = kAudioUnitSubType_HALOutput,
|
|
|
|
.componentManufacturer = kAudioUnitManufacturer_Apple,
|
|
|
|
};
|
|
|
|
|
|
|
|
Component comp = FindNextComponent(NULL, &desc);
|
|
|
|
if (comp == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
OSStatus res = OpenAComponent(comp, &dev->dev);
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
dev->dev_alive = true;
|
|
|
|
|
|
|
|
AudioStreamBasicDescription stream_desc = {
|
|
|
|
.mSampleRate = rate,
|
|
|
|
.mBitsPerChannel = sizeof(float) * CHAR_BIT,
|
|
|
|
.mChannelsPerFrame = 2,
|
|
|
|
.mBytesPerPacket = 2 * sizeof(float),
|
2011-08-08 17:41:55 +02:00
|
|
|
.mBytesPerFrame = 2 * sizeof(float),
|
2011-08-08 17:27:52 +02:00
|
|
|
.mFramesPerPacket = 1,
|
|
|
|
.mFormatID = kAudioFormatLinearPCM,
|
2011-08-08 17:31:03 +02:00
|
|
|
.mFormatFlags = kAudioFormatFlagIsFloat | kAudioFormatFlagIsPacked | (is_little_endian() ? 0 : kAudioFormatFlagIsBigEndian),
|
2011-08-08 17:27:52 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
res = AudioUnitSetProperty(dev->dev, kAudioUnitProperty_StreamFormat,
|
|
|
|
kAudioUnitScope_Input, 0, &stream_desc, sizeof(stream_desc));
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
AudioStreamBasicDescription real_desc;
|
|
|
|
UInt32 i_size = sizeof(real_desc);
|
|
|
|
res = AudioUnitGetProperty(dev->dev, kAudioUnitProperty_StreamFormat, kAudioUnitScope_Input, 0, &real_desc, &i_size);
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
2011-08-08 17:41:55 +02:00
|
|
|
SSNES_LOG("[CoreAudio]: Using output sample rate of %.1f Hz\n", (float)real_desc.mSampleRate);
|
2011-08-08 17:27:52 +02:00
|
|
|
g_settings.audio.out_rate = real_desc.mSampleRate;
|
|
|
|
|
|
|
|
if (real_desc.mChannelsPerFrame != stream_desc.mChannelsPerFrame)
|
|
|
|
goto error;
|
|
|
|
if (real_desc.mBitsPerChannel != stream_desc.mBitsPerChannel)
|
|
|
|
goto error;
|
|
|
|
if (real_desc.mFormatFlags != stream_desc.mFormatFlags)
|
|
|
|
goto error;
|
|
|
|
if (real_desc.mFormatID != stream_desc.mFormatID)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
AudioChannelLayout layout = {
|
|
|
|
.mChannelLayoutTag = kAudioChannelLayoutTag_UseChannelBitmap,
|
|
|
|
.mChannelBitmap = (1 << 2) - 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
res = AudioUnitSetProperty(dev->dev, kAudioUnitProperty_AudioChannelLayout,
|
|
|
|
kAudioUnitScope_Input, 0, &layout, sizeof(layout));
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
AURenderCallbackStruct cb = {
|
|
|
|
.inputProc = audio_cb,
|
|
|
|
.inputProcRefCon = dev,
|
|
|
|
};
|
|
|
|
|
|
|
|
res = AudioUnitSetProperty(dev->dev, kAudioUnitProperty_SetRenderCallback,
|
|
|
|
kAudioUnitScope_Input, 0, &cb, sizeof(cb));
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
res = AudioUnitInitialize(dev->dev);
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
size_t fifo_size = (latency * g_settings.audio.out_rate) / 1000;
|
2011-08-08 17:45:49 +02:00
|
|
|
fifo_size *= 2 * sizeof(float);
|
2011-08-08 17:27:52 +02:00
|
|
|
|
|
|
|
dev->buffer = fifo_new(fifo_size);
|
|
|
|
if (!dev->buffer)
|
|
|
|
goto error;
|
|
|
|
|
2011-08-08 17:45:49 +02:00
|
|
|
SSNES_LOG("[CoreAudio]: Using buffer size of %u bytes: (latency = %u ms)\n", (unsigned)fifo_size, latency);
|
2011-08-08 17:27:52 +02:00
|
|
|
|
|
|
|
res = AudioOutputUnitStart(dev->dev);
|
|
|
|
if (res != noErr)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return dev;
|
|
|
|
|
|
|
|
error:
|
|
|
|
SSNES_ERR("[CoreAudio]: Failed to initialize driver ...\n");
|
|
|
|
coreaudio_free(dev);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-11-02 19:31:36 +01:00
|
|
|
static ssize_t coreaudio_write(void *data, const void *buf_, size_t size)
|
2011-08-08 17:27:52 +02:00
|
|
|
{
|
|
|
|
coreaudio_t *dev = data;
|
|
|
|
|
|
|
|
const uint8_t *buf = buf_;
|
|
|
|
size_t written = 0;
|
|
|
|
|
|
|
|
while (size > 0)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&dev->lock);
|
|
|
|
|
|
|
|
size_t write_avail = fifo_write_avail(dev->buffer);
|
|
|
|
if (write_avail > size)
|
|
|
|
write_avail = size;
|
|
|
|
|
|
|
|
fifo_write(dev->buffer, buf, write_avail);
|
|
|
|
buf += write_avail;
|
|
|
|
written += write_avail;
|
|
|
|
size -= write_avail;
|
|
|
|
|
|
|
|
if (dev->nonblock)
|
|
|
|
{
|
|
|
|
pthread_mutex_unlock(&dev->lock);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (write_avail == 0)
|
|
|
|
pthread_cond_wait(&dev->cond, &dev->lock);
|
|
|
|
pthread_mutex_unlock(&dev->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
return written;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool coreaudio_stop(void *data)
|
|
|
|
{
|
|
|
|
coreaudio_t *dev = data;
|
|
|
|
return AudioOutputUnitStop(dev->dev) == noErr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void coreaudio_set_nonblock_state(void *data, bool state)
|
|
|
|
{
|
|
|
|
coreaudio_t *dev = data;
|
|
|
|
dev->nonblock = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool coreaudio_start(void *data)
|
|
|
|
{
|
|
|
|
coreaudio_t *dev = data;
|
|
|
|
return AudioOutputUnitStart(dev->dev) == noErr;
|
|
|
|
}
|
|
|
|
|
2011-08-08 17:45:49 +02:00
|
|
|
static bool coreaudio_use_float(void *data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-08-08 17:31:03 +02:00
|
|
|
const audio_driver_t audio_coreaudio = {
|
2011-08-08 17:27:52 +02:00
|
|
|
.init = coreaudio_init,
|
|
|
|
.write = coreaudio_write,
|
|
|
|
.stop = coreaudio_stop,
|
|
|
|
.start = coreaudio_start,
|
|
|
|
.set_nonblock_state = coreaudio_set_nonblock_state,
|
|
|
|
.free = coreaudio_free,
|
2011-08-08 17:45:49 +02:00
|
|
|
.use_float = coreaudio_use_float,
|
2011-08-08 17:27:52 +02:00
|
|
|
.ident = "coreaudio"
|
|
|
|
};
|
|
|
|
|