pulse: track loaded modules

Track the proxies we load in a module_info and free the proxy
on unload or exit.
This commit is contained in:
Wim Taymans 2020-08-21 15:40:19 +02:00
parent 4370675d2b
commit 0b588b2bfd
3 changed files with 62 additions and 5 deletions

View file

@ -64,6 +64,7 @@ static void context_unlink(pa_context *c)
pa_stream *s, *t; pa_stream *s, *t;
struct global *g; struct global *g;
pa_operation *o; pa_operation *o;
struct module_info *m;
pw_log_debug("context %p: unlink %d", c, c->state); pw_log_debug("context %p: unlink %d", c, c->state);
@ -88,6 +89,8 @@ static void context_unlink(pa_context *c)
spa_list_consume(o, &c->operations, link) spa_list_consume(o, &c->operations, link)
pa_operation_cancel(o); pa_operation_cancel(o);
spa_list_consume(m, &c->modules, link)
pw_proxy_destroy(m->proxy);
} }
void pa_context_set_state(pa_context *c, pa_context_state_t st) { void pa_context_set_state(pa_context *c, pa_context_state_t st) {
@ -1489,6 +1492,7 @@ pa_context *pa_context_new_with_proplist(pa_mainloop_api *mainloop, const char *
spa_list_init(&c->streams); spa_list_init(&c->streams);
spa_list_init(&c->operations); spa_list_init(&c->operations);
spa_list_init(&c->modules);
return c; return c;
} }

View file

@ -325,6 +325,13 @@ struct global {
}; };
}; };
struct module_info {
struct spa_list link; /* link in context modules */
uint32_t id;
struct pw_proxy *proxy;
struct spa_hook listener;
};
struct pa_context { struct pa_context {
int refcount; int refcount;
uint32_t client_index; uint32_t client_index;
@ -359,6 +366,7 @@ struct pa_context {
struct spa_list streams; struct spa_list streams;
struct spa_list operations; struct spa_list operations;
struct spa_list modules;
int no_fail:1; int no_fail:1;
int disconnect:1; int disconnect:1;

View file

@ -125,7 +125,7 @@ static int sink_callback(pa_context *c, struct global *g, struct sink_data *d)
i.channel_map = g->node_info.channel_map; i.channel_map = g->node_info.channel_map;
else else
pa_channel_map_init_auto(&i.channel_map, i.sample_spec.channels, PA_CHANNEL_MAP_OSS); pa_channel_map_init_auto(&i.channel_map, i.sample_spec.channels, PA_CHANNEL_MAP_OSS);
i.owner_module = 0; i.owner_module = g->id;
i.volume.channels = i.sample_spec.channels; i.volume.channels = i.sample_spec.channels;
for (n = 0; n < i.volume.channels; n++) for (n = 0; n < i.volume.channels; n++)
i.volume.values[n] = g->node_info.volume * g->node_info.channel_volumes[n] * PA_VOLUME_NORM; i.volume.values[n] = g->node_info.volume * g->node_info.channel_volumes[n] * PA_VOLUME_NORM;
@ -847,7 +847,7 @@ static int source_callback(pa_context *c, struct global *g, struct source_data *
i.channel_map = g->node_info.channel_map; i.channel_map = g->node_info.channel_map;
else else
pa_channel_map_init_auto(&i.channel_map, i.sample_spec.channels, PA_CHANNEL_MAP_OSS); pa_channel_map_init_auto(&i.channel_map, i.sample_spec.channels, PA_CHANNEL_MAP_OSS);
i.owner_module = 0; i.owner_module = g->id;
i.volume.channels = i.sample_spec.channels; i.volume.channels = i.sample_spec.channels;
for (n = 0; n < i.volume.channels; n++) for (n = 0; n < i.volume.channels; n++)
i.volume.values[n] = g->node_info.volume * g->node_info.channel_volumes[n] * PA_VOLUME_NORM; i.volume.values[n] = g->node_info.volume * g->node_info.channel_volumes[n] * PA_VOLUME_NORM;
@ -1402,6 +1402,16 @@ struct load_module {
struct spa_hook listener; struct spa_hook listener;
}; };
static struct module_info *find_module(pa_context *c, uint32_t idx)
{
struct module_info *m;
spa_list_for_each(m, &c->modules, link) {
if (m->id == idx)
return m;
}
return NULL;
}
static void on_load_module(pa_operation *o, void *userdata) static void on_load_module(pa_operation *o, void *userdata)
{ {
struct load_module *d = userdata; struct load_module *d = userdata;
@ -1417,11 +1427,37 @@ static void on_load_module(pa_operation *o, void *userdata)
spa_hook_remove(&d->listener); spa_hook_remove(&d->listener);
} }
static void module_proxy_removed(void *data)
{
struct module_info *m = data;
pw_proxy_destroy(m->proxy);
}
static void module_proxy_destroy(void *data)
{
struct module_info *m = data;
spa_hook_remove(&m->listener);
spa_list_remove(&m->link);
free(m);
}
static void module_proxy_bound(void *data, uint32_t global_id) static void module_proxy_bound(void *data, uint32_t global_id)
{ {
struct module_info *m;
pa_operation *o = data; pa_operation *o = data;
pa_context *c = o->context;
struct load_module *d = o->userdata; struct load_module *d = o->userdata;
static const struct pw_proxy_events proxy_events = {
.removed = module_proxy_removed,
.destroy = module_proxy_destroy,
};
d->idx = global_id; d->idx = global_id;
m = calloc(1, sizeof(struct module_info));
m->id = global_id;
m->proxy = d->proxy;
pw_proxy_add_listener(m->proxy, &m->listener, &proxy_events, m);
spa_list_append(&c->modules, &m->link);
on_load_module(o, d); on_load_module(o, d);
} }
@ -1431,6 +1467,7 @@ static void module_proxy_error(void *data, int seq, int res, const char *message
struct load_module *d = o->userdata; struct load_module *d = o->userdata;
d->error = res; d->error = res;
d->idx = PA_INVALID_INDEX; d->idx = PA_INVALID_INDEX;
pw_proxy_destroy(d->proxy);
on_load_module(o, d); on_load_module(o, d);
} }
@ -1544,13 +1581,21 @@ pa_operation* pa_context_unload_module(pa_context *c, uint32_t idx, pa_context_s
{ {
pa_operation *o; pa_operation *o;
struct success_ack *d; struct success_ack *d;
struct module_info *m;
int error;
pw_log_debug("context %p: %u", c, idx); pw_log_debug("context %p: %u", c, idx);
if ((m = find_module(c, idx)) != NULL) {
pw_proxy_destroy(m->proxy);
error = 0;
} else {
error = PA_ERR_NOENTITY;
}
o = pa_operation_new(c, NULL, on_success, sizeof(struct success_ack)); o = pa_operation_new(c, NULL, on_success, sizeof(struct success_ack));
d = o->userdata; d = o->userdata;
d->cb = cb; d->cb = cb;
d->error = PA_ERR_NOTIMPLEMENTED;
d->userdata = userdata; d->userdata = userdata;
d->error = error;
d->idx = idx; d->idx = idx;
pa_operation_sync(o); pa_operation_sync(o);
@ -1986,7 +2031,7 @@ static int sink_input_callback(pa_context *c, struct sink_input_data *d, struct
spa_zero(i); spa_zero(i);
i.index = g->id; i.index = g->id;
i.name = name; i.name = name;
i.owner_module = PA_INVALID_INDEX; i.owner_module = g->id;
i.client = g->node_info.client_id; i.client = g->node_info.client_id;
if (s) if (s)
i.sink = s->device_index; i.sink = s->device_index;
@ -2387,7 +2432,7 @@ static int source_output_callback(struct source_output_data *d, pa_context *c, s
spa_zero(i); spa_zero(i);
i.index = g->id; i.index = g->id;
i.name = name; i.name = name;
i.owner_module = PA_INVALID_INDEX; i.owner_module = g->id;
i.client = g->node_info.client_id; i.client = g->node_info.client_id;
if (s) if (s)
i.source = s->device_index; i.source = s->device_index;