2019-08-08 10:16:48 +02:00
|
|
|
/*
|
|
|
|
* virhostuptime.c: helper APIs for host uptime
|
|
|
|
*
|
|
|
|
* Copyright (C) 2019 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>
|
|
|
|
|
2020-09-01 13:27:44 +02:00
|
|
|
#ifdef WITH_GETUTXID
|
2019-08-08 10:16:48 +02:00
|
|
|
# include <utmpx.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "virhostuptime.h"
|
2019-10-14 16:37:03 +02:00
|
|
|
#include "viralloc.h"
|
|
|
|
#include "virfile.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
#include "virtime.h"
|
2019-08-08 10:16:48 +02:00
|
|
|
#include "virthread.h"
|
|
|
|
|
2021-02-01 13:42:04 +01:00
|
|
|
#include <math.h>
|
|
|
|
|
2019-10-14 16:37:03 +02:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
|
|
|
|
VIR_LOG_INIT("util.virhostuptime");
|
|
|
|
|
2019-08-08 10:16:48 +02:00
|
|
|
static unsigned long long bootTime;
|
|
|
|
static int bootTimeErrno;
|
|
|
|
static virOnceControl virHostGetBootTimeOnce = VIR_ONCE_CONTROL_INITIALIZER;
|
|
|
|
|
2019-10-14 16:37:03 +02:00
|
|
|
#if defined(__linux__)
|
|
|
|
# define UPTIME_FILE "/proc/uptime"
|
|
|
|
static int
|
|
|
|
virHostGetBootTimeProcfs(unsigned long long *btime)
|
|
|
|
{
|
|
|
|
unsigned long long now;
|
|
|
|
double up;
|
|
|
|
g_autofree char *buf = NULL;
|
|
|
|
char *tmp;
|
|
|
|
|
|
|
|
if (virTimeMillisNow(&now) < 0)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
/* 1KiB limit is more than enough. */
|
|
|
|
if (virFileReadAll(UPTIME_FILE, 1024, &buf) < 0)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
/* buf contains two doubles now:
|
|
|
|
* $uptime $idle_time
|
|
|
|
* We're interested only in the first one */
|
|
|
|
if (!(tmp = strchr(buf, ' '))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("uptime file has unexpected format '%s'"),
|
|
|
|
buf);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*tmp = '\0';
|
|
|
|
|
|
|
|
if (virStrToDouble(buf, NULL, &up) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unable to parse uptime value '%s'"),
|
|
|
|
buf);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2021-02-01 13:42:04 +01:00
|
|
|
*btime = llround(now / 1000 - up);
|
2019-10-14 16:37:03 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif /* defined(__linux__) */
|
|
|
|
|
2020-09-01 13:27:44 +02:00
|
|
|
#if defined(WITH_GETUTXID) || defined(__linux__)
|
2019-08-08 10:16:48 +02:00
|
|
|
static void
|
|
|
|
virHostGetBootTimeOnceInit(void)
|
|
|
|
{
|
2020-09-01 13:27:44 +02:00
|
|
|
# ifdef WITH_GETUTXID
|
2019-08-08 10:16:48 +02:00
|
|
|
struct utmpx id = {.ut_type = BOOT_TIME};
|
|
|
|
struct utmpx *res = NULL;
|
|
|
|
|
|
|
|
if (!(res = getutxid(&id))) {
|
|
|
|
bootTimeErrno = errno;
|
|
|
|
} else {
|
|
|
|
bootTime = res->ut_tv.tv_sec;
|
|
|
|
}
|
|
|
|
|
|
|
|
endutxent();
|
2020-09-01 13:27:44 +02:00
|
|
|
# endif /* WITH_GETUTXID */
|
2019-10-14 16:37:03 +02:00
|
|
|
|
2019-11-14 16:42:51 +01:00
|
|
|
# ifdef __linux__
|
2019-10-14 16:37:03 +02:00
|
|
|
if (bootTimeErrno != 0 || bootTime == 0)
|
|
|
|
bootTimeErrno = -virHostGetBootTimeProcfs(&bootTime);
|
2019-11-14 16:42:51 +01:00
|
|
|
# endif /* __linux__ */
|
2019-08-08 10:16:48 +02:00
|
|
|
}
|
|
|
|
|
2020-09-01 13:27:44 +02:00
|
|
|
#else /* !defined(WITH_GETUTXID) && !defined(__linux__) */
|
2019-08-08 10:16:48 +02:00
|
|
|
|
|
|
|
static void
|
|
|
|
virHostGetBootTimeOnceInit(void)
|
|
|
|
{
|
|
|
|
bootTimeErrno = ENOSYS;
|
|
|
|
}
|
2020-09-01 13:27:44 +02:00
|
|
|
#endif /* !defined(WITH_GETUTXID) && !defined(__linux__) */
|
2019-08-08 10:16:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* virHostGetBootTime:
|
|
|
|
* @when: UNIX timestamp of boot time
|
|
|
|
*
|
|
|
|
* Get a UNIX timestamp of host boot time and store it at @when.
|
|
|
|
*
|
|
|
|
* Return: 0 on success,
|
|
|
|
* -1 otherwise.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virHostGetBootTime(unsigned long long *when)
|
|
|
|
{
|
2019-12-19 10:11:04 +01:00
|
|
|
if (virHostBootTimeInit() < 0)
|
2019-08-08 10:16:48 +02:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (bootTimeErrno) {
|
|
|
|
errno = bootTimeErrno;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*when = bootTime;
|
|
|
|
return 0;
|
|
|
|
}
|
2019-12-19 10:11:04 +01:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virHostBootTimeInit(void)
|
|
|
|
{
|
|
|
|
if (virOnce(&virHostGetBootTimeOnce, virHostGetBootTimeOnceInit) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|