util: introduce object for holding a system inhibitor lock

The system inhibitor locks are currently handled by code in the
virNetDaemon class. The driver code invokes a callback provided
by the daemon when it wants to start or end inhibition.

When the first inhibition is started, the daemon will call out
to logind to apply it system wide.

This has many flaws

 * A single message is registered with logind regardless of
   what driver holds the inhibition
 * An inhibition of daemon shutdown can't be acquired
   without also inhibiting system shutdown
 * Config of the inhibitions cannot be tailored by the
   driver

The new virInhibitor object addresses these:

 * The object directly manages an inhibition with logind
   privately to the driver, enabling custom messages to
   be set.
 * It is possible to acquire an inhibition locally to the
   daemon without forwarding it to logind.

Reviewed-by: Michal Privoznik <mprivozn@redhat.com>
Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
This commit is contained in:
Daniel P. Berrangé 2024-12-16 15:19:34 +00:00
parent a0a8c95d07
commit d2e5aa4f4e
5 changed files with 281 additions and 0 deletions

View File

@ -293,6 +293,7 @@ src/util/virhostcpu.c
src/util/virhostmem.c
src/util/virhostuptime.c
src/util/viridentity.c
src/util/virinhibitor.c
src/util/virinitctl.c
src/util/viriscsi.c
src/util/virjson.c

View File

@ -2608,6 +2608,13 @@ virIdentitySetUserName;
virIdentitySetX509DName;
# util/virinhibitor.h
virInhibitorFree;
virInhibitorHold;
virInhibitorNew;
virInhibitorRelease;
# util/virinitctl.h
virInitctlFifos;
virInitctlSetRunLevel;

View File

@ -45,6 +45,7 @@ util_sources = [
'virhostmem.c',
'virhostuptime.c',
'viridentity.c',
'virinhibitor.c',
'virinitctl.c',
'viriscsi.c',
'virjson.c',

214
src/util/virinhibitor.c Normal file
View File

@ -0,0 +1,214 @@
/*
* virinhibitor.c: helper APIs for inhibiting host actions
*
* Copyright (C) 2024 Red Hat, Inc.
*
* 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.1 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 "virinhibitor.h"
#include "virgdbus.h"
#include "virsystemd.h"
#include "virfile.h"
#include "virlog.h"
#include "virenum.h"
#define VIR_FROM_THIS VIR_FROM_NONE
VIR_LOG_INIT("util.inhibitor");
struct _virInhibitor {
GMutex lock;
size_t count;
int fd;
char *what;
char *who;
char *why;
const char *mode;
virInhibitorAction action;
void *actionData;
};
VIR_ENUM_DECL(virInhibitorMode);
VIR_ENUM_IMPL(virInhibitorMode,
VIR_INHIBITOR_MODE_LAST,
"block", "delay");
#ifdef G_OS_UNIX
/* As per: https://www.freedesktop.org/wiki/Software/systemd/inhibit */
static int
virInhibitorAcquire(const char *what,
const char *who,
const char *why,
const char *mode,
int *inhibitorFD)
{
g_autoptr(GVariant) reply = NULL;
g_autoptr(GUnixFDList) replyFD = NULL;
g_autoptr(GVariant) message = NULL;
GDBusConnection *systemBus;
int fd;
int rc;
VIR_DEBUG("what=%s who=%s why=%s mode=%s",
NULLSTR(what), NULLSTR(who), NULLSTR(why), NULLSTR(mode));
if (!(systemBus = virGDBusGetSystemBus())) {
VIR_DEBUG("system dbus not available, skipping system inhibitor");
return 0;
}
if (virSystemdHasLogind() < 0) {
VIR_DEBUG("logind not available, skipping system inhibitor");
return 0;
}
message = g_variant_new("(ssss)", what, who, why, mode);
rc = virGDBusCallMethodWithFD(systemBus,
&reply,
G_VARIANT_TYPE("(h)"),
&replyFD,
NULL,
"org.freedesktop.login1",
"/org/freedesktop/login1",
"org.freedesktop.login1.Manager",
"Inhibit",
message,
NULL);
if (rc < 0)
return -1;
if (g_unix_fd_list_get_length(replyFD) <= 0) {
VIR_DEBUG("Missing inhibitor FD in logind reply");
return -1;
}
fd = g_unix_fd_list_get(replyFD, 0, NULL);
if (fd < 0) {
VIR_DEBUG("Unable to get inhibitor FD from logind reply");
return -1;
}
*inhibitorFD = fd;
VIR_DEBUG("Got inhibitor FD %d", fd);
return 0;
}
#endif
static char *
virInhibitorWhatFormat(virInhibitorWhat what)
{
const char *whatstr[] = {
"sleep",
"shutdown",
"idle",
"handle-power-key",
"handle-suspend-key",
"handle-hibernate-key",
"handle-lid-switch",
};
GString *str = g_string_new("");
size_t i;
for (i = 0; i < G_N_ELEMENTS(whatstr); i++) {
if (what & (1 << i)) {
if (str->len)
g_string_append(str, ":");
g_string_append(str, whatstr[i]);
}
}
return g_string_free(str, FALSE);
}
virInhibitor *virInhibitorNew(virInhibitorWhat what,
const char *who,
const char *why,
virInhibitorMode mode,
virInhibitorAction action,
void *actionData)
{
virInhibitor *inhibitor = g_new0(virInhibitor, 1);
inhibitor->fd = -1;
inhibitor->what = virInhibitorWhatFormat(what);
inhibitor->who = g_strdup(who);
inhibitor->why = g_strdup(why);
inhibitor->mode = virInhibitorModeTypeToString(mode);
inhibitor->action = action;
inhibitor->actionData = actionData;
return inhibitor;
}
void virInhibitorHold(virInhibitor *inhibitor)
{
g_mutex_lock(&inhibitor->lock);
if (inhibitor->count == 0) {
if (inhibitor->action) {
inhibitor->action(true, inhibitor->actionData);
}
#ifdef G_OS_UNIX
if (virInhibitorAcquire(
inhibitor->what, inhibitor->who, inhibitor->why,
inhibitor->mode, &inhibitor->fd) < 0) {
VIR_ERROR(_("Failed to acquire inhibitor: %1$s"),
virGetLastErrorMessage());
virResetLastError();
}
#else
VIR_DEBUG("No inhibitor implementation on non-UNIX platforms");
#endif
}
inhibitor->count++;
g_mutex_unlock(&inhibitor->lock);
}
void virInhibitorRelease(virInhibitor *inhibitor)
{
g_mutex_lock(&inhibitor->lock);
inhibitor->count--;
if (inhibitor->count == 0) {
VIR_FORCE_CLOSE(inhibitor->fd);
if (inhibitor->action) {
inhibitor->action(false, inhibitor->actionData);
}
}
g_mutex_unlock(&inhibitor->lock);
}
void virInhibitorFree(virInhibitor *inhibitor)
{
if (!inhibitor)
return;
g_free(inhibitor->what);
g_free(inhibitor->who);
g_free(inhibitor->why);
VIR_FORCE_CLOSE(inhibitor->fd);
g_free(inhibitor);
}

58
src/util/virinhibitor.h Normal file
View File

@ -0,0 +1,58 @@
/*
* virinhibitor.h: helper APIs for inhibiting host actions
*
* Copyright (C) 2024 Red Hat, Inc.
*
* 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.1 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/>.
*/
#pragma once
#include "internal.h"
typedef struct _virInhibitor virInhibitor;
typedef enum {
VIR_INHIBITOR_WHAT_NONE = 0,
VIR_INHIBITOR_WHAT_SLEEP = (1 << 1),
VIR_INHIBITOR_WHAT_SHUTDOWN = (1 << 2),
VIR_INHIBITOR_WHAT_IDLE = (1 << 3),
VIR_INHIBITOR_WHAT_POWER_KEY = (1 << 4),
VIR_INHIBITOR_WHAT_SUSPEND_KEY = (1 << 5),
VIR_INHIBITOR_WHAT_HIBERNATE_KEY = (1 << 6),
VIR_INHIBITOR_WHAT_LID_SWITCH = (1 << 7),
} virInhibitorWhat;
typedef enum {
VIR_INHIBITOR_MODE_BLOCK,
VIR_INHIBITOR_MODE_DELAY,
VIR_INHIBITOR_MODE_LAST
} virInhibitorMode;
typedef void (*virInhibitorAction)(bool inhibited,
void *opaque);
virInhibitor *virInhibitorNew(virInhibitorWhat what,
const char *who,
const char *why,
virInhibitorMode mode,
virInhibitorAction action,
void *actionData);
void virInhibitorHold(virInhibitor *inhibitor);
void virInhibitorRelease(virInhibitor *inhibitor);
void virInhibitorFree(virInhibitor *inhibitor);