mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	alsa-seq: unsubscribe when paused/suspended
When we are suspended or paused, unsubscribe from the ports so that we don't block the hardware devices. See #225
This commit is contained in:
		
							parent
							
								
									9b24a84ce6
								
							
						
					
					
						commit
						e6675ff2a8
					
				
					 3 changed files with 58 additions and 27 deletions
				
			
		| 
						 | 
					@ -374,18 +374,16 @@ static struct seq_port *alloc_port(struct seq_state *state, struct seq_stream *s
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return NULL;
 | 
						return NULL;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void free_port(struct seq_state *state, struct seq_port *port)
 | 
					static void free_port(struct seq_state *state, struct seq_port *port)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	spa_node_emit_port_info(&state->hooks,
 | 
						spa_node_emit_port_info(&state->hooks,
 | 
				
			||||||
			port->direction, port->id, NULL);
 | 
								port->direction, port->id, NULL);
 | 
				
			||||||
	port->valid = false;
 | 
						spa_zero(*port);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void init_port(struct seq_state *state, struct seq_port *port, const snd_seq_addr_t *addr)
 | 
					static void init_port(struct seq_state *state, struct seq_port *port, const snd_seq_addr_t *addr)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	snd_seq_port_subscribe_t* sub;
 | 
					 | 
				
			||||||
	int res;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	port->addr = *addr;
 | 
						port->addr = *addr;
 | 
				
			||||||
	port->info_all = SPA_PORT_CHANGE_MASK_FLAGS |
 | 
						port->info_all = SPA_PORT_CHANGE_MASK_FLAGS |
 | 
				
			||||||
			SPA_PORT_CHANGE_MASK_PROPS |
 | 
								SPA_PORT_CHANGE_MASK_PROPS |
 | 
				
			||||||
| 
						 | 
					@ -406,23 +404,7 @@ static void init_port(struct seq_state *state, struct seq_port *port, const snd_
 | 
				
			||||||
	spa_list_init(&port->free);
 | 
						spa_list_init(&port->free);
 | 
				
			||||||
	spa_list_init(&port->ready);
 | 
						spa_list_init(&port->ready);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	snd_seq_port_subscribe_alloca(&sub);
 | 
						spa_alsa_seq_activate_port(state, port, true);
 | 
				
			||||||
	if (port->direction == SPA_DIRECTION_OUTPUT) {
 | 
					 | 
				
			||||||
		snd_seq_port_subscribe_set_sender(sub, addr);
 | 
					 | 
				
			||||||
		snd_seq_port_subscribe_set_dest(sub, &state->event.addr);
 | 
					 | 
				
			||||||
	} else {
 | 
					 | 
				
			||||||
		snd_seq_port_subscribe_set_sender(sub, &state->event.addr);
 | 
					 | 
				
			||||||
		snd_seq_port_subscribe_set_dest(sub, addr);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	snd_seq_port_subscribe_set_time_update(sub, 1);
 | 
					 | 
				
			||||||
	snd_seq_port_subscribe_set_time_real(sub, 1);
 | 
					 | 
				
			||||||
	snd_seq_port_subscribe_set_queue(sub, state->event.queue_id);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if ((res = snd_seq_subscribe_port(state->event.hndl, sub)) < 0) {
 | 
					 | 
				
			||||||
                spa_log_error(state->log, "can't subscribe to %d:%d - %s",
 | 
					 | 
				
			||||||
				addr->client, addr->port, snd_strerror(res));
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	spa_log_debug(state->log, "connect: %d.%d: %d", addr->client, addr->port, res);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	emit_port_info(state, port, true);
 | 
						emit_port_info(state, port, true);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -376,6 +376,48 @@ static struct seq_port *find_port(struct seq_state *state,
 | 
				
			||||||
	return NULL;
 | 
						return NULL;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int spa_alsa_seq_activate_port(struct seq_state *state, struct seq_port *port, bool active)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int res;
 | 
				
			||||||
 | 
						snd_seq_port_subscribe_t* sub;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spa_log_debug(state->log, "activate: %d.%d: started:%d active:%d wanted:%d",
 | 
				
			||||||
 | 
								port->addr.client, port->addr.port, state->started, port->active, active);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (active && !state->started)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
						if (port->active == active)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						snd_seq_port_subscribe_alloca(&sub);
 | 
				
			||||||
 | 
						if (port->direction == SPA_DIRECTION_OUTPUT) {
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_sender(sub, &port->addr);
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_dest(sub, &state->event.addr);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_sender(sub, &state->event.addr);
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_dest(sub, &port->addr);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (active) {
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_time_update(sub, 1);
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_time_real(sub, 1);
 | 
				
			||||||
 | 
							snd_seq_port_subscribe_set_queue(sub, state->event.queue_id);
 | 
				
			||||||
 | 
							if ((res = snd_seq_subscribe_port(state->event.hndl, sub)) < 0) {
 | 
				
			||||||
 | 
								spa_log_error(state->log, "can't subscribe to %d:%d - %s",
 | 
				
			||||||
 | 
									port->addr.client, port->addr.port, snd_strerror(res));
 | 
				
			||||||
 | 
								active = false;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							if ((res = snd_seq_unsubscribe_port(state->event.hndl, sub)) < 0) {
 | 
				
			||||||
 | 
								spa_log_warn(state->log, "can't unsubscribe from %d:%d - %s",
 | 
				
			||||||
 | 
									port->addr.client, port->addr.port, snd_strerror(res));
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						spa_log_info(state->log, "activate: %d.%d: %d", port->addr.client, port->addr.port, active);
 | 
				
			||||||
 | 
						port->active = active;
 | 
				
			||||||
 | 
						return res;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct buffer *peek_buffer(struct seq_state *state,
 | 
					static struct buffer *peek_buffer(struct seq_state *state,
 | 
				
			||||||
		struct seq_port *port)
 | 
							struct seq_port *port)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -774,13 +816,15 @@ static void reset_buffers(struct seq_state *this, struct seq_port *port)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
static void reset_stream(struct seq_state *this, struct seq_stream *stream)
 | 
					static void reset_stream(struct seq_state *this, struct seq_stream *stream, bool active)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	uint32_t i;
 | 
						uint32_t i;
 | 
				
			||||||
	for (i = 0; i < MAX_PORTS; i++) {
 | 
						for (i = 0; i < MAX_PORTS; i++) {
 | 
				
			||||||
		struct seq_port *port = &stream->ports[i];
 | 
							struct seq_port *port = &stream->ports[i];
 | 
				
			||||||
		if (port->valid)
 | 
							if (port->valid) {
 | 
				
			||||||
			reset_buffers(this, port);
 | 
								reset_buffers(this, port);
 | 
				
			||||||
 | 
								spa_alsa_seq_activate_port(this, port, active);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -825,8 +869,10 @@ int spa_alsa_seq_start(struct seq_state *state)
 | 
				
			||||||
		state->threshold = state->duration;
 | 
							state->threshold = state->duration;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	reset_stream(state, &state->streams[SPA_DIRECTION_INPUT]);
 | 
						state->started = true;
 | 
				
			||||||
	reset_stream(state, &state->streams[SPA_DIRECTION_OUTPUT]);
 | 
					
 | 
				
			||||||
 | 
						reset_stream(state, &state->streams[SPA_DIRECTION_INPUT], true);
 | 
				
			||||||
 | 
						reset_stream(state, &state->streams[SPA_DIRECTION_OUTPUT], true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	state->source.func = alsa_on_timeout_event;
 | 
						state->source.func = alsa_on_timeout_event;
 | 
				
			||||||
	state->source.data = state;
 | 
						state->source.data = state;
 | 
				
			||||||
| 
						 | 
					@ -839,8 +885,6 @@ int spa_alsa_seq_start(struct seq_state *state)
 | 
				
			||||||
	init_loop(state);
 | 
						init_loop(state);
 | 
				
			||||||
	set_timers(state);
 | 
						set_timers(state);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	state->started = true;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -900,5 +944,8 @@ int spa_alsa_seq_pause(struct seq_state *state)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	state->started = false;
 | 
						state->started = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						reset_stream(state, &state->streams[SPA_DIRECTION_INPUT], false);
 | 
				
			||||||
 | 
						reset_stream(state, &state->streams[SPA_DIRECTION_OUTPUT], false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -86,6 +86,7 @@ struct seq_port {
 | 
				
			||||||
	struct spa_audio_info current_format;
 | 
						struct spa_audio_info current_format;
 | 
				
			||||||
	unsigned int have_format:1;
 | 
						unsigned int have_format:1;
 | 
				
			||||||
	unsigned int valid:1;
 | 
						unsigned int valid:1;
 | 
				
			||||||
 | 
						unsigned int active:1;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct seq_stream {
 | 
					struct seq_stream {
 | 
				
			||||||
| 
						 | 
					@ -173,6 +174,7 @@ int spa_alsa_seq_start(struct seq_state *state);
 | 
				
			||||||
int spa_alsa_seq_pause(struct seq_state *state);
 | 
					int spa_alsa_seq_pause(struct seq_state *state);
 | 
				
			||||||
int spa_alsa_seq_reassign_follower(struct seq_state *state);
 | 
					int spa_alsa_seq_reassign_follower(struct seq_state *state);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int spa_alsa_seq_activate_port(struct seq_state *state, struct seq_port *port, bool active);
 | 
				
			||||||
int spa_alsa_seq_recycle_buffer(struct seq_state *state, struct seq_port *port, uint32_t buffer_id);
 | 
					int spa_alsa_seq_recycle_buffer(struct seq_state *state, struct seq_port *port, uint32_t buffer_id);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int spa_alsa_seq_process(struct seq_state *state);
 | 
					int spa_alsa_seq_process(struct seq_state *state);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue