node_factory -> factory

Rename the node-factory to a more generic object factory.
Add type and version to create objects from a factory
Start on factory introspection
This commit is contained in:
Wim Taymans 2017-09-17 16:47:03 +02:00
parent 3c1957fd9d
commit 7f20e04803
24 changed files with 472 additions and 325 deletions

View file

@ -48,10 +48,11 @@ struct data {
static int make_node(struct data *data) static int make_node(struct data *data)
{ {
struct pw_node_factory *factory; struct pw_factory *factory;
struct pw_type *t = data->t;
struct pw_properties *props; struct pw_properties *props;
factory = pw_core_find_node_factory(data->core, "spa-node-factory"); factory = pw_core_find_factory(data->core, "spa-node-factory");
if (factory == NULL) if (factory == NULL)
return -1; return -1;
@ -63,7 +64,11 @@ static int make_node(struct data *data)
pw_properties_set(props, "pipewire.target.node", data->path); pw_properties_set(props, "pipewire.target.node", data->path);
} }
data->node = pw_node_factory_create_node(factory, NULL, data->factory, props); data->node = pw_factory_create_object(factory,
NULL,
t->node,
PW_VERSION_NODE,
props, SPA_ID_INVALID);
pw_remote_export(data->remote, data->node); pw_remote_export(data->remote, data->node);

View file

@ -447,7 +447,7 @@ static const struct spa_node impl_node = {
static void make_nodes(struct data *data) static void make_nodes(struct data *data)
{ {
struct pw_node_factory *factory; struct pw_factory *factory;
struct pw_properties *props; struct pw_properties *props;
data->node = pw_node_new(data->core, NULL, NULL, "SDL-sink", NULL, 0); data->node = pw_node_new(data->core, NULL, NULL, "SDL-sink", NULL, 0);
@ -456,10 +456,15 @@ static void make_nodes(struct data *data)
pw_node_register(data->node); pw_node_register(data->node);
factory = pw_core_find_node_factory(data->core, "spa-node-factory"); factory = pw_core_find_factory(data->core, "spa-node-factory");
props = pw_properties_new("spa.library.name", "v4l2/libspa-v4l2", props = pw_properties_new("spa.library.name", "v4l2/libspa-v4l2",
"spa.factory.name", "v4l2-source", NULL); "spa.factory.name", "v4l2-source", NULL);
data->v4l2 = pw_node_factory_create_node(factory, NULL, "v4l2-source", props); data->v4l2 = pw_factory_create_object(factory,
NULL,
data->t->node,
PW_VERSION_NODE,
props,
SPA_ID_INVALID);
data->link = pw_link_new(data->core, data->link = pw_link_new(data->core,
NULL, NULL,

View file

@ -34,36 +34,54 @@
struct pw_protocol *pw_protocol_native_ext_client_node_init(struct pw_core *core); struct pw_protocol *pw_protocol_native_ext_client_node_init(struct pw_core *core);
struct factory_data { struct factory_data {
struct pw_node_factory *this; struct pw_factory *this;
struct pw_properties *properties; struct pw_properties *properties;
struct spa_hook module_listener; struct spa_hook module_listener;
uint32_t type_client_node;
}; };
static struct pw_node *create_node(void *_data, static void *create_object(void *_data,
struct pw_resource *resource, struct pw_resource *resource,
const char *name, uint32_t type,
struct pw_properties *properties) uint32_t version,
struct pw_properties *properties,
uint32_t new_id)
{ {
struct pw_client_node *node; struct pw_client_node *node;
struct pw_resource *node_resource;
node = pw_client_node_new(resource, name, properties); if (resource == NULL)
goto no_resource;
node_resource = pw_resource_new(pw_resource_get_client(resource),
new_id, PW_PERM_RWX, type, version, 0);
if (node_resource == NULL)
goto no_mem;
node = pw_client_node_new(node_resource, properties);
if (node == NULL) if (node == NULL)
goto no_mem; goto no_mem;
return node->node; return node;
no_resource:
pw_log_error("client-node needs a resource");
pw_resource_error(resource, SPA_RESULT_INVALID_ARGUMENTS, "no resource");
goto done;
no_mem: no_mem:
pw_log_error("can't create node"); pw_log_error("can't create node");
pw_resource_error(resource, SPA_RESULT_NO_MEMORY, "no memory"); pw_resource_error(resource, SPA_RESULT_NO_MEMORY, "no memory");
goto done;
done:
if (properties) if (properties)
pw_properties_free(properties); pw_properties_free(properties);
return NULL; return NULL;
} }
static const struct pw_node_factory_implementation impl_factory = { static const struct pw_factory_implementation impl_factory = {
PW_VERSION_NODE_FACRORY_IMPLEMENTATION, PW_VERSION_FACTORY_IMPLEMENTATION,
.create_node = create_node, .create_object = create_object,
}; };
static void module_destroy(void *data) static void module_destroy(void *data)
@ -75,7 +93,7 @@ static void module_destroy(void *data)
if (d->properties) if (d->properties)
pw_properties_free(d->properties); pw_properties_free(d->properties);
pw_node_factory_destroy(d->this); pw_factory_destroy(d->this);
} }
const struct pw_module_events module_events = { const struct pw_module_events module_events = {
@ -86,26 +104,36 @@ const struct pw_module_events module_events = {
static bool module_init(struct pw_module *module, struct pw_properties *properties) static bool module_init(struct pw_module *module, struct pw_properties *properties)
{ {
struct pw_core *core = pw_module_get_core(module); struct pw_core *core = pw_module_get_core(module);
struct pw_node_factory *factory; struct pw_type *t = pw_core_get_type(core);
struct pw_factory *factory;
struct factory_data *data; struct factory_data *data;
uint32_t type_client_node;
factory = pw_node_factory_new(core, "client-node", sizeof(*data)); type_client_node = spa_type_map_get_id(t->map, PW_TYPE_INTERFACE__ClientNode);
factory = pw_factory_new(core,
"client-node",
type_client_node,
PW_VERSION_CLIENT_NODE,
NULL,
sizeof(*data));
if (factory == NULL) if (factory == NULL)
return false; return false;
data = pw_node_factory_get_user_data(factory); data = pw_factory_get_user_data(factory);
data->this = factory; data->this = factory;
data->properties = properties; data->properties = properties;
data->type_client_node = type_client_node;
pw_log_debug("module %p: new", module); pw_log_debug("module %p: new", module);
pw_node_factory_set_implementation(factory, pw_factory_set_implementation(factory,
&impl_factory, &impl_factory,
data); data);
pw_protocol_native_ext_client_node_init(core); pw_protocol_native_ext_client_node_init(core);
pw_node_factory_export(factory, NULL, pw_module_get_global(module)); pw_factory_register(factory, NULL, pw_module_get_global(module));
pw_module_add_listener(module, &data->module_listener, &module_events, data); pw_module_add_listener(module, &data->module_listener, &module_events, data);

View file

@ -1133,7 +1133,6 @@ static const struct pw_resource_events resource_events = {
* \memberof pw_client_node * \memberof pw_client_node
*/ */
struct pw_client_node *pw_client_node_new(struct pw_resource *resource, struct pw_client_node *pw_client_node_new(struct pw_resource *resource,
const char *name,
struct pw_properties *properties) struct pw_properties *properties)
{ {
struct impl *impl; struct impl *impl;
@ -1142,7 +1141,7 @@ struct pw_client_node *pw_client_node_new(struct pw_resource *resource,
struct pw_core *core = pw_client_get_core(client); struct pw_core *core = pw_client_get_core(client);
const struct spa_support *support; const struct spa_support *support;
uint32_t n_support; uint32_t n_support;
const char *name = "client-node";
impl = calloc(1, sizeof(struct impl)); impl = calloc(1, sizeof(struct impl));
if (impl == NULL) if (impl == NULL)

View file

@ -39,7 +39,6 @@ struct pw_client_node {
struct pw_client_node * struct pw_client_node *
pw_client_node_new(struct pw_resource *resource, pw_client_node_new(struct pw_resource *resource,
const char *name,
struct pw_properties *properties); struct pw_properties *properties);
void void

View file

@ -77,7 +77,6 @@ struct async_pending {
bool handled; bool handled;
char *handle; char *handle;
char *factory_name; char *factory_name;
char *name;
uint32_t type; uint32_t type;
uint32_t version; uint32_t version;
struct pw_properties *properties; struct pw_properties *properties;
@ -135,7 +134,6 @@ static void free_pending(struct async_pending *p)
spa_list_remove(&p->link); spa_list_remove(&p->link);
free(p->handle); free(p->handle);
free(p->factory_name); free(p->factory_name);
free(p->name);
if (p->properties) if (p->properties)
pw_properties_free(p->properties); pw_properties_free(p->properties);
free(p); free(p);
@ -305,9 +303,8 @@ portal_response(DBusConnection *connection, DBusMessage *msg, void *user_data)
pw_resource_do_parent(p->resource->resource, pw_resource_do_parent(p->resource->resource,
&p->resource->override, &p->resource->override,
struct pw_core_proxy_methods, struct pw_core_proxy_methods,
create_node, create_object,
p->factory_name, p->factory_name,
p->name,
p->type, p->type,
p->version, p->version,
&p->properties->dict, &p->properties->dict,
@ -325,13 +322,12 @@ portal_response(DBusConnection *connection, DBusMessage *msg, void *user_data)
} }
static void do_create_node(void *data, static void do_create_object(void *data,
const char *factory_name, const char *factory_name,
const char *name, uint32_t type,
uint32_t type, uint32_t version,
uint32_t version, const struct spa_dict *props,
const struct spa_dict *props, uint32_t new_id)
uint32_t new_id)
{ {
struct resource *resource = data; struct resource *resource = data;
struct client_info *cinfo = resource->cinfo; struct client_info *cinfo = resource->cinfo;
@ -350,9 +346,8 @@ static void do_create_node(void *data,
pw_resource_do_parent(resource->resource, pw_resource_do_parent(resource->resource,
&resource->override, &resource->override,
struct pw_core_proxy_methods, struct pw_core_proxy_methods,
create_node, create_object,
factory_name, factory_name,
name,
type, type,
version, version,
props, props,
@ -411,7 +406,6 @@ static void do_create_node(void *data,
p->handle = strdup(handle); p->handle = strdup(handle);
p->handled = false; p->handled = false;
p->factory_name = strdup(factory_name); p->factory_name = strdup(factory_name);
p->name = strdup(name);
p->type = type; p->type = type;
p->version = version; p->version = version;
p->properties = props ? pw_properties_new_dict(props) : NULL; p->properties = props ? pw_properties_new_dict(props) : NULL;
@ -479,7 +473,7 @@ do_create_link(void *data,
static const struct pw_core_proxy_methods core_override = { static const struct pw_core_proxy_methods core_override = {
PW_VERSION_CORE_PROXY_METHODS, PW_VERSION_CORE_PROXY_METHODS,
.create_node = do_create_node, .create_object = do_create_object,
.create_link = do_create_link, .create_link = do_create_link,
}; };

View file

@ -82,10 +82,10 @@ static void core_marshal_get_registry(void *object, uint32_t version, uint32_t n
} }
static void static void
core_marshal_create_node(void *object, core_marshal_create_object(void *object,
const char *factory_name, const char *name, const char *factory_name,
uint32_t type, uint32_t version, uint32_t type, uint32_t version,
const struct spa_dict *props, uint32_t new_id) const struct spa_dict *props, uint32_t new_id)
{ {
struct pw_proxy *proxy = object; struct pw_proxy *proxy = object;
struct spa_pod_builder *b; struct spa_pod_builder *b;
@ -99,7 +99,6 @@ core_marshal_create_node(void *object,
spa_pod_builder_add(b, spa_pod_builder_add(b,
SPA_POD_TYPE_STRUCT, &f, SPA_POD_TYPE_STRUCT, &f,
SPA_POD_TYPE_STRING, factory_name, SPA_POD_TYPE_STRING, factory_name,
SPA_POD_TYPE_STRING, name,
SPA_POD_TYPE_ID, type, SPA_POD_TYPE_ID, type,
SPA_POD_TYPE_INT, version, SPA_POD_TYPE_INT, version,
SPA_POD_TYPE_INT, n_items, 0); SPA_POD_TYPE_INT, n_items, 0);
@ -430,18 +429,17 @@ static bool core_demarshal_get_registry(void *object, void *data, size_t size)
return true; return true;
} }
static bool core_demarshal_create_node(void *object, void *data, size_t size) static bool core_demarshal_create_object(void *object, void *data, size_t size)
{ {
struct pw_resource *resource = object; struct pw_resource *resource = object;
struct spa_pod_iter it; struct spa_pod_iter it;
uint32_t version, type, new_id, i; uint32_t version, type, new_id, i;
const char *factory_name, *name; const char *factory_name;
struct spa_dict props; struct spa_dict props;
if (!spa_pod_iter_struct(&it, data, size) || if (!spa_pod_iter_struct(&it, data, size) ||
!spa_pod_iter_get(&it, !spa_pod_iter_get(&it,
SPA_POD_TYPE_STRING, &factory_name, SPA_POD_TYPE_STRING, &factory_name,
SPA_POD_TYPE_STRING, &name,
SPA_POD_TYPE_ID, &type, SPA_POD_TYPE_ID, &type,
SPA_POD_TYPE_INT, &version, SPA_POD_TYPE_INT, &version,
SPA_POD_TYPE_INT, &props.n_items, 0)) SPA_POD_TYPE_INT, &props.n_items, 0))
@ -458,7 +456,7 @@ static bool core_demarshal_create_node(void *object, void *data, size_t size)
SPA_POD_TYPE_INT, &new_id, 0)) SPA_POD_TYPE_INT, &new_id, 0))
return false; return false;
pw_resource_do(resource, struct pw_core_proxy_methods, create_node, factory_name, name, pw_resource_do(resource, struct pw_core_proxy_methods, create_object, factory_name,
type, version, type, version,
&props, new_id); &props, new_id);
return true; return true;
@ -904,7 +902,7 @@ static const struct pw_core_proxy_methods pw_protocol_native_core_method_marshal
&core_marshal_sync, &core_marshal_sync,
&core_marshal_get_registry, &core_marshal_get_registry,
&core_marshal_client_update, &core_marshal_client_update,
&core_marshal_create_node, &core_marshal_create_object,
&core_marshal_create_link &core_marshal_create_link
}; };
@ -913,7 +911,7 @@ static const struct pw_protocol_native_demarshal pw_protocol_native_core_method_
{ &core_demarshal_sync, 0, }, { &core_demarshal_sync, 0, },
{ &core_demarshal_get_registry, 0, }, { &core_demarshal_get_registry, 0, },
{ &core_demarshal_client_update, 0, }, { &core_demarshal_client_update, 0, },
{ &core_demarshal_create_node, PW_PROTOCOL_NATIVE_REMAP, }, { &core_demarshal_create_object, PW_PROTOCOL_NATIVE_REMAP, },
{ &core_demarshal_create_link, PW_PROTOCOL_NATIVE_REMAP, } { &core_demarshal_create_link, PW_PROTOCOL_NATIVE_REMAP, }
}; };

View file

@ -33,28 +33,34 @@
struct factory_data { struct factory_data {
struct pw_core *core; struct pw_core *core;
struct pw_node_factory *this; struct pw_factory *this;
struct pw_properties *properties; struct pw_properties *properties;
}; };
static struct pw_node *create_node(void *_data, static void *create_object(void *_data,
struct pw_resource *resource, struct pw_resource *resource,
const char *name, uint32_t type,
struct pw_properties *properties) uint32_t version,
struct pw_properties *properties,
uint32_t new_id)
{ {
struct factory_data *data = _data; struct factory_data *data = _data;
struct pw_node *node; struct pw_node *node;
const char *lib, *factory_name; const char *lib, *factory_name, *name;
if (properties == NULL) if (properties == NULL)
goto no_properties; goto no_properties;
lib = pw_properties_get(properties, "spa.library.name"); lib = pw_properties_get(properties, "spa.library.name");
factory_name = pw_properties_get(properties, "spa.factory.name"); factory_name = pw_properties_get(properties, "spa.factory.name");
name = pw_properties_get(properties, "name");
if (lib == NULL || factory_name == NULL) if (lib == NULL || factory_name == NULL)
goto no_properties; goto no_properties;
if (name == NULL)
name = "spa-node";
node = pw_spa_node_load(data->core, node = pw_spa_node_load(data->core,
NULL, NULL,
NULL, NULL,
@ -65,6 +71,12 @@ static struct pw_node *create_node(void *_data,
if (node == NULL) if (node == NULL)
goto no_mem; goto no_mem;
if (resource)
pw_global_bind(pw_node_get_global(node),
pw_resource_get_client(resource),
PW_PERM_RWX,
version, new_id);
return node; return node;
no_properties: no_properties:
@ -84,33 +96,39 @@ static struct pw_node *create_node(void *_data,
return NULL; return NULL;
} }
static const struct pw_node_factory_implementation impl_factory = { static const struct pw_factory_implementation impl_factory = {
PW_VERSION_NODE_FACRORY_IMPLEMENTATION, PW_VERSION_FACTORY_IMPLEMENTATION,
.create_node = create_node, .create_object = create_object,
}; };
static bool module_init(struct pw_module *module, struct pw_properties *properties) static bool module_init(struct pw_module *module, struct pw_properties *properties)
{ {
struct pw_core *core = pw_module_get_core(module); struct pw_core *core = pw_module_get_core(module);
struct pw_node_factory *factory; struct pw_type *t = pw_core_get_type(core);
struct pw_factory *factory;
struct factory_data *data; struct factory_data *data;
factory = pw_node_factory_new(core, "spa-node-factory", sizeof(*data)); factory = pw_factory_new(core,
"spa-node-factory",
t->node,
PW_VERSION_NODE,
NULL,
sizeof(*data));
if (factory == NULL) if (factory == NULL)
return false; return false;
data = pw_node_factory_get_user_data(factory); data = pw_factory_get_user_data(factory);
data->this = factory; data->this = factory;
data->core = core; data->core = core;
data->properties = properties; data->properties = properties;
pw_log_debug("module %p: new", module); pw_log_debug("module %p: new", module);
pw_node_factory_set_implementation(factory, pw_factory_set_implementation(factory,
&impl_factory, &impl_factory,
data); data);
pw_node_factory_export(factory, NULL, pw_module_get_global(module)); pw_factory_register(factory, NULL, pw_module_get_global(module));
return true; return true;
} }

View file

@ -164,31 +164,28 @@ static void core_get_registry(void *object, uint32_t version, uint32_t new_id)
} }
static void static void
core_create_node(void *object, core_create_object(void *object,
const char *factory_name, const char *factory_name,
const char *name, uint32_t type,
uint32_t type, uint32_t version,
uint32_t version, const struct spa_dict *props,
const struct spa_dict *props, uint32_t new_id)
uint32_t new_id)
{ {
struct pw_resource *resource = object; struct pw_resource *resource = object;
struct pw_resource *node_resource = NULL;
struct pw_client *client = resource->client; struct pw_client *client = resource->client;
struct pw_node_factory *factory; struct pw_factory *factory;
struct pw_node *node; void *obj;
struct pw_properties *properties; struct pw_properties *properties;
int res;
factory = pw_core_find_node_factory(client->core, factory_name); factory = pw_core_find_factory(client->core, factory_name);
if (factory == NULL) if (factory == NULL)
goto no_factory; goto no_factory;
if (type != client->core->type.node) { if (factory->type != type)
node_resource = pw_resource_new(client, new_id, PW_PERM_RWX, type, version, 0); goto wrong_type;
if (node_resource == NULL)
goto no_resource; if (factory->version < version)
} goto wrong_version;
if (props) { if (props) {
properties = pw_properties_new_dict(props); properties = pw_properties_new_dict(props);
@ -198,16 +195,10 @@ core_create_node(void *object,
properties = NULL; properties = NULL;
/* error will be posted */ /* error will be posted */
node = pw_node_factory_create_node(factory, node_resource, name, properties); obj = pw_factory_create_object(factory, resource, type, version, properties, new_id);
if (node == NULL) if (obj == NULL)
goto no_mem; goto no_mem;
if (type == client->core->type.node) {
res = pw_global_bind(pw_node_get_global(node), client, PW_PERM_RWX, version, new_id);
if (res < 0)
goto no_bind;
}
properties = NULL; properties = NULL;
done: done:
@ -218,22 +209,19 @@ core_create_node(void *object,
pw_core_resource_error(client->core_resource, pw_core_resource_error(client->core_resource,
resource->id, SPA_RESULT_INVALID_ARGUMENTS, "unknown factory name %s", factory_name); resource->id, SPA_RESULT_INVALID_ARGUMENTS, "unknown factory name %s", factory_name);
goto done; goto done;
wrong_version:
no_resource: wrong_type:
pw_log_error("can't create resource"); pw_log_error("invalid resource type/version");
goto no_mem; pw_core_resource_error(client->core_resource,
resource->id, SPA_RESULT_INCOMPATIBLE_VERSION, "wrong resource type/version");
goto done;
no_properties: no_properties:
pw_log_error("can't create properties"); pw_log_error("can't create properties");
pw_resource_destroy(node_resource);
goto no_mem; goto no_mem;
no_mem: no_mem:
pw_core_resource_error(client->core_resource, pw_core_resource_error(client->core_resource,
resource->id, SPA_RESULT_NO_MEMORY, "no memory"); resource->id, SPA_RESULT_NO_MEMORY, "no memory");
goto done; goto done;
no_bind:
pw_core_resource_error(client->core_resource,
resource->id, res, "can't bind node: %d", res);
goto done;
} }
static void static void
@ -343,7 +331,7 @@ static const struct pw_core_proxy_methods core_methods = {
.sync = core_sync, .sync = core_sync,
.get_registry = core_get_registry, .get_registry = core_get_registry,
.client_update = core_client_update, .client_update = core_client_update,
.create_node = core_create_node, .create_object = core_create_object,
.create_link = core_create_link .create_link = core_create_link
}; };
@ -451,7 +439,7 @@ struct pw_core *pw_core_new(struct pw_loop *main_loop, struct pw_properties *pro
spa_list_init(&this->module_list); spa_list_init(&this->module_list);
spa_list_init(&this->client_list); spa_list_init(&this->client_list);
spa_list_init(&this->node_list); spa_list_init(&this->node_list);
spa_list_init(&this->node_factory_list); spa_list_init(&this->factory_list);
spa_list_init(&this->link_list); spa_list_init(&this->link_list);
spa_hook_list_init(&this->listener_list); spa_hook_list_init(&this->listener_list);
@ -792,21 +780,21 @@ struct spa_format *pw_core_find_format(struct pw_core *core,
return NULL; return NULL;
} }
/** Find a node by name /** Find a factory by name
* *
* \param core the core object * \param core the core object
* \param name the name of the node to find * \param name the name of the factory to find
* *
* Find in the list of nodes registered in \a core for one with * Find in the list of factories registered in \a core for one with
* the given \a name. * the given \a name.
* *
* \memberof pw_core * \memberof pw_core
*/ */
struct pw_node_factory *pw_core_find_node_factory(struct pw_core *core, const char *name) struct pw_factory *pw_core_find_factory(struct pw_core *core, const char *name)
{ {
struct pw_node_factory *factory; struct pw_factory *factory;
spa_list_for_each(factory, &core->node_factory_list, link) { spa_list_for_each(factory, &core->factory_list, link) {
if (strcmp(factory->name, name) == 0) if (strcmp(factory->name, name) == 0)
return factory; return factory;
} }

View file

@ -41,7 +41,7 @@ struct pw_core;
#include <pipewire/global.h> #include <pipewire/global.h>
#include <pipewire/introspect.h> #include <pipewire/introspect.h>
#include <pipewire/loop.h> #include <pipewire/loop.h>
#include <pipewire/node-factory.h> #include <pipewire/factory.h>
#include <pipewire/port.h> #include <pipewire/port.h>
#include <pipewire/properties.h> #include <pipewire/properties.h>
#include <pipewire/type.h> #include <pipewire/type.h>
@ -172,9 +172,9 @@ pw_core_find_port(struct pw_core *core,
struct spa_format **format_filters, struct spa_format **format_filters,
char **error); char **error);
/** Find a node factory by name */ /** Find a factory by name */
struct pw_node_factory * struct pw_factory *
pw_core_find_node_factory(struct pw_core *core, const char *name); pw_core_find_factory(struct pw_core *core, const char *name);
#ifdef __cplusplus #ifdef __cplusplus
} }

143
src/pipewire/factory.c Normal file
View file

@ -0,0 +1,143 @@
/* PipeWire
* Copyright (C) 2016 Axis Communications AB
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "pipewire/pipewire.h"
#include "pipewire/factory.h"
#include "pipewire/private.h"
struct resource_data {
struct spa_hook resource_listener;
};
struct pw_factory *pw_factory_new(struct pw_core *core,
const char *name,
uint32_t type,
uint32_t version,
struct pw_properties *properties,
size_t user_data_size)
{
struct pw_factory *this;
this = calloc(1, sizeof(*this) + user_data_size);
this->core = core;
this->name = strdup(name);
this->type = type;
this->version = version;
this->properties = properties;
spa_list_init(&this->resource_list);
if (user_data_size > 0)
this->user_data = SPA_MEMBER(this, sizeof(*this), void);
pw_log_debug("factory %p: new %s", this, name);
return this;
}
void pw_factory_destroy(struct pw_factory *factory)
{
pw_log_debug("factory %p: destroy", factory);
if (factory->global) {
spa_list_remove(&factory->link);
pw_global_destroy(factory->global);
}
if (factory->name)
free((char *)factory->name);
free(factory);
}
static void factory_unbind_func(void *data)
{
struct pw_resource *resource = data;
spa_list_remove(&resource->link);
}
static const struct pw_resource_events resource_events = {
PW_VERSION_RESOURCE_EVENTS,
.destroy = factory_unbind_func,
};
static int
factory_bind_func(struct pw_global *global,
struct pw_client *client, uint32_t permissions,
uint32_t version, uint32_t id)
{
struct pw_factory *this = global->object;
struct pw_resource *resource;
struct resource_data *data;
resource = pw_resource_new(client, id, permissions, global->type, version, sizeof(*data));
if (resource == NULL)
goto no_mem;
data = pw_resource_get_user_data(resource);
pw_resource_add_listener(resource, &data->resource_listener, &resource_events, resource);
pw_log_debug("factory %p: bound to %d", this, resource->id);
spa_list_insert(this->resource_list.prev, &resource->link);
this->info.change_mask = ~0;
pw_factory_resource_info(resource, &this->info);
this->info.change_mask = 0;
return SPA_RESULT_OK;
no_mem:
pw_log_error("can't create factory resource");
pw_core_resource_error(client->core_resource,
client->core_resource->id, SPA_RESULT_NO_MEMORY, "no memory");
return SPA_RESULT_NO_MEMORY;
}
void pw_factory_register(struct pw_factory *factory,
struct pw_client *owner,
struct pw_global *parent)
{
struct pw_core *core = factory->core;
spa_list_append(&core->factory_list, &factory->link);
factory->global = pw_core_add_global(core, owner, parent,
core->type.factory, 0, factory_bind_func, factory);
}
void *pw_factory_get_user_data(struct pw_factory *factory)
{
return factory->user_data;
}
void pw_factory_set_implementation(struct pw_factory *factory,
const struct pw_factory_implementation *implementation,
void *data)
{
factory->implementation = implementation;
factory->implementation_data = data;
}
void *pw_factory_create_object(struct pw_factory *factory,
struct pw_resource *resource,
uint32_t type,
uint32_t version,
struct pw_properties *properties,
uint32_t new_id)
{
return factory->implementation->create_object(factory->implementation_data,
resource, type, version, properties, new_id);
}

87
src/pipewire/factory.h Normal file
View file

@ -0,0 +1,87 @@
/* PipeWire
* Copyright (C) 2016 Axis Communications AB
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifndef __PIPEWIRE_FACTORY_H__
#define __PIPEWIRE_FACTORY_H__
#ifdef __cplusplus
extern "C" {
#endif
#define PW_TYPE_INTERFACE__Factory PW_TYPE_INTERFACE_BASE "Factory"
#define PW_TYPE_FACTORY_BASE PW_TYPE_INTERFACE__Factory ":"
/** \class pw_factory
*
* \brief PipeWire factory interface.
*
* The factory is used to make objects on demand.
*/
struct pw_factory;
#include <pipewire/core.h>
#include <pipewire/client.h>
#include <pipewire/global.h>
#include <pipewire/properties.h>
#include <pipewire/resource.h>
struct pw_factory_implementation {
#define PW_VERSION_FACTORY_IMPLEMENTATION 0
uint32_t version;
/** The function to create an object from this factory */
void *(*create_object) (void *data,
struct pw_resource *resource,
uint32_t type,
uint32_t version,
struct pw_properties *properties,
uint32_t new_id);
};
struct pw_factory *pw_factory_new(struct pw_core *core,
const char *name,
uint32_t type,
uint32_t version,
struct pw_properties *properties,
size_t user_data_size);
void pw_factory_register(struct pw_factory *factory,
struct pw_client *owner,
struct pw_global *parent);
void pw_factory_destroy(struct pw_factory *factory);
void *pw_factory_get_user_data(struct pw_factory *factory);
void pw_factory_set_implementation(struct pw_factory *factory,
const struct pw_factory_implementation *implementation,
void *data);
void *pw_factory_create_object(struct pw_factory *factory,
struct pw_resource *resource,
uint32_t type,
uint32_t version,
struct pw_properties *properties,
uint32_t new_id);
#ifdef __cplusplus
}
#endif
#endif /* __PIPEWIRE_FACTORY_H__ */

View file

@ -37,6 +37,7 @@ struct pw_core_proxy;
struct pw_registry_proxy; struct pw_registry_proxy;
struct pw_module_proxy; struct pw_module_proxy;
struct pw_node_proxy; struct pw_node_proxy;
struct pw_factory_proxy;
struct pw_client_proxy; struct pw_client_proxy;
struct pw_link_proxy; struct pw_link_proxy;
@ -124,24 +125,22 @@ struct pw_core_proxy_methods {
*/ */
void (*client_update) (void *object, const struct spa_dict *props); void (*client_update) (void *object, const struct spa_dict *props);
/** /**
* Create a new node on the PipeWire server from a factory. * Create a new object on the PipeWire server from a factory.
* Use a \a factory_name of "client-node" to create a * Use a \a factory_name of "client-node" to create a
* \ref pw_client_node. * \ref pw_client_node.
* *
* \param factory_name the factory name to use * \param factory_name the factory name to use
* \param name the node name
* \param type the interface to bind to * \param type the interface to bind to
* \param version the version of the interface * \param version the version of the interface
* \param props extra properties * \param props extra properties
* \param new_id the client proxy id * \param new_id the client proxy id
*/ */
void (*create_node) (void *object, void (*create_object) (void *object,
const char *factory_name, const char *factory_name,
const char *name, uint32_t type,
uint32_t type, uint32_t version,
uint32_t version, const struct spa_dict *props,
const struct spa_dict *props, uint32_t new_id);
uint32_t new_id);
/** /**
* Create a new link between two node ports * Create a new link between two node ports
* *
@ -190,17 +189,16 @@ pw_core_proxy_client_update(struct pw_core_proxy *core, const struct spa_dict *p
} }
static inline void * static inline void *
pw_core_proxy_create_node(struct pw_core_proxy *core, pw_core_proxy_create_object(struct pw_core_proxy *core,
const char *factory_name, const char *factory_name,
const char *name, uint32_t type,
uint32_t type, uint32_t version,
uint32_t version, const struct spa_dict *props,
const struct spa_dict *props, size_t user_data_size)
size_t user_data_size)
{ {
struct pw_proxy *p = pw_proxy_new((struct pw_proxy*)core, type, user_data_size); struct pw_proxy *p = pw_proxy_new((struct pw_proxy*)core, type, user_data_size);
pw_proxy_do((struct pw_proxy*)core, struct pw_core_proxy_methods, create_node, factory_name, pw_proxy_do((struct pw_proxy*)core, struct pw_core_proxy_methods, create_object, factory_name,
name, type, version, props, pw_proxy_get_id(p)); type, version, props, pw_proxy_get_id(p));
return p; return p;
} }
@ -471,6 +469,35 @@ pw_node_proxy_add_listener(struct pw_node_proxy *node,
#define pw_node_resource_info(r,...) pw_resource_notify(r,struct pw_node_proxy_events,info,__VA_ARGS__) #define pw_node_resource_info(r,...) pw_resource_notify(r,struct pw_node_proxy_events,info,__VA_ARGS__)
#define PW_VERSION_FACTORY 0
#define PW_FACTORY_PROXY_EVENT_INFO 0
#define PW_FACTORY_PROXY_EVENT_NUM 1
/** Factory events */
struct pw_factory_proxy_events {
#define PW_VERSION_FACTORY_PROXY_EVENTS 0
uint32_t version;
/**
* Notify factory info
*
* \param info info about the factory
*/
void (*info) (void *object, struct pw_factory_info *info);
};
/** Factory */
static inline void
pw_factory_proxy_add_listener(struct pw_factory_proxy *factory,
struct spa_hook *listener,
const struct pw_factory_proxy_events *events,
void *data)
{
pw_proxy_add_proxy_listener((struct pw_proxy*)factory, listener, events, data);
}
#define pw_factory_resource_info(r,...) pw_resource_notify(r,struct pw_factory_proxy_events,info,__VA_ARGS__)
#define PW_VERSION_CLIENT 0 #define PW_VERSION_CLIENT 0
#define PW_CLIENT_PROXY_EVENT_INFO 0 #define PW_CLIENT_PROXY_EVENT_INFO 0

View file

@ -170,6 +170,23 @@ pw_node_info_update(struct pw_node_info *info,
void void
pw_node_info_free(struct pw_node_info *info); pw_node_info_free(struct pw_node_info *info);
/** The factory information. Extra information can be added in later versions \memberof pw_introspect */
struct pw_factory_info {
uint32_t id; /**< id of the global */
const char *name; /**< name the factory */
uint32_t type; /**< type of the factory */
uint32_t version; /**< version of the objects */
#define PW_FACTORY_CHANGE_MASK_PROPS (1 << 0)
uint64_t change_mask; /**< bitfield of changed fields since last call */
struct spa_dict *props; /**< the properties of the factory */
};
struct pw_factory_info *
pw_factory_info_update(struct pw_factory_info *info,
const struct pw_factory_info *update);
void
pw_factory_info_free(struct pw_factory_info *info);
/** The link information. Extra information can be added in later versions \memberof pw_introspect */ /** The link information. Extra information can be added in later versions \memberof pw_introspect */
struct pw_link_info { struct pw_link_info {

View file

@ -15,7 +15,7 @@ pipewire_headers = [
'mem.h', 'mem.h',
'module.h', 'module.h',
'node.h', 'node.h',
'node-factory.h', 'factory.h',
'pipewire.h', 'pipewire.h',
'port.h', 'port.h',
'properties.h', 'properties.h',
@ -45,7 +45,7 @@ pipewire_sources = [
'mem.c', 'mem.c',
'module.c', 'module.c',
'node.c', 'node.c',
'node-factory.c', 'factory.c',
'pipewire.c', 'pipewire.c',
'port.c', 'port.c',
'properties.c', 'properties.c',

View file

@ -123,7 +123,7 @@ module_bind_func(struct pw_global *global,
pw_log_debug("module %p: bound to %d", this, resource->id); pw_log_debug("module %p: bound to %d", this, resource->id);
spa_list_insert(this->resource_list.prev, &resource->link); spa_list_append(&this->resource_list, &resource->link);
this->info.change_mask = ~0; this->info.change_mask = ~0;
pw_module_resource_info(resource, &this->info); pw_module_resource_info(resource, &this->info);

View file

@ -1,85 +0,0 @@
/* PipeWire
* Copyright (C) 2016 Axis Communications AB
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "pipewire/pipewire.h"
#include "pipewire/node-factory.h"
#include "pipewire/private.h"
struct pw_node_factory *pw_node_factory_new(struct pw_core *core,
const char *name,
size_t user_data_size)
{
struct pw_node_factory *this;
this = calloc(1, sizeof(*this) + user_data_size);
this->core = core;
this->name = strdup(name);
if (user_data_size > 0)
this->user_data = SPA_MEMBER(this, sizeof(*this), void);
pw_log_debug("node-factory %p: new", this);
return this;
}
void pw_node_factory_destroy(struct pw_node_factory *factory)
{
pw_log_debug("node-factory %p: destroy", factory);
if (factory->global) {
spa_list_remove(&factory->link);
pw_global_destroy(factory->global);
}
if (factory->name)
free((char *)factory->name);
free(factory);
}
void pw_node_factory_export(struct pw_node_factory *factory,
struct pw_client *owner,
struct pw_global *parent)
{
struct pw_core *core = factory->core;
spa_list_append(&core->node_factory_list, &factory->link);
factory->global = pw_core_add_global(core, owner, parent, core->type.node_factory, 0, NULL, factory);
}
void *pw_node_factory_get_user_data(struct pw_node_factory *factory)
{
return factory->user_data;
}
void pw_node_factory_set_implementation(struct pw_node_factory *factory,
const struct pw_node_factory_implementation *implementation,
void *data)
{
factory->implementation = implementation;
factory->implementation_data = data;
}
struct pw_node *pw_node_factory_create_node(struct pw_node_factory *factory,
struct pw_resource *resource,
const char *name,
struct pw_properties *properties)
{
return factory->implementation->create_node(factory->implementation_data,
resource, name, properties);
}

View file

@ -1,80 +0,0 @@
/* PipeWire
* Copyright (C) 2016 Axis Communications AB
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifndef __PIPEWIRE_NODE_FACTORY_H__
#define __PIPEWIRE_NODE_FACTORY_H__
#ifdef __cplusplus
extern "C" {
#endif
#define PW_TYPE_INTERFACE__NodeFactory PW_TYPE_INTERFACE_BASE "NodeFactory"
#define PW_TYPE_NODE_FACTORY_BASE PW_TYPE_INTERFACE__NodeFactory ":"
/** \class pw_node_factory
*
* \brief PipeWire node factory interface.
*
* The factory object is used to make nodes on demand.
*/
struct pw_node_factory;
#include <pipewire/core.h>
#include <pipewire/client.h>
#include <pipewire/global.h>
#include <pipewire/properties.h>
#include <pipewire/resource.h>
struct pw_node_factory_implementation {
#define PW_VERSION_NODE_FACRORY_IMPLEMENTATION 0
uint32_t version;
/** The function to create a node from this factory */
struct pw_node *(*create_node) (void *data,
struct pw_resource *resource,
const char *name,
struct pw_properties *properties);
};
struct pw_node_factory *pw_node_factory_new(struct pw_core *core,
const char *name,
size_t user_data_size);
void pw_node_factory_export(struct pw_node_factory *factory,
struct pw_client *owner,
struct pw_global *parent);
void pw_node_factory_destroy(struct pw_node_factory *factory);
void *pw_node_factory_get_user_data(struct pw_node_factory *factory);
void pw_node_factory_set_implementation(struct pw_node_factory *factory,
const struct pw_node_factory_implementation *implementation,
void *data);
struct pw_node *pw_node_factory_create_node(struct pw_node_factory *factory,
struct pw_resource *resource,
const char *name,
struct pw_properties *properties);
#ifdef __cplusplus
}
#endif
#endif /* __PIPEWIRE_NODE_FACTORY_H__ */

View file

@ -130,7 +130,7 @@ struct pw_core {
struct spa_list global_list; /**< list of globals */ struct spa_list global_list; /**< list of globals */
struct spa_list client_list; /**< list of clients */ struct spa_list client_list; /**< list of clients */
struct spa_list node_list; /**< list of nodes */ struct spa_list node_list; /**< list of nodes */
struct spa_list node_factory_list; /**< list of node factories */ struct spa_list factory_list; /**< list of factories */
struct spa_list link_list; /**< list of links */ struct spa_list link_list; /**< list of links */
struct spa_hook_list listener_list; struct spa_hook_list listener_list;
@ -368,16 +368,22 @@ struct pw_stream {
struct spa_hook_list listener_list; struct spa_hook_list listener_list;
}; };
struct pw_node_factory { struct pw_factory {
struct pw_core *core; /**< the core */ struct pw_core *core; /**< the core */
struct spa_list link; /**< link in core node_factory_list */ struct spa_list link; /**< link in core node_factory_list */
struct pw_global *global; /**< global for this factory */ struct pw_global *global; /**< global for this factory */
struct pw_factory_info info; /**< introspectable factory info */
const char *name; /**< the factory name */ const char *name; /**< the factory name */
uint32_t type; /**< the type produced by the factory*/
uint32_t version; /**< the version of the produced object */
struct pw_properties *properties; /**< properties of the factory */
const struct pw_node_factory_implementation *implementation; const struct pw_factory_implementation *implementation;
void *implementation_data; void *implementation_data;
struct spa_list resource_list; /**< The list of resources of this factory */
void *user_data; void *user_data;
}; };

View file

@ -1084,13 +1084,12 @@ struct pw_proxy *pw_remote_export(struct pw_remote *remote,
struct pw_proxy *proxy; struct pw_proxy *proxy;
struct node_data *data; struct node_data *data;
proxy = pw_core_proxy_create_node(remote->core_proxy, proxy = pw_core_proxy_create_object(remote->core_proxy,
"client-node", "client-node",
"client-node", impl->type_client_node,
impl->type_client_node, PW_VERSION_CLIENT_NODE,
PW_VERSION_CLIENT_NODE, &node->properties->dict,
&node->properties->dict, sizeof(struct node_data));
sizeof(struct node_data));
if (proxy == NULL) if (proxy == NULL)
return NULL; return NULL;

View file

@ -957,8 +957,7 @@ pw_stream_connect(struct pw_stream *stream,
if (flags & PW_STREAM_FLAG_AUTOCONNECT) if (flags & PW_STREAM_FLAG_AUTOCONNECT)
pw_properties_set(stream->properties, "pipewire.autoconnect", "1"); pw_properties_set(stream->properties, "pipewire.autoconnect", "1");
impl->node_proxy = pw_core_proxy_create_node(stream->remote->core_proxy, impl->node_proxy = pw_core_proxy_create_object(stream->remote->core_proxy,
"client-node",
"client-node", "client-node",
impl->type_client_node, impl->type_client_node,
PW_VERSION_CLIENT_NODE, PW_VERSION_CLIENT_NODE,

View file

@ -42,7 +42,7 @@ void pw_type_init(struct pw_type *type)
type->core = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Core); type->core = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Core);
type->registry = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Registry); type->registry = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Registry);
type->node = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Node); type->node = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Node);
type->node_factory = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__NodeFactory); type->factory = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Factory);
type->link = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Link); type->link = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Link);
type->client = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Client); type->client = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Client);
type->module = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Module); type->module = spa_type_map_get_id(type->map, PW_TYPE_INTERFACE__Module);

View file

@ -51,7 +51,7 @@ struct pw_type {
uint32_t core; uint32_t core;
uint32_t registry; uint32_t registry;
uint32_t node; uint32_t node;
uint32_t node_factory; uint32_t factory;
uint32_t link; uint32_t link;
uint32_t client; uint32_t client;
uint32_t module; uint32_t module;

View file

@ -811,7 +811,7 @@ static bool do_info(struct data *data, const char *cmd, char *args, char **error
static bool do_create_node(struct data *data, const char *cmd, char *args, char **error) static bool do_create_node(struct data *data, const char *cmd, char *args, char **error)
{ {
struct remote_data *rd = data->current; struct remote_data *rd = data->current;
char *a[3]; char *a[2];
int n; int n;
uint32_t id; uint32_t id;
struct pw_type *t = data->t; struct pw_type *t = data->t;
@ -819,18 +819,18 @@ static bool do_create_node(struct data *data, const char *cmd, char *args, char
struct pw_properties *props = NULL; struct pw_properties *props = NULL;
struct proxy_data *pd; struct proxy_data *pd;
n = pw_split_ip(args, WHITESPACE, 3, a); n = pw_split_ip(args, WHITESPACE, 2, a);
if (n < 2) { if (n < 1) {
asprintf(error, "%s <factory-name> <name> [<properties>]", cmd); asprintf(error, "%s <factory-name> [<properties>]", cmd);
return false; return false;
} }
if (n == 3) if (n == 2)
props = parse_props(a[2]); props = parse_props(a[1]);
proxy = pw_core_proxy_create_node(rd->core_proxy, a[0], a[1], proxy = pw_core_proxy_create_object(rd->core_proxy, a[0],
t->node, PW_VERSION_NODE, t->node, PW_VERSION_NODE,
props ? &props->dict : NULL, props ? &props->dict : NULL,
sizeof(struct proxy_data)); sizeof(struct proxy_data));
pd = pw_proxy_get_user_data(proxy); pd = pw_proxy_get_user_data(proxy);
pd->rd = rd; pd->rd = rd;