
- Add a new variable "name" in WpEventHook and use it to log all the hooks(by name) picked up in _push_event(). This gives a clear picture if hook is registered for a given event. - Form a name for an event and a chain of events for an event run, log both of them. This gives a clear picture of the events executed and order in which they are dispatched. - Similarly build hooks chain and print it in _source_dispatch(), this gives a clear picture of the hooks picked and the order in which they are dispatched. - Log only the dispatchable(with hooks) events, this de-clutters the log messages.
336 lines
11 KiB
C
336 lines
11 KiB
C
/* WirePlumber
|
|
*
|
|
* Copyright © 2021 Collabora Ltd.
|
|
* @author George Kiagiadakis <george.kiagiadakis@collabora.com>
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
#include <wp/wp.h>
|
|
#include <spa/utils/defs.h>
|
|
#include <pipewire/keys.h>
|
|
#include "module-default-nodes/common.h"
|
|
|
|
/*
|
|
* Module Provides the APIs to query the default device nodes. Module looks at
|
|
* the default metadata to know the default devices.
|
|
*/
|
|
|
|
typedef struct _WpDefaultNode WpDefaultNode;
|
|
struct _WpDefaultNode
|
|
{
|
|
gchar *value;
|
|
gchar *config_value;
|
|
};
|
|
|
|
struct _WpDefaultNodesApi
|
|
{
|
|
WpPlugin parent;
|
|
|
|
WpDefaultNode defaults[N_DEFAULT_NODES];
|
|
WpObjectManager *om;
|
|
};
|
|
|
|
enum {
|
|
ACTION_GET_DEFAULT_NODE,
|
|
ACTION_GET_DEFAULT_CONFIGURED_NODE_NAME,
|
|
ACTION_SET_DEFAULT_CONFIGURED_NODE_NAME,
|
|
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 void
|
|
sync_changed_notification (WpCore * core, GAsyncResult * res,
|
|
WpDefaultNodesApi * self)
|
|
{
|
|
g_autoptr (GError) error = NULL;
|
|
if (!wp_core_sync_finish (core, res, &error)) {
|
|
wp_warning_object (self, "core sync error: %s", error->message);
|
|
return;
|
|
}
|
|
|
|
g_signal_emit (self, signals[SIGNAL_CHANGED], 0);
|
|
return;
|
|
}
|
|
|
|
static void
|
|
schedule_changed_notification (WpDefaultNodesApi *self)
|
|
{
|
|
g_autoptr (WpCore) core = wp_object_get_core (WP_OBJECT (self));
|
|
g_return_if_fail (core);
|
|
// Event-Stack TBD: do we need to retain this behavior? or push this as a
|
|
// event & hook pair on to event stack
|
|
wp_core_sync_closure (core, NULL, g_cclosure_new_object (
|
|
G_CALLBACK (sync_changed_notification), G_OBJECT (self)));
|
|
}
|
|
|
|
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);
|
|
|
|
if (subject != 0)
|
|
return;
|
|
|
|
for (gint i = 0; i < N_DEFAULT_NODES; i++) {
|
|
if (!g_strcmp0 (key, DEFAULT_KEY[i])) {
|
|
if (value && !g_strcmp0 (type, "Spa:String:JSON")) {
|
|
g_autoptr (WpSpaJson) json = wp_spa_json_new_from_string (value);
|
|
g_autofree gchar *name = NULL;
|
|
if (wp_spa_json_object_get (json, "name", "s", &name, NULL)) {
|
|
wp_debug_object (m, "'%s' changed from %s -> '%s'", key, name,
|
|
self->defaults[i].value);
|
|
g_clear_pointer (&self->defaults[i].value, g_free);
|
|
|
|
self->defaults[i].value = g_strdup (name);
|
|
}
|
|
}
|
|
|
|
schedule_changed_notification (self);
|
|
break;
|
|
} else if (!g_strcmp0 (key, DEFAULT_CONFIG_KEY[i])) {
|
|
|
|
if (value && !g_strcmp0 (type, "Spa:String:JSON")) {
|
|
g_autoptr (WpSpaJson) json = wp_spa_json_new_from_string (value);
|
|
g_autofree gchar *name = NULL;
|
|
if (wp_spa_json_object_get (json, "name", "s", &name, NULL)){
|
|
wp_debug_object (m, "'%s' changed from %s -> '%s'", key, name,
|
|
self->defaults[i].config_value);
|
|
g_clear_pointer (&self->defaults[i].config_value, g_free);
|
|
|
|
self->defaults[i].config_value = g_strdup (name);
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
on_metadata_changed_hook (WpEvent *event, gpointer d)
|
|
{
|
|
WpDefaultNodesApi * self = WP_DEFAULT_NODES_API (d);
|
|
g_autoptr (GObject) subject = wp_event_get_subject (event);
|
|
WpMetadata *m = WP_METADATA (subject);
|
|
g_autoptr (WpProperties) p = wp_event_get_properties (event);
|
|
guint32 subject_id = atoi (wp_properties_get (p, "event.subject.id"));
|
|
const gchar *key = wp_properties_get (p, "event.subject.key");
|
|
const gchar *type = wp_properties_get (p, "event.subject.spa_type");
|
|
const gchar *value = wp_properties_get (p, "event.subject.value");
|
|
|
|
on_metadata_changed (m, subject_id, key, type, value, self);
|
|
}
|
|
|
|
static void
|
|
on_metadata_added (WpEvent *event, gpointer d)
|
|
{
|
|
WpDefaultNodesApi * self = WP_DEFAULT_NODES_API (d);
|
|
g_autoptr (GObject) subject = wp_event_get_subject (event);
|
|
WpMetadata *obj = WP_METADATA (subject);
|
|
|
|
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_id;
|
|
const gchar *key, *type, *value;
|
|
wp_metadata_iterator_item_extract (&val, &subject_id, &key, &type, &value);
|
|
on_metadata_changed (WP_METADATA (obj), subject_id, key, type, value, self);
|
|
}
|
|
}
|
|
}
|
|
|
|
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);
|
|
g_autoptr (WpEventDispatcher) dispatcher =
|
|
wp_event_dispatcher_get_instance (core);
|
|
g_autoptr (WpEventHook) hook = NULL;
|
|
g_return_if_fail (dispatcher);
|
|
|
|
/* default metadata added */
|
|
hook = wp_simple_event_hook_new ("default-nodes-api-on-metadata-added", 10,
|
|
WP_EVENT_HOOK_EXEC_TYPE_ON_EVENT,
|
|
g_cclosure_new ((GCallback) on_metadata_added, self, NULL));
|
|
wp_interest_event_hook_add_interest (WP_INTEREST_EVENT_HOOK (hook),
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "event.type", "=s", "object-added",
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "event.subject.type", "=s", "metadata",
|
|
WP_CONSTRAINT_TYPE_PW_GLOBAL_PROPERTY, "metadata.name", "=s", "default",
|
|
NULL);
|
|
wp_event_dispatcher_register_hook (dispatcher, hook);
|
|
g_clear_object(&hook);
|
|
|
|
/* default metadata changed */
|
|
hook = wp_simple_event_hook_new ("default-nodes-api-on-metadata-changed", 10,
|
|
WP_EVENT_HOOK_EXEC_TYPE_ON_EVENT,
|
|
g_cclosure_new ((GCallback) on_metadata_changed_hook, self, NULL));
|
|
wp_interest_event_hook_add_interest (WP_INTEREST_EVENT_HOOK (hook),
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "event.type", "=s", "object-changed",
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "event.subject.type", "=s", "metadata",
|
|
WP_CONSTRAINT_TYPE_PW_GLOBAL_PROPERTY, "metadata.name", "=s", "default",
|
|
NULL);
|
|
wp_event_dispatcher_register_hook (dispatcher, hook);
|
|
g_clear_object(&hook);
|
|
|
|
/* Create the metadata 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, "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);
|
|
|
|
for (guint i = 0; i < N_DEFAULT_NODES; i++) {
|
|
g_clear_pointer (&self->defaults[i].value, g_free);
|
|
g_clear_pointer (&self->defaults[i].config_value, 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, NODE_TYPE_STR[i])) {
|
|
node_t = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (node_t != -1 && self->defaults[node_t].value) {
|
|
g_autoptr (WpIterator) it = NULL;
|
|
g_auto (GValue) val = G_VALUE_INIT;
|
|
it = wp_object_manager_new_filtered_iterator (self->om,
|
|
WP_TYPE_NODE,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY,
|
|
PW_KEY_NODE_NAME, "=s", self->defaults[node_t].value,
|
|
NULL);
|
|
for (; wp_iterator_next (it, &val); g_value_unset (&val)) {
|
|
WpNode *node = g_value_get_object (&val);
|
|
const gchar *mc = wp_pipewire_object_get_property (
|
|
WP_PIPEWIRE_OBJECT (node), PW_KEY_MEDIA_CLASS);
|
|
if (!g_str_has_prefix (mc, "Stream/"))
|
|
return wp_proxy_get_bound_id (WP_PROXY (node));
|
|
}
|
|
}
|
|
return SPA_ID_INVALID;
|
|
}
|
|
|
|
static gchar *
|
|
wp_default_nodes_api_get_default_configured_node_name (WpDefaultNodesApi * self,
|
|
const gchar * media_class)
|
|
{
|
|
for (gint i = 0; i < N_DEFAULT_NODES; i++)
|
|
if (!g_strcmp0 (media_class, NODE_TYPE_STR[i]) &&
|
|
self->defaults[i].config_value)
|
|
return g_strdup (self->defaults[i].config_value);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static gboolean
|
|
wp_default_nodes_api_set_default_configured_node_name (WpDefaultNodesApi * self,
|
|
const gchar * media_class, const gchar * name)
|
|
{
|
|
g_autoptr (WpMetadata) m = wp_object_manager_lookup (self->om,
|
|
WP_TYPE_METADATA, NULL);
|
|
if (!m)
|
|
return FALSE;
|
|
|
|
for (gint i = 0; i < N_DEFAULT_NODES; i++) {
|
|
if (!g_strcmp0 (media_class, NODE_TYPE_STR[i])) {
|
|
if (name) {
|
|
g_autofree gchar *v = g_strdup_printf ("{ \"name\": \"%s\" }", name);
|
|
wp_metadata_set (m, 0, DEFAULT_CONFIG_KEY[i], "Spa:String:JSON", v);
|
|
} else {
|
|
wp_metadata_set (m, 0, DEFAULT_CONFIG_KEY[i], NULL, NULL);
|
|
}
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
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[ACTION_GET_DEFAULT_CONFIGURED_NODE_NAME] = g_signal_new_class_handler (
|
|
"get-default-configured-node-name", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
(GCallback) wp_default_nodes_api_get_default_configured_node_name,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_STRING, 1, G_TYPE_STRING);
|
|
|
|
signals[ACTION_SET_DEFAULT_CONFIGURED_NODE_NAME] = g_signal_new_class_handler (
|
|
"set-default-configured-node-name", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
(GCallback) wp_default_nodes_api_set_default_configured_node_name,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_BOOLEAN, 2, G_TYPE_STRING, 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;
|
|
}
|