diff --git a/modules/meson.build b/modules/meson.build index 56c01cc1e79c41c670bbfc5a4e900da5132b9731..51a012651ba179bb083f525a24f4fc0acc3378ae 100644 --- a/modules/meson.build +++ b/modules/meson.build @@ -165,3 +165,14 @@ shared_library( install_dir : wireplumber_module_dir, dependencies : [wp_dep, pipewire_dep, wplua_dep], ) + +shared_library( + 'wireplumber-module-mixer-api', + [ + 'module-mixer-api.c', + ], + c_args : [common_c_args, '-DG_LOG_DOMAIN="m-mixer-api"'], + install : true, + install_dir : wireplumber_module_dir, + dependencies : [wp_dep, pipewire_dep], +) diff --git a/modules/module-mixer-api.c b/modules/module-mixer-api.c new file mode 100644 index 0000000000000000000000000000000000000000..6e865732823fa916caf08ba705ce61d35f86e9cb --- /dev/null +++ b/modules/module-mixer-api.c @@ -0,0 +1,469 @@ +/* WirePlumber + * + * Copyright © 2021 Collabora Ltd. + * @author George Kiagiadakis <george.kiagiadakis@collabora.com> + * + * SPDX-License-Identifier: MIT + */ + +#include <wp/wp.h> +#include <pipewire/pipewire.h> + +#include <spa/pod/iter.h> +#include <spa/param/audio/raw.h> + +struct volume { + uint8_t channels; + float values[SPA_AUDIO_MAX_CHANNELS]; +}; + +struct channel_map { + uint8_t channels; + uint32_t map[SPA_AUDIO_MAX_CHANNELS]; +}; + +struct node_info { + guint32 seq; + + guint32 device_id; + gint32 route_index; + gint32 route_device; + + struct volume volume; + struct channel_map map; + bool mute; + float base; + float step; +}; + +struct _WpMixerApi +{ + WpPlugin parent; + WpObjectManager *om; + GHashTable *node_infos; + guint32 seq; +}; + +enum { + ACTION_SET_VOLUME, + ACTION_GET_VOLUME, + SIGNAL_CHANGED, + N_SIGNALS +}; + +static guint signals[N_SIGNALS] = {0}; + +G_DECLARE_FINAL_TYPE (WpMixerApi, wp_mixer_api, WP, MIXER_API, WpPlugin) +G_DEFINE_TYPE (WpMixerApi, wp_mixer_api, WP_TYPE_PLUGIN) + +static void +wp_mixer_api_init (WpMixerApi * self) +{ +} + +static void +node_info_fill (struct node_info * info, WpSpaPod * props) +{ + g_autoptr (WpSpaPod) channelVolumes = NULL; + g_autoptr (WpSpaPod) channelMap = NULL; + + wp_spa_pod_get_object (props, NULL, + "mute", "b", &info->mute, + "volumeBase", "f", &info->base, + "volumeStep", "f", &info->step, + "channelVolumes", "P", &channelVolumes, + "channelMap", "P", &channelMap, + NULL); + + if (channelVolumes) + info->volume.channels = spa_pod_copy_array ( + wp_spa_pod_get_spa_pod (channelVolumes), SPA_TYPE_Float, + info->volume.values, SPA_AUDIO_MAX_CHANNELS); + + if (channelMap) + info->map.channels = spa_pod_copy_array ( + wp_spa_pod_get_spa_pod (channelMap), SPA_TYPE_Id, + info->map.map, SPA_AUDIO_MAX_CHANNELS); +} + +static void +collect_node_info (WpMixerApi * self, struct node_info *info, + WpPipewireObject * node) +{ + g_autoptr (WpPipewireObject) dev = NULL; + const gchar *str = NULL; + gboolean have_volume = FALSE; + + info->device_id = SPA_ID_INVALID; + info->route_index = -1; + info->route_device = -1; + + if ((str = wp_pipewire_object_get_property (node, PW_KEY_DEVICE_ID))) { + dev = wp_object_manager_lookup (self->om, WP_TYPE_DEVICE, + WP_CONSTRAINT_TYPE_G_PROPERTY, "bound-id", "=s", str, NULL); + } + + if (dev && (str = wp_pipewire_object_get_property (node, "card.profile.device"))) { + gint32 p_device = atoi (str); + g_autoptr (WpIterator) it = NULL; + g_auto (GValue) val = G_VALUE_INIT; + + it = wp_pipewire_object_enum_params_sync (dev, "Route", NULL); + for (; wp_iterator_next (it, &val); g_value_unset (&val)) { + WpSpaPod *param = g_value_get_boxed (&val); + gint32 r_index, r_device; + g_autoptr (WpSpaPod) props = NULL; + + if (!wp_spa_pod_get_object (param, NULL, + "index", "i", &r_index, + "device", "i", &r_device, + "props", "P", &props, + NULL)) + continue; + if (r_device != p_device) + continue; + + if (props) { + info->device_id = wp_proxy_get_bound_id (WP_PROXY (dev)); + info->route_index = r_index; + info->route_device = r_device; + node_info_fill (info, props); + have_volume = TRUE; + } + } + } + + if (!have_volume) { + g_autoptr (WpIterator) it = NULL; + g_auto (GValue) val = G_VALUE_INIT; + + it = wp_pipewire_object_enum_params_sync (node, "Props", NULL); + for (; wp_iterator_next (it, &val); g_value_unset (&val)) { + WpSpaPod *param = g_value_get_boxed (&val); + node_info_fill (info, param); + } + } +} + +static void on_objects_changed (WpObjectManager * om, WpMixerApi * self); + +static void +on_sync_done (WpCore * core, GAsyncResult * res, WpMixerApi * self) +{ + g_autoptr (GError) error = NULL; + if (!wp_core_sync_finish (core, res, &error)) + wp_warning_object (core, "sync error: %s", error->message); + if (self->om) { + on_objects_changed (self->om, self); + } +} + +static void +on_params_changed (WpPipewireObject * obj, guint param_id, WpMixerApi * self) +{ + if ((WP_IS_NODE (obj) && param_id == SPA_PARAM_Props) || + (WP_IS_DEVICE (obj) && param_id == SPA_PARAM_Route)) { + g_autoptr (WpCore) core = wp_object_get_core (WP_OBJECT (self)); + wp_core_sync (core, NULL, (GAsyncReadyCallback) on_sync_done, self); + } +} + +static void +on_objects_changed (WpObjectManager * om, WpMixerApi * self) +{ + g_autoptr (WpIterator) it = + wp_object_manager_new_filtered_iterator (om, WP_TYPE_NODE, NULL); + g_auto (GValue) val = G_VALUE_INIT; + GHashTableIter infos_it; + struct node_info *info; + struct node_info old; + + self->seq++; + + for (; wp_iterator_next (it, &val); g_value_unset (&val)) { + WpPipewireObject *node = g_value_get_object (&val); + guint id = wp_proxy_get_bound_id (WP_PROXY (node)); + + info = g_hash_table_lookup (self->node_infos, GUINT_TO_POINTER (id)); + if (!info) { + info = g_slice_new0 (struct node_info); + g_hash_table_insert (self->node_infos, GUINT_TO_POINTER (id), info); + } + info->seq = self->seq; + + old = *info; + collect_node_info (self, info, node); + if (memcmp (&old, info, sizeof (struct node_info)) != 0) { + wp_debug_object (self, "node %u changed volume props", id); + g_signal_emit (self, signals[SIGNAL_CHANGED], 0, id); + } + } + + /* remove node_info of nodes that were removed from the object manager */ + g_hash_table_iter_init (&infos_it, self->node_infos); + while (g_hash_table_iter_next (&infos_it, NULL, (gpointer *) &info)) { + if (info->seq != self->seq) + g_hash_table_iter_remove (&infos_it); + } +} + +static void +on_object_added (WpObjectManager * om, WpProxy * obj, WpMixerApi * self) +{ + g_signal_connect (obj, "params-changed", G_CALLBACK (on_params_changed), self); +} + +static void +on_object_removed (WpObjectManager * om, WpProxy * obj, WpMixerApi * self) +{ + g_signal_handlers_disconnect_by_func (obj, G_CALLBACK (on_params_changed), self); +} + +static void +node_info_free (gpointer info) +{ + g_slice_free (struct node_info, info); +} + +static void +on_om_installed (WpObjectManager * om, WpMixerApi * self) +{ + wp_object_update_features (WP_OBJECT (self), WP_PLUGIN_FEATURE_ENABLED, 0); +} + +static void +wp_mixer_api_enable (WpPlugin * plugin, WpTransition * transition) +{ + WpMixerApi * self = WP_MIXER_API (plugin); + g_autoptr (WpCore) core = wp_object_get_core (WP_OBJECT (plugin)); + g_return_if_fail (core); + + self->node_infos = g_hash_table_new_full (g_direct_hash, g_direct_equal, + NULL, node_info_free); + + self->om = wp_object_manager_new (); + wp_object_manager_add_interest (self->om, WP_TYPE_NODE, + WP_CONSTRAINT_TYPE_PW_GLOBAL_PROPERTY, "media.class", "#s", "*Audio*", + NULL); + wp_object_manager_add_interest (self->om, WP_TYPE_DEVICE, + WP_CONSTRAINT_TYPE_PW_GLOBAL_PROPERTY, "media.class", "=s", "Audio/Device", + NULL); + wp_object_manager_request_object_features (self->om, + WP_TYPE_GLOBAL_PROXY, WP_OBJECT_FEATURES_ALL); + g_signal_connect_object (self->om, "objects-changed", + G_CALLBACK (on_objects_changed), self, 0); + g_signal_connect_object (self->om, "object-added", + G_CALLBACK (on_object_added), self, 0); + g_signal_connect_object (self->om, "object-removed", + G_CALLBACK (on_object_removed), 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_mixer_api_disable (WpPlugin * plugin) +{ + WpMixerApi * self = WP_MIXER_API (plugin); + + { + g_autoptr (WpIterator) it = wp_object_manager_new_iterator (self->om); + g_auto (GValue) val = G_VALUE_INIT; + + for (; wp_iterator_next (it, &val); g_value_unset (&val)) { + WpProxy *obj = g_value_get_object (&val); + on_object_removed (self->om, obj, self); + } + } + + g_clear_object (&self->om); + g_clear_pointer (&self->node_infos, g_hash_table_unref); +} + +static gboolean +wp_mixer_api_set_volume (WpMixerApi * self, guint32 id, GVariant * vvolume) +{ + struct node_info *info = + g_hash_table_lookup (self->node_infos, GUINT_TO_POINTER (id)); + struct volume new_volume = {0}; + gint mute = -1; + WpSpaIdTable t_audioChannel = + wp_spa_id_table_from_name ("Spa:Enum:AudioChannel"); + + if (!info || !vvolume) + return FALSE; + + if (g_variant_is_of_type (vvolume, G_VARIANT_TYPE_DOUBLE)) { + gdouble val = g_variant_get_double (vvolume); + new_volume = info->volume; + for (uint i = 0; i < new_volume.channels; i++) + new_volume.values[i] = val; + } + else if (g_variant_is_of_type (vvolume, G_VARIANT_TYPE_VARDICT)) { + GVariantIter iter; + const gchar *idx_str; + GVariant *v; + + g_variant_lookup (vvolume, "mute", "b", &mute); + if (g_variant_lookup (vvolume, "channelVolumes", "a{sv}", &iter)) { + /* keep the existing volume values for unspecified channels */ + new_volume = info->volume; + + while (g_variant_iter_loop (&iter, "{&sv}", &idx_str, &v)) { + guint index = atoi (idx_str); + const gchar *channel_str = NULL; + WpSpaIdValue channel = NULL; + gdouble val; + + if (g_variant_lookup (v, "channel", "&s", &channel_str)) { + channel = wp_spa_id_table_find_value_from_short_name ( + t_audioChannel, channel_str); + if (!channel) + wp_message_object (self, "invalid channel: %s", channel_str); + } + + if (channel) { + for (uint i = 0; i < info->map.channels; i++) + if (info->map.map[i] == wp_spa_id_value_number (channel)) { + index = i; + break; + } + } + + if (index >= MIN(new_volume.channels, SPA_AUDIO_MAX_CHANNELS)) { + wp_message_object (self, "invalid channel index: %u", index); + continue; + } + + if (g_variant_lookup (v, "volume", "d", &val)) { + new_volume.values[index] = val; + } + } + } + } else { + return FALSE; + } + + /* set param */ + g_autoptr (WpSpaPod) props = NULL; + g_autoptr (WpSpaPodBuilder) b = + wp_spa_pod_builder_new_object ("Spa:Pod:Object:Param:Props", "Props"); + + if (new_volume.channels > 0) + wp_spa_pod_builder_add (b, "channelVolumes", "a", + sizeof(float), SPA_TYPE_Float, + new_volume.channels, new_volume.values, NULL); + if (mute != -1) + wp_spa_pod_builder_add (b, "mute", "b", (mute == TRUE), NULL); + + props = wp_spa_pod_builder_end (b); + + if (info->device_id != SPA_ID_INVALID) { + WpPipewireObject *device = wp_object_manager_lookup (self->om, + WP_TYPE_DEVICE, WP_CONSTRAINT_TYPE_G_PROPERTY, + "bound-id", "=u", info->device_id, NULL); + g_return_val_if_fail (device != NULL, FALSE); + + wp_pipewire_object_set_param (device, "Route", 0, wp_spa_pod_new_object ( + "Spa:Pod:Object:Param:Route", "Route", + "index", "i", info->route_index, + "device", "i", info->route_device, + "props", "P", props, + "save", "b", true, + NULL)); + } else { + WpPipewireObject *node = wp_object_manager_lookup (self->om, + WP_TYPE_NODE, WP_CONSTRAINT_TYPE_G_PROPERTY, + "bound-id", "=u", id, NULL); + g_return_val_if_fail (node != NULL, FALSE); + + wp_pipewire_object_set_param (node, "Props", 0, g_steal_pointer (&props)); + } + + return TRUE; +} + +static GVariant * +wp_mixer_api_get_volume (WpMixerApi * self, guint32 id) +{ + struct node_info *info = + g_hash_table_lookup (self->node_infos, GUINT_TO_POINTER (id)); + g_auto (GVariantBuilder) b = + G_VARIANT_BUILDER_INIT (G_VARIANT_TYPE_VARDICT); + g_auto (GVariantBuilder) b_vol = + G_VARIANT_BUILDER_INIT (G_VARIANT_TYPE_VARDICT); + WpSpaIdTable t_audioChannel = + wp_spa_id_table_from_name ("Spa:Enum:AudioChannel"); + + if (!info) + return NULL; + + g_variant_builder_add (&b, "{sv}", "id", g_variant_new_uint32 (id)); + g_variant_builder_add (&b, "{sv}", "mute", g_variant_new_boolean (info->mute)); + g_variant_builder_add (&b, "{sv}", "base", g_variant_new_double (info->base)); + g_variant_builder_add (&b, "{sv}", "step", g_variant_new_double (info->step)); + + for (guint i = 0; i < info->volume.channels; i++) { + gchar index_str[10]; + g_auto (GVariantBuilder) b_vol_nested = + G_VARIANT_BUILDER_INIT (G_VARIANT_TYPE_VARDICT); + + g_variant_builder_add (&b_vol_nested, "{sv}", + "volume", g_variant_new_double (info->volume.values[i])); + + if (i < info->map.channels) { + WpSpaIdValue v = + wp_spa_id_table_find_value (t_audioChannel, info->map.map[i]); + if (v) { + const gchar *channel_str = wp_spa_id_value_short_name (v); + g_variant_builder_add (&b_vol_nested, "{sv}", + "channel", g_variant_new_string (channel_str)); + } + } + + g_snprintf (index_str, 10, "%u", i); + g_variant_builder_add (&b_vol, "{sv}", index_str, + g_variant_builder_end (&b_vol_nested)); + } + + g_variant_builder_add (&b, "{sv}", + "channelVolumes", g_variant_builder_end (&b_vol)); + return g_variant_builder_end (&b); +} + +static void +wp_mixer_api_class_init (WpMixerApiClass * klass) +{ + WpPluginClass *plugin_class = (WpPluginClass *) klass; + + plugin_class->enable = wp_mixer_api_enable; + plugin_class->disable = wp_mixer_api_disable; + + signals[ACTION_SET_VOLUME] = g_signal_new_class_handler ( + "set-volume", G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, + (GCallback) wp_mixer_api_set_volume, + NULL, NULL, NULL, + G_TYPE_BOOLEAN, 2, G_TYPE_UINT, G_TYPE_VARIANT); + + signals[ACTION_GET_VOLUME] = g_signal_new_class_handler ( + "get-volume", G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION, + (GCallback) wp_mixer_api_get_volume, + NULL, NULL, NULL, + G_TYPE_VARIANT, 1, G_TYPE_UINT); + + signals[SIGNAL_CHANGED] = g_signal_new ( + "changed", G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 1, G_TYPE_UINT); +} + +WP_PLUGIN_EXPORT gboolean +wireplumber__module_init (WpCore * core, GVariant * args, GError ** error) +{ + wp_plugin_register (g_object_new (wp_mixer_api_get_type (), + "name", "mixer-api", + "core", core, + NULL)); + return TRUE; +}