mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-22 17:35:59 +03:00
97770d5aeb
Add some debug checks for possibly failing syscalls.
196 lines
4.4 KiB
C
196 lines
4.4 KiB
C
/*
|
|
* Copyright (C) 2015 Red Hat, Inc.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*/
|
|
|
|
#include "lib/misc/lib.h"
|
|
#include "lib/commands/toolcontext.h"
|
|
#include "lib/notify/lvmnotify.h"
|
|
|
|
#define LVM_DBUS_DESTINATION "com.redhat.lvmdbus1"
|
|
#define LVM_DBUS_PATH "/com/redhat/lvmdbus1/Manager"
|
|
#define LVM_DBUS_INTERFACE "com.redhat.lvmdbus1.Manager"
|
|
#define LVM_DBUS_LOCK_FILE "/var/lock/lvm/lvmdbusd"
|
|
#define LVM_DBUS_LOCK_FILE_ENV_KEY "LVM_DBUSD_LOCKFILE"
|
|
#define SD_BUS_SYSTEMD_NO_SUCH_UNIT_ERROR "org.freedesktop.systemd1.NoSuchUnit"
|
|
#define SD_BUS_DBUS_SERVICE_UNKNOWN_ERROR "org.freedesktop.DBus.Error.ServiceUnknown"
|
|
|
|
#ifdef NOTIFYDBUS_SUPPORT
|
|
#include <systemd/sd-bus.h>
|
|
#include <fcntl.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
|
|
int lvmnotify_is_supported(void)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int lvmdbusd_running(void)
|
|
{
|
|
int fd = 0;
|
|
int rc = 0;
|
|
int errno_cpy = 0;
|
|
int running = 0;
|
|
const char *lockfile = NULL;
|
|
|
|
/*
|
|
* lvm dbusd uses a lock file with a lock on it, thus to determine if the daemon is running
|
|
* requires that you attempt to lock the file as well. Thus the existence of the file does
|
|
* not mean it's running, but the absence of the file does indicate it's not running.
|
|
*
|
|
* See lvmdbusd for more details.
|
|
*/
|
|
|
|
lockfile = getenv(LVM_DBUS_LOCK_FILE_ENV_KEY);
|
|
if (!lockfile) {
|
|
lockfile = LVM_DBUS_LOCK_FILE;
|
|
}
|
|
|
|
errno = 0;
|
|
fd = open(lockfile, O_RDWR);
|
|
if (-1 == fd) {
|
|
errno_cpy = errno;
|
|
if (errno_cpy == ENOENT) {
|
|
return 0;
|
|
} else {
|
|
/* Safest option is to return running when we encounter unexpected errors */
|
|
log_debug_dbus("Unexpected errno: %d on lockfile open, returning running", errno_cpy);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* Need to ensure we close lock FD now */
|
|
errno = 0;
|
|
rc = lockf(fd, F_TLOCK|F_TEST, 0);
|
|
if (-1 != rc) {
|
|
/* Not locked, thus not running */
|
|
running = 0;
|
|
} else {
|
|
errno_cpy = errno;
|
|
if (errno_cpy == EACCES || errno_cpy == EAGAIN) {
|
|
/* Locked, so daemon is running */
|
|
running = 1;
|
|
} else {
|
|
log_debug_dbus("Unexpected errno: %d on lockf, returning running", errno_cpy);
|
|
running = 1 ;
|
|
}
|
|
}
|
|
|
|
if (close(fd))
|
|
log_sys_debug("close", lockfile);
|
|
|
|
return running;
|
|
}
|
|
|
|
|
|
void lvmnotify_send(struct cmd_context *cmd)
|
|
{
|
|
static const char _dbus_notification_failed_msg[] = "D-Bus notification failed";
|
|
sd_bus *bus = NULL;
|
|
sd_bus_message *m = NULL;
|
|
sd_bus_error error = SD_BUS_ERROR_NULL;
|
|
const char *cmd_name;
|
|
int ret;
|
|
int result = 0;
|
|
|
|
if (!cmd->vg_notify && !cmd->lv_notify && !cmd->pv_notify)
|
|
return;
|
|
|
|
cmd->vg_notify = 0;
|
|
cmd->lv_notify = 0;
|
|
cmd->pv_notify = 0;
|
|
|
|
/* If lvmdbusd isn't running, don't notify as you will start it as it will auto activate */
|
|
if (!lvmdbusd_running()) {
|
|
log_debug_dbus("dbus damon not running, not notifying");
|
|
return;
|
|
}
|
|
|
|
cmd_name = get_cmd_name();
|
|
|
|
ret = sd_bus_open_system(&bus);
|
|
if (ret < 0) {
|
|
log_debug_dbus("Failed to connect to dbus: %d", ret);
|
|
return;
|
|
}
|
|
|
|
log_debug_dbus("Nofify dbus at %s.", LVM_DBUS_DESTINATION);
|
|
|
|
ret = sd_bus_call_method(bus,
|
|
LVM_DBUS_DESTINATION,
|
|
LVM_DBUS_PATH,
|
|
LVM_DBUS_INTERFACE,
|
|
"ExternalEvent",
|
|
&error,
|
|
&m,
|
|
"s",
|
|
cmd_name);
|
|
|
|
if (ret < 0) {
|
|
if (sd_bus_error_has_name(&error, SD_BUS_SYSTEMD_NO_SUCH_UNIT_ERROR) ||
|
|
sd_bus_error_has_name(&error, SD_BUS_DBUS_SERVICE_UNKNOWN_ERROR))
|
|
log_debug_dbus("%s: %s", _dbus_notification_failed_msg, error.message);
|
|
else
|
|
log_warn("WARNING: %s: %s", _dbus_notification_failed_msg, error.message);
|
|
goto out;
|
|
}
|
|
|
|
ret = sd_bus_message_read(m, "i", &result);
|
|
if (ret < 0)
|
|
log_debug_dbus("Failed to parse dbus response message: %d", ret);
|
|
if (result)
|
|
log_debug_dbus("Bad return value from dbus service: %d", result);
|
|
out:
|
|
sd_bus_error_free(&error);
|
|
sd_bus_message_unref(m);
|
|
sd_bus_flush_close_unref(bus);
|
|
}
|
|
|
|
void set_vg_notify(struct cmd_context *cmd)
|
|
{
|
|
cmd->vg_notify = 1;
|
|
}
|
|
|
|
void set_lv_notify(struct cmd_context *cmd)
|
|
{
|
|
cmd->lv_notify = 1;
|
|
}
|
|
|
|
void set_pv_notify(struct cmd_context *cmd)
|
|
{
|
|
cmd->pv_notify = 1;
|
|
}
|
|
|
|
#else
|
|
|
|
int lvmnotify_is_supported(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void lvmnotify_send(struct cmd_context *cmd)
|
|
{
|
|
}
|
|
|
|
void set_vg_notify(struct cmd_context *cmd)
|
|
{
|
|
}
|
|
|
|
void set_lv_notify(struct cmd_context *cmd)
|
|
{
|
|
}
|
|
|
|
void set_pv_notify(struct cmd_context *cmd)
|
|
{
|
|
}
|
|
|
|
#endif
|
|
|