jack: add support for timebase master and sync

This commit is contained in:
Wim Taymans 2019-08-28 13:55:50 +02:00
parent a52ad27a16
commit 0c2c2c72d0

View file

@ -261,10 +261,11 @@ struct client {
void *xrun_arg; void *xrun_arg;
JackSyncCallback sync_callback; JackSyncCallback sync_callback;
void *sync_arg; void *sync_arg;
uint32_t sync_version;
unsigned int sync_emit:1; unsigned int sync_emit:1;
JackTimebaseCallback timebase_callback; JackTimebaseCallback timebase_callback;
void *timebase_arg; void *timebase_arg;
unsigned int timebase_emit:1; unsigned int timebase_new_pos:1;
struct spa_io_position *position; struct spa_io_position *position;
uint32_t sample_rate; uint32_t sample_rate;
@ -812,6 +813,7 @@ on_rtsocket_condition(void *data, int fd, uint32_t mask)
uint32_t buffer_size, sample_rate; uint32_t buffer_size, sample_rate;
struct link *l; struct link *l;
struct spa_io_position *pos = c->position; struct spa_io_position *pos = c->position;
struct pw_node_activation *a = c->driver_activation;
if (read(fd, &cmd, sizeof(cmd)) != sizeof(cmd)) if (read(fd, &cmd, sizeof(cmd)) != sizeof(cmd))
pw_log_warn(NAME" %p: read failed %m", c); pw_log_warn(NAME" %p: read failed %m", c);
@ -845,14 +847,19 @@ on_rtsocket_condition(void *data, int fd, uint32_t mask)
jack_state = position_to_jack(pos, &jack_position); jack_state = position_to_jack(pos, &jack_position);
if (c->sync_callback && c->sync_emit) { if (a) {
c->sync_callback(jack_state, if (a->sync_version != c->sync_version) {
&jack_position, c->sync_arg); c->sync_emit = true;
c->timebase_new_pos = true;
c->sync_version = a->sync_version;
}
if (c->sync_emit) {
if (c->sync_callback &&
c->sync_callback(jack_state, &jack_position, c->sync_arg)) {
ATOMIC_DEC(a->sync_pending);
}
c->sync_emit = false; c->sync_emit = false;
} }
if (c->driver_activation) {
struct pw_node_activation *a = c->driver_activation;
if (c->xrun_count != a->xrun_count && if (c->xrun_count != a->xrun_count &&
c->xrun_count != 0 && c->xrun_callback) c->xrun_count != 0 && c->xrun_callback)
c->xrun_callback(c->xrun_arg); c->xrun_callback(c->xrun_arg);
@ -866,14 +873,14 @@ on_rtsocket_condition(void *data, int fd, uint32_t mask)
if (c->process_callback) if (c->process_callback)
c->process_callback(c->buffer_size, c->process_arg); c->process_callback(c->buffer_size, c->process_arg);
if (c->timebase_callback) { if (c->timebase_callback && a->segment_master[1] == c->node_id) {
c->timebase_callback(jack_state, c->timebase_callback(jack_state,
buffer_size, buffer_size,
&jack_position, &jack_position,
c->timebase_emit, c->timebase_new_pos,
c->timebase_arg); c->timebase_arg);
c->timebase_emit = false; c->timebase_new_pos = false;
debug_position(c, &jack_position); debug_position(c, &jack_position);
jack_to_position(&jack_position, pos); jack_to_position(&jack_position, pos);
@ -2076,7 +2083,8 @@ int jack_activate (jack_client_t *client)
pw_thread_loop_lock(c->context.loop); pw_thread_loop_lock(c->context.loop);
pw_client_node_proxy_set_active(c->node_proxy, true); pw_client_node_proxy_set_active(c->node_proxy, true);
c->timebase_emit = true; c->timebase_new_pos = true;
c->sync_emit = true;
res = do_sync(c); res = do_sync(c);
@ -2097,7 +2105,8 @@ int jack_deactivate (jack_client_t *client)
pw_thread_loop_lock(c->context.loop); pw_thread_loop_lock(c->context.loop);
pw_client_node_proxy_set_active(c->node_proxy, false); pw_client_node_proxy_set_active(c->node_proxy, false);
c->timebase_emit = false; c->timebase_new_pos = false;
c->sync_emit = false;
res = do_sync(c); res = do_sync(c);
@ -3268,9 +3277,20 @@ SPA_EXPORT
int jack_release_timebase (jack_client_t *client) int jack_release_timebase (jack_client_t *client)
{ {
struct client *c = (struct client *) client; struct client *c = (struct client *) client;
struct pw_node_activation *a = c->driver_activation;
if (a == NULL)
return -EIO;
if (!ATOMIC_CAS(a->segment_master[1], c->node_id, 0))
return -EINVAL;
SPA_FLAG_UNSET(a->position.segments[0].valid, SPA_IO_SEGMENT_VALID_BAR);
c->timebase_callback = NULL; c->timebase_callback = NULL;
c->timebase_arg = NULL; c->timebase_arg = NULL;
c->timebase_emit = false; c->timebase_new_pos = false;
return 0; return 0;
} }
@ -3280,9 +3300,21 @@ int jack_set_sync_callback (jack_client_t *client,
void *arg) void *arg)
{ {
struct client *c = (struct client *) client; struct client *c = (struct client *) client;
struct pw_node_activation *a = c->driver_activation;
if (a == NULL)
return -EIO;
if (sync_callback && c->sync_callback == NULL) {
ATOMIC_INC(a->sync_total);
} else if (sync_callback == NULL && c->sync_callback) {
ATOMIC_DEC(a->sync_total);
}
c->sync_callback = sync_callback; c->sync_callback = sync_callback;
c->sync_arg = arg; c->sync_arg = arg;
c->sync_emit = true; c->sync_emit = true;
return 0; return 0;
} }
@ -3290,8 +3322,15 @@ SPA_EXPORT
int jack_set_sync_timeout (jack_client_t *client, int jack_set_sync_timeout (jack_client_t *client,
jack_time_t timeout) jack_time_t timeout)
{ {
pw_log_warn(NAME" %p: not implemented %lu", client, timeout); struct client *c = (struct client *) client;
return -ENOTSUP; struct pw_node_activation *a = c->driver_activation;
if (a == NULL)
return -EIO;
ATOMIC_STORE(a->sync_timeout, timeout);
return 0;
} }
SPA_EXPORT SPA_EXPORT
@ -3306,12 +3345,22 @@ int jack_set_timebase_callback (jack_client_t *client,
if (a == NULL) if (a == NULL)
return -EIO; return -EIO;
/* was ok */
if (ATOMIC_LOAD(a->segment_master[1]) == c->node_id)
return 0;
/* try to become master */
if (conditional) {
if (!ATOMIC_CAS(a->segment_master[1], 0, c->node_id))
return -EBUSY;
} else {
ATOMIC_STORE(a->segment_master[1], c->node_id);
}
c->timebase_callback = timebase_callback; c->timebase_callback = timebase_callback;
c->timebase_arg = arg; c->timebase_arg = arg;
c->timebase_emit = true; c->timebase_new_pos = true;
return 0; return 0;
} }
@ -3382,14 +3431,14 @@ int jack_transport_reposition (jack_client_t *client,
pw_log_debug("frame:%u", pos->frame); pw_log_debug("frame:%u", pos->frame);
do { do {
seq1 = SEQ_WRITE(&a->pending.seq); seq1 = SEQ_WRITE(a->pending.seq);
a->pending.change_mask |= PW_NODE_ACTIVATION_UPDATE_REPOSITION; a->pending.change_mask |= PW_NODE_ACTIVATION_UPDATE_REPOSITION;
a->pending.segment.flags = 0; a->pending.segment.flags = 0;
a->pending.segment.start = 0; a->pending.segment.start = 0;
a->pending.segment.duration = 0; a->pending.segment.duration = 0;
a->pending.segment.position = pos->frame; a->pending.segment.position = pos->frame;
a->pending.segment.rate = 1.0; a->pending.segment.rate = 1.0;
seq2 = SEQ_WRITE(&a->pending.seq); seq2 = SEQ_WRITE(a->pending.seq);
} while (!SEQ_WRITE_SUCCESS(seq1, seq2)); } while (!SEQ_WRITE_SUCCESS(seq1, seq2));
return 0; return 0;
@ -3400,10 +3449,10 @@ static void update_state(struct pw_node_activation *a, enum spa_io_position_stat
uint32_t seq1, seq2; uint32_t seq1, seq2;
do { do {
seq1 = SEQ_WRITE(&a->pending.seq); seq1 = SEQ_WRITE(a->pending.seq);
a->pending.change_mask |= PW_NODE_ACTIVATION_UPDATE_STATE; a->pending.change_mask |= PW_NODE_ACTIVATION_UPDATE_STATE;
a->pending.state = state; a->pending.state = state;
seq2 = SEQ_WRITE(&a->pending.seq); seq2 = SEQ_WRITE(a->pending.seq);
} while (!SEQ_WRITE_SUCCESS(seq1, seq2)); } while (!SEQ_WRITE_SUCCESS(seq1, seq2));
} }