mirror of
https://gitlab.freedesktop.org/pipewire/wireplumber.git
synced 2025-12-25 00:50:04 +01:00
Syslog calls this level "notice" and I prefer it because we use it to display significant messages that are not warnings, but they are not really "standard", as GLib wants them to be. There is nothing "standard" about log messages in general. Also, make these notice messages be enabled at debug level 2, together with warnings. The default log.level is 2 and it is a good idea to show notices by default too. Finally, show them in the log with "N" and also change criticals to be shown with "E", meaning "error"... Then promote G_LOG_LEVEL_ERROR messages to be shown with "F", meaning "fatal", because in fact these messages are always fatal and always call abort(). Still, keep the term "critical" in the functions to make sure that whoever uses them is aware that this level is only for critical conditions and not suitable to display any kind of error.
276 lines
7.4 KiB
C
276 lines
7.4 KiB
C
/* WirePlumber
|
|
*
|
|
* Copyright © 2021 Collabora Ltd.
|
|
* @author George Kiagiadakis <george.kiagiadakis@collabora.com>
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
#include "plugin.h"
|
|
#include "reserve-device.h"
|
|
#include "reserve-device-enums.h"
|
|
|
|
WP_LOG_TOPIC (log_topic_rd, "m-reserve-device")
|
|
|
|
G_DEFINE_TYPE (WpReserveDevicePlugin, wp_reserve_device_plugin, WP_TYPE_PLUGIN)
|
|
|
|
enum
|
|
{
|
|
ACTION_CREATE_RESERVATION,
|
|
ACTION_DESTROY_RESERVATION,
|
|
ACTION_GET_RESERVATION,
|
|
ACTION_GET_DBUS,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
static void
|
|
rd_unref (gpointer data)
|
|
{
|
|
WpReserveDevice *rd = data;
|
|
g_signal_emit_by_name (rd, "release");
|
|
g_object_unref (rd);
|
|
}
|
|
|
|
static void
|
|
clear_reservation (WpReserveDevicePlugin *self)
|
|
{
|
|
g_hash_table_remove_all (self->reserve_devices);
|
|
g_clear_object (&self->manager);
|
|
}
|
|
|
|
static void
|
|
on_dbus_state_changed (GObject * obj, GParamSpec * spec,
|
|
WpReserveDevicePlugin *self)
|
|
{
|
|
WpDBusState state = wp_dbus_get_state (self->dbus);
|
|
switch (state) {
|
|
case WP_DBUS_STATE_CONNECTED: {
|
|
g_autoptr (GDBusConnection) conn = NULL;
|
|
|
|
conn = wp_dbus_get_connection (self->dbus);
|
|
g_return_if_fail (conn);
|
|
|
|
self->manager = g_dbus_object_manager_server_new (
|
|
FDO_RESERVE_DEVICE1_PATH);
|
|
g_dbus_object_manager_server_set_connection (self->manager, conn);
|
|
break;
|
|
}
|
|
|
|
case WP_DBUS_STATE_CONNECTING:
|
|
case WP_DBUS_STATE_CLOSED:
|
|
clear_reservation (self);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_init (WpReserveDevicePlugin * self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_constructed (GObject *object)
|
|
{
|
|
WpReserveDevicePlugin *self = WP_RESERVE_DEVICE_PLUGIN (object);
|
|
g_autoptr (WpCore) core = wp_object_get_core (WP_OBJECT (self));
|
|
|
|
self->reserve_devices = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
NULL, rd_unref);
|
|
|
|
self->dbus = wp_dbus_get_instance (core, G_BUS_TYPE_SESSION);
|
|
g_signal_connect_object (self->dbus, "notify::state",
|
|
G_CALLBACK (on_dbus_state_changed), self, 0);
|
|
|
|
G_OBJECT_CLASS (wp_reserve_device_plugin_parent_class)->constructed (object);
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_finalize (GObject * object)
|
|
{
|
|
WpReserveDevicePlugin *self = WP_RESERVE_DEVICE_PLUGIN (object);
|
|
|
|
g_clear_pointer (&self->reserve_devices, g_hash_table_unref);
|
|
g_clear_object (&self->dbus);
|
|
|
|
G_OBJECT_CLASS (wp_reserve_device_plugin_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
on_dbus_activated (GObject * obj, GAsyncResult * res, gpointer user_data)
|
|
{
|
|
WpTransition * transition = WP_TRANSITION (user_data);
|
|
WpReserveDevicePlugin * self = wp_transition_get_source_object (transition);
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
if (!wp_object_activate_finish (WP_OBJECT (obj), res, &error)) {
|
|
wp_transition_return_error (transition, g_steal_pointer (&error));
|
|
return;
|
|
}
|
|
|
|
wp_object_update_features (WP_OBJECT (self), WP_PLUGIN_FEATURE_ENABLED, 0);
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_enable (WpPlugin * plugin, WpTransition * transition)
|
|
{
|
|
WpReserveDevicePlugin *self = WP_RESERVE_DEVICE_PLUGIN (plugin);
|
|
|
|
/* make sure dbus always activated */
|
|
g_return_if_fail (self->dbus);
|
|
wp_object_activate (WP_OBJECT (self->dbus), WP_OBJECT_FEATURES_ALL, NULL,
|
|
(GAsyncReadyCallback) on_dbus_activated, transition);
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_disable (WpPlugin * plugin)
|
|
{
|
|
WpReserveDevicePlugin *self = WP_RESERVE_DEVICE_PLUGIN (plugin);
|
|
|
|
clear_reservation (self);
|
|
|
|
wp_object_update_features (WP_OBJECT (self), 0, WP_PLUGIN_FEATURE_ENABLED);
|
|
}
|
|
|
|
static gpointer
|
|
wp_reserve_device_plugin_create_reservation (WpReserveDevicePlugin *self,
|
|
const gchar *name, const gchar *app_name, const gchar *app_dev_name,
|
|
gint priority)
|
|
{
|
|
WpDBusState state = wp_dbus_get_state (self->dbus);
|
|
if (state != WP_DBUS_STATE_CONNECTED) {
|
|
wp_notice_object (self, "not connected to D-Bus");
|
|
return NULL;
|
|
}
|
|
|
|
WpReserveDevice *rd = g_object_new (wp_reserve_device_get_type (),
|
|
"plugin", self,
|
|
"name", name,
|
|
"application-name", app_name,
|
|
"application-device-name", app_dev_name,
|
|
"priority", priority,
|
|
NULL);
|
|
|
|
/* use rd->name to avoid copying @em name again */
|
|
g_hash_table_replace (self->reserve_devices, rd->name, rd);
|
|
|
|
return g_object_ref (rd);
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_destroy_reservation (WpReserveDevicePlugin *self,
|
|
const gchar *name)
|
|
{
|
|
WpDBusState state = wp_dbus_get_state (self->dbus);
|
|
if (state != WP_DBUS_STATE_CONNECTED) {
|
|
wp_notice_object (self, "not connected to D-Bus");
|
|
return;
|
|
}
|
|
g_hash_table_remove (self->reserve_devices, name);
|
|
}
|
|
|
|
static gpointer
|
|
wp_reserve_device_plugin_get_reservation (WpReserveDevicePlugin *self,
|
|
const gchar *name)
|
|
{
|
|
WpDBusState state = wp_dbus_get_state (self->dbus);
|
|
if (state != WP_DBUS_STATE_CONNECTED) {
|
|
wp_notice_object (self, "not connected to D-Bus");
|
|
return NULL;
|
|
}
|
|
|
|
WpReserveDevice *rd = g_hash_table_lookup (self->reserve_devices, name);
|
|
return rd ? g_object_ref (rd) : NULL;
|
|
}
|
|
|
|
static gpointer
|
|
wp_reserve_device_plugin_get_dbus (WpReserveDevicePlugin *self)
|
|
{
|
|
return self->dbus ? g_object_ref (self->dbus) : NULL;
|
|
}
|
|
|
|
static void
|
|
wp_reserve_device_plugin_class_init (WpReserveDevicePluginClass * klass)
|
|
{
|
|
GObjectClass *object_class = (GObjectClass *) klass;
|
|
WpPluginClass *plugin_class = (WpPluginClass *) klass;
|
|
|
|
object_class->constructed = wp_reserve_device_plugin_constructed;
|
|
object_class->finalize = wp_reserve_device_plugin_finalize;
|
|
|
|
plugin_class->enable = wp_reserve_device_plugin_enable;
|
|
plugin_class->disable = wp_reserve_device_plugin_disable;
|
|
|
|
/**
|
|
* WpReserveDevicePlugin::create-reservation:
|
|
*
|
|
* @brief
|
|
* @em name:
|
|
* @em app_name:
|
|
* @em app_dev_name:
|
|
* @em priority:
|
|
*
|
|
* Returns: (transfer full): the reservation object
|
|
*/
|
|
signals[ACTION_CREATE_RESERVATION] = g_signal_new_class_handler (
|
|
"create-reservation", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
(GCallback) wp_reserve_device_plugin_create_reservation,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_OBJECT, 4, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT);
|
|
|
|
/**
|
|
* WpReserveDevicePlugin::destroy-reservation:
|
|
*
|
|
* @brief
|
|
* @em name:
|
|
*
|
|
*/
|
|
signals[ACTION_DESTROY_RESERVATION] = g_signal_new_class_handler (
|
|
"destroy-reservation", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
(GCallback) wp_reserve_device_plugin_destroy_reservation,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1, G_TYPE_STRING);
|
|
|
|
/**
|
|
* WpReserveDevicePlugin::get-reservation:
|
|
*
|
|
* @brief
|
|
* @em name:
|
|
*
|
|
* Returns: (transfer full): the reservation object
|
|
*/
|
|
signals[ACTION_GET_RESERVATION] = g_signal_new_class_handler (
|
|
"get-reservation", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
(GCallback) wp_reserve_device_plugin_get_reservation,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_OBJECT, 1, G_TYPE_STRING);
|
|
|
|
/**
|
|
* WpReserveDevicePlugin::get-dbus:
|
|
*
|
|
* Returns: (transfer full): the dbus object
|
|
*/
|
|
signals[ACTION_GET_DBUS] = g_signal_new_class_handler (
|
|
"get-dbus", G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
(GCallback) wp_reserve_device_plugin_get_dbus,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_OBJECT, 0);
|
|
|
|
}
|
|
|
|
WP_PLUGIN_EXPORT GObject *
|
|
wireplumber__module_init (WpCore * core, GVariant * args, GError ** error)
|
|
{
|
|
return G_OBJECT (g_object_new (wp_reserve_device_plugin_get_type (),
|
|
"name", "reserve-device",
|
|
"core", core,
|
|
NULL));
|
|
}
|