2015-07-25 01:43:14 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015 Andrew Kelley
|
|
|
|
*
|
|
|
|
* This file is part of libsoundio, which is MIT licensed.
|
|
|
|
* See http://opensource.org/licenses/MIT
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "jack.hpp"
|
|
|
|
#include "soundio.hpp"
|
2015-07-25 02:16:48 +00:00
|
|
|
#include "atomics.hpp"
|
2015-07-27 18:27:41 +00:00
|
|
|
#include "list.hpp"
|
2015-07-25 01:43:14 +00:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2015-07-25 02:16:48 +00:00
|
|
|
static atomic_flag global_msg_callback_flag = ATOMIC_FLAG_INIT;
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
struct SoundIoJackPort {
|
2015-07-28 18:28:07 +00:00
|
|
|
const char *full_name;
|
|
|
|
int full_name_len;
|
2015-07-27 18:27:41 +00:00
|
|
|
const char *name;
|
|
|
|
int name_len;
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoChannelId channel_id;
|
2015-07-27 18:27:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct SoundIoJackClient {
|
|
|
|
const char *name;
|
|
|
|
int name_len;
|
|
|
|
bool is_physical;
|
|
|
|
SoundIoDevicePurpose purpose;
|
2015-07-27 23:37:45 +00:00
|
|
|
int port_count;
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoJackPort ports[SOUNDIO_MAX_CHANNELS];
|
2015-07-27 18:27:41 +00:00
|
|
|
};
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
static void flush_events_jack(struct SoundIoPrivate *si) {
|
|
|
|
SoundIo *soundio = &si->pub;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 05:00:33 +00:00
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
bool change_devices = false;
|
|
|
|
bool cb_shutdown = false;
|
2015-07-25 05:00:33 +00:00
|
|
|
SoundIoDevicesInfo *old_devices_info = nullptr;
|
|
|
|
|
|
|
|
soundio_os_mutex_lock(sij->mutex);
|
|
|
|
|
|
|
|
if (sij->ready_devices_info) {
|
|
|
|
old_devices_info = si->safe_devices_info;
|
|
|
|
si->safe_devices_info = sij->ready_devices_info;
|
|
|
|
sij->ready_devices_info = nullptr;
|
2015-07-28 20:36:31 +00:00
|
|
|
change_devices = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sij->is_shutdown && !sij->emitted_shutdown_cb) {
|
|
|
|
sij->emitted_shutdown_cb = true;
|
|
|
|
cb_shutdown = true;
|
2015-07-25 05:00:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
soundio_os_mutex_unlock(sij->mutex);
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
if (change_devices)
|
2015-07-25 05:00:33 +00:00
|
|
|
soundio->on_devices_change(soundio);
|
|
|
|
soundio_destroy_devices_info(old_devices_info);
|
2015-07-28 20:36:31 +00:00
|
|
|
|
|
|
|
if (cb_shutdown)
|
|
|
|
soundio->on_backend_disconnect(soundio);
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
static void wait_events_jack(struct SoundIoPrivate *si) {
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 05:00:33 +00:00
|
|
|
flush_events_jack(si);
|
|
|
|
soundio_os_mutex_lock(sij->mutex);
|
|
|
|
soundio_os_cond_wait(sij->cond, sij->mutex);
|
|
|
|
soundio_os_mutex_unlock(sij->mutex);
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
static void wakeup_jack(struct SoundIoPrivate *si) {
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 05:00:33 +00:00
|
|
|
soundio_os_mutex_lock(sij->mutex);
|
|
|
|
soundio_os_cond_signal(sij->cond, sij->mutex);
|
|
|
|
soundio_os_mutex_unlock(sij->mutex);
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-27 23:37:45 +00:00
|
|
|
static int outstream_process_callback(jack_nframes_t nframes, void *arg) {
|
|
|
|
SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
outstream->write_callback(outstream, nframes);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void outstream_destroy_jack(struct SoundIoPrivate *is, struct SoundIoOutStreamPrivate *os) {
|
2015-07-28 00:06:12 +00:00
|
|
|
SoundIoOutStreamJack *osj = &os->backend_data.jack;
|
2015-07-25 01:43:14 +00:00
|
|
|
|
2015-07-27 23:37:45 +00:00
|
|
|
jack_client_close(osj->client);
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
static SoundIoDeviceJackPort *find_port_matching_channel(SoundIoDevice *device, SoundIoChannelId id) {
|
|
|
|
SoundIoDevicePrivate *dev = (SoundIoDevicePrivate *)device;
|
|
|
|
SoundIoDeviceJack *dj = &dev->backend_data.jack;
|
|
|
|
|
|
|
|
for (int ch = 0; ch < device->current_layout.channel_count; ch += 1) {
|
|
|
|
SoundIoChannelId chan_id = device->current_layout.channels[ch];
|
|
|
|
if (chan_id == id)
|
|
|
|
return &dj->ports[ch];
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static int outstream_xrun_callback(void *arg) {
|
|
|
|
SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
outstream->underflow_callback(outstream);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_buffer_size_callback(jack_nframes_t nframes, void *arg) {
|
|
|
|
SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
SoundIo *soundio = outstream->device->soundio;
|
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)soundio;
|
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
if ((jack_nframes_t)sij->buffer_size == nframes) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
outstream->error_callback(outstream, SoundIoErrorStreaming);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_sample_rate_callback(jack_nframes_t nframes, void *arg) {
|
|
|
|
SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
if (nframes == (jack_nframes_t)outstream->sample_rate) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
outstream->error_callback(outstream, SoundIoErrorStreaming);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void outstream_shutdown_callback(void *arg) {
|
|
|
|
SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
outstream->error_callback(outstream, SoundIoErrorStreaming);
|
|
|
|
}
|
|
|
|
|
2015-07-27 23:37:45 +00:00
|
|
|
static int outstream_open_jack(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
2015-07-28 20:36:31 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-28 00:06:12 +00:00
|
|
|
SoundIoOutStreamJack *osj = &os->backend_data.jack;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoDevice *device = outstream->device;
|
|
|
|
SoundIoDevicePrivate *dev = (SoundIoDevicePrivate *)device;
|
|
|
|
SoundIoDeviceJack *dj = &dev->backend_data.jack;
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
if (sij->is_shutdown)
|
|
|
|
return SoundIoErrorBackendDisconnected;
|
2015-07-27 23:37:45 +00:00
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
outstream->buffer_duration = device->buffer_duration_current;
|
|
|
|
outstream->period_duration = 0.0;
|
|
|
|
outstream->prebuf_duration = 0.0;
|
2015-07-27 23:37:45 +00:00
|
|
|
|
|
|
|
jack_status_t status;
|
|
|
|
osj->client = jack_client_open(outstream->name, JackNoStartServer, &status);
|
|
|
|
if (!osj->client) {
|
|
|
|
outstream_destroy_jack(si, os);
|
|
|
|
assert(!(status & JackInvalidOption));
|
|
|
|
if (status & JackShmFailure)
|
|
|
|
return SoundIoErrorSystemResources;
|
|
|
|
if (status & JackNoSuchClient)
|
|
|
|
return SoundIoErrorNoSuchClient;
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
int err;
|
|
|
|
if ((err = jack_set_process_callback(osj->client, outstream_process_callback, os))) {
|
|
|
|
outstream_destroy_jack(si, os);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
2015-07-28 20:36:31 +00:00
|
|
|
if ((err = jack_set_buffer_size_callback(osj->client, outstream_buffer_size_callback, os))) {
|
|
|
|
outstream_destroy_jack(si, os);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if ((err = jack_set_sample_rate_callback(osj->client, outstream_sample_rate_callback, os))) {
|
|
|
|
outstream_destroy_jack(si, os);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if ((err = jack_set_xrun_callback(osj->client, outstream_xrun_callback, os))) {
|
|
|
|
outstream_destroy_jack(si, os);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
jack_on_shutdown(osj->client, outstream_shutdown_callback, os);
|
2015-07-27 23:37:45 +00:00
|
|
|
|
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
// register ports and map channels
|
|
|
|
int connected_count = 0;
|
2015-07-27 23:37:45 +00:00
|
|
|
for (int ch = 0; ch < outstream->layout.channel_count; ch += 1) {
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoChannelId my_channel_id = outstream->layout.channels[ch];
|
|
|
|
const char *channel_name = soundio_get_channel_name(my_channel_id);
|
2015-07-27 23:37:45 +00:00
|
|
|
unsigned long flags = JackPortIsOutput;
|
|
|
|
if (!outstream->non_terminal_hint)
|
|
|
|
flags |= JackPortIsTerminal;
|
2015-07-28 18:28:07 +00:00
|
|
|
jack_port_t *jport = jack_port_register(osj->client, channel_name, JACK_DEFAULT_AUDIO_TYPE, flags, 0);
|
|
|
|
if (!jport) {
|
2015-07-27 23:37:45 +00:00
|
|
|
outstream_destroy_jack(si, os);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoOutStreamJackPort *osjp = &osj->ports[ch];
|
|
|
|
osjp->source_port = jport;
|
|
|
|
// figure out which dest port this connects to
|
|
|
|
SoundIoDeviceJackPort *djp = find_port_matching_channel(device, my_channel_id);
|
|
|
|
if (djp) {
|
|
|
|
osjp->dest_port_name = djp->full_name;
|
|
|
|
osjp->dest_port_name_len = djp->full_name_len;
|
|
|
|
connected_count += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If nothing got connected, channel layouts aren't working. Just send the
|
|
|
|
// data in the order of the ports.
|
|
|
|
if (connected_count == 0) {
|
|
|
|
outstream->layout_error = SoundIoErrorIncompatibleDevice;
|
|
|
|
|
|
|
|
int ch_count = min(outstream->layout.channel_count, dj->port_count);
|
|
|
|
for (int ch = 0; ch < ch_count; ch += 1) {
|
|
|
|
SoundIoOutStreamJackPort *osjp = &osj->ports[ch];
|
|
|
|
SoundIoDeviceJackPort *djp = &dj->ports[ch];
|
|
|
|
osjp->dest_port_name = djp->full_name;
|
|
|
|
osjp->dest_port_name_len = djp->full_name_len;
|
|
|
|
}
|
2015-07-27 23:37:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-27 23:37:45 +00:00
|
|
|
static int outstream_pause_jack(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os, bool pause) {
|
2015-07-28 00:06:12 +00:00
|
|
|
SoundIoOutStreamJack *osj = &os->backend_data.jack;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
2015-07-28 20:36:31 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
if (sij->is_shutdown)
|
|
|
|
return SoundIoErrorBackendDisconnected;
|
|
|
|
|
2015-07-27 23:37:45 +00:00
|
|
|
int err;
|
|
|
|
if (pause) {
|
|
|
|
if ((err = jack_deactivate(osj->client)))
|
|
|
|
return SoundIoErrorStreaming;
|
|
|
|
} else {
|
|
|
|
if ((err = jack_activate(osj->client)))
|
|
|
|
return SoundIoErrorStreaming;
|
|
|
|
|
|
|
|
for (int ch = 0; ch < outstream->layout.channel_count; ch += 1) {
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoOutStreamJackPort *osjp = &osj->ports[ch];
|
|
|
|
const char *dest_port_name = osjp->dest_port_name;
|
|
|
|
// allow unconnected ports
|
|
|
|
if (!dest_port_name)
|
|
|
|
continue;
|
|
|
|
const char *source_port_name = jack_port_name(osjp->source_port);
|
|
|
|
if ((err = jack_connect(osj->client, source_port_name, dest_port_name)))
|
|
|
|
return SoundIoErrorStreaming;
|
2015-07-27 23:37:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_start_jack(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
|
|
|
return outstream_pause_jack(si, os, false);
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
static int outstream_begin_write_jack(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os,
|
2015-07-25 01:43:14 +00:00
|
|
|
SoundIoChannelArea **out_areas, int *frame_count)
|
|
|
|
{
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
SoundIoOutStreamJack *osj = &os->backend_data.jack;
|
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
assert(*frame_count <= sij->buffer_size);
|
|
|
|
|
|
|
|
for (int ch = 0; ch < outstream->layout.channel_count; ch += 1) {
|
|
|
|
SoundIoOutStreamJackPort *osjp = &osj->ports[ch];
|
|
|
|
if (!(osj->areas[ch].ptr = (char*)jack_port_get_buffer(osjp->source_port, *frame_count)))
|
|
|
|
return SoundIoErrorStreaming;
|
|
|
|
|
|
|
|
osj->areas[ch].step = outstream->bytes_per_sample;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_areas = osj->areas;
|
|
|
|
|
|
|
|
return 0;
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_end_write_jack(struct SoundIoPrivate *, struct SoundIoOutStreamPrivate *, int frame_count) {
|
2015-07-28 18:28:07 +00:00
|
|
|
return 0;
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_clear_buffer_jack(struct SoundIoPrivate *, struct SoundIoOutStreamPrivate *) {
|
2015-07-28 20:36:31 +00:00
|
|
|
// JACK does not support `prebuf` which is the same as a `prebuf` value of 0,
|
|
|
|
// which means that clearing the buffer is always successful and does nothing.
|
|
|
|
return 0;
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static void instream_destroy_jack(struct SoundIoPrivate *, struct SoundIoInStreamPrivate *) {
|
|
|
|
soundio_panic("TODO destroy instream");
|
|
|
|
}
|
2015-07-25 01:43:14 +00:00
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static int instream_open_jack(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
|
|
|
//SoundIoInStream *outstream = &is->pub;
|
|
|
|
//SoundIoInStreamJack *isj = &is->backend_data.jack;
|
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
if (sij->is_shutdown) {
|
|
|
|
instream_destroy_jack(si, is);
|
|
|
|
return SoundIoErrorBackendDisconnected;
|
|
|
|
}
|
2015-07-27 23:37:45 +00:00
|
|
|
soundio_panic("TODO open instream");
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static int instream_start_jack(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
if (sij->is_shutdown)
|
|
|
|
return SoundIoErrorBackendDisconnected;
|
2015-07-25 01:43:14 +00:00
|
|
|
|
2015-07-27 23:37:45 +00:00
|
|
|
soundio_panic("TODO start instream");
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static int instream_begin_read_jack(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is,
|
2015-07-25 01:43:14 +00:00
|
|
|
SoundIoChannelArea **out_areas, int *frame_count)
|
|
|
|
{
|
2015-07-27 23:37:45 +00:00
|
|
|
soundio_panic("TODO begin read");
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static int instream_end_read_jack(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
2015-07-27 23:37:45 +00:00
|
|
|
soundio_panic("TODO end read");
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
static int instream_pause_jack(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is, bool pause) {
|
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
if (sij->is_shutdown)
|
|
|
|
return SoundIoErrorBackendDisconnected;
|
2015-07-27 23:37:45 +00:00
|
|
|
soundio_panic("TODO pause");
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
static void split_str(const char *input_str, int input_str_len, char c,
|
|
|
|
const char **out_1, int *out_len_1, const char **out_2, int *out_len_2)
|
|
|
|
{
|
|
|
|
*out_1 = input_str;
|
|
|
|
while (*input_str) {
|
|
|
|
if (*input_str == c) {
|
|
|
|
*out_len_1 = input_str - *out_1;
|
|
|
|
*out_2 = input_str + 1;
|
|
|
|
*out_len_2 = input_str_len - 1 - *out_len_1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
input_str += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static SoundIoJackClient *find_or_create_client(SoundIoList<SoundIoJackClient> *clients,
|
|
|
|
SoundIoDevicePurpose purpose, bool is_physical, const char *client_name, int client_name_len)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < clients->length; i += 1) {
|
|
|
|
SoundIoJackClient *client = &clients->at(i);
|
|
|
|
if (client->is_physical == is_physical &&
|
|
|
|
client->purpose == purpose &&
|
2015-07-28 21:53:54 +00:00
|
|
|
soundio_streql(client->name, client->name_len, client_name, client_name_len))
|
2015-07-27 18:27:41 +00:00
|
|
|
{
|
|
|
|
return client;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int err;
|
|
|
|
if ((err = clients->add_one()))
|
|
|
|
return nullptr;
|
|
|
|
SoundIoJackClient *client = &clients->last();
|
|
|
|
client->is_physical = is_physical;
|
|
|
|
client->purpose = purpose;
|
|
|
|
client->name = client_name;
|
|
|
|
client->name_len = client_name_len;
|
2015-07-27 23:37:45 +00:00
|
|
|
client->port_count = 0;
|
2015-07-27 18:27:41 +00:00
|
|
|
return client;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *dupe_str(const char *str, int str_len) {
|
|
|
|
char *out = allocate_nonzero<char>(str_len + 1);
|
|
|
|
if (!out)
|
|
|
|
return nullptr;
|
|
|
|
memcpy(out, str, str_len);
|
|
|
|
out[str_len] = 0;
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
static void destruct_device(SoundIoDevicePrivate *dp) {
|
|
|
|
SoundIoDeviceJack *dj = &dp->backend_data.jack;
|
|
|
|
for (int i = 0; i < dj->port_count; i += 1) {
|
|
|
|
SoundIoDeviceJackPort *djp = &dj->ports[i];
|
|
|
|
destroy(djp->full_name);
|
|
|
|
}
|
|
|
|
deallocate(dj->ports, dj->port_count);
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
static int refresh_devices(SoundIoPrivate *si) {
|
|
|
|
SoundIo *soundio = &si->pub;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 05:00:33 +00:00
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
if (sij->is_shutdown)
|
|
|
|
return SoundIoErrorBackendDisconnected;
|
|
|
|
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
SoundIoDevicesInfo *devices_info = create<SoundIoDevicesInfo>();
|
|
|
|
if (!devices_info)
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
|
|
|
|
devices_info->default_output_index = -1;
|
|
|
|
devices_info->default_input_index = -1;
|
|
|
|
const char **port_names = jack_get_ports(sij->client, nullptr, nullptr, 0);
|
|
|
|
if (!port_names) {
|
|
|
|
destroy(devices_info);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
SoundIoList<SoundIoJackClient> clients;
|
2015-07-25 05:00:33 +00:00
|
|
|
const char **port_name_ptr = port_names;
|
|
|
|
while (*port_name_ptr) {
|
2015-07-27 18:27:41 +00:00
|
|
|
const char *client_and_port_name = *port_name_ptr;
|
2015-07-28 18:28:07 +00:00
|
|
|
int client_and_port_name_len = strlen(client_and_port_name);
|
2015-07-27 18:27:41 +00:00
|
|
|
jack_port_t *jport = jack_port_by_name(sij->client, client_and_port_name);
|
|
|
|
int flags = jack_port_flags(jport);
|
2015-07-27 23:37:45 +00:00
|
|
|
|
|
|
|
const char *port_type = jack_port_type(jport);
|
|
|
|
if (strcmp(port_type, JACK_DEFAULT_AUDIO_TYPE) != 0) {
|
|
|
|
// we don't know how to support such a port
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
SoundIoDevicePurpose purpose = (flags & JackPortIsInput) ?
|
|
|
|
SoundIoDevicePurposeOutput : SoundIoDevicePurposeInput;
|
|
|
|
bool is_physical = flags & JackPortIsPhysical;
|
|
|
|
|
|
|
|
const char *client_name = nullptr;
|
|
|
|
const char *port_name = nullptr;
|
|
|
|
int client_name_len;
|
|
|
|
int port_name_len;
|
2015-07-28 18:28:07 +00:00
|
|
|
split_str(client_and_port_name, client_and_port_name_len, ':',
|
2015-07-27 18:27:41 +00:00
|
|
|
&client_name, &client_name_len, &port_name, &port_name_len);
|
|
|
|
if (!client_name || !port_name) {
|
|
|
|
// device does not have colon, skip it
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
SoundIoJackClient *client = find_or_create_client(&clients, purpose, is_physical,
|
|
|
|
client_name, client_name_len);
|
|
|
|
if (!client) {
|
|
|
|
jack_free(port_names);
|
|
|
|
destroy(devices_info);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-27 23:37:45 +00:00
|
|
|
if (client->port_count >= SOUNDIO_MAX_CHANNELS) {
|
|
|
|
// we hit the channel limit, skip the leftovers
|
|
|
|
continue;
|
2015-07-27 18:27:41 +00:00
|
|
|
}
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJackPort *port = &client->ports[client->port_count++];
|
2015-07-28 18:28:07 +00:00
|
|
|
port->full_name = client_and_port_name;
|
|
|
|
port->full_name_len = client_and_port_name_len;
|
2015-07-27 18:27:41 +00:00
|
|
|
port->name = port_name;
|
|
|
|
port->name_len = port_name_len;
|
2015-07-28 18:28:07 +00:00
|
|
|
port->channel_id = soundio_parse_channel_id(port_name, port_name_len);
|
2015-07-27 18:27:41 +00:00
|
|
|
|
|
|
|
port_name_ptr += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < clients.length; i += 1) {
|
|
|
|
SoundIoJackClient *client = &clients.at(i);
|
2015-07-27 23:37:45 +00:00
|
|
|
if (client->port_count <= 0)
|
2015-07-27 18:27:41 +00:00
|
|
|
continue;
|
2015-07-25 05:00:33 +00:00
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoDevicePrivate *dev = create<SoundIoDevicePrivate>();
|
|
|
|
if (!dev) {
|
2015-07-25 05:00:33 +00:00
|
|
|
jack_free(port_names);
|
|
|
|
destroy(devices_info);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-28 18:28:07 +00:00
|
|
|
SoundIoDevice *device = &dev->pub;
|
|
|
|
SoundIoDeviceJack *dj = &dev->backend_data.jack;
|
2015-07-27 23:37:45 +00:00
|
|
|
int description_len = client->name_len + 3 + 2 * client->port_count;
|
|
|
|
for (int port_index = 0; port_index < client->port_count; port_index += 1) {
|
|
|
|
SoundIoJackPort *port = &client->ports[port_index];
|
2015-07-27 18:27:41 +00:00
|
|
|
description_len += port->name_len;
|
|
|
|
}
|
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
dev->destruct = destruct_device;
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
device->ref_count = 1;
|
|
|
|
device->soundio = soundio;
|
|
|
|
device->is_raw = false;
|
2015-07-27 18:27:41 +00:00
|
|
|
device->purpose = client->purpose;
|
|
|
|
device->name = dupe_str(client->name, client->name_len);
|
|
|
|
device->description = allocate<char>(description_len);
|
2015-07-25 05:00:33 +00:00
|
|
|
device->layout_count = 1;
|
|
|
|
device->layouts = create<SoundIoChannelLayout>();
|
|
|
|
device->format_count = 1;
|
|
|
|
device->formats = create<SoundIoFormat>();
|
2015-07-27 18:27:41 +00:00
|
|
|
device->current_format = SoundIoFormatFloat32NE;
|
|
|
|
device->sample_rate_min = sij->sample_rate;
|
|
|
|
device->sample_rate_max = sij->sample_rate;
|
|
|
|
device->sample_rate_current = sij->sample_rate;
|
|
|
|
device->buffer_duration_min = sij->buffer_size / (double) sij->sample_rate;
|
|
|
|
device->buffer_duration_max = device->buffer_duration_min;
|
|
|
|
device->buffer_duration_current = device->buffer_duration_min;
|
2015-07-28 18:28:07 +00:00
|
|
|
dj->port_count = client->port_count;
|
|
|
|
dj->ports = allocate<SoundIoDeviceJackPort>(dj->port_count);
|
2015-07-25 05:00:33 +00:00
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
if (!device->name || !device->description || !device->layouts || !device->formats || !dj->ports) {
|
2015-07-25 05:00:33 +00:00
|
|
|
jack_free(port_names);
|
|
|
|
soundio_device_unref(device);
|
|
|
|
destroy(devices_info);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
for (int port_index = 0; port_index < client->port_count; port_index += 1) {
|
|
|
|
SoundIoJackPort *port = &client->ports[port_index];
|
|
|
|
SoundIoDeviceJackPort *djp = &dj->ports[port_index];
|
|
|
|
djp->full_name = dupe_str(port->full_name, port->full_name_len);
|
|
|
|
djp->full_name_len = port->full_name_len;
|
|
|
|
djp->channel_id = port->channel_id;
|
|
|
|
|
|
|
|
if (!djp->full_name) {
|
|
|
|
jack_free(port_names);
|
|
|
|
soundio_device_unref(device);
|
|
|
|
destroy(devices_info);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
memcpy(device->description, client->name, client->name_len);
|
|
|
|
memcpy(&device->description[client->name_len], ": ", 2);
|
|
|
|
int index = client->name_len + 2;
|
2015-07-27 23:37:45 +00:00
|
|
|
for (int port_index = 0; port_index < client->port_count; port_index += 1) {
|
|
|
|
SoundIoJackPort *port = &client->ports[port_index];
|
2015-07-27 18:27:41 +00:00
|
|
|
memcpy(&device->description[index], port->name, port->name_len);
|
|
|
|
index += port->name_len;
|
2015-07-27 23:37:45 +00:00
|
|
|
if (port_index + 1 < client->port_count) {
|
2015-07-27 18:27:41 +00:00
|
|
|
memcpy(&device->description[index], ", ", 2);
|
|
|
|
index += 2;
|
|
|
|
}
|
|
|
|
}
|
2015-07-25 05:00:33 +00:00
|
|
|
|
2015-07-28 18:28:07 +00:00
|
|
|
device->current_layout.channel_count = client->port_count;
|
|
|
|
bool any_invalid = false;
|
|
|
|
for (int port_index = 0; port_index < client->port_count; port_index += 1) {
|
|
|
|
SoundIoJackPort *port = &client->ports[port_index];
|
|
|
|
device->current_layout.channels[port_index] = port->channel_id;
|
|
|
|
any_invalid = any_invalid || (port->channel_id == SoundIoChannelIdInvalid);
|
|
|
|
}
|
|
|
|
if (any_invalid) {
|
|
|
|
const struct SoundIoChannelLayout *layout = soundio_channel_layout_get_default(client->port_count);
|
|
|
|
if (layout)
|
|
|
|
device->current_layout = *layout;
|
2015-07-27 18:27:41 +00:00
|
|
|
} else {
|
2015-07-28 18:28:07 +00:00
|
|
|
soundio_channel_layout_detect_builtin(&device->current_layout);
|
2015-07-27 18:27:41 +00:00
|
|
|
}
|
2015-07-25 05:00:33 +00:00
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
device->layouts[0] = device->current_layout;
|
|
|
|
device->formats[0] = device->current_format;
|
2015-07-25 05:00:33 +00:00
|
|
|
|
|
|
|
SoundIoList<SoundIoDevice *> *device_list;
|
2015-07-27 18:27:41 +00:00
|
|
|
if (device->purpose == SoundIoDevicePurposeOutput) {
|
2015-07-25 05:00:33 +00:00
|
|
|
device_list = &devices_info->output_devices;
|
2015-07-27 18:27:41 +00:00
|
|
|
if (devices_info->default_output_index < 0 && client->is_physical)
|
2015-07-25 05:00:33 +00:00
|
|
|
devices_info->default_output_index = device_list->length;
|
|
|
|
} else {
|
2015-07-27 18:27:41 +00:00
|
|
|
assert(device->purpose == SoundIoDevicePurposeInput);
|
2015-07-25 05:00:33 +00:00
|
|
|
device_list = &devices_info->input_devices;
|
2015-07-27 18:27:41 +00:00
|
|
|
if (devices_info->default_input_index < 0 && client->is_physical)
|
2015-07-25 05:00:33 +00:00
|
|
|
devices_info->default_input_index = device_list->length;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (device_list->append(device)) {
|
|
|
|
soundio_device_unref(device);
|
|
|
|
destroy(devices_info);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
jack_free(port_names);
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
soundio_os_mutex_lock(sij->mutex);
|
|
|
|
soundio_destroy_devices_info(sij->ready_devices_info);
|
|
|
|
sij->ready_devices_info = devices_info;
|
|
|
|
soundio_os_cond_signal(sij->cond, sij->mutex);
|
|
|
|
soundio->on_events_signal(soundio);
|
|
|
|
soundio_os_mutex_unlock(sij->mutex);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int buffer_size_callback(jack_nframes_t nframes, void *arg) {
|
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 05:00:33 +00:00
|
|
|
sij->buffer_size = nframes;
|
|
|
|
if (sij->initialized)
|
|
|
|
refresh_devices(si);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sample_rate_callback(jack_nframes_t nframes, void *arg) {
|
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 05:00:33 +00:00
|
|
|
sij->sample_rate = nframes;
|
|
|
|
if (sij->initialized)
|
|
|
|
refresh_devices(si);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void port_registration_callback(jack_port_id_t port_id, int reg, void *arg) {
|
2015-07-27 18:27:41 +00:00
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-27 18:27:41 +00:00
|
|
|
if (sij->initialized)
|
|
|
|
refresh_devices(si);
|
2015-07-25 05:00:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int port_rename_calllback(jack_port_id_t port_id,
|
|
|
|
const char *old_name, const char *new_name, void *arg)
|
|
|
|
{
|
2015-07-27 18:27:41 +00:00
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-27 18:27:41 +00:00
|
|
|
if (sij->initialized)
|
|
|
|
refresh_devices(si);
|
2015-07-25 05:00:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void shutdown_callback(void *arg) {
|
2015-07-28 20:36:31 +00:00
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
|
|
|
SoundIo *soundio = &si->pub;
|
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
|
|
|
soundio_os_mutex_lock(sij->mutex);
|
|
|
|
sij->is_shutdown = true;
|
|
|
|
soundio->on_events_signal(soundio);
|
|
|
|
soundio_os_mutex_unlock(sij->mutex);
|
2015-07-25 05:00:33 +00:00
|
|
|
}
|
2015-07-25 01:43:14 +00:00
|
|
|
|
|
|
|
static void destroy_jack(SoundIoPrivate *si) {
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 01:43:14 +00:00
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
if (sij->client)
|
|
|
|
jack_client_close(sij->client);
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
if (sij->cond)
|
|
|
|
soundio_os_cond_destroy(sij->cond);
|
|
|
|
|
|
|
|
if (sij->mutex)
|
|
|
|
soundio_os_mutex_destroy(sij->mutex);
|
|
|
|
|
|
|
|
soundio_destroy_devices_info(sij->ready_devices_info);
|
2015-07-25 01:43:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_jack_init(struct SoundIoPrivate *si) {
|
2015-07-27 23:37:45 +00:00
|
|
|
SoundIoJack *sij = &si->backend_data.jack;
|
2015-07-25 01:43:14 +00:00
|
|
|
SoundIo *soundio = &si->pub;
|
2015-07-25 02:16:48 +00:00
|
|
|
|
|
|
|
if (!global_msg_callback_flag.test_and_set()) {
|
|
|
|
if (soundio->jack_error_callback)
|
|
|
|
jack_set_error_function(soundio->jack_error_callback);
|
|
|
|
if (soundio->jack_info_callback)
|
|
|
|
jack_set_info_function(soundio->jack_info_callback);
|
|
|
|
global_msg_callback_flag.clear();
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
sij->mutex = soundio_os_mutex_create();
|
|
|
|
if (!sij->mutex) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
sij->cond = soundio_os_cond_create();
|
|
|
|
if (!sij->cond) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-07-27 18:27:41 +00:00
|
|
|
// We pass JackNoStartServer due to
|
|
|
|
// https://github.com/jackaudio/jack2/issues/138
|
2015-07-25 01:43:14 +00:00
|
|
|
jack_status_t status;
|
|
|
|
sij->client = jack_client_open(soundio->app_name, JackNoStartServer, &status);
|
|
|
|
if (!sij->client) {
|
|
|
|
destroy_jack(si);
|
|
|
|
assert(!(status & JackInvalidOption));
|
|
|
|
if (status & JackShmFailure)
|
|
|
|
return SoundIoErrorSystemResources;
|
|
|
|
if (status & JackNoSuchClient)
|
|
|
|
return SoundIoErrorNoSuchClient;
|
|
|
|
|
|
|
|
return SoundIoErrorInitAudioBackend;
|
|
|
|
}
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
int err;
|
|
|
|
if ((err = jack_set_buffer_size_callback(sij->client, buffer_size_callback, si))) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorInitAudioBackend;
|
|
|
|
}
|
|
|
|
if ((err = jack_set_sample_rate_callback(sij->client, sample_rate_callback, si))) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorInitAudioBackend;
|
|
|
|
}
|
|
|
|
if ((err = jack_set_port_registration_callback(sij->client, port_registration_callback, si))) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorInitAudioBackend;
|
|
|
|
}
|
|
|
|
if ((err = jack_set_port_rename_callback(sij->client, port_rename_calllback, si))) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorInitAudioBackend;
|
|
|
|
}
|
|
|
|
jack_on_shutdown(sij->client, shutdown_callback, si);
|
|
|
|
|
2015-07-28 20:36:31 +00:00
|
|
|
sij->buffer_size = jack_get_buffer_size(sij->client);
|
|
|
|
sij->sample_rate = jack_get_sample_rate(sij->client);
|
|
|
|
|
2015-07-25 05:00:33 +00:00
|
|
|
if ((err = jack_activate(sij->client))) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return SoundIoErrorInitAudioBackend;
|
|
|
|
}
|
|
|
|
|
|
|
|
sij->initialized = true;
|
|
|
|
if ((err = refresh_devices(si))) {
|
|
|
|
destroy_jack(si);
|
|
|
|
return err;
|
|
|
|
}
|
2015-07-25 01:43:14 +00:00
|
|
|
|
|
|
|
si->destroy = destroy_jack;
|
|
|
|
si->flush_events = flush_events_jack;
|
|
|
|
si->wait_events = wait_events_jack;
|
|
|
|
si->wakeup = wakeup_jack;
|
|
|
|
|
|
|
|
si->outstream_open = outstream_open_jack;
|
|
|
|
si->outstream_destroy = outstream_destroy_jack;
|
|
|
|
si->outstream_start = outstream_start_jack;
|
|
|
|
si->outstream_begin_write = outstream_begin_write_jack;
|
|
|
|
si->outstream_end_write = outstream_end_write_jack;
|
|
|
|
si->outstream_clear_buffer = outstream_clear_buffer_jack;
|
|
|
|
si->outstream_pause = outstream_pause_jack;
|
|
|
|
|
|
|
|
si->instream_open = instream_open_jack;
|
|
|
|
si->instream_destroy = instream_destroy_jack;
|
|
|
|
si->instream_start = instream_start_jack;
|
|
|
|
si->instream_begin_read = instream_begin_read_jack;
|
|
|
|
si->instream_end_read = instream_end_read_jack;
|
|
|
|
si->instream_pause = instream_pause_jack;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|