2023-08-25 22:40:20 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2021 SUSE Software Solutions Germany GmbH
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "wayland/meta-wayland-idle-inhibit.h"
|
|
|
|
|
|
|
|
#include <wayland-server.h>
|
|
|
|
|
|
|
|
#include "backends/meta-backend-private.h"
|
|
|
|
#include "backends/meta-logical-monitor.h"
|
|
|
|
#include "backends/meta-settings-private.h"
|
|
|
|
#include "wayland/meta-wayland-private.h"
|
|
|
|
#include "wayland/meta-wayland-versions.h"
|
|
|
|
|
|
|
|
#include "idle-inhibit-unstable-v1-server-protocol.h"
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
typedef enum _IdleState
|
|
|
|
{
|
2023-09-01 21:32:06 +00:00
|
|
|
IDLE_STATE_INITIALIZING,
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
IDLE_STATE_UNINHIBITED,
|
|
|
|
IDLE_STATE_INHIBITING,
|
|
|
|
IDLE_STATE_INHIBITED,
|
|
|
|
IDLE_STATE_UNINHIBITING,
|
|
|
|
} IdleState;
|
|
|
|
|
2023-08-25 22:40:20 +00:00
|
|
|
struct _MetaWaylandIdleInhibitor
|
|
|
|
{
|
|
|
|
GDBusProxy *session_proxy;
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
struct wl_resource *resource;
|
|
|
|
|
|
|
|
MetaSurfaceActor *actor;
|
2023-08-25 22:40:20 +00:00
|
|
|
gulong is_obscured_changed_handler;
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
gulong actor_destroyed_handler_id;
|
|
|
|
|
|
|
|
MetaWaylandSurface *surface;
|
2023-08-28 09:52:12 +00:00
|
|
|
gulong surface_destroy_handler_id;
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
gulong actor_changed_handler_id;
|
|
|
|
|
|
|
|
uint32_t cookie;
|
|
|
|
IdleState state;
|
2023-08-25 22:40:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct _MetaWaylandIdleInhibitor MetaWaylandIdleInhibitor;
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
static void update_inhibitation (MetaWaylandIdleInhibitor *inhibitor);
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_wayland_inhibitor_free (MetaWaylandIdleInhibitor *inhibitor)
|
|
|
|
{
|
|
|
|
g_clear_signal_handler (&inhibitor->is_obscured_changed_handler,
|
|
|
|
inhibitor->actor);
|
|
|
|
g_clear_signal_handler (&inhibitor->actor_destroyed_handler_id,
|
|
|
|
inhibitor->actor);
|
|
|
|
g_clear_signal_handler (&inhibitor->actor_changed_handler_id,
|
|
|
|
inhibitor->surface);
|
|
|
|
g_clear_signal_handler (&inhibitor->surface_destroy_handler_id,
|
|
|
|
inhibitor->surface);
|
|
|
|
|
|
|
|
g_free (inhibitor);
|
|
|
|
}
|
|
|
|
|
2023-08-25 22:40:20 +00:00
|
|
|
static void
|
|
|
|
inhibit_completed (GObject *source,
|
|
|
|
GAsyncResult *res,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor = user_data;
|
|
|
|
g_autoptr (GVariant) ret = NULL;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
ret = g_dbus_proxy_call_finish (G_DBUS_PROXY (source), res, &error);
|
|
|
|
if (!ret)
|
|
|
|
{
|
|
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
|
|
g_warning ("Failed to inhibit: %s", error->message);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
g_warn_if_fail (inhibitor->state == IDLE_STATE_INHIBITING);
|
|
|
|
|
2023-08-25 22:40:20 +00:00
|
|
|
g_variant_get (ret, "(u)", &inhibitor->cookie);
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
inhibitor->state = IDLE_STATE_INHIBITED;
|
2023-08-25 22:40:20 +00:00
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
update_inhibitation (inhibitor);
|
|
|
|
}
|
2023-08-25 22:40:20 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
uninhibit_completed (GObject *source,
|
|
|
|
GAsyncResult *res,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor = user_data;
|
|
|
|
g_autoptr (GVariant) ret = NULL;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
ret = g_dbus_proxy_call_finish (G_DBUS_PROXY (source), res, &error);
|
|
|
|
if (!ret)
|
|
|
|
{
|
|
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
|
|
g_warning ("Failed to uninhibit: %s", error->message);
|
|
|
|
return;
|
|
|
|
}
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
|
|
|
|
if (!inhibitor)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_warn_if_fail (inhibitor->state == IDLE_STATE_UNINHIBITING);
|
|
|
|
inhibitor->state = IDLE_STATE_UNINHIBITED;
|
|
|
|
|
|
|
|
update_inhibitation (inhibitor);
|
2023-08-25 22:40:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
update_inhibitation (MetaWaylandIdleInhibitor *inhibitor)
|
|
|
|
{
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
gboolean should_inhibit;
|
2023-08-25 22:40:20 +00:00
|
|
|
|
|
|
|
if (!inhibitor->session_proxy)
|
|
|
|
return;
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
if (!inhibitor->surface ||
|
2023-09-13 15:58:23 +00:00
|
|
|
!inhibitor->resource ||
|
|
|
|
!inhibitor->actor)
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
{
|
|
|
|
should_inhibit = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-09-13 15:58:23 +00:00
|
|
|
if (meta_surface_actor_is_effectively_obscured (inhibitor->actor))
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
should_inhibit = FALSE;
|
|
|
|
else
|
|
|
|
should_inhibit = TRUE;
|
|
|
|
}
|
2023-08-25 22:40:20 +00:00
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
switch (inhibitor->state)
|
2023-08-25 22:40:20 +00:00
|
|
|
{
|
2023-09-01 21:32:06 +00:00
|
|
|
case IDLE_STATE_INITIALIZING:
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
case IDLE_STATE_UNINHIBITED:
|
|
|
|
if (!inhibitor->resource)
|
2023-08-25 22:40:20 +00:00
|
|
|
{
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
meta_wayland_inhibitor_free (inhibitor);
|
|
|
|
return;
|
2023-08-25 22:40:20 +00:00
|
|
|
}
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
|
|
|
|
if (!should_inhibit)
|
|
|
|
return;
|
|
|
|
|
|
|
|
break;
|
|
|
|
case IDLE_STATE_INHIBITED:
|
|
|
|
if (should_inhibit)
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
case IDLE_STATE_INHIBITING:
|
|
|
|
case IDLE_STATE_UNINHIBITING:
|
|
|
|
/* Update inhibitation after current asynchronous call completes. */
|
|
|
|
return;
|
2023-08-25 22:40:20 +00:00
|
|
|
}
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
|
|
|
|
if (should_inhibit)
|
|
|
|
{
|
|
|
|
g_dbus_proxy_call (G_DBUS_PROXY (inhibitor->session_proxy),
|
|
|
|
"Inhibit",
|
|
|
|
g_variant_new ("(ss)", "mutter", "idle-inhibit"),
|
|
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
|
|
-1,
|
|
|
|
NULL,
|
|
|
|
inhibit_completed,
|
|
|
|
inhibitor);
|
|
|
|
inhibitor->state = IDLE_STATE_INHIBITING;
|
|
|
|
}
|
|
|
|
else
|
2023-08-25 22:40:20 +00:00
|
|
|
{
|
|
|
|
g_dbus_proxy_call (G_DBUS_PROXY (inhibitor->session_proxy),
|
|
|
|
"UnInhibit",
|
|
|
|
g_variant_new ("(u)", inhibitor->cookie),
|
|
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
|
|
-1,
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
NULL,
|
2023-08-25 22:40:20 +00:00
|
|
|
uninhibit_completed,
|
|
|
|
inhibitor);
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
inhibitor->state = IDLE_STATE_UNINHIBITING;
|
2023-08-25 22:40:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
is_obscured_changed (MetaSurfaceActor *actor,
|
|
|
|
GParamSpec *pspec,
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor)
|
|
|
|
{
|
|
|
|
update_inhibitation (inhibitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
inhibitor_proxy_completed (GObject *source,
|
|
|
|
GAsyncResult *res,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor = user_data;
|
|
|
|
GDBusProxy *proxy;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
proxy = g_dbus_proxy_new_finish (res, &error);
|
|
|
|
if (!proxy)
|
|
|
|
{
|
|
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
|
|
|
{
|
|
|
|
g_warning ("Failed to obtain org.freedesktop.ScreenSaver proxy: %s",
|
|
|
|
error->message);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
inhibitor->session_proxy = proxy;
|
2023-09-01 21:32:06 +00:00
|
|
|
inhibitor->state = IDLE_STATE_UNINHIBITED;
|
2023-08-25 22:40:20 +00:00
|
|
|
|
|
|
|
update_inhibitation (inhibitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
idle_inhibit_destroy (struct wl_client *client,
|
|
|
|
struct wl_resource *resource)
|
|
|
|
{
|
|
|
|
wl_resource_destroy (resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
idle_inhibitor_destructor (struct wl_resource *resource)
|
|
|
|
{
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor = wl_resource_get_user_data (resource);
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
switch (inhibitor->state)
|
2023-08-28 09:52:12 +00:00
|
|
|
{
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
case IDLE_STATE_UNINHIBITED:
|
|
|
|
meta_wayland_inhibitor_free (inhibitor);
|
|
|
|
return;
|
2023-09-01 21:32:06 +00:00
|
|
|
case IDLE_STATE_INITIALIZING:
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
case IDLE_STATE_INHIBITED:
|
|
|
|
case IDLE_STATE_INHIBITING:
|
|
|
|
case IDLE_STATE_UNINHIBITING:
|
|
|
|
inhibitor->resource = NULL;
|
|
|
|
break;
|
2023-08-28 09:52:12 +00:00
|
|
|
}
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
update_inhibitation (inhibitor);
|
2023-08-25 22:40:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2023-08-28 09:52:12 +00:00
|
|
|
on_surface_destroyed (MetaWaylandSurface *surface,
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor)
|
2023-08-25 22:40:20 +00:00
|
|
|
{
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
g_clear_signal_handler (&inhibitor->is_obscured_changed_handler,
|
|
|
|
inhibitor->actor);
|
|
|
|
g_clear_signal_handler (&inhibitor->actor_destroyed_handler_id,
|
|
|
|
inhibitor->actor);
|
|
|
|
inhibitor->actor = NULL;
|
|
|
|
g_clear_signal_handler (&inhibitor->actor_changed_handler_id,
|
|
|
|
inhibitor->surface);
|
|
|
|
g_clear_signal_handler (&inhibitor->surface_destroy_handler_id,
|
|
|
|
inhibitor->surface);
|
2023-08-25 22:40:20 +00:00
|
|
|
inhibitor->surface = NULL;
|
|
|
|
}
|
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
static void
|
|
|
|
on_actor_destroyed (MetaSurfaceActor *actor,
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor)
|
|
|
|
{
|
|
|
|
g_warn_if_fail (actor == inhibitor->actor);
|
|
|
|
|
|
|
|
g_clear_signal_handler (&inhibitor->is_obscured_changed_handler, actor);
|
|
|
|
g_clear_signal_handler (&inhibitor->actor_destroyed_handler_id, actor);
|
|
|
|
inhibitor->actor = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
attach_actor (MetaWaylandIdleInhibitor *inhibitor)
|
|
|
|
{
|
|
|
|
inhibitor->actor = meta_wayland_surface_get_actor (inhibitor->surface);
|
2023-12-06 14:34:47 +00:00
|
|
|
|
|
|
|
if (inhibitor->actor)
|
|
|
|
{
|
|
|
|
inhibitor->is_obscured_changed_handler =
|
|
|
|
g_signal_connect (inhibitor->actor, "notify::is-obscured",
|
|
|
|
G_CALLBACK (is_obscured_changed), inhibitor);
|
|
|
|
inhibitor->actor_destroyed_handler_id =
|
|
|
|
g_signal_connect (inhibitor->actor, "destroy",
|
|
|
|
G_CALLBACK (on_actor_destroyed), inhibitor);
|
|
|
|
}
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_actor_changed (MetaWaylandSurface *surface,
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor)
|
|
|
|
{
|
|
|
|
g_clear_signal_handler (&inhibitor->is_obscured_changed_handler,
|
|
|
|
inhibitor->surface);
|
|
|
|
g_clear_signal_handler (&inhibitor->actor_destroyed_handler_id,
|
|
|
|
inhibitor->surface);
|
|
|
|
attach_actor (inhibitor);
|
|
|
|
}
|
|
|
|
|
2023-08-25 22:40:20 +00:00
|
|
|
static const struct zwp_idle_inhibitor_v1_interface meta_wayland_idle_inhibitor_interface =
|
|
|
|
{
|
|
|
|
idle_inhibit_destroy,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
idle_inhibit_manager_create_inhibitor (struct wl_client *client,
|
|
|
|
struct wl_resource *resource,
|
|
|
|
uint32_t id,
|
|
|
|
struct wl_resource *surface_resource)
|
|
|
|
{
|
|
|
|
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
|
|
|
|
MetaWaylandIdleInhibitor *inhibitor;
|
|
|
|
struct wl_resource *inhibitor_resource;
|
|
|
|
|
|
|
|
inhibitor_resource = wl_resource_create (client,
|
|
|
|
&zwp_idle_inhibitor_v1_interface,
|
|
|
|
wl_resource_get_version (resource),
|
|
|
|
id);
|
|
|
|
|
|
|
|
inhibitor = g_new0 (MetaWaylandIdleInhibitor, 1);
|
|
|
|
inhibitor->surface = surface;
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
inhibitor->resource = inhibitor_resource;
|
2023-08-25 22:40:20 +00:00
|
|
|
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
attach_actor (inhibitor);
|
|
|
|
|
|
|
|
inhibitor->actor_changed_handler_id =
|
|
|
|
g_signal_connect (surface, "actor-changed",
|
|
|
|
G_CALLBACK (on_actor_changed), inhibitor);
|
|
|
|
inhibitor->surface_destroy_handler_id =
|
|
|
|
g_signal_connect (surface, "destroy",
|
|
|
|
G_CALLBACK (on_surface_destroyed), inhibitor);
|
2023-08-25 22:40:20 +00:00
|
|
|
|
|
|
|
g_dbus_proxy_new_for_bus (G_BUS_TYPE_SESSION,
|
|
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
|
|
NULL,
|
|
|
|
"org.freedesktop.ScreenSaver",
|
|
|
|
"/org/freedesktop/ScreenSaver",
|
|
|
|
"org.freedesktop.ScreenSaver",
|
wayland/idle-inhibit: Add state tracking to fix races
This changes how state is tracked by introducing an explicit state. We
need this since we use asynchronous calls to the out of process
component that handles actual inhibitation, including idleness.
This means if inhibitations changes rapidly, we might end up with an
incorrect state if we e.g. try to uninhibit while we're currently trying
to inhibit.
This is done by adding a state variable that accounts for the pending
state, as well as the active state, with a function that looks at the
current conditions to derive what state we should be in, and what state
we are in, to decide what the next action should be.
For example, if we're trying to inhibit, but now wants to uninhibit,
we'll wait for the inhibit call to complete, recheck what we want, which
would result in an async uninhibit call being made.
Fixes: 388b534062 ("wayland: Implement idle inhibit protocol")
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3219>
2023-08-28 14:15:01 +00:00
|
|
|
NULL,
|
2023-08-25 22:40:20 +00:00
|
|
|
inhibitor_proxy_completed,
|
|
|
|
inhibitor);
|
|
|
|
|
|
|
|
wl_resource_set_implementation (inhibitor_resource,
|
|
|
|
&meta_wayland_idle_inhibitor_interface,
|
|
|
|
inhibitor,
|
|
|
|
idle_inhibitor_destructor);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const struct zwp_idle_inhibit_manager_v1_interface meta_wayland_idle_inhibit_manager_interface =
|
|
|
|
{
|
|
|
|
idle_inhibit_destroy,
|
|
|
|
idle_inhibit_manager_create_inhibitor,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
bind_idle_inhibit (struct wl_client *client,
|
|
|
|
void *data,
|
|
|
|
uint32_t version,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
|
|
|
struct wl_resource *resource;
|
|
|
|
|
|
|
|
resource = wl_resource_create (client,
|
|
|
|
&zwp_idle_inhibit_manager_v1_interface,
|
|
|
|
version, id);
|
|
|
|
|
|
|
|
wl_resource_set_implementation (resource,
|
|
|
|
&meta_wayland_idle_inhibit_manager_interface,
|
|
|
|
NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_wayland_idle_inhibit_init (MetaWaylandCompositor *compositor)
|
|
|
|
{
|
|
|
|
if (wl_global_create (compositor->wayland_display,
|
|
|
|
&zwp_idle_inhibit_manager_v1_interface,
|
|
|
|
META_ZWP_IDLE_INHIBIT_V1_VERSION,
|
|
|
|
NULL,
|
|
|
|
bind_idle_inhibit) == NULL)
|
|
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
|
|
}
|