Files
wireplumber/modules/module-default-nodes-api.c

207 lines
5.7 KiB
C

/* WirePlumber
*
* Copyright © 2021 Collabora Ltd.
* @author George Kiagiadakis <george.kiagiadakis@collabora.com>
*
* SPDX-License-Identifier: MIT
*/
#include <wp/wp.h>
#include <pipewire/keys.h>
#include "module-default-nodes/common.h"
struct _WpDefaultNodesApi
{
WpPlugin parent;
gchar *defaults[N_DEFAULT_NODES];
WpObjectManager *om;
GSource *idle_source;
};
enum {
ACTION_GET_DEFAULT_NODE,
SIGNAL_CHANGED,
N_SIGNALS
};
static guint signals[N_SIGNALS] = {0};
G_DECLARE_FINAL_TYPE (WpDefaultNodesApi, wp_default_nodes_api,
WP, DEFAULT_NODES_API, WpPlugin)
G_DEFINE_TYPE (WpDefaultNodesApi, wp_default_nodes_api, WP_TYPE_PLUGIN)
static void
wp_default_nodes_api_init (WpDefaultNodesApi * self)
{
}
static gboolean
emit_changed_cb (gpointer data)
{
WpDefaultNodesApi *self = data;
g_signal_emit (self, signals[SIGNAL_CHANGED], 0);
g_clear_pointer (&self->idle_source, g_source_unref);
return G_SOURCE_REMOVE;
}
static void
schedule_changed_notification (WpDefaultNodesApi *self)
{
g_autoptr (WpCore) core = wp_object_get_core (WP_OBJECT (self));
g_return_if_fail (core);
wp_core_idle_add (core, &self->idle_source, emit_changed_cb, self, NULL);
}
static void
on_metadata_changed (WpMetadata *m, guint32 subject,
const gchar *key, const gchar *type, const gchar *value, gpointer d)
{
WpDefaultNodesApi * self = WP_DEFAULT_NODES_API (d);
gint node_t = -1;
gchar name[1024];
if (subject == 0) {
for (gint i = 0; i < N_DEFAULT_NODES; i++) {
if (!g_strcmp0 (key, DEFAULT_KEY[i])) {
node_t = i;
break;
}
}
}
if (node_t != -1) {
g_clear_pointer (&self->defaults[node_t], g_free);
if (value && !g_strcmp0 (type, "Spa:String:JSON") &&
json_object_find (value, "name", name, sizeof(name)) == 0)
{
self->defaults[node_t] = g_strdup (name);
}
wp_debug_object (m, "changed '%s' -> '%s'", key, self->defaults[node_t]);
schedule_changed_notification (self);
}
}
static void
on_metadata_added (WpObjectManager *om, WpObject *obj, WpDefaultNodesApi * self)
{
if (WP_IS_METADATA (obj)) {
g_autoptr (WpIterator) it = wp_metadata_new_iterator (WP_METADATA (obj), 0);
g_auto (GValue) val = G_VALUE_INIT;
for (; wp_iterator_next (it, &val); g_value_unset (&val)) {
guint32 subject;
const gchar *key, *type, *value;
wp_metadata_iterator_item_extract (&val, &subject, &key, &type, &value);
on_metadata_changed (WP_METADATA (obj), subject, key, type, value, self);
}
g_signal_connect_object (obj, "changed",
G_CALLBACK (on_metadata_changed), self, 0);
}
}
static void
on_om_installed (WpObjectManager * om, WpDefaultNodesApi * self)
{
wp_object_update_features (WP_OBJECT (self), WP_PLUGIN_FEATURE_ENABLED, 0);
}
static void
wp_default_nodes_api_enable (WpPlugin * plugin, WpTransition * transition)
{
WpDefaultNodesApi * self = WP_DEFAULT_NODES_API (plugin);
g_autoptr (WpCore) core = wp_object_get_core (WP_OBJECT (plugin));
g_return_if_fail (core);
/* Create the metadatas object manager */
self->om = wp_object_manager_new ();
wp_object_manager_add_interest (self->om, WP_TYPE_METADATA,
WP_CONSTRAINT_TYPE_PW_GLOBAL_PROPERTY, "metadata.name", "=s", "default",
NULL);
wp_object_manager_add_interest (self->om, WP_TYPE_NODE, NULL);
wp_object_manager_request_object_features (self->om,
WP_TYPE_METADATA, WP_OBJECT_FEATURES_ALL);
wp_object_manager_request_object_features (self->om,
WP_TYPE_NODE, WP_PIPEWIRE_OBJECT_FEATURES_MINIMAL);
g_signal_connect_object (self->om, "object-added",
G_CALLBACK (on_metadata_added), self, 0);
g_signal_connect_object (self->om, "installed",
G_CALLBACK (on_om_installed), self, 0);
wp_core_install_object_manager (core, self->om);
}
static void
wp_default_nodes_api_disable (WpPlugin * plugin)
{
WpDefaultNodesApi * self = WP_DEFAULT_NODES_API (plugin);
if (self->idle_source)
g_source_destroy (self->idle_source);
g_clear_pointer (&self->idle_source, g_source_unref);
for (guint i = 0; i < N_DEFAULT_NODES; i++)
g_clear_pointer (&self->defaults[i], g_free);
g_clear_object (&self->om);
}
static guint
wp_default_nodes_api_get_default_node (WpDefaultNodesApi * self,
const gchar * media_class)
{
gint node_t = -1;
for (gint i = 0; i < N_DEFAULT_NODES; i++) {
if (!g_strcmp0 (media_class, MEDIA_CLASS[i])) {
node_t = i;
break;
}
}
if (node_t != -1 && self->defaults[node_t]) {
g_autoptr (WpNode) node = wp_object_manager_lookup (self->om,
WP_TYPE_NODE,
WP_CONSTRAINT_TYPE_PW_PROPERTY,
PW_KEY_NODE_NAME, "=s", self->defaults[node_t],
WP_CONSTRAINT_TYPE_PW_PROPERTY,
PW_KEY_MEDIA_CLASS, "=s", MEDIA_CLASS[node_t],
NULL);
if (node)
return wp_proxy_get_bound_id (WP_PROXY (node));
}
return SPA_ID_INVALID;
}
static void
wp_default_nodes_api_class_init (WpDefaultNodesApiClass * klass)
{
WpPluginClass *plugin_class = (WpPluginClass *) klass;
plugin_class->enable = wp_default_nodes_api_enable;
plugin_class->disable = wp_default_nodes_api_disable;
signals[ACTION_GET_DEFAULT_NODE] = g_signal_new_class_handler (
"get-default-node", G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
(GCallback) wp_default_nodes_api_get_default_node,
NULL, NULL, NULL,
G_TYPE_UINT, 1, G_TYPE_STRING);
signals[SIGNAL_CHANGED] = g_signal_new (
"changed", G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
}
WP_PLUGIN_EXPORT gboolean
wireplumber__module_init (WpCore * core, GVariant * args, GError ** error)
{
wp_plugin_register (g_object_new (wp_default_nodes_api_get_type (),
"name", "default-nodes-api",
"core", core,
NULL));
return TRUE;
}