mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	jack: update for sync
This commit is contained in:
		
							parent
							
								
									0c2c2c72d0
								
							
						
					
					
						commit
						b4243991b3
					
				
					 1 changed files with 134 additions and 79 deletions
				
			
		| 
						 | 
					@ -255,17 +255,16 @@ struct client {
 | 
				
			||||||
	void *portregistration_arg;
 | 
						void *portregistration_arg;
 | 
				
			||||||
	JackPortConnectCallback connect_callback;
 | 
						JackPortConnectCallback connect_callback;
 | 
				
			||||||
	void *connect_arg;
 | 
						void *connect_arg;
 | 
				
			||||||
 | 
						JackPortRenameCallback rename_callback;
 | 
				
			||||||
 | 
						void *rename_arg;
 | 
				
			||||||
	JackGraphOrderCallback graph_callback;
 | 
						JackGraphOrderCallback graph_callback;
 | 
				
			||||||
	void *graph_arg;
 | 
						void *graph_arg;
 | 
				
			||||||
	JackXRunCallback xrun_callback;
 | 
						JackXRunCallback xrun_callback;
 | 
				
			||||||
	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;
 | 
					 | 
				
			||||||
	JackTimebaseCallback timebase_callback;
 | 
						JackTimebaseCallback timebase_callback;
 | 
				
			||||||
	void *timebase_arg;
 | 
						void *timebase_arg;
 | 
				
			||||||
	unsigned int timebase_new_pos:1;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	struct spa_io_position *position;
 | 
						struct spa_io_position *position;
 | 
				
			||||||
	uint32_t sample_rate;
 | 
						uint32_t sample_rate;
 | 
				
			||||||
| 
						 | 
					@ -813,7 +812,8 @@ 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;
 | 
							struct pw_node_activation *activation = c->activation;
 | 
				
			||||||
 | 
							struct pw_node_activation *driver = 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);
 | 
				
			||||||
| 
						 | 
					@ -826,8 +826,8 @@ on_rtsocket_condition(void *data, int fd, uint32_t mask)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		nsec = pos->clock.nsec;
 | 
							nsec = pos->clock.nsec;
 | 
				
			||||||
		c->activation->status = PW_NODE_ACTIVATION_AWAKE;
 | 
							activation->status = PW_NODE_ACTIVATION_AWAKE;
 | 
				
			||||||
		c->activation->awake_time = nsec;
 | 
							activation->awake_time = nsec;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		buffer_size = pos->clock.duration;
 | 
							buffer_size = pos->clock.duration;
 | 
				
			||||||
		if (buffer_size != c->buffer_size) {
 | 
							if (buffer_size != c->buffer_size) {
 | 
				
			||||||
| 
						 | 
					@ -847,23 +847,16 @@ 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 (a) {
 | 
							if (driver) {
 | 
				
			||||||
			if (a->sync_version != c->sync_version) {
 | 
								if (activation->pending_sync) {
 | 
				
			||||||
				c->sync_emit = true;
 | 
									if (c->sync_callback == NULL ||
 | 
				
			||||||
				c->timebase_new_pos = true;
 | 
									    c->sync_callback(jack_state, &jack_position, c->sync_arg))
 | 
				
			||||||
				c->sync_version = a->sync_version;
 | 
										activation->pending_sync = false;
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			if (c->sync_emit) {
 | 
								if (c->xrun_count != driver->xrun_count &&
 | 
				
			||||||
				if (c->sync_callback &&
 | 
					 | 
				
			||||||
				    c->sync_callback(jack_state, &jack_position, c->sync_arg)) {
 | 
					 | 
				
			||||||
					ATOMIC_DEC(a->sync_pending);
 | 
					 | 
				
			||||||
				}
 | 
					 | 
				
			||||||
				c->sync_emit = false;
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			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);
 | 
				
			||||||
			c->xrun_count = a->xrun_count;
 | 
								c->xrun_count = driver->xrun_count;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		pw_log_trace(NAME" %p: do process %"PRIu64" %d %d %d %"PRIi64" %f", c,
 | 
							pw_log_trace(NAME" %p: do process %"PRIu64" %d %d %d %"PRIi64" %f", c,
 | 
				
			||||||
| 
						 | 
					@ -873,24 +866,28 @@ 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 && a->segment_master[1] == c->node_id) {
 | 
							if (c->timebase_callback && driver && driver->segment_master[1] == c->node_id) {
 | 
				
			||||||
 | 
								if (activation->pending_new_pos ||
 | 
				
			||||||
 | 
								    jack_state == JackTransportRolling ||
 | 
				
			||||||
 | 
								    jack_state == JackTransportLooping) {
 | 
				
			||||||
				c->timebase_callback(jack_state,
 | 
									c->timebase_callback(jack_state,
 | 
				
			||||||
						     buffer_size,
 | 
											     buffer_size,
 | 
				
			||||||
						     &jack_position,
 | 
											     &jack_position,
 | 
				
			||||||
					     c->timebase_new_pos,
 | 
											     activation->pending_new_pos,
 | 
				
			||||||
						     c->timebase_arg);
 | 
											     c->timebase_arg);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			c->timebase_new_pos = false;
 | 
									activation->pending_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);
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
		process_tee(c);
 | 
							process_tee(c);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		clock_gettime(CLOCK_MONOTONIC, &ts);
 | 
							clock_gettime(CLOCK_MONOTONIC, &ts);
 | 
				
			||||||
		nsec = SPA_TIMESPEC_TO_NSEC(&ts);
 | 
							nsec = SPA_TIMESPEC_TO_NSEC(&ts);
 | 
				
			||||||
		c->activation->status = PW_NODE_ACTIVATION_FINISHED;
 | 
							activation->status = PW_NODE_ACTIVATION_FINISHED;
 | 
				
			||||||
		c->activation->finish_time = nsec;
 | 
							activation->finish_time = nsec;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		cmd = 1;
 | 
							cmd = 1;
 | 
				
			||||||
		pw_array_for_each(l, &c->links) {
 | 
							pw_array_for_each(l, &c->links) {
 | 
				
			||||||
| 
						 | 
					@ -985,7 +982,6 @@ static int client_node_set_param(void *object,
 | 
				
			||||||
static int update_driver_activation(struct client *c)
 | 
					static int update_driver_activation(struct client *c)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct link *link;
 | 
						struct link *link;
 | 
				
			||||||
 | 
					 | 
				
			||||||
	pw_log_debug(NAME" %p: driver %d", c, c->driver_id);
 | 
						pw_log_debug(NAME" %p: driver %d", c, c->driver_id);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	link = find_activation(&c->links, c->driver_id);
 | 
						link = find_activation(&c->links, c->driver_id);
 | 
				
			||||||
| 
						 | 
					@ -2078,44 +2074,51 @@ SPA_EXPORT
 | 
				
			||||||
int jack_activate (jack_client_t *client)
 | 
					int jack_activate (jack_client_t *client)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
	int res = 0;
 | 
						int res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (c->active)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	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_new_pos = true;
 | 
					 | 
				
			||||||
	c->sync_emit = true;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	res = do_sync(c);
 | 
						res = do_sync(c);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	pw_thread_loop_unlock(c->context.loop);
 | 
						pw_thread_loop_unlock(c->context.loop);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (res > 0)
 | 
						if (res < 0)
 | 
				
			||||||
		c->active = true;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return res;
 | 
							return res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						c->activation->pending_new_pos = true;
 | 
				
			||||||
 | 
						c->activation->pending_sync = true;
 | 
				
			||||||
 | 
						c->active = true;
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
int jack_deactivate (jack_client_t *client)
 | 
					int jack_deactivate (jack_client_t *client)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
	int res = 0;
 | 
						int res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!c->active)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	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_new_pos = false;
 | 
						c->activation->pending_new_pos = false;
 | 
				
			||||||
	c->sync_emit = false;
 | 
						c->activation->pending_sync = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	res = do_sync(c);
 | 
						res = do_sync(c);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	pw_thread_loop_unlock(c->context.loop);
 | 
						pw_thread_loop_unlock(c->context.loop);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (res > 0)
 | 
						if (res < 0)
 | 
				
			||||||
		c->active = false;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return res;
 | 
							return res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						c->active = false;
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
| 
						 | 
					@ -2191,18 +2194,28 @@ void jack_on_shutdown (jack_client_t *client,
 | 
				
			||||||
                       JackShutdownCallback shutdown_callback, void *arg)
 | 
					                       JackShutdownCallback shutdown_callback, void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
		c->shutdown_callback = shutdown_callback;
 | 
							c->shutdown_callback = shutdown_callback;
 | 
				
			||||||
		c->shutdown_arg = arg;
 | 
							c->shutdown_arg = arg;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
void jack_on_info_shutdown (jack_client_t *client,
 | 
					void jack_on_info_shutdown (jack_client_t *client,
 | 
				
			||||||
                            JackInfoShutdownCallback shutdown_callback, void *arg)
 | 
					                            JackInfoShutdownCallback shutdown_callback, void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
		c->info_shutdown_callback = shutdown_callback;
 | 
							c->info_shutdown_callback = shutdown_callback;
 | 
				
			||||||
		c->info_shutdown_arg = arg;
 | 
							c->info_shutdown_arg = arg;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
int jack_set_process_callback (jack_client_t *client,
 | 
					int jack_set_process_callback (jack_client_t *client,
 | 
				
			||||||
| 
						 | 
					@ -2231,6 +2244,10 @@ int jack_set_freewheel_callback (jack_client_t *client,
 | 
				
			||||||
                                 void *arg)
 | 
					                                 void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->freewheel_callback = freewheel_callback;
 | 
						c->freewheel_callback = freewheel_callback;
 | 
				
			||||||
	c->freewheel_arg = arg;
 | 
						c->freewheel_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2242,6 +2259,10 @@ int jack_set_buffer_size_callback (jack_client_t *client,
 | 
				
			||||||
                                   void *arg)
 | 
					                                   void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->bufsize_callback = bufsize_callback;
 | 
						c->bufsize_callback = bufsize_callback;
 | 
				
			||||||
	c->bufsize_arg = arg;
 | 
						c->bufsize_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2253,6 +2274,10 @@ int jack_set_sample_rate_callback (jack_client_t *client,
 | 
				
			||||||
                                   void *arg)
 | 
					                                   void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->srate_callback = srate_callback;
 | 
						c->srate_callback = srate_callback;
 | 
				
			||||||
	c->srate_arg = arg;
 | 
						c->srate_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2264,6 +2289,10 @@ int jack_set_client_registration_callback (jack_client_t *client,
 | 
				
			||||||
                                            registration_callback, void *arg)
 | 
					                                            registration_callback, void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->registration_callback = registration_callback;
 | 
						c->registration_callback = registration_callback;
 | 
				
			||||||
	c->registration_arg = arg;
 | 
						c->registration_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2275,6 +2304,10 @@ int jack_set_port_registration_callback (jack_client_t *client,
 | 
				
			||||||
                                          registration_callback, void *arg)
 | 
					                                          registration_callback, void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->portregistration_callback = registration_callback;
 | 
						c->portregistration_callback = registration_callback;
 | 
				
			||||||
	c->portregistration_arg = arg;
 | 
						c->portregistration_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2287,6 +2320,10 @@ int jack_set_port_connect_callback (jack_client_t *client,
 | 
				
			||||||
                                    connect_callback, void *arg)
 | 
					                                    connect_callback, void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->connect_callback = connect_callback;
 | 
						c->connect_callback = connect_callback;
 | 
				
			||||||
	c->connect_arg = arg;
 | 
						c->connect_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2294,9 +2331,16 @@ int jack_set_port_connect_callback (jack_client_t *client,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
int jack_set_port_rename_callback (jack_client_t *client,
 | 
					int jack_set_port_rename_callback (jack_client_t *client,
 | 
				
			||||||
                                   JackPortRenameCallback
 | 
					                                   JackPortRenameCallback rename_callback,
 | 
				
			||||||
                                   rename_callback, void *arg)
 | 
									   void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						c->rename_callback = rename_callback;
 | 
				
			||||||
 | 
						c->rename_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2306,6 +2350,10 @@ int jack_set_graph_order_callback (jack_client_t *client,
 | 
				
			||||||
                                   void *data)
 | 
					                                   void *data)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -1;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->graph_callback = graph_callback;
 | 
						c->graph_callback = graph_callback;
 | 
				
			||||||
	c->graph_arg = data;
 | 
						c->graph_arg = data;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2316,6 +2364,10 @@ int jack_set_xrun_callback (jack_client_t *client,
 | 
				
			||||||
                            JackXRunCallback xrun_callback, void *arg)
 | 
					                            JackXRunCallback xrun_callback, void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -1;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	c->xrun_callback = xrun_callback;
 | 
						c->xrun_callback = xrun_callback;
 | 
				
			||||||
	c->xrun_arg = arg;
 | 
						c->xrun_arg = arg;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -2326,6 +2378,11 @@ int jack_set_latency_callback (jack_client_t *client,
 | 
				
			||||||
			       JackLatencyCallback latency_callback,
 | 
								       JackLatencyCallback latency_callback,
 | 
				
			||||||
			       void *data)
 | 
								       void *data)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	pw_log_warn(NAME" %p: not implemented", client);
 | 
						pw_log_warn(NAME" %p: not implemented", client);
 | 
				
			||||||
	return -ENOTSUP;
 | 
						return -ENOTSUP;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -3289,7 +3346,7 @@ int jack_release_timebase (jack_client_t *client)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	c->timebase_callback = NULL;
 | 
						c->timebase_callback = NULL;
 | 
				
			||||||
	c->timebase_arg = NULL;
 | 
						c->timebase_arg = NULL;
 | 
				
			||||||
	c->timebase_new_pos = false;
 | 
						c->activation->pending_new_pos = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -3299,22 +3356,16 @@ int jack_set_sync_callback (jack_client_t *client,
 | 
				
			||||||
			    JackSyncCallback sync_callback,
 | 
								    JackSyncCallback sync_callback,
 | 
				
			||||||
			    void *arg)
 | 
								    void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						int res;
 | 
				
			||||||
	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;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if ((res = jack_activate(client)) < 0)
 | 
				
			||||||
 | 
							return res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						c->activation->pending_sync = true;
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3339,6 +3390,7 @@ int  jack_set_timebase_callback (jack_client_t *client,
 | 
				
			||||||
				 JackTimebaseCallback timebase_callback,
 | 
									 JackTimebaseCallback timebase_callback,
 | 
				
			||||||
				 void *arg)
 | 
									 void *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						int res;
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
	struct pw_node_activation *a = c->driver_activation;
 | 
						struct pw_node_activation *a = c->driver_activation;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3359,7 +3411,11 @@ int  jack_set_timebase_callback (jack_client_t *client,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	c->timebase_callback = timebase_callback;
 | 
						c->timebase_callback = timebase_callback;
 | 
				
			||||||
	c->timebase_arg = arg;
 | 
						c->timebase_arg = arg;
 | 
				
			||||||
	c->timebase_new_pos = true;
 | 
					
 | 
				
			||||||
 | 
						if ((res = jack_activate(client)) < 0)
 | 
				
			||||||
 | 
							return res;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						c->activation->pending_new_pos = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -3444,14 +3500,18 @@ int  jack_transport_reposition (jack_client_t *client,
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void update_state(struct pw_node_activation *a, enum spa_io_position_state state)
 | 
					static void update_command(struct client *c, uint32_t command)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct pw_node_activation *a = c->driver_activation;
 | 
				
			||||||
	uint32_t seq1, seq2;
 | 
						uint32_t seq1, seq2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!a)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	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_COMMAND;
 | 
				
			||||||
		a->pending.state = state;
 | 
							a->pending.command = command;
 | 
				
			||||||
		seq2 = SEQ_WRITE(a->pending.seq);
 | 
							seq2 = SEQ_WRITE(a->pending.seq);
 | 
				
			||||||
	} while (!SEQ_WRITE_SUCCESS(seq1, seq2));
 | 
						} while (!SEQ_WRITE_SUCCESS(seq1, seq2));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -3460,24 +3520,14 @@ SPA_EXPORT
 | 
				
			||||||
void jack_transport_start (jack_client_t *client)
 | 
					void jack_transport_start (jack_client_t *client)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
	struct pw_node_activation *a = c->driver_activation;
 | 
						update_command(c, PW_NODE_ACTIVATION_COMMAND_START);
 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!a)
 | 
					 | 
				
			||||||
		return;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	update_state(a, SPA_IO_POSITION_STATE_STARTING);
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
void jack_transport_stop (jack_client_t *client)
 | 
					void jack_transport_stop (jack_client_t *client)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct client *c = (struct client *) client;
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
	struct pw_node_activation *a = c->driver_activation;
 | 
						update_command(c, PW_NODE_ACTIVATION_COMMAND_STOP);
 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!a)
 | 
					 | 
				
			||||||
		return;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	update_state(a, SPA_IO_POSITION_STATE_STOPPED);
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SPA_EXPORT
 | 
					SPA_EXPORT
 | 
				
			||||||
| 
						 | 
					@ -3503,6 +3553,11 @@ int jack_set_session_callback (jack_client_t       *client,
 | 
				
			||||||
                               JackSessionCallback  session_callback,
 | 
					                               JackSessionCallback  session_callback,
 | 
				
			||||||
                               void                *arg)
 | 
					                               void                *arg)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct client *c = (struct client *) client;
 | 
				
			||||||
 | 
						if (c->active) {
 | 
				
			||||||
 | 
							pw_log_error(NAME" %p: can't set callback on active client", c);
 | 
				
			||||||
 | 
							return -EIO;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	pw_log_warn(NAME" %p: not implemented", client);
 | 
						pw_log_warn(NAME" %p: not implemented", client);
 | 
				
			||||||
	return -ENOTSUP;
 | 
						return -ENOTSUP;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue