2015-07-01 08:02:44 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015 Andrew Kelley
|
|
|
|
*
|
|
|
|
* This file is part of libsoundio, which is MIT licensed.
|
|
|
|
* See http://opensource.org/licenses/MIT
|
|
|
|
*/
|
|
|
|
|
2015-07-01 08:20:26 +00:00
|
|
|
#include "soundio.hpp"
|
2015-07-01 08:02:44 +00:00
|
|
|
#include "util.hpp"
|
|
|
|
#include "dummy.hpp"
|
2015-07-02 09:53:08 +00:00
|
|
|
#include "os.hpp"
|
2015-07-02 10:48:27 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#ifdef SOUNDIO_HAVE_PULSEAUDIO
|
|
|
|
#include "pulseaudio.hpp"
|
|
|
|
#endif
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-07 09:55:32 +00:00
|
|
|
#ifdef SOUNDIO_HAVE_ALSA
|
|
|
|
#include "alsa.hpp"
|
|
|
|
#endif
|
|
|
|
|
2015-07-01 09:37:51 +00:00
|
|
|
#include <string.h>
|
2015-07-01 08:02:44 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
2015-07-01 08:24:57 +00:00
|
|
|
const char *soundio_error_string(int error) {
|
|
|
|
switch ((enum SoundIoError)error) {
|
|
|
|
case SoundIoErrorNone: return "(no error)";
|
|
|
|
case SoundIoErrorNoMem: return "out of memory";
|
|
|
|
case SoundIoErrorInitAudioBackend: return "unable to initialize audio backend";
|
|
|
|
case SoundIoErrorSystemResources: return "system resource not available";
|
|
|
|
case SoundIoErrorOpeningDevice: return "unable to open device";
|
|
|
|
}
|
2015-07-01 20:24:47 +00:00
|
|
|
soundio_panic("invalid error enum value: %d", error);
|
2015-07-01 08:24:57 +00:00
|
|
|
}
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
int soundio_get_bytes_per_sample(enum SoundIoSampleFormat sample_format) {
|
|
|
|
switch (sample_format) {
|
|
|
|
case SoundIoSampleFormatUInt8: return 1;
|
|
|
|
case SoundIoSampleFormatInt16: return 2;
|
|
|
|
case SoundIoSampleFormatInt24: return 3;
|
|
|
|
case SoundIoSampleFormatInt32: return 4;
|
|
|
|
case SoundIoSampleFormatFloat: return 4;
|
|
|
|
case SoundIoSampleFormatDouble: return 8;
|
2015-07-01 20:24:47 +00:00
|
|
|
case SoundIoSampleFormatInvalid: soundio_panic("invalid sample format");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
2015-07-01 20:24:47 +00:00
|
|
|
soundio_panic("invalid sample format");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const char * soundio_sample_format_string(enum SoundIoSampleFormat sample_format) {
|
|
|
|
switch (sample_format) {
|
|
|
|
case SoundIoSampleFormatUInt8: return "unsigned 8-bit integer";
|
|
|
|
case SoundIoSampleFormatInt16: return "signed 16-bit integer";
|
|
|
|
case SoundIoSampleFormatInt24: return "signed 24-bit integer";
|
|
|
|
case SoundIoSampleFormatInt32: return "signed 32-bit integer";
|
|
|
|
case SoundIoSampleFormatFloat: return "32-bit float";
|
|
|
|
case SoundIoSampleFormatDouble: return "64-bit float";
|
|
|
|
case SoundIoSampleFormatInvalid: return "invalid sample format";
|
|
|
|
}
|
2015-07-01 20:24:47 +00:00
|
|
|
soundio_panic("invalid sample format");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const char *soundio_backend_name(enum SoundIoBackend backend) {
|
|
|
|
switch (backend) {
|
2015-07-01 09:53:53 +00:00
|
|
|
case SoundIoBackendNone: return "(none)";
|
2015-07-01 08:02:44 +00:00
|
|
|
case SoundIoBackendPulseAudio: return "PulseAudio";
|
2015-07-07 09:55:32 +00:00
|
|
|
case SoundIoBackendAlsa: return "ALSA";
|
2015-07-01 08:02:44 +00:00
|
|
|
case SoundIoBackendDummy: return "Dummy";
|
|
|
|
}
|
2015-07-01 20:24:47 +00:00
|
|
|
soundio_panic("invalid backend enum value: %d", (int)backend);
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_destroy(struct SoundIo *soundio) {
|
|
|
|
if (!soundio)
|
|
|
|
return;
|
|
|
|
|
2015-07-01 09:53:53 +00:00
|
|
|
soundio_disconnect(soundio);
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
destroy(soundio);
|
|
|
|
}
|
|
|
|
|
2015-07-04 10:37:50 +00:00
|
|
|
static void default_on_devices_change(struct SoundIo *) { }
|
2015-07-04 21:20:52 +00:00
|
|
|
static void default_on_events_signal(struct SoundIo *) { }
|
2015-07-04 10:37:50 +00:00
|
|
|
|
2015-07-01 09:53:53 +00:00
|
|
|
struct SoundIo * soundio_create(void) {
|
2015-07-02 09:53:08 +00:00
|
|
|
soundio_os_init();
|
2015-07-01 08:02:44 +00:00
|
|
|
struct SoundIo *soundio = create<SoundIo>();
|
|
|
|
if (!soundio) {
|
|
|
|
soundio_destroy(soundio);
|
2015-07-01 09:53:53 +00:00
|
|
|
return NULL;
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
2015-07-04 10:37:50 +00:00
|
|
|
soundio->on_devices_change = default_on_devices_change;
|
2015-07-04 21:20:52 +00:00
|
|
|
soundio->on_events_signal = default_on_events_signal;
|
2015-07-01 09:53:53 +00:00
|
|
|
return soundio;
|
|
|
|
}
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-01 09:53:53 +00:00
|
|
|
int soundio_connect(struct SoundIo *soundio) {
|
2015-07-01 08:02:44 +00:00
|
|
|
int err;
|
|
|
|
|
2015-07-02 10:48:27 +00:00
|
|
|
#ifdef SOUNDIO_HAVE_PULSEAUDIO
|
2015-07-01 09:53:53 +00:00
|
|
|
soundio->current_backend = SoundIoBackendPulseAudio;
|
2015-07-01 08:02:44 +00:00
|
|
|
err = soundio_pulseaudio_init(soundio);
|
2015-07-04 21:20:52 +00:00
|
|
|
if (!err)
|
|
|
|
return 0;
|
2015-07-01 08:02:44 +00:00
|
|
|
if (err != SoundIoErrorInitAudioBackend) {
|
2015-07-01 09:53:53 +00:00
|
|
|
soundio_disconnect(soundio);
|
2015-07-01 08:02:44 +00:00
|
|
|
return err;
|
|
|
|
}
|
2015-07-02 10:48:27 +00:00
|
|
|
#endif
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-07 09:55:32 +00:00
|
|
|
#ifdef SOUNDIO_HAVE_ALSA
|
|
|
|
soundio->current_backend = SoundIoBackendAlsa;
|
|
|
|
err = soundio_alsa_init(soundio);
|
|
|
|
if (!err)
|
|
|
|
return 0;
|
|
|
|
if (err != SoundIoErrorInitAudioBackend) {
|
|
|
|
soundio_disconnect(soundio);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-07-01 09:53:53 +00:00
|
|
|
soundio->current_backend = SoundIoBackendDummy;
|
2015-07-01 08:02:44 +00:00
|
|
|
err = soundio_dummy_init(soundio);
|
|
|
|
if (err) {
|
2015-07-01 09:53:53 +00:00
|
|
|
soundio_disconnect(soundio);
|
2015-07-01 08:02:44 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-01 09:53:53 +00:00
|
|
|
void soundio_disconnect(struct SoundIo *soundio) {
|
|
|
|
if (soundio->destroy)
|
|
|
|
soundio->destroy(soundio);
|
|
|
|
assert(!soundio->backend_data);
|
|
|
|
|
|
|
|
soundio->current_backend = SoundIoBackendNone;
|
|
|
|
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_destroy_devices_info(soundio->safe_devices_info);
|
|
|
|
soundio->safe_devices_info = nullptr;
|
2015-07-01 09:53:53 +00:00
|
|
|
|
|
|
|
soundio->destroy = nullptr;
|
|
|
|
soundio->flush_events = nullptr;
|
|
|
|
soundio->wait_events = nullptr;
|
|
|
|
soundio->wakeup = nullptr;
|
|
|
|
|
|
|
|
soundio->output_device_init = nullptr;
|
|
|
|
soundio->output_device_destroy = nullptr;
|
|
|
|
soundio->output_device_start = nullptr;
|
|
|
|
soundio->output_device_free_count = nullptr;
|
|
|
|
soundio->output_device_begin_write = nullptr;
|
|
|
|
soundio->output_device_write = nullptr;
|
|
|
|
soundio->output_device_clear_buffer = nullptr;
|
|
|
|
|
|
|
|
soundio->input_device_init = nullptr;
|
|
|
|
soundio->input_device_destroy = nullptr;
|
|
|
|
soundio->input_device_start = nullptr;
|
|
|
|
soundio->input_device_peek = nullptr;
|
|
|
|
soundio->input_device_drop = nullptr;
|
|
|
|
soundio->input_device_clear_buffer = nullptr;
|
|
|
|
}
|
|
|
|
|
2015-07-01 08:02:44 +00:00
|
|
|
void soundio_flush_events(struct SoundIo *soundio) {
|
2015-07-04 21:20:52 +00:00
|
|
|
assert(soundio->flush_events);
|
2015-07-01 08:02:44 +00:00
|
|
|
if (soundio->flush_events)
|
|
|
|
soundio->flush_events(soundio);
|
|
|
|
}
|
2015-07-01 08:20:26 +00:00
|
|
|
|
|
|
|
int soundio_get_input_device_count(struct SoundIo *soundio) {
|
2015-07-04 21:20:52 +00:00
|
|
|
soundio_flush_events(soundio);
|
2015-07-01 08:20:26 +00:00
|
|
|
assert(soundio->safe_devices_info);
|
|
|
|
return soundio->safe_devices_info->input_devices.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_get_output_device_count(struct SoundIo *soundio) {
|
2015-07-04 21:20:52 +00:00
|
|
|
soundio_flush_events(soundio);
|
2015-07-01 08:20:26 +00:00
|
|
|
assert(soundio->safe_devices_info);
|
|
|
|
return soundio->safe_devices_info->output_devices.length;
|
|
|
|
}
|
2015-07-01 08:24:57 +00:00
|
|
|
|
|
|
|
int soundio_get_default_input_device_index(struct SoundIo *soundio) {
|
2015-07-04 21:20:52 +00:00
|
|
|
soundio_flush_events(soundio);
|
2015-07-01 08:24:57 +00:00
|
|
|
assert(soundio->safe_devices_info);
|
|
|
|
return soundio->safe_devices_info->default_input_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_get_default_output_device_index(struct SoundIo *soundio) {
|
2015-07-04 21:20:52 +00:00
|
|
|
soundio_flush_events(soundio);
|
2015-07-01 08:24:57 +00:00
|
|
|
assert(soundio->safe_devices_info);
|
|
|
|
return soundio->safe_devices_info->default_output_index;
|
|
|
|
}
|
2015-07-01 08:29:35 +00:00
|
|
|
|
|
|
|
struct SoundIoDevice *soundio_get_input_device(struct SoundIo *soundio, int index) {
|
|
|
|
assert(soundio->safe_devices_info);
|
|
|
|
assert(index >= 0);
|
|
|
|
assert(index < soundio->safe_devices_info->input_devices.length);
|
|
|
|
SoundIoDevice *device = soundio->safe_devices_info->input_devices.at(index);
|
|
|
|
soundio_device_ref(device);
|
|
|
|
return device;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct SoundIoDevice *soundio_get_output_device(struct SoundIo *soundio, int index) {
|
|
|
|
assert(soundio->safe_devices_info);
|
|
|
|
assert(index >= 0);
|
|
|
|
assert(index < soundio->safe_devices_info->output_devices.length);
|
|
|
|
SoundIoDevice *device = soundio->safe_devices_info->output_devices.at(index);
|
|
|
|
soundio_device_ref(device);
|
|
|
|
return device;
|
|
|
|
}
|
2015-07-01 09:37:51 +00:00
|
|
|
|
|
|
|
// the name is the identifier for the device. UTF-8 encoded
|
|
|
|
const char *soundio_device_name(const struct SoundIoDevice *device) {
|
|
|
|
return device->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
// UTF-8 encoded
|
|
|
|
const char *soundio_device_description(const struct SoundIoDevice *device) {
|
|
|
|
return device->description;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum SoundIoDevicePurpose soundio_device_purpose(const struct SoundIoDevice *device) {
|
|
|
|
return device->purpose;
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct SoundIoChannelLayout *soundio_device_channel_layout(const struct SoundIoDevice *device) {
|
|
|
|
return &device->channel_layout;
|
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_device_sample_rate(const struct SoundIoDevice *device) {
|
|
|
|
return device->default_sample_rate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_device_unref(struct SoundIoDevice *device) {
|
|
|
|
if (!device)
|
|
|
|
return;
|
|
|
|
|
|
|
|
device->ref_count -= 1;
|
|
|
|
assert(device->ref_count >= 0);
|
|
|
|
|
|
|
|
if (device->ref_count == 0) {
|
|
|
|
free(device->name);
|
|
|
|
free(device->description);
|
|
|
|
destroy(device);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_device_ref(struct SoundIoDevice *device) {
|
|
|
|
device->ref_count += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_wait_events(struct SoundIo *soundio) {
|
|
|
|
soundio->wait_events(soundio);
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_wakeup(struct SoundIo *soundio) {
|
|
|
|
soundio->wakeup(soundio);
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_output_device_fill_with_silence(struct SoundIoOutputDevice *output_device) {
|
|
|
|
char *buffer;
|
|
|
|
int requested_frame_count = soundio_output_device_free_count(output_device);
|
|
|
|
while (requested_frame_count > 0) {
|
|
|
|
int frame_count = requested_frame_count;
|
|
|
|
soundio_output_device_begin_write(output_device, &buffer, &frame_count);
|
|
|
|
memset(buffer, 0, frame_count * output_device->bytes_per_frame);
|
|
|
|
soundio_output_device_write(output_device, buffer, frame_count);
|
|
|
|
requested_frame_count -= frame_count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_output_device_free_count(struct SoundIoOutputDevice *output_device) {
|
|
|
|
SoundIo *soundio = output_device->device->soundio;
|
|
|
|
return soundio->output_device_free_count(soundio, output_device);
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_output_device_begin_write(struct SoundIoOutputDevice *output_device,
|
|
|
|
char **data, int *frame_count)
|
|
|
|
{
|
|
|
|
SoundIo *soundio = output_device->device->soundio;
|
|
|
|
soundio->output_device_begin_write(soundio, output_device, data, frame_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_output_device_write(struct SoundIoOutputDevice *output_device,
|
|
|
|
char *data, int frame_count)
|
|
|
|
{
|
|
|
|
SoundIo *soundio = output_device->device->soundio;
|
|
|
|
soundio->output_device_write(soundio, output_device, data, frame_count);
|
|
|
|
}
|
2015-07-04 09:57:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
int soundio_output_device_create(struct SoundIoDevice *device,
|
|
|
|
enum SoundIoSampleFormat sample_format,
|
|
|
|
double latency, void *userdata,
|
|
|
|
void (*write_callback)(struct SoundIoOutputDevice *, int frame_count),
|
|
|
|
void (*underrun_callback)(struct SoundIoOutputDevice *),
|
|
|
|
struct SoundIoOutputDevice **out_output_device)
|
|
|
|
{
|
|
|
|
*out_output_device = nullptr;
|
|
|
|
|
|
|
|
SoundIoOutputDevice *output_device = create<SoundIoOutputDevice>();
|
|
|
|
if (!output_device) {
|
|
|
|
soundio_output_device_destroy(output_device);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
soundio_device_ref(device);
|
|
|
|
output_device->device = device;
|
|
|
|
output_device->userdata = userdata;
|
|
|
|
output_device->write_callback = write_callback;
|
|
|
|
output_device->underrun_callback = underrun_callback;
|
|
|
|
output_device->sample_format = sample_format;
|
|
|
|
output_device->latency = latency;
|
|
|
|
output_device->bytes_per_frame = soundio_get_bytes_per_frame(sample_format,
|
|
|
|
device->channel_layout.channel_count);
|
|
|
|
|
|
|
|
SoundIo *soundio = device->soundio;
|
|
|
|
int err = soundio->output_device_init(soundio, output_device);
|
|
|
|
if (err) {
|
|
|
|
soundio_output_device_destroy(output_device);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_output_device = output_device;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_output_device_destroy(SoundIoOutputDevice *output_device) {
|
|
|
|
if (!output_device)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SoundIo *soundio = output_device->device->soundio;
|
|
|
|
|
|
|
|
if (soundio->output_device_destroy)
|
|
|
|
soundio->output_device_destroy(soundio, output_device);
|
|
|
|
|
|
|
|
soundio_device_unref(output_device->device);
|
|
|
|
destroy(output_device);
|
|
|
|
}
|
2015-07-04 10:37:50 +00:00
|
|
|
|
|
|
|
int soundio_output_device_start(struct SoundIoOutputDevice *output_device) {
|
|
|
|
SoundIo *soundio = output_device->device->soundio;
|
|
|
|
return soundio->output_device_start(soundio, output_device);
|
|
|
|
}
|
2015-07-06 08:05:22 +00:00
|
|
|
|
|
|
|
int soundio_input_device_create(struct SoundIoDevice *device,
|
|
|
|
enum SoundIoSampleFormat sample_format, double latency, void *userdata,
|
|
|
|
void (*read_callback)(struct SoundIoInputDevice *),
|
|
|
|
struct SoundIoInputDevice **out_input_device)
|
|
|
|
{
|
|
|
|
*out_input_device = nullptr;
|
|
|
|
|
|
|
|
SoundIoInputDevice *sid = create<SoundIoInputDevice>();
|
|
|
|
if (!sid) {
|
|
|
|
soundio_input_device_destroy(sid);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
soundio_device_ref(device);
|
|
|
|
sid->device = device;
|
|
|
|
sid->userdata = userdata;
|
|
|
|
sid->read_callback = read_callback;
|
|
|
|
sid->sample_format = sample_format;
|
|
|
|
sid->latency = latency;
|
|
|
|
sid->bytes_per_frame = soundio_get_bytes_per_frame(sample_format,
|
|
|
|
device->channel_layout.channel_count);
|
|
|
|
|
|
|
|
SoundIo *soundio = device->soundio;
|
|
|
|
int err = soundio->input_device_init(soundio, sid);
|
|
|
|
if (err) {
|
|
|
|
soundio_input_device_destroy(sid);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_input_device = sid;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_input_device_start(struct SoundIoInputDevice *input_device) {
|
|
|
|
SoundIo *soundio = input_device->device->soundio;
|
|
|
|
return soundio->input_device_start(soundio, input_device);
|
|
|
|
}
|
|
|
|
|
|
|
|
void soundio_input_device_destroy(struct SoundIoInputDevice *input_device) {
|
|
|
|
if (!input_device)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SoundIo *soundio = input_device->device->soundio;
|
|
|
|
|
|
|
|
if (soundio->input_device_destroy)
|
|
|
|
soundio->input_device_destroy(soundio, input_device);
|
|
|
|
|
|
|
|
soundio_device_unref(input_device->device);
|
|
|
|
destroy(input_device);
|
|
|
|
}
|
2015-07-07 09:55:32 +00:00
|
|
|
|
|
|
|
void soundio_destroy_devices_info(SoundIoDevicesInfo *devices_info) {
|
|
|
|
if (!devices_info)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (int i = 0; i < devices_info->input_devices.length; i += 1)
|
|
|
|
soundio_device_unref(devices_info->input_devices.at(i));
|
|
|
|
for (int i = 0; i < devices_info->output_devices.length; i += 1)
|
|
|
|
soundio_device_unref(devices_info->output_devices.at(i));
|
|
|
|
|
|
|
|
destroy(devices_info);
|
|
|
|
}
|