Files
wireplumber/modules/module-default-nodes-api.c
Ashok Sidipotu ff833b138d event-dispatcher: log event and hook names and chains
- 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.
2023-04-17 07:47:09 -04:00

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;
}