Beast Jack driver (Re: Lockfree ringbuffer review)
- From: Stefan Westerfeld <stefan space twc de>
- To: Tim Janik <timj gtk org>
- Cc: beast gnome org, paul linuxaudiosystems com
- Subject: Beast Jack driver (Re: Lockfree ringbuffer review)
- Date: Thu, 1 Jun 2006 22:11:18 +0200
Hi!
On Thu, Jun 01, 2006 at 05:14:15PM +0200, Tim Janik wrote:
> On Wed, 31 May 2006, Stefan Westerfeld wrote:
> >On Tue, May 30, 2006 at 05:13:10PM +0200, Tim Janik wrote:
> >Well, I thought of it, but copy can copy anything, be it bytes or
> >std::strings. The ringbuffer is a template (so you can instantiate it
> >with floats or bytes or anything else), so just plugging bseblockutils.h
> >won't work.
>
> i really think using a template here is overkill. it's highly unlikely
> that we'll use this ringbuffer with ints or doubles. and using floats
> allowes optimizations that currently are not possible.
>
> [...]
>
> >We could use template specialization to
> >enforce that if you're instantiating a float ringbuffer, then
> >bseblockutils will be used. But it wouldn't make things faster.
>
> well, what's the use case for a non-float ringbuffer?
I've just read my way through the new jack midi API, and we'll need a
ring buffer to get the midi events from the realtime thread, where they
will be delivered by jack, and our midi thread. So yes, there is a use
case, which will probably have unsigned char as data type, or even
something like struct MidiEvent.
> >The benchmarks on my AMD64 suggest that replacing memcpy in
> >bseblockutils.h by SSE instructions will not have a positive effect on
> >performance. I don't know about the other Intel/AMD systems though.
>
> that is unrelated to the ring buffer. if SIMD copying was slower than
> non-SIMD copying, this has to be fixed as part of an optimization of
> the blockutils implementation, not by the ring buffer (or by the ring
> buffer not using blockutils).
Ok, ok... new version uses template specialization, so we have fast_copy
now, which will use Bse::Block::copy if the data type permits doing so.
> >Finally, if we were to optimize the JACK driver for maximum performance,
> >a more important factor would probably be that the ringbuffer even has
> >this API.
>
> sorry? what API are you talking about?
> void bse_block_copy_float (guint n_values,
> gfloat *values,
> const gfloat *ivalues);
> copying?
>
> >It means that we need to deinterleave and reinterleave the
> >data JACK supplies on the stack, and then read/write it to the
> >ringbuffer. If the ringbuffer would offer an API to access its memory
> >directly, the extra copy would disappear. But it makes the code harder
> >to maintain and for now my first priority has been getting things
> >implemented correctly.
>
> i don't quite understand this point.
> are you saying that replacing memcpy() by bse_block_copy_float() can't
> be done due to interleaving? :-O
No. It can be replaced, and the new version has it. However, our data
flow looks like this:
input
jack thread process() callback supplies N separate buffers
|
V
interleave on the stack (manually implemented, in process()) *jack-thread*
|
V
copy into the input ringbuffer (input ringbuffer write) *jack-thread*
|
V
copy in bse engine supplied buffer (input ringbuffer read) *engine-thread*
|
V
[ beast processing ] *engine-thread*
|
V
copy from bse engine supplied buffer (output ringbuffer write) *engine-thread*
|
V
copy onto stack buffer (output ringbuffer read) *jack-thread*
|
V
deinterleave into seperate buffers (manually) *jack-thread*
|
V
end of jack thread process() callback
output
What I was pointing out to is that if the ringbuffer provided an API to
access the data which is stored in the ringbuffer directly, then we
could avoid some copies (if you wonder how such an API would look, read
http://www.portaudio.com/trac/browser/portaudio/branches/v19-devel/src/hostapi/coreaudio/ringbuffer.c
and look at RingBuffer_GetWriteRegions). However if we get everything we
want - with the number of copies we're making right now - I am perfectly
happy. Every sensible bse network will eat much more cycles than we can
possibly spend in those extra Bse::Block::copy() calls.
> >namespace {
> >
> >/**
> >* The FrameRingBuffer class implements a ringbuffer for the communication
> >* between two threads. One thread - the producer thread - may only write
> >* data to the ringbuffer. The other thread - the consumer thread - may
> >* only read data from the ringbuffer.
> >*
> >* Given that these two threads only use the appropriate functions, no
> >* other synchronization is required to ensure that the data gets safely
> >* from the producer thread to the consumer thread. However, all operations
> >* that are provided by the ringbuffer are non-blocking, so that you may
> >* need a condition or other synchronization primitive if you want the
> >* producer and/or consumer to block if the ringbuffer is full/empty.
>
> s/if/on the event that/ ?
I think not (but I am no native speaker). What I want to say is this:
Suppose the ringbuffer is empty. The consumer will, when calling the
read function, just read 0 frames. A common scenario is that the
consumer will want to wait until data is available. To implement this,
the consumer will need to use a synchonization mechanism (i.e.
condition).
> >*
> >* Implementation: the synchronization between the two threads is only
> >* implemented by two index variables (read_frame_pos and write_frame_pos)
> >* for which atomic integer reads and writes are required. Since the
> >* producer thread only modifies the write_frame_pos and the consumer thread
> >* only modifies the read_frame_pos, no compare-and-swap or similar
> >* operations are needed to avoid concurrent writes.
> >*/
>
> good comment btw, i now have an idea on why/how things are supposed
> to work and can scrutinize your implementaiton accordingly ;)
>
> >template<class T>
> >class FrameRingBuffer {
> > //BIRNET_PRIVATE_COPY (FrameRingBuffer);
>
> what's this comment for?
I wanted to call BIRNET_PRIVATE_COPY, but it didn't work, so I put //
marks there, so it compiled. So can I use BIRNET_PRIVATE_COPY or do I
need to put the code BIRNET_PRIVATE_COPY would put there if it worked
here manually?
> > // the extra element allows us to see the difference between an
> > empty/full ringbuffer
>
> s/element/frame/ right?
Right.
> > /**
> > * Writes data to the ringbuffer; if there is not enough data
>
> s/enough data/enough free space/ if i understand correctly.
>
> > * in the ringbuffer, the function will not block.
>
> hm, it be better to say what the function does rather than saying
> "it will not block", i.e. what it not does.
> afaiu, this should be:
>
> * Write data to the ringbuffer; if there is not enough free space
> * in the ringbuffer, the function will return with the amount of
> * frames consumed by a partial write.
Well, it's similar now:
* Write data to the ringbuffer; if there is not enough free space
* in the ringbuffer, the function will return the amount of frames
* consumed by a partial write (without blocking).
I wanted to keep the remark that it doesn't block there. I also adapted
the read description in a similar way:
* Read data from the ringbuffer; if there is not enough data
* in the ringbuffer, the function will return the number of frames
* that could be read without blocking.
> > /**
> > * Returns the maximum number of frames that the ringbuffer can contain.
> > *
> > * This function can be called from any thread.
> > */
> > guint
> > size() const
> > {
> > // the extra element allows us to see the difference between an
> > empty/full ringbuffer
> > return (m_buffer.size() - 1) / m_elements_per_frame;
> > }
>
> ugh this is mighty confusing.
> you should get rid of this function or at least clearly rename it.
> seeing it, i had to go back and read your code all over to figure whether
> you
> used size() in every place or m_buffer.size().
> if you need a name, call it total_n_frames(), allthough it seems this
> function
> isn't used/needed at all.
Well, its not used for internal purposes. But its provided for the code
which uses the ringbuffer. Basically if I am implementing C++ code for
some data structure that has a size (as the ringbuffer does), then I
should also provide an accessor that can be used to determine the size.
Just as std::vector has a resize() and size() function.
> > /**
> > * Clears the ringbuffer.
>
> s/Clears/Clear/, applies to lots of other comments as well.
>
> > *
> > * This function is @emph{not} threadsafe, and can not be used while
>
> well, depends on what you mean by "threadsafe". i'd simply leave that
> out and just write:
>
> * Clear the ringbuffer.
> * This function may not be used while
> * either the producer thread or the consumer thread are modifying
> * the ringbuffer.
>
> (also fixed "can" vs. "may")
Oh yes, your comment sounds better. Definitely. Fixed.
> > void
> > clear()
> > {
> > Atomic::int_set (&m_read_frame_pos, 0);
> > Atomic::int_set (&m_write_frame_pos, 0);
> > }
>
> is this really needed outside of resize() though?
The jack driver needs it (to get into a sane state after a dropout).
> > m_buffer.resize ((n_frames + 1) * m_elements_per_frame);
> > clear();
>
> looks to me like you could simply inline resetting of the
> indices here.
If there is a function anyway, I might as well use it.
> hm, don't you want review on the rest of the driver as well?
> (that'd prolly also be the part that is more interesting for Paul)
Definitely. So here is an updated version of the ringbuffer and the
whole driver source. I think the driver already works pretty well, its
just that there are quite a few FIXMEs in there.
As I am only posting the C++ source to the list, if anyone is so brave
and wants to compile the driver (you need beast CVS for this), I've put
the whole driver to:
http://space.twc.de/~stefan/download/20060601-preview-bse-jack-0.7.0.tar.gz
(sorry that it extracts to the wrong directory, but I just used make
dist...).
Anyway, here is the C++ source (the interesting part, for the review):
/* BSE - Bedevilled Sound Engine
* Copyright (C) 2004 Tim Janik
* Copyright (C) 2006 Stefan Westerfeld
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This program 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this program; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
* Boston, MA 02111-1307, USA.
*/
#include "configure.h"
#include "bsepcmdevice-jack.hh"
#include <bse/bseblockutils.hh>
#include <bse/gsldatautils.h>
#include <jack/jack.h>
#include <string.h>
#include <errno.h>
#include <signal.h>
#include <set>
#include <string>
#include <map>
#include <vector>
using namespace Birnet;
using std::vector;
using std::set;
using std::map;
using std::string;
using std::min;
using std::max;
using std::copy;
using Bse::Block;
namespace {
/**
* This function uses std::copy to copy the n_values of data from ivalues
* to ovalues. If a specialized version is available in bseblockutils,
* then this - usually faster - version will be used.
*
* The data in ivalues and ovalues may not overlap.
*/
template<class Data> void
fast_copy (guint n_values,
Data *ovalues,
const Data *ivalues)
{
copy (ivalues, ivalues + n_values, ovalues);
}
template<> void
fast_copy (guint n_values,
float *ovalues,
const float *ivalues)
{
Block::copy (n_values, ovalues, ivalues);
}
template<> void
fast_copy (guint n_values,
guint32 *ovalues,
const guint32 *ivalues)
{
Block::copy (n_values, ovalues, ivalues);
}
/**
* The FrameRingBuffer class implements a ringbuffer for the communication
* between two threads. One thread - the producer thread - may only write
* data to the ringbuffer. The other thread - the consumer thread - may
* only read data from the ringbuffer.
*
* Given that these two threads only use the appropriate functions, no
* other synchronization is required to ensure that the data gets safely
* from the producer thread to the consumer thread. However, all operations
* that are provided by the ringbuffer are non-blocking, so that you may
* need a condition or other synchronization primitive if you want the
* producer and/or consumer to block if the ringbuffer is full/empty.
*
* Implementation: the synchronization between the two threads is only
* implemented by two index variables (read_frame_pos and write_frame_pos)
* for which atomic integer reads and writes are required. Since the
* producer thread only modifies the write_frame_pos and the consumer thread
* only modifies the read_frame_pos, no compare-and-swap or similar
* operations are needed to avoid concurrent writes.
*/
template<class T>
class FrameRingBuffer {
//BIRNET_PRIVATE_COPY (FrameRingBuffer);
private:
vector<T> m_buffer;
volatile int m_read_frame_pos;
volatile int m_write_frame_pos;
guint m_elements_per_frame;
void
write_memory_barrier()
{
static volatile int dummy = 0;
/*
* writing this dummy integer should ensure that all prior writes
* are committed to memory
*/
Atomic::int_set (&dummy, 0x12345678);
}
public:
FrameRingBuffer (guint n_frames = 0,
guint elements_per_frame = 1)
{
resize (n_frames, elements_per_frame);
}
/**
* @returns the number of frames that are available for reading
*
* Check available read space in the ringbuffer.
* This function may only be called from the consumer thread.
*/
guint
get_readable_frames()
{
int wpos = Atomic::int_get (&m_write_frame_pos);
int rpos = Atomic::int_get (&m_read_frame_pos);
int size = m_buffer.size() / m_elements_per_frame;
if (wpos < rpos) /* wpos == rpos -> empty ringbuffer */
wpos += size;
return wpos - rpos;
}
/**
* @returns the number of successfully read frames
*
* Read data from the ringbuffer; if there is not enough data
* in the ringbuffer, the function will return the number of frames
* that could be read without blocking.
*
* This function should be called from the consumer thread.
*/
guint
read (guint n_frames,
T *frames)
{
int rpos = Atomic::int_get (&m_read_frame_pos);
guint size = m_buffer.size() / m_elements_per_frame;
guint can_read = min (get_readable_frames(), n_frames);
guint read1 = min (can_read, size - rpos) * m_elements_per_frame;
fast_copy (read1, frames, &m_buffer[rpos * m_elements_per_frame]);
guint read2 = can_read * m_elements_per_frame - read1;
fast_copy (read2, frames + read1, &m_buffer[0]);
Atomic::int_set (&m_read_frame_pos, (rpos + can_read) % size);
return can_read;
}
/**
* @returns the number of frames that can be written
*
* Check available write space in the ringbuffer.
* This function should be called from the producer thread.
*/
guint
get_writable_frames()
{
int wpos = Atomic::int_get (&m_write_frame_pos);
int rpos = Atomic::int_get (&m_read_frame_pos);
guint size = m_buffer.size() / m_elements_per_frame;
if (rpos <= wpos) /* wpos == rpos -> empty ringbuffer */
rpos += size;
// the extra frame allows us to see the difference between an empty/full ringbuffer
return rpos - wpos - 1;
}
/**
* @returns the number of successfully written frames
*
* Write data to the ringbuffer; if there is not enough free space
* in the ringbuffer, the function will return the amount of frames
* consumed by a partial write (without blocking).
*
* This function may only be called from the producer thread.
*/
guint
write (guint n_frames,
const T *frames)
{
int wpos = Atomic::int_get (&m_write_frame_pos);
guint size = m_buffer.size() / m_elements_per_frame;
guint can_write = min (get_writable_frames(), n_frames);
guint write1 = min (can_write, size - wpos) * m_elements_per_frame;
fast_copy (write1, &m_buffer[wpos * m_elements_per_frame], frames);
guint write2 = can_write * m_elements_per_frame - write1;
fast_copy (write2, &m_buffer[0], frames + write1);
// It is important that the data from the previous writes get committed
// to memory *before* the index variable is updated. Otherwise, the
// consumer thread could be reading invalid data, if the index variable
// got written before the rest of the data (when unordered writes are
// performed).
write_memory_barrier();
Atomic::int_set (&m_write_frame_pos, (wpos + can_write) % size);
return can_write;
}
/**
* @returns the maximum number of frames that the ringbuffer can contain
*
* This function can be called from any thread.
*/
guint
get_total_n_frames() const
{
// the extra frame allows us to see the difference between an empty/full ringbuffer
return (m_buffer.size() - 1) / m_elements_per_frame;
}
/**
* @returns the number of elements that are part of one frame
*
* This function can be called from any thread.
*/
guint
get_elements_per_frame() const
{
return m_elements_per_frame;
}
/**
* Clear the ringbuffer.
*
* This function may not be used while either the producer thread or
* the consumer thread are modifying the ringbuffer.
*/
void
clear()
{
Atomic::int_set (&m_read_frame_pos, 0);
Atomic::int_set (&m_write_frame_pos, 0);
}
/**
* Resize and clear the ringbuffer.
*
* This function may not be used while either the producer thread or
* the consumer thread are modifying the ringbuffer.
*/
void
resize (guint n_frames,
guint elements_per_frame = 1)
{
m_elements_per_frame = elements_per_frame;
// the extra frame allows us to see the difference between an empty/full ringbuffer
m_buffer.resize ((n_frames + 1) * m_elements_per_frame);
clear();
}
};
static BIRNET_MSG_TYPE_DEFINE (debug_pcm, "pcm", BIRNET_MSG_DEBUG, NULL);
#define DEBUG(...) sfi_debug (debug_pcm, __VA_ARGS__)
/* --- JACK PCM handle --- */
enum
{
CALLBACK_STATE_INACTIVE = 0,
CALLBACK_STATE_ACTIVE = 1,
CALLBACK_STATE_PLEASE_TERMINATE = 2,
CALLBACK_STATE_TERMINATED = 3
};
struct JackPcmHandle
{
BsePcmHandle handle;
JackPcmHandle (jack_client_t *jack_client)
: jack_client (jack_client)
{
memset (&handle, 0, sizeof (handle));
callback_state = ixruns = pixruns = oxruns = poxruns = 0;
is_down = false;
}
vector<jack_port_t *> input_ports;
vector<jack_port_t *> output_ports;
guint buffer_frames; /* input/output ringbuffer size in frames */
Cond cond;
Mutex cond_mutex;
jack_client_t *jack_client;
FrameRingBuffer<float> input_ringbuffer;
FrameRingBuffer<float> output_ringbuffer;
int callback_state;
int ixruns;
int oxruns;
int pixruns;
int poxruns;
bool is_down;
};
}
/* --- prototypes --- */
static void bse_pcm_device_jack_class_init (BsePcmDeviceJACKClass *klass);
static void bse_pcm_device_jack_init (BsePcmDeviceJACK *self);
static gsize jack_device_read (BsePcmHandle *handle,
gfloat *values);
static void jack_device_write (BsePcmHandle *handle,
const gfloat *values);
static gboolean jack_device_check_io (BsePcmHandle *handle,
glong *tiumeoutp);
static guint jack_device_latency (BsePcmHandle *handle);
static void jack_device_retrigger (JackPcmHandle *jack);
/* --- define object type and export to BSE --- */
static const char type_blurb[] = ("PCM driver implementation for JACK Audio Connection Kit "
"(http://jackaudio.org)");
BSE_REGISTER_OBJECT (BsePcmDeviceJACK, BsePcmDevice, NULL, "", type_blurb, NULL, bse_pcm_device_jack_class_init, NULL, bse_pcm_device_jack_init);
BSE_DEFINE_EXPORTS (__FILE__);
/* --- variables --- */
static gpointer parent_class = NULL;
/* --- functions --- */
static void
bse_pcm_device_jack_init (BsePcmDeviceJACK *self)
{
/* FIXME: move signal handling somewhere else */
sigset_t signal_mask;
sigemptyset (&signal_mask);
sigaddset (&signal_mask, SIGPIPE);
int rc = pthread_sigmask (SIG_BLOCK, &signal_mask, NULL);
if (rc != 0)
g_printerr ("jack driver: pthread_sigmask for SIGPIPE failed: %s\n", strerror (errno));
jack_status_t status;
self->jack_client = jack_client_open ("beast", JackNoStartServer, &status); /* FIXME: translations(?) */
DEBUG ("attaching to JACK server returned status: %d\n", status);
/* FIXME: check status for proper error reporting
* FIXME: what about server name? (necessary if more than one jackd instance is running)
* FIXME: get rid of device unloading, so that the jackd connection can be kept initialized
*
* maybe move this to a different function
*/
}
namespace {
struct DeviceDetails {
guint ports, input_ports, output_ports, physical_ports, terminal_ports;
vector<string> input_port_names;
vector<string> output_port_names;
DeviceDetails() : ports (0), input_ports (0), output_ports (0), physical_ports (0), terminal_ports (0)
{
}
};
}
static map<string, DeviceDetails>
query_jack_devices (BsePcmDeviceJACK *self)
{
map<string, DeviceDetails> devices;
/* FIXME: we need to make a difference here between
* - jackd is running, but no output ports can be found
* - jackd is not running
*/
const char **jack_ports = self->jack_client ? jack_get_ports (self->jack_client, NULL, NULL, 0) : NULL;
if (jack_ports)
{
for (guint i = 0; jack_ports[i]; i++)
{
const char *end = strchr (jack_ports[i], ':');
if (end)
{
string device_name (jack_ports[i], end);
DeviceDetails &details = devices[device_name];
details.ports++;
jack_port_t *jack_port = jack_port_by_name (self->jack_client, jack_ports[i]);
if (jack_port)
{
int flags = jack_port_flags (jack_port);
if (flags & JackPortIsInput)
{
details.input_ports++;
details.input_port_names.push_back (jack_ports[i]);
}
if (flags & JackPortIsOutput)
{
details.output_ports++;
details.output_port_names.push_back (jack_ports[i]);
}
if (flags & JackPortIsPhysical)
details.physical_ports++;
if (flags & JackPortIsTerminal)
details.terminal_ports++;
}
}
}
free (jack_ports);
}
return devices;
}
static SfiRing*
bse_pcm_device_jack_list_devices (BseDevice *device)
{
BsePcmDeviceJACK *self = BSE_PCM_DEVICE_JACK (device);
map<string, DeviceDetails> devices = query_jack_devices (self);
SfiRing *ring = NULL;
for (map<string, DeviceDetails>::iterator di = devices.begin(); di != devices.end(); di++)
{
const string &device_name = di->first;
DeviceDetails &details = di->second;
BseDeviceEntry *entry;
entry = bse_device_group_entry_new (device, g_strdup (device_name.c_str()),
g_strdup ("JACK Devices"),
g_strdup_printf ("%s (%d*input %d*output)%s",
device_name.c_str(),
details.input_ports,
details.output_ports,
details.physical_ports == details.ports ?
" [ hardware ]" : ""));
/* ensure that alsa_pcm always is the first item listed (it is the default device) */
if (device_name == "alsa_pcm")
ring = sfi_ring_prepend (ring, entry);
else
ring = sfi_ring_append (ring, entry);
}
if (!ring)
ring = sfi_ring_append (ring, bse_device_error_new (device, g_strdup_printf ("No devices found")));
return ring;
}
static void
jack_shutdown_callback (void *jack_handle_ptr)
{
JackPcmHandle *jack = static_cast <JackPcmHandle *> (jack_handle_ptr);
AutoLocker cond_locker (jack->cond_mutex);
jack->is_down = true;
jack->cond.signal();
}
static int
jack_process_callback (jack_nframes_t n_frames,
void *jack_handle_ptr)
{
typedef jack_default_audio_sample_t sample_t;
JackPcmHandle *jack = static_cast <JackPcmHandle *> (jack_handle_ptr);
int callback_state = Atomic::int_get (&jack->callback_state);
/* still waiting for initialization to complete */
if (callback_state == CALLBACK_STATE_ACTIVE)
{
bool have_cond_lock = jack->cond_mutex.trylock();
guint n_channels = jack->handle.n_channels;
float buffer[n_frames * n_channels];
if (jack->handle.readable)
{
/* interleave input data for processing in the engine thread */
g_assert (jack->input_ports.size() == n_channels);
for (guint ch = 0; ch < n_channels; ch++)
{
sample_t *values = (sample_t *) jack_port_get_buffer (jack->input_ports[ch], n_frames);
float *p = &buffer[ch];
for (guint i = 0; i < n_frames; i++)
{
*p = *values++;
p += n_channels;
}
}
guint frames_written = jack->input_ringbuffer.write (n_frames, buffer);
if (frames_written != n_frames)
Atomic::int_add (&jack->ixruns, 1); /* input underrun detected */
}
if (jack->handle.writable)
{
guint read_frames = jack->output_ringbuffer.read (n_frames, buffer);
if (read_frames != n_frames)
{
Atomic::int_add (&jack->oxruns, 1); /* output underrun detected */
Block::fill ((n_frames - read_frames) * n_channels, &buffer[read_frames * n_channels], 0.0);
}
/* deinterleave output data for processing in the engine thread */
g_assert (jack->output_ports.size() == n_channels);
for (guint ch = 0; ch < n_channels; ch++)
{
sample_t *values = (sample_t *) jack_port_get_buffer (jack->output_ports[ch], n_frames);
sample_t *values_end = values + n_frames;
float *p = &buffer[ch];
while (values < values_end)
{
*values++ = *p;
p += n_channels;
}
}
}
/*
* as we can't (always) lock our data structures from the jack realtime
* thread, using a condition introduces the possibility for a race:
*
* normal operation:
*
* [bse thread] writes some data to output_ringbuffer
* [bse thread] checks remaining space, there is no room left
* [bse thread] sleeps on the condition
* [jack thread] reads some data from output_ringbuffer
* [jack thread] signals the condition
* [bse thread] continues to write some data to output_ringbuffer
*
* race condition:
*
* [bse thread] writes some data to output_ringbuffer
* [bse thread] checks remaining space, there is no room left
* [jack thread] reads some data from output_ringbuffer
* [jack thread] signals the condition
* [bse thread] sleeps on the condition
*
* since the jack callback gets called periodically, the bse thread will
* never starve, though, in the worst case it will just miss a frame
*
* so we absolutely exclude the possibility for priority inversion (by
* using trylock instead of lock), by introducing extra latency in
* some extremely rare cases
*/
if (have_cond_lock)
jack->cond_mutex.unlock();
}
else
{
for (guint ch = 0; ch < jack->input_ports.size(); ch++)
{
sample_t *values = (sample_t *) jack_port_get_buffer (jack->input_ports[ch], n_frames);
Block::fill (n_frames, values, 0.0);
}
for (guint ch = 0; ch < jack->output_ports.size(); ch++)
{
sample_t *values = (sample_t *) jack_port_get_buffer (jack->output_ports[ch], n_frames);
Block::fill (n_frames, values, 0.0);
}
}
jack->cond.signal();
if (callback_state == CALLBACK_STATE_PLEASE_TERMINATE)
{
Atomic::int_set (&jack->callback_state, CALLBACK_STATE_TERMINATED);
return 1;
}
return 0;
}
static void
terminate_and_free_jack (JackPcmHandle *jack)
{
g_return_if_fail (jack->jack_client != NULL);
Atomic::int_set (&jack->callback_state, CALLBACK_STATE_PLEASE_TERMINATE);
while (Atomic::int_get (&jack->callback_state) == CALLBACK_STATE_PLEASE_TERMINATE)
{
AutoLocker cond_locker (jack->cond_mutex);
if (jack->is_down) /* if jack is already gone, then forget it */
Atomic::int_set (&jack->callback_state, CALLBACK_STATE_TERMINATED);
else
jack->cond.wait_timed (jack->cond_mutex, 100000);
}
for (guint ch = 0; ch < jack->input_ports.size(); ch++)
jack_port_disconnect (jack->jack_client, jack->input_ports[ch]);
for (guint ch = 0; ch < jack->output_ports.size(); ch++)
jack_port_disconnect (jack->jack_client, jack->output_ports[ch]);
jack_deactivate (jack->jack_client);
delete jack;
}
static BseErrorType
bse_pcm_device_jack_open (BseDevice *device,
gboolean require_readable,
gboolean require_writable,
guint n_args,
const gchar **args)
{
BsePcmDeviceJACK *self = BSE_PCM_DEVICE_JACK (device);
if (!self->jack_client)
return BSE_ERROR_FILE_OPEN_FAILED;
JackPcmHandle *jack = new JackPcmHandle (self->jack_client);
BsePcmHandle *handle = &jack->handle;
handle->readable = require_readable;
handle->writable = require_writable;
handle->n_channels = BSE_PCM_DEVICE (device)->req_n_channels;
/* try setup */
BseErrorType error = BSE_ERROR_NONE;
const char *dname = (n_args == 1) ? args[0] : "alsa_pcm";
handle->mix_freq = jack_get_sample_rate (self->jack_client);
Atomic::int_set (&jack->callback_state, CALLBACK_STATE_INACTIVE);
for (guint i = 0; i < handle->n_channels; i++)
{
const int port_name_size = jack_port_name_size();
char port_name[port_name_size];
jack_port_t *port;
snprintf (port_name, port_name_size, "in_%u", i);
port = jack_port_register (self->jack_client, port_name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
if (port)
jack->input_ports.push_back (port);
else
error = BSE_ERROR_FILE_OPEN_FAILED;
snprintf (port_name, port_name_size, "out_%u", i);
port = jack_port_register (self->jack_client, port_name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
if (port)
jack->output_ports.push_back (port);
else
error = BSE_ERROR_FILE_OPEN_FAILED;
}
/* activate */
jack_set_process_callback (self->jack_client, jack_process_callback, jack);
jack_on_shutdown (self->jack_client, jack_shutdown_callback, jack);
jack_activate (self->jack_client);
/* connect ports */
map<string, DeviceDetails> devices = query_jack_devices (self);
map<string, DeviceDetails>::const_iterator di;
di = devices.find (dname);
if (di != devices.end())
{
const DeviceDetails &details = di->second;
for (guint ch = 0; ch < handle->n_channels; ch++)
{
if (details.output_ports > ch)
jack_connect (self->jack_client, details.output_port_names[ch].c_str(), jack_port_name (jack->input_ports[ch]));
if (details.input_ports > ch)
jack_connect (self->jack_client, jack_port_name (jack->output_ports[ch]), details.input_port_names[ch].c_str());
}
}
/* setup buffer size */
// keep at least two jack callback sizes for dropout free audio
guint min_buffer_frames = jack_get_buffer_size (self->jack_client) * 2;
// keep an extra engine buffer size (this compensates also for cases where the
// engine buffer size is not a 2^N value, which would otherwise cause the
// buffer never to be fully filled with 2 periods of data)
min_buffer_frames += BSE_PCM_DEVICE (device)->req_block_length;
// honor the user defined latency specification
//
// FIXME: should we try to tweak local buffering so that it corresponds
// to the user defined latency, or global buffering (including the jack
// buffer)? we do the former, since it is easier
guint user_buffer_frames = BSE_PCM_DEVICE (device)->req_latency_ms * handle->mix_freq / 1000;
guint buffer_frames = max (min_buffer_frames, user_buffer_frames);
jack->input_ringbuffer.resize (buffer_frames, handle->n_channels);
jack->output_ringbuffer.resize (buffer_frames, handle->n_channels);
jack->buffer_frames = jack->output_ringbuffer.get_writable_frames();
// the ringbuffer should be exactly as big as requested
g_assert (jack->buffer_frames == buffer_frames);
DEBUG ("ringbuffer size = %.3fms", jack->buffer_frames / double (handle->mix_freq) * 1000);
/* setup PCM handle or shutdown */
if (!error)
{
bse_device_set_opened (device, dname, handle->readable, handle->writable);
if (handle->readable)
handle->read = jack_device_read;
if (handle->writable)
handle->write = jack_device_write;
handle->check_io = jack_device_check_io;
handle->latency = jack_device_latency;
BSE_PCM_DEVICE (device)->handle = handle;
/* will prevent dropouts at initialization, when no data is there at all */
jack_device_retrigger (jack);
jack_device_latency (handle); // debugging only: print latency values
}
else
{
terminate_and_free_jack (jack);
}
DEBUG ("JACK: opening PCM \"%s\" readupble=%d writable=%d: %s", dname, require_readable, require_writable, bse_error_blurb (error));
return error;
}
static void
bse_pcm_device_jack_close (BseDevice *device)
{
JackPcmHandle *jack = (JackPcmHandle*) BSE_PCM_DEVICE (device)->handle;
BSE_PCM_DEVICE (device)->handle = NULL;
terminate_and_free_jack (jack);
}
static void
bse_pcm_device_jack_finalize (GObject *object)
{
BsePcmDeviceJACK *self = BSE_PCM_DEVICE_JACK (object);
if (self->jack_client)
{
jack_client_close (self->jack_client);
self->jack_client = NULL;
}
/* chain parent class' handler */
G_OBJECT_CLASS (parent_class)->finalize (object);
}
static void
jack_device_retrigger (JackPcmHandle *jack)
{
g_return_if_fail (jack->jack_client != NULL);
/*
* we need to reset the active flag to false here, as we modify the
* buffers in a non threadsafe way; this is why we also wait for
* the condition, to ensure that the jack callback really isn't
* active any more
*/
Atomic::int_set (&jack->callback_state, CALLBACK_STATE_INACTIVE);
/* usually should not timeout, but be notified by the jack callback */
jack->cond_mutex.lock();
if (!jack->is_down)
jack->cond.wait_timed (jack->cond_mutex, 100000);
jack->cond_mutex.unlock();
/* jack_ringbuffer_reset is not threadsafe! */
jack->input_ringbuffer.clear();
jack->output_ringbuffer.clear();
/* initialize output ringbuffer with silence */
vector<float> silence (jack->buffer_frames * jack->handle.n_channels);
guint frames_written = jack->output_ringbuffer.write (jack->buffer_frames, &silence[0]);
g_assert (frames_written == jack->buffer_frames);
DEBUG ("jack_device_retrigger: %d frames written", frames_written);
}
static gboolean
jack_device_check_io (BsePcmHandle *handle,
glong *timeoutp)
{
JackPcmHandle *jack = (JackPcmHandle*) handle;
g_return_val_if_fail (jack->jack_client != NULL, false);
/*
int ixruns = Atomic::int_get (&jack->ixruns);
int oxruns = Atomic::int_get (&jack->oxruns);
if (jack->poxruns != oxruns || jack->pixruns != ixruns)
{
printf ("beast caused jack input xruns %d, output xruns %d\n", ixruns, oxruns);
jack->poxruns = oxruns;
jack->pixruns = ixruns;
jack_device_retrigger (jack);
}
*/
/* (FIXME?)
*
* we can not guarantee that beast will read the data from the jack ringbuffer,
* since this depends on the presence of a module that actually reads data in
* the synthesis graph : so we simply ignore the ixruns variable; I am not sure
* if thats the right thing to do, though
*/
int oxruns = Atomic::int_get (&jack->oxruns);
if (jack->poxruns != oxruns)
{
g_printerr ("%d beast jack driver xruns\n", oxruns);
jack->poxruns = oxruns;
jack_device_retrigger (jack);
}
Atomic::int_set (&jack->callback_state, CALLBACK_STATE_ACTIVE);
guint n_frames_avail = min (jack->output_ringbuffer.get_writable_frames(), jack->input_ringbuffer.get_readable_frames());
/* check whether data can be processed */
if (n_frames_avail >= handle->block_length)
return TRUE; /* need processing */
/* calculate timeout until processing is possible or needed */
guint diff_frames = handle->block_length - n_frames_avail;
*timeoutp = diff_frames * 1000 / handle->mix_freq;
return FALSE;
}
static guint
jack_device_latency (BsePcmHandle *handle)
{
JackPcmHandle *jack = (JackPcmHandle*) handle;
jack_nframes_t rlatency = 0, wlatency = 0;
g_return_val_if_fail (jack->jack_client != NULL, 0);
/* FIXME: the API of this function is broken, because you can't use its result
* to sync for instance the play position pointer with the screen
*
* why? because it doesn't return you rlatency and wlatency seperately, but
* only the sum of both
*
* when using jack, there is no guarantee that rlatency and wlatency are equal
*/
for (guint i = 0; i < jack->input_ports.size(); i++)
{
jack_nframes_t latency = jack_port_get_total_latency (jack->jack_client, jack->input_ports[i]);
rlatency = max (rlatency, latency);
}
for (guint i = 0; i < jack->output_ports.size(); i++)
{
jack_nframes_t latency = jack_port_get_total_latency (jack->jack_client, jack->output_ports[i]);
wlatency = max (wlatency, latency);
}
guint total_latency = 2 * jack->buffer_frames + rlatency + wlatency;
DEBUG ("rlatency=%.3f ms wlatency=%.3f ms ringbuffer=%.3f ms total_latency=%.3f ms",
rlatency / double (handle->mix_freq) * 1000,
wlatency / double (handle->mix_freq) * 1000,
jack->buffer_frames / double (handle->mix_freq) * 1000,
total_latency / double (handle->mix_freq) * 1000);
return total_latency;
}
static gsize
jack_device_read (BsePcmHandle *handle,
gfloat *values)
{
JackPcmHandle *jack = (JackPcmHandle*) handle;
g_return_val_if_fail (jack->jack_client != NULL, 0);
g_return_val_if_fail (Atomic::int_get (&jack->callback_state) == CALLBACK_STATE_ACTIVE, 0);
guint frames_left = handle->block_length;
while (frames_left)
{
guint frames_read = jack->input_ringbuffer.read (frames_left, values);
frames_left -= frames_read;
values += frames_read * handle->n_channels;
if (frames_left)
{
AutoLocker cond_locker (jack->cond_mutex);
/* usually should not timeout, but be notified by the jack callback */
if (!jack->input_ringbuffer.get_readable_frames())
jack->cond.wait_timed (jack->cond_mutex, 100000);
if (jack->is_down)
{
/*
* FIXME: we need a way to indicate an error here; beast should provide
* an adequate reaction in case the JACK server is down (it should stop
* playing the file, and show a dialog, if JACK can not be reconnected)
*
* if we have a way to abort processing, this if can be moved above
* the condition wait; however, right now moving it there means that
* beast will render the output as fast as possible when jack dies, and
* this will look like a machine lockup
*/
Block::fill (frames_left * handle->n_channels, values, 0.0); /* <- remove me once we can indicate errors */
return handle->block_length * handle->n_channels;
}
}
}
return handle->block_length * handle->n_channels;
}
static void
jack_device_write (BsePcmHandle *handle,
const gfloat *values)
{
JackPcmHandle *jack = (JackPcmHandle*) handle;
g_return_if_fail (jack->jack_client != NULL);
g_return_if_fail (Atomic::int_get (&jack->callback_state) == CALLBACK_STATE_ACTIVE);
guint frames_left = handle->block_length;
while (frames_left)
{
guint frames_written = jack->output_ringbuffer.write (frames_left, values);
frames_left -= frames_written;
values += frames_written * handle->n_channels;
if (frames_left)
{
AutoLocker cond_locker (jack->cond_mutex);
/* usually should not timeout, but be notified by the jack callback */
if (!jack->output_ringbuffer.get_writable_frames())
jack->cond.wait_timed (jack->cond_mutex, 100000);
if (jack->is_down)
{
/*
* FIXME: we need a way to indicate an error here; beast should provide
* an adequate reaction in case the JACK server is down (it should stop
* playing the file, and show a dialog, if JACK can not be reconnected)
*
* if we have a way to abort processing, this if can be moved above
* the condition wait; however, right now moving it there means that
* beast will render the output as fast as possible when jack dies, and
* this will look like a machine lockup
*/
return;
}
}
}
}
static void
bse_pcm_device_jack_class_init (BsePcmDeviceJACKClass *klass)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
BseDeviceClass *device_class = BSE_DEVICE_CLASS (klass);
parent_class = g_type_class_peek_parent (klass);
gobject_class->finalize = bse_pcm_device_jack_finalize;
device_class->list_devices = bse_pcm_device_jack_list_devices;
const gchar *name = "jack";
const gchar *syntax = _("device");
const gchar *info = g_intern_printf (/* TRANSLATORS: keep this text to 70 chars in width */
_("JACK Audio Connection Kit PCM driver (http://jackaudio.org)\n"));
/* Being BSE_RATING_PREFFERED + 1, should do the right thing: if jackd is running,
* use it, if jackd isn't running (we'll get an error upon initialization,
* then), use the next best driver.
*
* Right now we will never start the jackd process ourselves.
*/
bse_device_class_setup (klass, BSE_RATING_PREFERRED + 1, name, syntax, info);
device_class->open = bse_pcm_device_jack_open;
device_class->close = bse_pcm_device_jack_close;
}
Cu... Stefan
--
Stefan Westerfeld, Hamburg/Germany, http://space.twc.de/~stefan
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]