587dbb
From 2e500304e304e45042a59855319ff0379b1978b3 Mon Sep 17 00:00:00 2001
587dbb
From: Bastien Nocera <hadess@hadess.net>
587dbb
Date: Tue, 27 Jul 2021 17:24:17 +0200
587dbb
Subject: [PATCH 1/4] tests: Remove unused constant in GMemoryMonitor test
587dbb
587dbb
---
587dbb
 gio/tests/memory-monitor-dbus.py.in | 3 ---
587dbb
 1 file changed, 3 deletions(-)
587dbb
587dbb
diff --git a/gio/tests/memory-monitor-dbus.py.in b/gio/tests/memory-monitor-dbus.py.in
587dbb
index cd16cf4e3..7823e7309 100755
587dbb
--- a/gio/tests/memory-monitor-dbus.py.in
587dbb
+++ b/gio/tests/memory-monitor-dbus.py.in
587dbb
@@ -31,9 +31,6 @@ try:
587dbb
 
587dbb
     dbus.mainloop.glib.DBusGMainLoop(set_as_default=True)
587dbb
 
587dbb
-    # XDG_DESKTOP_PORTAL_PATH = os.path.expanduser("~/.cache/jhbuild/build/xdg-desktop-portal/xdg-desktop-portal")
587dbb
-    XDG_DESKTOP_PORTAL_PATH = "@libexecdir@/xdg-desktop-portal"
587dbb
-
587dbb
     class TestLowMemoryMonitor(dbusmock.DBusTestCase):
587dbb
         '''Test GMemoryMonitorDBus'''
587dbb
 
587dbb
-- 
587dbb
GitLab
587dbb
587dbb
587dbb
From a7000cd989438b01e599b2cfa8b6d5a360bfd102 Mon Sep 17 00:00:00 2001
587dbb
From: Bastien Nocera <hadess@hadess.net>
587dbb
Date: Wed, 28 Jul 2021 15:10:16 +0200
587dbb
Subject: [PATCH 2/4] gio: g_clear_signal_handler() can handle NULL args
587dbb
587dbb
---
587dbb
 gio/gmemorymonitordbus.c | 6 ++----
587dbb
 1 file changed, 2 insertions(+), 4 deletions(-)
587dbb
587dbb
diff --git a/gio/gmemorymonitordbus.c b/gio/gmemorymonitordbus.c
587dbb
index a34a58d3b..08dc53df1 100644
587dbb
--- a/gio/gmemorymonitordbus.c
587dbb
+++ b/gio/gmemorymonitordbus.c
587dbb
@@ -115,8 +115,7 @@ lmm_vanished_cb (GDBusConnection *connection,
587dbb
 {
587dbb
   GMemoryMonitorDBus *dbus = user_data;
587dbb
 
587dbb
-  if (dbus->proxy != NULL)
587dbb
-    g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
+  g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
   g_clear_object (&dbus->proxy);
587dbb
 }
587dbb
 
587dbb
@@ -143,8 +142,7 @@ g_memory_monitor_dbus_finalize (GObject *object)
587dbb
 {
587dbb
   GMemoryMonitorDBus *dbus = G_MEMORY_MONITOR_DBUS (object);
587dbb
 
587dbb
-  if (dbus->proxy != NULL)
587dbb
-    g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
+  g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
   g_clear_object (&dbus->proxy);
587dbb
   g_clear_handle_id (&dbus->watch_id, g_bus_unwatch_name);
587dbb
 
587dbb
-- 
587dbb
GitLab
587dbb
587dbb
587dbb
From 92399e7114e590f0371b1a5d71f478f840cb4074 Mon Sep 17 00:00:00 2001
587dbb
From: Bastien Nocera <hadess@hadess.net>
587dbb
Date: Wed, 28 Jul 2021 15:30:15 +0200
587dbb
Subject: [PATCH 3/4] gio: Do not block when low-memory-monitor daemon appears
587dbb
587dbb
---
587dbb
 gio/gmemorymonitordbus.c | 42 +++++++++++++++++++++++++++-------------
587dbb
 1 file changed, 29 insertions(+), 13 deletions(-)
587dbb
587dbb
diff --git a/gio/gmemorymonitordbus.c b/gio/gmemorymonitordbus.c
587dbb
index 08dc53df1..739b83214 100644
587dbb
--- a/gio/gmemorymonitordbus.c
587dbb
+++ b/gio/gmemorymonitordbus.c
587dbb
@@ -25,6 +25,7 @@
587dbb
 #include "giomodule-priv.h"
587dbb
 #include "glibintl.h"
587dbb
 #include "glib/gstdio.h"
587dbb
+#include "gcancellable.h"
587dbb
 #include "gdbusproxy.h"
587dbb
 #include "gdbusnamewatching.h"
587dbb
 
587dbb
@@ -38,6 +39,7 @@ struct _GMemoryMonitorDBus
587dbb
   GObject parent_instance;
587dbb
 
587dbb
   guint watch_id;
587dbb
+  GCancellable *cancellable;
587dbb
   GDBusProxy *proxy;
587dbb
   gulong signal_id;
587dbb
 };
587dbb
@@ -77,24 +79,15 @@ proxy_signal_cb (GDBusProxy         *proxy,
587dbb
 }
587dbb
 
587dbb
 static void
587dbb
-lmm_appeared_cb (GDBusConnection *connection,
587dbb
-                 const gchar     *name,
587dbb
-                 const gchar     *name_owner,
587dbb
-                 gpointer         user_data)
587dbb
+lmm_proxy_cb (GObject      *source_object,
587dbb
+              GAsyncResult *res,
587dbb
+              gpointer      user_data)
587dbb
 {
587dbb
   GMemoryMonitorDBus *dbus = user_data;
587dbb
   GDBusProxy *proxy;
587dbb
   GError *error = NULL;
587dbb
 
587dbb
-  proxy = g_dbus_proxy_new_for_bus_sync (G_BUS_TYPE_SYSTEM,
587dbb
-                                         G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START,
587dbb
-                                         NULL,
587dbb
-                                         "org.freedesktop.LowMemoryMonitor",
587dbb
-                                         "/org/freedesktop/LowMemoryMonitor",
587dbb
-                                         "org.freedesktop.LowMemoryMonitor",
587dbb
-                                         NULL,
587dbb
-                                         &error);
587dbb
-
587dbb
+  proxy = g_dbus_proxy_new_finish (res, &error);
587dbb
   if (!proxy)
587dbb
     {
587dbb
       g_debug ("Failed to create LowMemoryMonitor D-Bus proxy: %s",
587dbb
@@ -106,6 +99,26 @@ lmm_appeared_cb (GDBusConnection *connection,
587dbb
   dbus->signal_id = g_signal_connect (G_OBJECT (proxy), "g-signal",
587dbb
                                       G_CALLBACK (proxy_signal_cb), dbus);
587dbb
   dbus->proxy = proxy;
587dbb
+
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+lmm_appeared_cb (GDBusConnection *connection,
587dbb
+                 const gchar     *name,
587dbb
+                 const gchar     *name_owner,
587dbb
+                 gpointer         user_data)
587dbb
+{
587dbb
+  GMemoryMonitorDBus *dbus = user_data;
587dbb
+
587dbb
+  g_dbus_proxy_new (connection,
587dbb
+                    G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START,
587dbb
+                    NULL,
587dbb
+                    "org.freedesktop.LowMemoryMonitor",
587dbb
+                    "/org/freedesktop/LowMemoryMonitor",
587dbb
+                    "org.freedesktop.LowMemoryMonitor",
587dbb
+                    dbus->cancellable,
587dbb
+                    lmm_proxy_cb,
587dbb
+                    dbus);
587dbb
 }
587dbb
 
587dbb
 static void
587dbb
@@ -126,6 +139,7 @@ g_memory_monitor_dbus_initable_init (GInitable     *initable,
587dbb
 {
587dbb
   GMemoryMonitorDBus *dbus = G_MEMORY_MONITOR_DBUS (initable);
587dbb
 
587dbb
+  dbus->cancellable = g_cancellable_new ();
587dbb
   dbus->watch_id = g_bus_watch_name (G_BUS_TYPE_SYSTEM,
587dbb
                                      "org.freedesktop.LowMemoryMonitor",
587dbb
                                      G_BUS_NAME_WATCHER_FLAGS_AUTO_START,
587dbb
@@ -142,6 +156,8 @@ g_memory_monitor_dbus_finalize (GObject *object)
587dbb
 {
587dbb
   GMemoryMonitorDBus *dbus = G_MEMORY_MONITOR_DBUS (object);
587dbb
 
587dbb
+  g_cancellable_cancel (dbus->cancellable);
587dbb
+  g_clear_object (&dbus->cancellable);
587dbb
   g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
   g_clear_object (&dbus->proxy);
587dbb
   g_clear_handle_id (&dbus->watch_id, g_bus_unwatch_name);
587dbb
-- 
587dbb
GitLab
587dbb
587dbb
587dbb
From 889bdb994fed44344a84ad01aa5633a1b1b62b19 Mon Sep 17 00:00:00 2001
587dbb
From: Patrick Griffis <pgriffis@igalia.com>
587dbb
Date: Tue, 20 Jul 2021 16:04:31 -0500
587dbb
Subject: [PATCH 4/4] Add GPowerProfileMonitor
587dbb
587dbb
---
587dbb
 docs/reference/gio/gio-docs.xml            |   1 +
587dbb
 docs/reference/gio/gio-sections-common.txt |  18 ++
587dbb
 docs/reference/gio/meson.build             |   1 +
587dbb
 gio/gio.h                                  |   1 +
587dbb
 gio/giomodule.c                            |   7 +
587dbb
 gio/gpowerprofilemonitor.c                 | 141 ++++++++++++
587dbb
 gio/gpowerprofilemonitor.h                 |  63 ++++++
587dbb
 gio/gpowerprofilemonitordbus.c             | 240 +++++++++++++++++++++
587dbb
 gio/gpowerprofilemonitordbus.h             |  32 +++
587dbb
 gio/meson.build                            |   3 +
587dbb
 gio/tests/meson.build                      |   1 +
587dbb
 gio/tests/power-profile-monitor.c          |  79 +++++++
587dbb
 12 files changed, 587 insertions(+)
587dbb
 create mode 100644 gio/gpowerprofilemonitor.c
587dbb
 create mode 100644 gio/gpowerprofilemonitor.h
587dbb
 create mode 100644 gio/gpowerprofilemonitordbus.c
587dbb
 create mode 100644 gio/gpowerprofilemonitordbus.h
587dbb
 create mode 100644 gio/tests/power-profile-monitor.c
587dbb
587dbb
diff --git a/docs/reference/gio/gio-docs.xml b/docs/reference/gio/gio-docs.xml
587dbb
index a09d6d31d..b01133900 100644
587dbb
--- a/docs/reference/gio/gio-docs.xml
587dbb
+++ b/docs/reference/gio/gio-docs.xml
587dbb
@@ -238,6 +238,7 @@
587dbb
         <xi:include href="xml/gmenuexporter.xml"/>
587dbb
         <xi:include href="xml/gdbusmenumodel.xml"/>
587dbb
         <xi:include href="xml/gnotification.xml"/>
587dbb
+        <xi:include href="xml/gpowerprofilemonitor.xml"/>
587dbb
     </chapter>
587dbb
     <chapter id="extending">
587dbb
         <title>Extending GIO</title>
587dbb
diff --git a/docs/reference/gio/gio-sections-common.txt b/docs/reference/gio/gio-sections-common.txt
587dbb
index 250491a42..a7addedc2 100644
587dbb
--- a/docs/reference/gio/gio-sections-common.txt
587dbb
+++ b/docs/reference/gio/gio-sections-common.txt
587dbb
@@ -4247,6 +4247,24 @@ G_NETWORK_MONITOR_GET_INTERFACE
587dbb
 g_network_connectivity_get_type
587dbb
 </SECTION>
587dbb
 
587dbb
+<SECTION>
587dbb
+<FILE>gpowerprofilemonitor</FILE>
587dbb
+<TITLE>GPowerProfileMonitor</TITLE>
587dbb
+GPowerProfileMonitor
587dbb
+GPowerProfileMonitorInterface
587dbb
+G_POWER_PROFILE_MONITOR_EXTENSION_POINT_NAME
587dbb
+g_power_profile_monitor_dup_default
587dbb
+g_power_profile_monitor_get_power_saver_enabled
587dbb
+<SUBSECTION Standard>
587dbb
+g_power_profile_monitor_get_type
587dbb
+G_TYPE_POWER_PROFILE_MONITOR
587dbb
+G_POWER_PROFILE_MONITOR
587dbb
+G_IS_POWER_PROFILE_MONITOR
587dbb
+G_POWER_PROFILE_MONITOR_GET_INTERFACE
587dbb
+G_TYPE_POWER_PROFILE_LEVEL
587dbb
+g_power_profile_level_get_type
587dbb
+</SECTION>
587dbb
+
587dbb
 <SECTION>
587dbb
 <FILE>gmenuexporter</FILE>
587dbb
 g_dbus_connection_export_menu_model
587dbb
diff --git a/docs/reference/gio/meson.build b/docs/reference/gio/meson.build
587dbb
index 4d0364819..fbabd25ca 100644
587dbb
--- a/docs/reference/gio/meson.build
587dbb
+++ b/docs/reference/gio/meson.build
587dbb
@@ -65,6 +65,7 @@ if get_option('gtk_doc')
587dbb
     'gopenuriportal.h',
587dbb
     'gpollfilemonitor.h',
587dbb
     'gportalsupport.h',
587dbb
+    'gpowerprofilemonitordbus.h',
587dbb
     'gproxyresolverportal.h',
587dbb
     'gregistrysettingsbackend.h',
587dbb
     'gresourcefile.h',
587dbb
diff --git a/gio/gio.h b/gio/gio.h
587dbb
index f5d2dd5a3..e9afab666 100644
587dbb
--- a/gio/gio.h
587dbb
+++ b/gio/gio.h
587dbb
@@ -120,6 +120,7 @@
587dbb
 #include <gio/gpollableinputstream.h>
587dbb
 #include <gio/gpollableoutputstream.h>
587dbb
 #include <gio/gpollableutils.h>
587dbb
+#include <gio/gpowerprofilemonitor.h>
587dbb
 #include <gio/gpropertyaction.h>
587dbb
 #include <gio/gproxy.h>
587dbb
 #include <gio/gproxyaddress.h>
587dbb
diff --git a/gio/giomodule.c b/gio/giomodule.c
587dbb
index c1d451b5c..dfd895717 100644
587dbb
--- a/gio/giomodule.c
587dbb
+++ b/gio/giomodule.c
587dbb
@@ -48,6 +48,8 @@
587dbb
 #include "gmemorymonitor.h"
587dbb
 #include "gmemorymonitorportal.h"
587dbb
 #include "gmemorymonitordbus.h"
587dbb
+#include "gpowerprofilemonitor.h"
587dbb
+#include "gpowerprofilemonitordbus.h"
587dbb
 #ifdef G_OS_WIN32
587dbb
 #include "gregistrysettingsbackend.h"
587dbb
 #include "giowin32-priv.h"
587dbb
@@ -1077,6 +1079,7 @@ extern GType _g_network_monitor_nm_get_type (void);
587dbb
 
587dbb
 extern GType g_memory_monitor_dbus_get_type (void);
587dbb
 extern GType g_memory_monitor_portal_get_type (void);
587dbb
+extern GType g_power_profile_monitor_dbus_get_type (void);
587dbb
 
587dbb
 #ifdef G_OS_UNIX
587dbb
 extern GType g_fdo_notification_backend_get_type (void);
587dbb
@@ -1187,6 +1190,9 @@ _g_io_modules_ensure_extension_points_registered (void)
587dbb
 
587dbb
       ep = g_io_extension_point_register (G_MEMORY_MONITOR_EXTENSION_POINT_NAME);
587dbb
       g_io_extension_point_set_required_type (ep, G_TYPE_MEMORY_MONITOR);
587dbb
+
587dbb
+      ep = g_io_extension_point_register (G_POWER_PROFILE_MONITOR_EXTENSION_POINT_NAME);
587dbb
+      g_io_extension_point_set_required_type (ep, G_TYPE_POWER_PROFILE_MONITOR);
587dbb
     }
587dbb
   
587dbb
   G_UNLOCK (registered_extensions);
587dbb
@@ -1272,6 +1278,7 @@ _g_io_modules_ensure_loaded (void)
587dbb
       g_type_ensure (g_null_settings_backend_get_type ());
587dbb
       g_type_ensure (g_memory_settings_backend_get_type ());
587dbb
       g_type_ensure (g_keyfile_settings_backend_get_type ());
587dbb
+      g_type_ensure (g_power_profile_monitor_dbus_get_type ());
587dbb
 #if defined(HAVE_INOTIFY_INIT1)
587dbb
       g_type_ensure (g_inotify_file_monitor_get_type ());
587dbb
 #endif
587dbb
diff --git a/gio/gpowerprofilemonitor.c b/gio/gpowerprofilemonitor.c
587dbb
new file mode 100644
587dbb
index 000000000..f5028b3e8
587dbb
--- /dev/null
587dbb
+++ b/gio/gpowerprofilemonitor.c
587dbb
@@ -0,0 +1,141 @@
587dbb
+/* GIO - GLib Input, Output and Streaming Library
587dbb
+ *
587dbb
+ * Copyright 2019 Red Hat, Inc
587dbb
+ * Copyright 2021 Igalia S.L.
587dbb
+ *
587dbb
+ * This library is free software; you can redistribute it and/or
587dbb
+ * modify it under the terms of the GNU Lesser General Public
587dbb
+ * License as published by the Free Software Foundation; either
587dbb
+ * version 2.1 of the License, or (at your option) any later version.
587dbb
+ *
587dbb
+ * This library is distributed in the hope that it will be useful,
587dbb
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
587dbb
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
587dbb
+ * Lesser General Public License for more details.
587dbb
+ *
587dbb
+ * You should have received a copy of the GNU Lesser General
587dbb
+ * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
587dbb
+ */
587dbb
+
587dbb
+#include "config.h"
587dbb
+#include "glib.h"
587dbb
+#include "glibintl.h"
587dbb
+
587dbb
+#include "gpowerprofilemonitor.h"
587dbb
+#include "ginetaddress.h"
587dbb
+#include "ginetsocketaddress.h"
587dbb
+#include "ginitable.h"
587dbb
+#include "gioenumtypes.h"
587dbb
+#include "giomodule-priv.h"
587dbb
+#include "gtask.h"
587dbb
+
587dbb
+/**
587dbb
+ * SECTION:gpowerprofilemonitor
587dbb
+ * @title: GPowerProfileMonitor
587dbb
+ * @short_description: Power profile monitor
587dbb
+ * @include: gio/gio.h
587dbb
+ *
587dbb
+ * #GPowerProfileMonitor makes it possible for applications as well as OS components
587dbb
+ * to monitor system power profiles and act upon them. It currently only exports
587dbb
+ * whether the system is in “Power Saver” mode (known as “Low Power” mode on
587dbb
+ * some systems).
587dbb
+ *
587dbb
+ * When in “Low Power” mode, it is recommended that applications:
587dbb
+ * - disabling automatic downloads
587dbb
+ * - reduce the rate of refresh from online sources such as calendar or
587dbb
+ *   email synchronisation
587dbb
+ * - if the application has expensive visual effects, reduce them
587dbb
+ *
587dbb
+ * It is also likely that OS components providing services to applications will
587dbb
+ * lower their own background activity, for the sake of the system.
587dbb
+ *
587dbb
+ * There are a variety of tools that exist for power consumption analysis, but those
587dbb
+ * usually depend on the OS and hardware used. On Linux, one could use `upower` to
587dbb
+ * monitor the battery discharge rate, `powertop` to check on the background activity
587dbb
+ * or activity at all), `sysprof` to inspect CPU usage, and `intel_gpu_time` to
587dbb
+ * profile GPU usage.
587dbb
+ *
587dbb
+ * Don't forget to disconnect the #GPowerProfileMonitor::notify::power-saver-enabled
587dbb
+ * signal, and unref the #GPowerProfileMonitor itself when exiting.
587dbb
+ *
587dbb
+ * Since: 2.70
587dbb
+ */
587dbb
+
587dbb
+/**
587dbb
+ * GPowerProfileMonitor:
587dbb
+ *
587dbb
+ * #GPowerProfileMonitor monitors system power profile and notifies on
587dbb
+ * changes.
587dbb
+ *
587dbb
+ * Since: 2.70
587dbb
+ */
587dbb
+
587dbb
+/**
587dbb
+ * GPowerProfileMonitorInterface:
587dbb
+ * @g_iface: The parent interface.
587dbb
+ *
587dbb
+ * The virtual function table for #GPowerProfileMonitor.
587dbb
+ *
587dbb
+ * Since: 2.70
587dbb
+ */
587dbb
+
587dbb
+G_DEFINE_INTERFACE_WITH_CODE (GPowerProfileMonitor, g_power_profile_monitor, G_TYPE_OBJECT,
587dbb
+                              g_type_interface_add_prerequisite (g_define_type_id, G_TYPE_INITABLE))
587dbb
+
587dbb
+
587dbb
+/**
587dbb
+ * g_power_profile_monitor_dup_default:
587dbb
+ *
587dbb
+ * Gets a reference to the default #GPowerProfileMonitor for the system.
587dbb
+ *
587dbb
+ * Returns: (not nullable) (transfer full): a new reference to the default #GPowerProfileMonitor
587dbb
+ *
587dbb
+ * Since: 2.70
587dbb
+ */
587dbb
+GPowerProfileMonitor *
587dbb
+g_power_profile_monitor_dup_default (void)
587dbb
+{
587dbb
+  return g_object_ref (_g_io_module_get_default (G_POWER_PROFILE_MONITOR_EXTENSION_POINT_NAME,
587dbb
+                                                 "GIO_USE_POWER_PROFILE_MONITOR",
587dbb
+                                                 NULL));
587dbb
+}
587dbb
+
587dbb
+/**
587dbb
+ * g_power_profile_monitor_get_power_saver_enabled:
587dbb
+ * @monitor: a #GPowerProfileMonitor
587dbb
+ *
587dbb
+ * Gets whether the system is in “Power Saver” mode.
587dbb
+ *
587dbb
+ * You are expected to listen to the
587dbb
+ * #GPowerProfileMonitor::notify::power-saver-enabled signal to know when the profile has
587dbb
+ * changed.
587dbb
+ *
587dbb
+ * Returns: Whether the system is in “Power Saver” mode.
587dbb
+ *
587dbb
+ * Since: 2.70
587dbb
+ */
587dbb
+gboolean
587dbb
+g_power_profile_monitor_get_power_saver_enabled (GPowerProfileMonitor *monitor)
587dbb
+{
587dbb
+  gboolean enabled;
587dbb
+  g_object_get (monitor, "power-saver-enabled", &enabled, NULL);
587dbb
+  return enabled;
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_default_init (GPowerProfileMonitorInterface *iface)
587dbb
+{
587dbb
+  /**
587dbb
+   * GPowerProfileMonitor:power-saver-enabled:
587dbb
+   *
587dbb
+   * Whether “Power Saver” mode is enabled on the system.
587dbb
+   *
587dbb
+   * Since: 2.70
587dbb
+   */
587dbb
+  g_object_interface_install_property (iface,
587dbb
+                                       g_param_spec_boolean ("power-saver-enabled",
587dbb
+                                                             "power-saver-enabled",
587dbb
+                                                             "Power Saver Enabled",
587dbb
+                                                             FALSE,
587dbb
+                                                             G_PARAM_READABLE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY));
587dbb
+}
587dbb
diff --git a/gio/gpowerprofilemonitor.h b/gio/gpowerprofilemonitor.h
587dbb
new file mode 100644
587dbb
index 000000000..0891fc3dc
587dbb
--- /dev/null
587dbb
+++ b/gio/gpowerprofilemonitor.h
587dbb
@@ -0,0 +1,63 @@
587dbb
+/* GIO - GLib Input, Output and Streaming Library
587dbb
+ *
587dbb
+ * Copyright 2019 Red Hat, Inc.
587dbb
+ * Copyright 2021 Igalia S.L.
587dbb
+ *
587dbb
+ * This library is free software; you can redistribute it and/or
587dbb
+ * modify it under the terms of the GNU Lesser General Public
587dbb
+ * License as published by the Free Software Foundation; either
587dbb
+ * version 2.1 of the License, or (at your option) any later version.
587dbb
+ *
587dbb
+ * This library is distributed in the hope that it will be useful,
587dbb
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
587dbb
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
587dbb
+ * Lesser General Public License for more details.
587dbb
+ *
587dbb
+ * You should have received a copy of the GNU Lesser General
587dbb
+ * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
587dbb
+ */
587dbb
+
587dbb
+#ifndef __G_POWER_PROFILE_MONITOR_H__
587dbb
+#define __G_POWER_PROFILE_MONITOR_H__
587dbb
+
587dbb
+#if !defined (__GIO_GIO_H_INSIDE__) && !defined (GIO_COMPILATION)
587dbb
+#error "Only <gio/gio.h> can be included directly."
587dbb
+#endif
587dbb
+
587dbb
+#include <gio/giotypes.h>
587dbb
+
587dbb
+G_BEGIN_DECLS
587dbb
+
587dbb
+/**
587dbb
+ * G_POWER_PROFILE_MONITOR_EXTENSION_POINT_NAME:
587dbb
+ *
587dbb
+ * Extension point for power profile usage monitoring functionality.
587dbb
+ * See [Extending GIO][extending-gio].
587dbb
+ *
587dbb
+ * Since: 2.70
587dbb
+ */
587dbb
+#define G_POWER_PROFILE_MONITOR_EXTENSION_POINT_NAME "gio-power-profile-monitor"
587dbb
+
587dbb
+#define G_TYPE_POWER_PROFILE_MONITOR             (g_power_profile_monitor_get_type ())
587dbb
+GLIB_AVAILABLE_IN_2_70
587dbb
+G_DECLARE_INTERFACE (GPowerProfileMonitor, g_power_profile_monitor, g, power_profile_monitor, GObject)
587dbb
+
587dbb
+#define G_POWER_PROFILE_MONITOR(o)               (G_TYPE_CHECK_INSTANCE_CAST ((o), G_TYPE_POWER_PROFILE_MONITOR, GPowerProfileMonitor))
587dbb
+#define G_IS_POWER_PROFILE_MONITOR(o)            (G_TYPE_CHECK_INSTANCE_TYPE ((o), G_TYPE_POWER_PROFILE_MONITOR))
587dbb
+#define G_POWER_PROFILE_MONITOR_GET_INTERFACE(o) (G_TYPE_INSTANCE_GET_INTERFACE ((o), G_TYPE_POWER_PROFILE_MONITOR, GPowerProfileMonitorInterface))
587dbb
+
587dbb
+struct _GPowerProfileMonitorInterface
587dbb
+{
587dbb
+  /*< private >*/
587dbb
+  GTypeInterface g_iface;
587dbb
+};
587dbb
+
587dbb
+GLIB_AVAILABLE_IN_2_70
587dbb
+GPowerProfileMonitor      *g_power_profile_monitor_dup_default              (void);
587dbb
+
587dbb
+GLIB_AVAILABLE_IN_2_70
587dbb
+gboolean                   g_power_profile_monitor_get_power_saver_enabled  (GPowerProfileMonitor *monitor);
587dbb
+
587dbb
+G_END_DECLS
587dbb
+
587dbb
+#endif /* __G_POWER_PROFILE_MONITOR_H__ */
587dbb
diff --git a/gio/gpowerprofilemonitordbus.c b/gio/gpowerprofilemonitordbus.c
587dbb
new file mode 100644
587dbb
index 000000000..8bbfe3acc
587dbb
--- /dev/null
587dbb
+++ b/gio/gpowerprofilemonitordbus.c
587dbb
@@ -0,0 +1,240 @@
587dbb
+/* GIO - GLib Input, Output and Streaming Library
587dbb
+ *
587dbb
+ * Copyright 2019 Red Hat, Inc.
587dbb
+ * Copyrgith 2021 Igalia S.L.
587dbb
+ *
587dbb
+ * This library is free software; you can redistribute it and/or
587dbb
+ * modify it under the terms of the GNU Lesser General Public
587dbb
+ * License as published by the Free Software Foundation; either
587dbb
+ * version 2.1 of the License, or (at your option) any later version.
587dbb
+ *
587dbb
+ * This library is distributed in the hope that it will be useful,
587dbb
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
587dbb
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
587dbb
+ * Lesser General Public License for more details.
587dbb
+ *
587dbb
+ * You should have received a copy of the GNU Lesser General
587dbb
+ * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
587dbb
+ */
587dbb
+
587dbb
+#include "config.h"
587dbb
+
587dbb
+#include "gpowerprofilemonitor.h"
587dbb
+#include "gpowerprofilemonitordbus.h"
587dbb
+#include "gioerror.h"
587dbb
+#include "ginitable.h"
587dbb
+#include "giomodule-priv.h"
587dbb
+#include "glibintl.h"
587dbb
+#include "glib/gstdio.h"
587dbb
+#include "gcancellable.h"
587dbb
+#include "gdbusproxy.h"
587dbb
+#include "gdbusnamewatching.h"
587dbb
+
587dbb
+#define G_POWER_PROFILE_MONITOR_DBUS_GET_INITABLE_IFACE(o) (G_TYPE_INSTANCE_GET_INTERFACE ((o), G_TYPE_INITABLE, GInitable))
587dbb
+
587dbb
+static void g_power_profile_monitor_dbus_iface_init (GPowerProfileMonitorInterface *iface);
587dbb
+static void g_power_profile_monitor_dbus_initable_iface_init (GInitableIface *iface);
587dbb
+
587dbb
+struct _GPowerProfileMonitorDBus
587dbb
+{
587dbb
+  GObject parent_instance;
587dbb
+
587dbb
+  guint watch_id;
587dbb
+  GCancellable *cancellable;
587dbb
+  GDBusProxy *proxy;
587dbb
+  gulong signal_id;
587dbb
+
587dbb
+  gboolean power_saver_enabled;
587dbb
+};
587dbb
+
587dbb
+typedef enum
587dbb
+{
587dbb
+  PROP_POWER_SAVER_ENABLED = 1,
587dbb
+} GPowerProfileMonitorDBusProperty;
587dbb
+
587dbb
+#define POWERPROFILES_DBUS_NAME "net.hadess.PowerProfiles"
587dbb
+#define POWERPROFILES_DBUS_IFACE "net.hadess.PowerProfiles"
587dbb
+#define POWERPROFILES_DBUS_PATH "/net/hadess/PowerProfiles"
587dbb
+
587dbb
+G_DEFINE_TYPE_WITH_CODE (GPowerProfileMonitorDBus, g_power_profile_monitor_dbus, G_TYPE_OBJECT,
587dbb
+                         G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE,
587dbb
+                                                g_power_profile_monitor_dbus_initable_iface_init)
587dbb
+                         G_IMPLEMENT_INTERFACE (G_TYPE_POWER_PROFILE_MONITOR,
587dbb
+                                                g_power_profile_monitor_dbus_iface_init)
587dbb
+                         _g_io_modules_ensure_extension_points_registered ();
587dbb
+                         g_io_extension_point_implement (G_POWER_PROFILE_MONITOR_EXTENSION_POINT_NAME,
587dbb
+                                                         g_define_type_id,
587dbb
+                                                         "dbus",
587dbb
+                                                         30))
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_dbus_init (GPowerProfileMonitorDBus *dbus)
587dbb
+{
587dbb
+  dbus->power_saver_enabled = FALSE;
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+ppd_properties_changed_cb (GDBusProxy *proxy,
587dbb
+                           GVariant   *changed_properties,
587dbb
+                           GStrv      *invalidated_properties,
587dbb
+                           gpointer    user_data)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = user_data;
587dbb
+  const char *active_profile;
587dbb
+  gboolean enabled;
587dbb
+
587dbb
+  if (!g_variant_lookup (changed_properties, "ActiveProfile", "&s", &active_profile))
587dbb
+    return;
587dbb
+
587dbb
+  enabled = g_strcmp0 (active_profile, "power-saver") == 0;
587dbb
+  if (enabled == dbus->power_saver_enabled)
587dbb
+    return;
587dbb
+
587dbb
+  dbus->power_saver_enabled = enabled;
587dbb
+  g_object_notify (G_OBJECT (dbus), "power-saver-enabled");
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+ppd_proxy_cb (GObject      *source_object,
587dbb
+              GAsyncResult *res,
587dbb
+              gpointer      user_data)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = user_data;
587dbb
+  GVariant *active_profile_variant;
587dbb
+  GDBusProxy *proxy;
587dbb
+  GError *error = NULL;
587dbb
+  const char *active_profile;
587dbb
+  gboolean power_saver_enabled;
587dbb
+
587dbb
+  proxy = g_dbus_proxy_new_finish (res, &error);
587dbb
+  if (!proxy)
587dbb
+    {
587dbb
+      g_debug ("GPowerProfileMonitorDBus: Failed to create PowerProfiles D-Bus proxy: %s",
587dbb
+               error->message);
587dbb
+      g_error_free (error);
587dbb
+      return;
587dbb
+    }
587dbb
+
587dbb
+  active_profile_variant = g_dbus_proxy_get_cached_property (proxy, "ActiveProfile");
587dbb
+  if (g_variant_is_of_type (active_profile_variant, G_VARIANT_TYPE_STRING))
587dbb
+    {
587dbb
+      active_profile = g_variant_get_string (active_profile_variant, NULL);
587dbb
+      power_saver_enabled = g_strcmp0 (active_profile, "power-saver") == 0;
587dbb
+      if (power_saver_enabled != dbus->power_saver_enabled)
587dbb
+        {
587dbb
+          dbus->power_saver_enabled = power_saver_enabled;
587dbb
+          g_object_notify (G_OBJECT (dbus), "power-saver-enabled");
587dbb
+        }
587dbb
+    }
587dbb
+
587dbb
+  dbus->signal_id = g_signal_connect (G_OBJECT (proxy), "g-properties-changed",
587dbb
+                                      G_CALLBACK (ppd_properties_changed_cb), dbus);
587dbb
+  dbus->proxy = g_steal_pointer (&proxy);
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+ppd_appeared_cb (GDBusConnection *connection,
587dbb
+                 const gchar     *name,
587dbb
+                 const gchar     *name_owner,
587dbb
+                 gpointer         user_data)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = user_data;
587dbb
+
587dbb
+  g_dbus_proxy_new (connection,
587dbb
+                    G_DBUS_PROXY_FLAGS_NONE,
587dbb
+                    NULL,
587dbb
+                    POWERPROFILES_DBUS_NAME,
587dbb
+                    POWERPROFILES_DBUS_PATH,
587dbb
+                    POWERPROFILES_DBUS_IFACE,
587dbb
+                    dbus->cancellable,
587dbb
+                    ppd_proxy_cb,
587dbb
+                    dbus);
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+ppd_vanished_cb (GDBusConnection *connection,
587dbb
+                 const gchar     *name,
587dbb
+                 gpointer         user_data)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = user_data;
587dbb
+
587dbb
+  g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
+  g_clear_object (&dbus->proxy);
587dbb
+
587dbb
+  dbus->power_saver_enabled = FALSE;
587dbb
+  g_object_notify (G_OBJECT (dbus), "power-saver-enabled");
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_dbus_get_property (GObject    *object,
587dbb
+                                           guint       prop_id,
587dbb
+                                           GValue     *value,
587dbb
+                                           GParamSpec *pspec)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = G_POWER_PROFILE_MONITOR_DBUS (object);
587dbb
+
587dbb
+  switch ((GPowerProfileMonitorDBusProperty) prop_id)
587dbb
+    {
587dbb
+    case PROP_POWER_SAVER_ENABLED:
587dbb
+      g_value_set_boolean (value, dbus->power_saver_enabled);
587dbb
+      break;
587dbb
+
587dbb
+    default:
587dbb
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
587dbb
+    }
587dbb
+}
587dbb
+
587dbb
+static gboolean
587dbb
+g_power_profile_monitor_dbus_initable_init (GInitable     *initable,
587dbb
+                                            GCancellable  *cancellable,
587dbb
+                                            GError       **error)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = G_POWER_PROFILE_MONITOR_DBUS (initable);
587dbb
+
587dbb
+  dbus->cancellable = g_cancellable_new ();
587dbb
+  dbus->watch_id = g_bus_watch_name (G_BUS_TYPE_SYSTEM,
587dbb
+                                     POWERPROFILES_DBUS_NAME,
587dbb
+                                     G_BUS_NAME_WATCHER_FLAGS_AUTO_START,
587dbb
+                                     ppd_appeared_cb,
587dbb
+                                     ppd_vanished_cb,
587dbb
+                                     dbus,
587dbb
+                                     NULL);
587dbb
+
587dbb
+  return TRUE;
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_dbus_finalize (GObject *object)
587dbb
+{
587dbb
+  GPowerProfileMonitorDBus *dbus = G_POWER_PROFILE_MONITOR_DBUS (object);
587dbb
+
587dbb
+  g_cancellable_cancel (dbus->cancellable);
587dbb
+  g_clear_object (&dbus->cancellable);
587dbb
+  g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
587dbb
+  g_clear_object (&dbus->proxy);
587dbb
+  g_clear_handle_id (&dbus->watch_id, g_bus_unwatch_name);
587dbb
+
587dbb
+  G_OBJECT_CLASS (g_power_profile_monitor_dbus_parent_class)->finalize (object);
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_dbus_class_init (GPowerProfileMonitorDBusClass *nl_class)
587dbb
+{
587dbb
+  GObjectClass *gobject_class = G_OBJECT_CLASS (nl_class);
587dbb
+
587dbb
+  gobject_class->get_property = g_power_profile_monitor_dbus_get_property;
587dbb
+  gobject_class->finalize = g_power_profile_monitor_dbus_finalize;
587dbb
+
587dbb
+  g_object_class_override_property (gobject_class, PROP_POWER_SAVER_ENABLED, "power-saver-enabled");
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_dbus_iface_init (GPowerProfileMonitorInterface *monitor_iface)
587dbb
+{
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+g_power_profile_monitor_dbus_initable_iface_init (GInitableIface *iface)
587dbb
+{
587dbb
+  iface->init = g_power_profile_monitor_dbus_initable_init;
587dbb
+}
587dbb
diff --git a/gio/gpowerprofilemonitordbus.h b/gio/gpowerprofilemonitordbus.h
587dbb
new file mode 100644
587dbb
index 000000000..ecf7246d1
587dbb
--- /dev/null
587dbb
+++ b/gio/gpowerprofilemonitordbus.h
587dbb
@@ -0,0 +1,32 @@
587dbb
+/* GIO - GLib Input, Output and Streaming Library
587dbb
+ *
587dbb
+ * Copyright 2019 Red Hat, Inc.
587dbb
+ * Copyright 2021 Igalia S.L.
587dbb
+ *
587dbb
+ * This library is free software; you can redistribute it and/or
587dbb
+ * modify it under the terms of the GNU Lesser General Public
587dbb
+ * License as published by the Free Software Foundation; either
587dbb
+ * version 2.1 of the License, or (at your option) any later version.
587dbb
+ *
587dbb
+ * This library is distributed in the hope that it will be useful,
587dbb
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
587dbb
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
587dbb
+ * Lesser General Public License for more details.
587dbb
+ *
587dbb
+ * You should have received a copy of the GNU Lesser General
587dbb
+ * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
587dbb
+ */
587dbb
+
587dbb
+#ifndef __G_POWER_PROFILE_MONITOR_DBUS_H__
587dbb
+#define __G_POWER_PROFILE_MONITOR_DBUS_H__
587dbb
+
587dbb
+#include <glib-object.h>
587dbb
+
587dbb
+G_BEGIN_DECLS
587dbb
+
587dbb
+#define G_TYPE_POWER_PROFILE_MONITOR_DBUS         (g_power_profile_monitor_dbus_get_type ())
587dbb
+G_DECLARE_FINAL_TYPE (GPowerProfileMonitorDBus, g_power_profile_monitor_dbus, G, POWER_PROFILE_MONITOR_DBUS, GObject)
587dbb
+
587dbb
+G_END_DECLS
587dbb
+
587dbb
+#endif /* __G_POWER_PROFILE_MONITOR_DBUS_H__ */
587dbb
diff --git a/gio/meson.build b/gio/meson.build
587dbb
index 49a37a7bd..d5838ed8a 100644
587dbb
--- a/gio/meson.build
587dbb
+++ b/gio/meson.build
587dbb
@@ -533,6 +533,8 @@ gio_sources = files(
587dbb
   'gpollableoutputstream.c',
587dbb
   'gpollableutils.c',
587dbb
   'gpollfilemonitor.c',
587dbb
+  'gpowerprofilemonitor.c',
587dbb
+  'gpowerprofilemonitordbus.c',
587dbb
   'gproxy.c',
587dbb
   'gproxyaddress.c',
587dbb
   'gproxyaddressenumerator.c',
587dbb
@@ -673,6 +675,7 @@ gio_headers = files(
587dbb
   'gpollableinputstream.h',
587dbb
   'gpollableoutputstream.h',
587dbb
   'gpollableutils.h',
587dbb
+  'gpowerprofilemonitor.h',
587dbb
   'gproxy.h',
587dbb
   'gproxyaddress.h',
587dbb
   'gproxyaddressenumerator.h',
587dbb
diff --git a/gio/tests/meson.build b/gio/tests/meson.build
587dbb
index 98d1401d0..fc2055101 100644
587dbb
--- a/gio/tests/meson.build
587dbb
+++ b/gio/tests/meson.build
587dbb
@@ -75,6 +75,7 @@ gio_tests = {
587dbb
   'network-monitor-race' : {},
587dbb
   'permission' : {},
587dbb
   'pollable' : {'dependencies' : [libdl_dep]},
587dbb
+  'power-profile-monitor' : {},
587dbb
   'proxy-test' : {},
587dbb
   'readwrite' : {},
587dbb
   'simple-async-result' : {},
587dbb
diff --git a/gio/tests/power-profile-monitor.c b/gio/tests/power-profile-monitor.c
587dbb
new file mode 100644
587dbb
index 000000000..bb32f181f
587dbb
--- /dev/null
587dbb
+++ b/gio/tests/power-profile-monitor.c
587dbb
@@ -0,0 +1,79 @@
587dbb
+/* GIO - GLib Input, Output and Streaming Library
587dbb
+ *
587dbb
+ * Copyright 2021 Igalia S.L.
587dbb
+ *
587dbb
+ * This library is free software; you can redistribute it and/or
587dbb
+ * modify it under the terms of the GNU Lesser General Public
587dbb
+ * License as published by the Free Software Foundation; either
587dbb
+ * version 2.1 of the License, or (at your option) any later version.
587dbb
+ *
587dbb
+ * This library is distributed in the hope that it will be useful,
587dbb
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
587dbb
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
587dbb
+ * Lesser General Public License for more details.
587dbb
+ *
587dbb
+ * You should have received a copy of the GNU Lesser General
587dbb
+ * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
587dbb
+ */
587dbb
+
587dbb
+#include <gio/gio.h>
587dbb
+
587dbb
+static void
587dbb
+test_dup_default (void)
587dbb
+{
587dbb
+  GPowerProfileMonitor *monitor;
587dbb
+
587dbb
+  monitor = g_power_profile_monitor_dup_default ();
587dbb
+  g_assert_nonnull (monitor);
587dbb
+  g_object_unref (monitor);
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+power_saver_enabled_cb (GPowerProfileMonitor *monitor,
587dbb
+                        GParamSpec           *pspec,
587dbb
+                        gpointer              user_data)
587dbb
+{
587dbb
+  gboolean enabled;
587dbb
+
587dbb
+  enabled = g_power_profile_monitor_get_power_saver_enabled (monitor);
587dbb
+  g_debug ("Power Saver %s (%d)", enabled ? "enabled" : "disabled", enabled);
587dbb
+}
587dbb
+
587dbb
+static void
587dbb
+do_watch_power_profile (void)
587dbb
+{
587dbb
+  GPowerProfileMonitor *monitor;
587dbb
+  GMainLoop *loop;
587dbb
+  gulong signal_id;
587dbb
+
587dbb
+  monitor = g_power_profile_monitor_dup_default ();
587dbb
+  signal_id = g_signal_connect (G_OBJECT (monitor), "notify::power-saver-enabled",
587dbb
+		                G_CALLBACK (power_saver_enabled_cb), NULL);
587dbb
+
587dbb
+  loop = g_main_loop_new (NULL, TRUE);
587dbb
+  g_main_loop_run (loop);
587dbb
+
587dbb
+  g_signal_handler_disconnect (monitor, signal_id);
587dbb
+  g_object_unref (monitor);
587dbb
+  g_main_loop_unref (loop);
587dbb
+}
587dbb
+
587dbb
+int
587dbb
+main (int argc, char **argv)
587dbb
+{
587dbb
+  int ret;
587dbb
+
587dbb
+  if (argc == 2 && !strcmp (argv[1], "--watch"))
587dbb
+    {
587dbb
+      do_watch_power_profile ();
587dbb
+      return 0;
587dbb
+    }
587dbb
+
587dbb
+  g_test_init (&argc, &argv, NULL);
587dbb
+
587dbb
+  g_test_add_func ("/power-profile-monitor/default", test_dup_default);
587dbb
+
587dbb
+  ret = g_test_run ();
587dbb
+
587dbb
+  return ret;
587dbb
+}
587dbb
-- 
587dbb
GitLab
587dbb