mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-10-29 05:40:27 -04:00
Use the timer queue for scheduling stream and object data timeouts. This avoids allocating timerfds for these timeouts and the timer queue can handle many timeouts more efficiently.
1034 lines
24 KiB
C
1034 lines
24 KiB
C
/* PipeWire */
|
|
/* SPDX-FileCopyrightText: Copyright © 2020 Wim Taymans */
|
|
/* SPDX-License-Identifier: MIT */
|
|
|
|
#include "manager.h"
|
|
|
|
#include <spa/pod/iter.h>
|
|
#include <spa/pod/parser.h>
|
|
#include <spa/utils/result.h>
|
|
#include <spa/utils/string.h>
|
|
#include <pipewire/extensions/metadata.h>
|
|
|
|
#include "log.h"
|
|
#include "module-protocol-pulse/server.h"
|
|
|
|
#define manager_emit_sync(m) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, sync, 0)
|
|
#define manager_emit_added(m,o) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, added, 0, o)
|
|
#define manager_emit_updated(m,o) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, updated, 0, o)
|
|
#define manager_emit_removed(m,o) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, removed, 0, o)
|
|
#define manager_emit_metadata(m,o,s,k,t,v) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, metadata,0,o,s,k,t,v)
|
|
#define manager_emit_disconnect(m) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, disconnect, 0)
|
|
#define manager_emit_object_data_timeout(m,o,k) spa_hook_list_call(&(m)->hooks, struct pw_manager_events, object_data_timeout,0,o,k)
|
|
|
|
struct object;
|
|
|
|
struct manager {
|
|
struct pw_manager this;
|
|
|
|
struct pw_loop *loop;
|
|
struct pw_timer_queue *timer_queue;
|
|
|
|
struct spa_hook core_listener;
|
|
struct spa_hook registry_listener;
|
|
int sync_seq;
|
|
|
|
struct spa_hook_list hooks;
|
|
};
|
|
|
|
struct object_info {
|
|
const char *type;
|
|
uint32_t version;
|
|
const void *events;
|
|
void (*init) (struct object *object);
|
|
void (*destroy) (struct object *object);
|
|
};
|
|
|
|
struct object_data {
|
|
struct spa_list link;
|
|
struct object *object;
|
|
const char *key;
|
|
size_t size;
|
|
struct pw_timer timer;
|
|
};
|
|
|
|
struct object {
|
|
struct pw_manager_object this;
|
|
|
|
struct manager *manager;
|
|
|
|
const struct object_info *info;
|
|
|
|
int changed;
|
|
struct spa_list pending_list;
|
|
|
|
struct spa_hook proxy_listener;
|
|
struct spa_hook object_listener;
|
|
|
|
struct spa_list data_list;
|
|
};
|
|
|
|
static int core_sync(struct manager *m)
|
|
{
|
|
m->sync_seq = pw_core_sync(m->this.core, PW_ID_CORE, m->sync_seq);
|
|
pw_log_debug("sync start %u", m->sync_seq);
|
|
return m->sync_seq;
|
|
}
|
|
|
|
static uint32_t clear_params(struct spa_list *param_list, uint32_t id)
|
|
{
|
|
struct pw_manager_param *p, *t;
|
|
uint32_t count = 0;
|
|
|
|
spa_list_for_each_safe(p, t, param_list, link) {
|
|
if (id == SPA_ID_INVALID || p->id == id) {
|
|
spa_list_remove(&p->link);
|
|
free(p);
|
|
count++;
|
|
}
|
|
}
|
|
return count;
|
|
}
|
|
|
|
static struct pw_manager_param *add_param(struct spa_list *params,
|
|
int seq, uint32_t id, const struct spa_pod *param)
|
|
{
|
|
struct pw_manager_param *p;
|
|
|
|
if (id == SPA_ID_INVALID) {
|
|
if (param == NULL || !spa_pod_is_object(param)) {
|
|
errno = EINVAL;
|
|
return NULL;
|
|
}
|
|
id = SPA_POD_OBJECT_ID(param);
|
|
}
|
|
|
|
p = malloc(sizeof(*p) + (param != NULL ? SPA_POD_SIZE(param) : 0));
|
|
if (p == NULL)
|
|
return NULL;
|
|
|
|
p->id = id;
|
|
p->seq = seq;
|
|
if (param != NULL) {
|
|
p->param = SPA_PTROFF(p, sizeof(*p), struct spa_pod);
|
|
memcpy(p->param, param, SPA_POD_SIZE(param));
|
|
} else {
|
|
clear_params(params, id);
|
|
p->param = NULL;
|
|
}
|
|
spa_list_append(params, &p->link);
|
|
|
|
return p;
|
|
}
|
|
|
|
static bool has_param(struct spa_list *param_list, struct pw_manager_param *p)
|
|
{
|
|
struct pw_manager_param *t;
|
|
spa_list_for_each(t, param_list, link) {
|
|
if (p->id == t->id &&
|
|
SPA_POD_SIZE(p->param) == SPA_POD_SIZE(t->param) &&
|
|
memcmp(p->param, t->param, SPA_POD_SIZE(p->param)) == 0)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static struct object *find_object_by_id(struct manager *m, uint32_t id)
|
|
{
|
|
struct object *o;
|
|
spa_list_for_each(o, &m->this.object_list, this.link) {
|
|
if (o->this.id == id)
|
|
return o;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void object_update_params(struct object *o)
|
|
{
|
|
struct pw_manager_param *p, *t;
|
|
uint32_t i;
|
|
|
|
for (i = 0; i < o->this.n_params; i++) {
|
|
spa_list_for_each_safe(p, t, &o->pending_list, link) {
|
|
if (p->id == o->this.params[i].id &&
|
|
p->seq != o->this.params[i].seq &&
|
|
p->param != NULL) {
|
|
spa_list_remove(&p->link);
|
|
free(p);
|
|
}
|
|
}
|
|
}
|
|
|
|
spa_list_consume(p, &o->pending_list, link) {
|
|
spa_list_remove(&p->link);
|
|
if (p->param == NULL) {
|
|
clear_params(&o->this.param_list, p->id);
|
|
free(p);
|
|
} else {
|
|
spa_list_append(&o->this.param_list, &p->link);
|
|
}
|
|
}
|
|
}
|
|
static void object_reset_params(struct object *o)
|
|
{
|
|
uint32_t i;
|
|
for (i = 0; i < o->this.n_params; i++)
|
|
o->this.params[i].user = 0;
|
|
}
|
|
|
|
static void object_data_free(struct object_data *d)
|
|
{
|
|
spa_list_remove(&d->link);
|
|
pw_timer_queue_cancel(&d->timer);
|
|
free(d);
|
|
}
|
|
|
|
static void object_destroy(struct object *o)
|
|
{
|
|
struct manager *m = o->manager;
|
|
struct object_data *d;
|
|
spa_list_remove(&o->this.link);
|
|
m->this.n_objects--;
|
|
if (o->this.proxy)
|
|
pw_proxy_destroy(o->this.proxy);
|
|
pw_properties_free(o->this.props);
|
|
if (o->this.message_object_path)
|
|
free(o->this.message_object_path);
|
|
clear_params(&o->this.param_list, SPA_ID_INVALID);
|
|
clear_params(&o->pending_list, SPA_ID_INVALID);
|
|
spa_list_consume(d, &o->data_list, link)
|
|
object_data_free(d);
|
|
free(o);
|
|
}
|
|
|
|
/* core */
|
|
static const struct object_info core_info = {
|
|
.type = PW_TYPE_INTERFACE_Core,
|
|
.version = PW_VERSION_CORE,
|
|
};
|
|
|
|
/* client */
|
|
static void client_event_info(void *data, const struct pw_client_info *info)
|
|
{
|
|
struct object *o = data;
|
|
int changed = 0;
|
|
|
|
pw_log_debug("object %p: id:%d change-mask:%08"PRIx64, o, o->this.id, info->change_mask);
|
|
|
|
info = o->this.info = pw_client_info_merge(o->this.info, info, o->changed == 0);
|
|
if (info == NULL)
|
|
return;
|
|
|
|
if (info->change_mask & PW_CLIENT_CHANGE_MASK_PROPS)
|
|
changed++;
|
|
|
|
if (changed) {
|
|
o->changed += changed;
|
|
core_sync(o->manager);
|
|
}
|
|
}
|
|
|
|
static const struct pw_client_events client_events = {
|
|
PW_VERSION_CLIENT_EVENTS,
|
|
.info = client_event_info,
|
|
};
|
|
|
|
static void client_destroy(struct object *o)
|
|
{
|
|
if (o->this.info) {
|
|
pw_client_info_free(o->this.info);
|
|
o->this.info = NULL;
|
|
}
|
|
}
|
|
|
|
static const struct object_info client_info = {
|
|
.type = PW_TYPE_INTERFACE_Client,
|
|
.version = PW_VERSION_CLIENT,
|
|
.events = &client_events,
|
|
.destroy = client_destroy,
|
|
};
|
|
|
|
/* module */
|
|
static void module_event_info(void *data, const struct pw_module_info *info)
|
|
{
|
|
struct object *o = data;
|
|
int changed = 0;
|
|
|
|
pw_log_debug("object %p: id:%d change-mask:%08"PRIx64, o, o->this.id, info->change_mask);
|
|
|
|
info = o->this.info = pw_module_info_merge(o->this.info, info, o->changed == 0);
|
|
if (info == NULL)
|
|
return;
|
|
|
|
if (info->change_mask & PW_MODULE_CHANGE_MASK_PROPS)
|
|
changed++;
|
|
|
|
if (changed) {
|
|
o->changed += changed;
|
|
core_sync(o->manager);
|
|
}
|
|
}
|
|
|
|
static const struct pw_module_events module_events = {
|
|
PW_VERSION_MODULE_EVENTS,
|
|
.info = module_event_info,
|
|
};
|
|
|
|
static void module_destroy(struct object *o)
|
|
{
|
|
if (o->this.info) {
|
|
pw_module_info_free(o->this.info);
|
|
o->this.info = NULL;
|
|
}
|
|
}
|
|
|
|
static const struct object_info module_info = {
|
|
.type = PW_TYPE_INTERFACE_Module,
|
|
.version = PW_VERSION_MODULE,
|
|
.events = &module_events,
|
|
.destroy = module_destroy,
|
|
};
|
|
|
|
/* device */
|
|
static void device_event_info(void *data, const struct pw_device_info *info)
|
|
{
|
|
struct object *o = data;
|
|
uint32_t i, changed = 0;
|
|
bool enumerate = false;
|
|
|
|
pw_log_debug("object %p: id:%d change-mask:%08"PRIx64, o, o->this.id, info->change_mask);
|
|
|
|
info = o->this.info = pw_device_info_merge(o->this.info, info, o->changed == 0);
|
|
if (info == NULL)
|
|
return;
|
|
|
|
o->this.n_params = info->n_params;
|
|
o->this.params = info->params;
|
|
|
|
if (info->change_mask & PW_DEVICE_CHANGE_MASK_PROPS)
|
|
changed++;
|
|
|
|
if (info->change_mask & PW_DEVICE_CHANGE_MASK_PARAMS) {
|
|
for (i = 0; i < info->n_params; i++) {
|
|
uint32_t id = info->params[i].id;
|
|
int res;
|
|
|
|
if (info->params[i].user == 0)
|
|
continue;
|
|
|
|
switch (id) {
|
|
case SPA_PARAM_EnumProfile:
|
|
case SPA_PARAM_Profile:
|
|
case SPA_PARAM_EnumRoute:
|
|
changed++;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
enumerate = true;
|
|
|
|
add_param(&o->pending_list, info->params[i].seq, id, NULL);
|
|
if (!(info->params[i].flags & SPA_PARAM_INFO_READ))
|
|
continue;
|
|
|
|
res = pw_device_enum_params((struct pw_device*)o->this.proxy,
|
|
++info->params[i].seq, id, 0, -1, NULL);
|
|
if (SPA_RESULT_IS_ASYNC(res))
|
|
info->params[i].seq = res;
|
|
}
|
|
}
|
|
if (changed || enumerate) {
|
|
o->changed += changed;
|
|
core_sync(o->manager);
|
|
}
|
|
}
|
|
static struct object *find_device(struct manager *m, uint32_t card_id, uint32_t device)
|
|
{
|
|
struct object *o;
|
|
|
|
spa_list_for_each(o, &m->this.object_list, this.link) {
|
|
struct pw_node_info *info;
|
|
const char *str;
|
|
|
|
if (!spa_streq(o->this.type, PW_TYPE_INTERFACE_Node))
|
|
continue;
|
|
|
|
if ((info = o->this.info) != NULL &&
|
|
(str = spa_dict_lookup(info->props, PW_KEY_DEVICE_ID)) != NULL &&
|
|
(uint32_t)atoi(str) == card_id &&
|
|
(str = spa_dict_lookup(info->props, "card.profile.device")) != NULL &&
|
|
(uint32_t)atoi(str) == device)
|
|
return o;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void device_event_param(void *data, int seq,
|
|
uint32_t id, uint32_t index, uint32_t next,
|
|
const struct spa_pod *param)
|
|
{
|
|
struct object *o = data, *dev;
|
|
struct manager *m = o->manager;
|
|
struct pw_manager_param *p;
|
|
|
|
p = add_param(&o->pending_list, seq, id, param);
|
|
if (p == NULL)
|
|
return;
|
|
|
|
if ((id == SPA_PARAM_Route || id == SPA_PARAM_EnumRoute) &&
|
|
!has_param(&o->this.param_list, p)) {
|
|
uint32_t idx, device;
|
|
if (spa_pod_parse_object(param,
|
|
SPA_TYPE_OBJECT_ParamRoute, NULL,
|
|
SPA_PARAM_ROUTE_index, SPA_POD_Int(&idx),
|
|
SPA_PARAM_ROUTE_device, SPA_POD_Int(&device)) < 0)
|
|
return;
|
|
|
|
if ((dev = find_device(m, o->this.id, device)) != NULL) {
|
|
dev->changed++;
|
|
core_sync(o->manager);
|
|
}
|
|
}
|
|
}
|
|
|
|
static const struct pw_device_events device_events = {
|
|
PW_VERSION_DEVICE_EVENTS,
|
|
.info = device_event_info,
|
|
.param = device_event_param,
|
|
};
|
|
|
|
static void device_destroy(struct object *o)
|
|
{
|
|
if (o->this.info) {
|
|
pw_device_info_free(o->this.info);
|
|
o->this.info = NULL;
|
|
}
|
|
}
|
|
|
|
static const struct object_info device_info = {
|
|
.type = PW_TYPE_INTERFACE_Device,
|
|
.version = PW_VERSION_DEVICE,
|
|
.events = &device_events,
|
|
.destroy = device_destroy,
|
|
};
|
|
|
|
/* node */
|
|
static void node_event_info(void *data, const struct pw_node_info *info)
|
|
{
|
|
struct object *o = data;
|
|
uint32_t i, changed = 0;
|
|
bool enumerate = false;
|
|
|
|
pw_log_debug("object %p: id:%d change-mask:%08"PRIx64, o, o->this.id, info->change_mask);
|
|
|
|
info = o->this.info = pw_node_info_merge(o->this.info, info, o->changed == 0);
|
|
if (info == NULL)
|
|
return;
|
|
|
|
o->this.n_params = info->n_params;
|
|
o->this.params = info->params;
|
|
|
|
if (info->change_mask & PW_NODE_CHANGE_MASK_STATE)
|
|
changed++;
|
|
|
|
if (info->change_mask & PW_NODE_CHANGE_MASK_PROPS)
|
|
changed++;
|
|
|
|
if (info->change_mask & PW_NODE_CHANGE_MASK_PARAMS) {
|
|
for (i = 0; i < info->n_params; i++) {
|
|
uint32_t id = info->params[i].id;
|
|
int res;
|
|
|
|
if (info->params[i].user == 0)
|
|
continue;
|
|
|
|
switch (id) {
|
|
case SPA_PARAM_Props:
|
|
case SPA_PARAM_PropInfo:
|
|
case SPA_PARAM_Format:
|
|
case SPA_PARAM_EnumFormat:
|
|
/* also emit changed for the Latency param because the stream might
|
|
* now be linked. FIXME, we should check if a new link is made for
|
|
* a stream and only emit a changed event in that case. */
|
|
case SPA_PARAM_Latency:
|
|
changed++;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
enumerate = true;
|
|
|
|
add_param(&o->pending_list, info->params[i].seq, id, NULL);
|
|
if (!(info->params[i].flags & SPA_PARAM_INFO_READ))
|
|
continue;
|
|
|
|
res = pw_node_enum_params((struct pw_node*)o->this.proxy,
|
|
++info->params[i].seq, id, 0, -1, NULL);
|
|
if (SPA_RESULT_IS_ASYNC(res))
|
|
info->params[i].seq = res;
|
|
}
|
|
}
|
|
if (changed || enumerate) {
|
|
o->changed += changed;
|
|
core_sync(o->manager);
|
|
}
|
|
}
|
|
|
|
static void node_event_param(void *data, int seq,
|
|
uint32_t id, uint32_t index, uint32_t next,
|
|
const struct spa_pod *param)
|
|
{
|
|
struct object *o = data;
|
|
add_param(&o->pending_list, seq, id, param);
|
|
}
|
|
|
|
static const struct pw_node_events node_events = {
|
|
PW_VERSION_NODE_EVENTS,
|
|
.info = node_event_info,
|
|
.param = node_event_param,
|
|
};
|
|
|
|
static void node_destroy(struct object *o)
|
|
{
|
|
if (o->this.info) {
|
|
pw_node_info_free(o->this.info);
|
|
o->this.info = NULL;
|
|
}
|
|
}
|
|
|
|
static const struct object_info node_info = {
|
|
.type = PW_TYPE_INTERFACE_Node,
|
|
.version = PW_VERSION_NODE,
|
|
.events = &node_events,
|
|
.destroy = node_destroy,
|
|
};
|
|
|
|
/* link */
|
|
static const struct object_info link_info = {
|
|
.type = PW_TYPE_INTERFACE_Link,
|
|
.version = PW_VERSION_LINK,
|
|
};
|
|
|
|
/* metadata */
|
|
static int metadata_property(void *data,
|
|
uint32_t subject,
|
|
const char *key,
|
|
const char *type,
|
|
const char *value)
|
|
{
|
|
struct object *o = data;
|
|
struct manager *m = o->manager;
|
|
manager_emit_metadata(m, &o->this, subject, key, type, value);
|
|
return 0;
|
|
}
|
|
|
|
static const struct pw_metadata_events metadata_events = {
|
|
PW_VERSION_METADATA_EVENTS,
|
|
.property = metadata_property,
|
|
};
|
|
|
|
static void metadata_init(struct object *object)
|
|
{
|
|
struct object *o = object;
|
|
struct manager *m = o->manager;
|
|
o->this.creating = false;
|
|
manager_emit_added(m, &o->this);
|
|
}
|
|
|
|
static const struct object_info metadata_info = {
|
|
.type = PW_TYPE_INTERFACE_Metadata,
|
|
.version = PW_VERSION_METADATA,
|
|
.events = &metadata_events,
|
|
.init = metadata_init,
|
|
};
|
|
|
|
static const struct object_info *objects[] =
|
|
{
|
|
&core_info,
|
|
&module_info,
|
|
&client_info,
|
|
&device_info,
|
|
&node_info,
|
|
&link_info,
|
|
&metadata_info,
|
|
};
|
|
|
|
static const struct object_info *find_info(const char *type, uint32_t version)
|
|
{
|
|
SPA_FOR_EACH_ELEMENT_VAR(objects, i) {
|
|
if (spa_streq((*i)->type, type) &&
|
|
(*i)->version <= version)
|
|
return *i;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
destroy_removed(void *data)
|
|
{
|
|
struct object *o = data;
|
|
pw_proxy_destroy(o->this.proxy);
|
|
}
|
|
|
|
static void
|
|
destroy_proxy(void *data)
|
|
{
|
|
struct object *o = data;
|
|
|
|
spa_assert(o->info);
|
|
|
|
if (o->info->events)
|
|
spa_hook_remove(&o->object_listener);
|
|
spa_hook_remove(&o->proxy_listener);
|
|
|
|
if (o->info->destroy)
|
|
o->info->destroy(o);
|
|
|
|
o->this.proxy = NULL;
|
|
}
|
|
|
|
static const struct pw_proxy_events proxy_events = {
|
|
PW_VERSION_PROXY_EVENTS,
|
|
.removed = destroy_removed,
|
|
.destroy = destroy_proxy,
|
|
};
|
|
|
|
static void registry_event_global(void *data, uint32_t id,
|
|
uint32_t permissions, const char *type, uint32_t version,
|
|
const struct spa_dict *props)
|
|
{
|
|
struct manager *m = data;
|
|
struct object *o;
|
|
const struct object_info *info;
|
|
const char *str;
|
|
struct pw_proxy *proxy;
|
|
|
|
info = find_info(type, version);
|
|
if (info == NULL)
|
|
return;
|
|
|
|
proxy = pw_registry_bind(m->this.registry,
|
|
id, type, info->version, 0);
|
|
if (proxy == NULL)
|
|
return;
|
|
|
|
o = calloc(1, sizeof(*o));
|
|
if (o == NULL) {
|
|
pw_log_error("can't alloc object for %u %s/%d: %m", id, type, version);
|
|
pw_proxy_destroy(proxy);
|
|
return;
|
|
}
|
|
str = props ? spa_dict_lookup(props, PW_KEY_OBJECT_SERIAL) : NULL;
|
|
if (!spa_atou64(str, &o->this.serial, 0))
|
|
o->this.serial = SPA_ID_INVALID;
|
|
|
|
o->this.id = id;
|
|
o->this.permissions = permissions;
|
|
o->this.type = info->type;
|
|
o->this.version = version;
|
|
o->this.index = o->this.serial < (1ULL<<32) ? o->this.serial : SPA_ID_INVALID;
|
|
o->this.props = props ? pw_properties_new_dict(props) : NULL;
|
|
o->this.proxy = proxy;
|
|
o->this.creating = true;
|
|
spa_list_init(&o->this.param_list);
|
|
spa_list_init(&o->pending_list);
|
|
spa_list_init(&o->data_list);
|
|
|
|
o->manager = m;
|
|
o->info = info;
|
|
spa_list_append(&m->this.object_list, &o->this.link);
|
|
m->this.n_objects++;
|
|
|
|
if (info->events)
|
|
pw_proxy_add_object_listener(proxy,
|
|
&o->object_listener,
|
|
o->info->events, o);
|
|
pw_proxy_add_listener(proxy,
|
|
&o->proxy_listener,
|
|
&proxy_events, o);
|
|
|
|
if (info->init)
|
|
info->init(o);
|
|
|
|
core_sync(m);
|
|
}
|
|
|
|
static void registry_event_global_remove(void *data, uint32_t id)
|
|
{
|
|
struct manager *m = data;
|
|
struct object *o;
|
|
|
|
if ((o = find_object_by_id(m, id)) == NULL)
|
|
return;
|
|
|
|
o->this.removing = true;
|
|
|
|
if (!o->this.creating) {
|
|
o->this.change_mask = ~0;
|
|
manager_emit_removed(m, &o->this);
|
|
}
|
|
object_destroy(o);
|
|
}
|
|
|
|
static const struct pw_registry_events registry_events = {
|
|
PW_VERSION_REGISTRY_EVENTS,
|
|
.global = registry_event_global,
|
|
.global_remove = registry_event_global_remove,
|
|
};
|
|
|
|
static void on_core_info(void *data, const struct pw_core_info *info)
|
|
{
|
|
struct manager *m = data;
|
|
m->this.info = pw_core_info_merge(m->this.info, info, true);
|
|
}
|
|
|
|
static void on_core_done(void *data, uint32_t id, int seq)
|
|
{
|
|
struct manager *m = data;
|
|
struct object *o;
|
|
|
|
if (id == PW_ID_CORE) {
|
|
if (m->sync_seq != seq)
|
|
return;
|
|
|
|
pw_log_debug("sync end %u/%u", m->sync_seq, seq);
|
|
|
|
manager_emit_sync(m);
|
|
|
|
spa_list_for_each(o, &m->this.object_list, this.link)
|
|
object_update_params(o);
|
|
|
|
spa_list_for_each(o, &m->this.object_list, this.link) {
|
|
if (o->this.creating) {
|
|
o->this.creating = false;
|
|
manager_emit_added(m, &o->this);
|
|
o->changed = 0;
|
|
} else if (o->changed > 0) {
|
|
manager_emit_updated(m, &o->this);
|
|
o->changed = 0;
|
|
}
|
|
object_reset_params(o);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void on_core_error(void *data, uint32_t id, int seq, int res, const char *message)
|
|
{
|
|
struct manager *m = data;
|
|
|
|
if (id == PW_ID_CORE && res == -EPIPE) {
|
|
pw_log_debug("connection error: %d, %s", res, message);
|
|
manager_emit_disconnect(m);
|
|
}
|
|
}
|
|
|
|
static const struct pw_core_events core_events = {
|
|
PW_VERSION_CORE_EVENTS,
|
|
.done = on_core_done,
|
|
.info = on_core_info,
|
|
.error = on_core_error
|
|
};
|
|
|
|
struct pw_manager *pw_manager_new(struct pw_core *core)
|
|
{
|
|
struct manager *m;
|
|
struct pw_context *context;
|
|
|
|
m = calloc(1, sizeof(*m));
|
|
if (m == NULL)
|
|
return NULL;
|
|
|
|
m->this.core = core;
|
|
m->this.registry = pw_core_get_registry(m->this.core,
|
|
PW_VERSION_REGISTRY, 0);
|
|
if (m->this.registry == NULL) {
|
|
free(m);
|
|
return NULL;
|
|
}
|
|
|
|
context = pw_core_get_context(core);
|
|
m->loop = pw_context_get_main_loop(context);
|
|
m->timer_queue = pw_context_get_timer_queue(context);
|
|
|
|
spa_hook_list_init(&m->hooks);
|
|
|
|
spa_list_init(&m->this.object_list);
|
|
|
|
pw_core_add_listener(m->this.core,
|
|
&m->core_listener,
|
|
&core_events, m);
|
|
pw_registry_add_listener(m->this.registry,
|
|
&m->registry_listener,
|
|
®istry_events, m);
|
|
|
|
return &m->this;
|
|
}
|
|
|
|
void pw_manager_add_listener(struct pw_manager *manager,
|
|
struct spa_hook *listener,
|
|
const struct pw_manager_events *events, void *data)
|
|
{
|
|
struct manager *m = SPA_CONTAINER_OF(manager, struct manager, this);
|
|
spa_hook_list_append(&m->hooks, listener, events, data);
|
|
core_sync(m);
|
|
}
|
|
|
|
int pw_manager_set_metadata(struct pw_manager *manager,
|
|
struct pw_manager_object *metadata,
|
|
uint32_t subject, const char *key, const char *type,
|
|
const char *format, ...)
|
|
{
|
|
struct manager *m = SPA_CONTAINER_OF(manager, struct manager, this);
|
|
struct object *s;
|
|
va_list args;
|
|
char buf[1024];
|
|
char *value;
|
|
|
|
if ((s = find_object_by_id(m, subject)) == NULL)
|
|
return -ENOENT;
|
|
if (!SPA_FLAG_IS_SET(s->this.permissions, PW_PERM_M))
|
|
return -EACCES;
|
|
|
|
if (metadata == NULL)
|
|
return -ENOTSUP;
|
|
if (!SPA_FLAG_IS_SET(metadata->permissions, PW_PERM_W|PW_PERM_X))
|
|
return -EACCES;
|
|
if (metadata->proxy == NULL)
|
|
return -ENOENT;
|
|
|
|
if (type != NULL) {
|
|
va_start(args, format);
|
|
vsnprintf(buf, sizeof(buf), format, args);
|
|
va_end(args);
|
|
value = buf;
|
|
} else {
|
|
spa_assert(format == NULL);
|
|
value = NULL;
|
|
}
|
|
|
|
pw_metadata_set_property((struct pw_metadata*)metadata->proxy,
|
|
subject, key, type, value);
|
|
return 0;
|
|
}
|
|
|
|
int pw_manager_for_each_object(struct pw_manager *manager,
|
|
int (*callback) (void *data, struct pw_manager_object *object),
|
|
void *data)
|
|
{
|
|
struct manager *m = SPA_CONTAINER_OF(manager, struct manager, this);
|
|
struct object *o;
|
|
int res;
|
|
|
|
spa_list_for_each(o, &m->this.object_list, this.link) {
|
|
if (o->this.creating || o->this.removing)
|
|
continue;
|
|
if ((res = callback(data, &o->this)) != 0)
|
|
return res;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void pw_manager_destroy(struct pw_manager *manager)
|
|
{
|
|
struct manager *m = SPA_CONTAINER_OF(manager, struct manager, this);
|
|
struct object *o;
|
|
|
|
spa_hook_list_clean(&m->hooks);
|
|
|
|
spa_hook_remove(&m->core_listener);
|
|
|
|
spa_list_consume(o, &m->this.object_list, this.link)
|
|
object_destroy(o);
|
|
|
|
spa_hook_remove(&m->registry_listener);
|
|
pw_proxy_destroy((struct pw_proxy*)m->this.registry);
|
|
|
|
if (m->this.info)
|
|
pw_core_info_free(m->this.info);
|
|
|
|
free(m);
|
|
}
|
|
|
|
static struct object_data *object_find_data(struct object *o, const char *key)
|
|
{
|
|
struct object_data *d;
|
|
spa_list_for_each(d, &o->data_list, link) {
|
|
if (spa_streq(d->key, key))
|
|
return d;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void *pw_manager_object_add_data(struct pw_manager_object *obj, const char *key, size_t size)
|
|
{
|
|
struct object *o = SPA_CONTAINER_OF(obj, struct object, this);
|
|
struct object_data *d;
|
|
|
|
d = object_find_data(o, key);
|
|
if (d != NULL) {
|
|
if (d->size == size)
|
|
goto done;
|
|
object_data_free(d);
|
|
}
|
|
|
|
d = calloc(1, sizeof(struct object_data) + size);
|
|
if (d == NULL)
|
|
return NULL;
|
|
|
|
d->object = o;
|
|
d->key = key;
|
|
d->size = size;
|
|
|
|
spa_list_append(&o->data_list, &d->link);
|
|
|
|
done:
|
|
return SPA_PTROFF(d, sizeof(struct object_data), void);
|
|
}
|
|
|
|
static void object_data_timeout(void *data)
|
|
{
|
|
struct object_data *d = data;
|
|
struct object *o = d->object;
|
|
struct manager *m = o->manager;
|
|
|
|
pw_log_debug("manager:%p object id:%d data '%s' lifetime ends",
|
|
m, o->this.id, d->key);
|
|
|
|
manager_emit_object_data_timeout(m, &o->this, d->key);
|
|
}
|
|
|
|
void *pw_manager_object_add_temporary_data(struct pw_manager_object *obj, const char *key,
|
|
size_t size, uint64_t lifetime_nsec)
|
|
{
|
|
struct object *o = SPA_CONTAINER_OF(obj, struct object, this);
|
|
struct object_data *d;
|
|
void *data;
|
|
|
|
data = pw_manager_object_add_data(obj, key, size);
|
|
if (data == NULL)
|
|
return NULL;
|
|
|
|
d = SPA_PTROFF(data, -sizeof(struct object_data), void);
|
|
|
|
pw_timer_queue_add(o->manager->timer_queue, &d->timer, NULL,
|
|
lifetime_nsec, object_data_timeout, d);
|
|
|
|
return data;
|
|
}
|
|
|
|
void *pw_manager_object_get_data(struct pw_manager_object *obj, const char *id)
|
|
{
|
|
struct object *o = SPA_CONTAINER_OF(obj, struct object, this);
|
|
struct object_data *d = object_find_data(o, id);
|
|
|
|
return d ? SPA_PTROFF(d, sizeof(*d), void) : NULL;
|
|
}
|
|
|
|
int pw_manager_sync(struct pw_manager *manager)
|
|
{
|
|
struct manager *m = SPA_CONTAINER_OF(manager, struct manager, this);
|
|
return core_sync(m);
|
|
}
|
|
|
|
bool pw_manager_object_is_client(struct pw_manager_object *o)
|
|
{
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Client);
|
|
}
|
|
|
|
bool pw_manager_object_is_module(struct pw_manager_object *o)
|
|
{
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Module);
|
|
}
|
|
|
|
bool pw_manager_object_is_card(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Device) &&
|
|
o->props != NULL &&
|
|
(str = pw_properties_get(o->props, PW_KEY_MEDIA_CLASS)) != NULL &&
|
|
spa_streq(str, "Audio/Device");
|
|
}
|
|
|
|
bool pw_manager_object_is_sink(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
o->props != NULL &&
|
|
(str = pw_properties_get(o->props, PW_KEY_MEDIA_CLASS)) != NULL &&
|
|
(spa_streq(str, "Audio/Sink") || spa_streq(str, "Audio/Duplex"));
|
|
}
|
|
|
|
bool pw_manager_object_is_source(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
o->props != NULL &&
|
|
(str = pw_properties_get(o->props, PW_KEY_MEDIA_CLASS)) != NULL &&
|
|
(spa_streq(str, "Audio/Source") ||
|
|
spa_streq(str, "Audio/Duplex") ||
|
|
spa_streq(str, "Audio/Source/Virtual"));
|
|
}
|
|
|
|
bool pw_manager_object_is_monitor(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
o->props != NULL &&
|
|
(str = pw_properties_get(o->props, PW_KEY_MEDIA_CLASS)) != NULL &&
|
|
(spa_streq(str, "Audio/Sink"));
|
|
}
|
|
|
|
bool pw_manager_object_is_virtual(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
struct pw_node_info *info;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
(info = o->info) != NULL && info->props != NULL &&
|
|
(str = spa_dict_lookup(info->props, PW_KEY_NODE_VIRTUAL)) != NULL &&
|
|
pw_properties_parse_bool(str);
|
|
}
|
|
|
|
bool pw_manager_object_is_network(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
struct pw_node_info *info;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
(info = o->info) != NULL && info->props != NULL &&
|
|
(str = spa_dict_lookup(info->props, PW_KEY_NODE_NETWORK)) != NULL &&
|
|
pw_properties_parse_bool(str);
|
|
}
|
|
|
|
bool pw_manager_object_is_source_or_monitor(struct pw_manager_object *o)
|
|
{
|
|
return pw_manager_object_is_source(o) || pw_manager_object_is_monitor(o);
|
|
}
|
|
|
|
bool pw_manager_object_is_sink_input(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
o->props != NULL &&
|
|
(str = pw_properties_get(o->props, PW_KEY_MEDIA_CLASS)) != NULL &&
|
|
spa_streq(str, "Stream/Output/Audio");
|
|
}
|
|
|
|
bool pw_manager_object_is_source_output(struct pw_manager_object *o)
|
|
{
|
|
const char *str;
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Node) &&
|
|
o->props != NULL &&
|
|
(str = pw_properties_get(o->props, PW_KEY_MEDIA_CLASS)) != NULL &&
|
|
spa_streq(str, "Stream/Input/Audio");
|
|
}
|
|
|
|
bool pw_manager_object_is_recordable(struct pw_manager_object *o)
|
|
{
|
|
return pw_manager_object_is_source(o) || pw_manager_object_is_sink(o) || pw_manager_object_is_sink_input(o);
|
|
}
|
|
|
|
bool pw_manager_object_is_link(struct pw_manager_object *o)
|
|
{
|
|
return spa_streq(o->type, PW_TYPE_INTERFACE_Link);
|
|
}
|