2015-06-21 18:20:04 +10:00
|
|
|
/*
|
|
|
|
* uptime - uptime related functions - part of procps
|
|
|
|
*
|
|
|
|
* Copyright (C) 1992-1998 Michael K. Johnson <johnsonm@redhat.com>
|
|
|
|
* Copyright (C) ???? Larry Greenfield <greenfie@gauss.rutgers.edu>
|
|
|
|
* Copyright (C) 1993 J. Cowley
|
|
|
|
* Copyright (C) 1998-2003 Albert Cahalan
|
2021-01-21 17:37:48 +11:00
|
|
|
* Copyright (C) 2015 Craig Small <csmall@dropbear.xyz>
|
2015-06-21 18:20:04 +10:00
|
|
|
*
|
|
|
|
* 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <locale.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <utmp.h>
|
|
|
|
|
2021-01-19 00:00:00 -06:00
|
|
|
#include <proc/misc.h>
|
2015-06-21 18:20:04 +10:00
|
|
|
#include "procps-private.h"
|
|
|
|
|
|
|
|
#define UPTIME_FILE "/proc/uptime"
|
|
|
|
|
|
|
|
static __thread char upbuf[256];
|
|
|
|
static __thread char shortbuf[256];
|
2015-06-21 18:20:04 +10:00
|
|
|
|
|
|
|
static int count_users(void)
|
|
|
|
{
|
|
|
|
int numuser = 0;
|
|
|
|
struct utmp *ut;
|
|
|
|
|
|
|
|
setutent();
|
|
|
|
while ((ut = getutent())) {
|
2015-06-29 22:09:59 +10:00
|
|
|
if ((ut->ut_type == USER_PROCESS) && (ut->ut_name[0] != '\0'))
|
|
|
|
numuser++;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
endutent();
|
|
|
|
|
|
|
|
return numuser;
|
|
|
|
}
|
2015-06-29 22:09:59 +10:00
|
|
|
|
2015-06-21 18:20:04 +10:00
|
|
|
/*
|
|
|
|
* uptime:
|
|
|
|
*
|
|
|
|
* Find the uptime and idle time of the system.
|
|
|
|
* These numbers are found in /proc/uptime
|
|
|
|
* Unlike other procps functions this closes the file each time
|
|
|
|
* Either uptime_secs or idle_secs can be null
|
|
|
|
*
|
2016-05-01 16:50:25 +10:00
|
|
|
* Returns: 0 on success and <0 on failure
|
2015-06-21 18:20:04 +10:00
|
|
|
*/
|
2015-06-29 22:09:59 +10:00
|
|
|
PROCPS_EXPORT int procps_uptime(
|
|
|
|
double *restrict uptime_secs,
|
|
|
|
double *restrict idle_secs)
|
2015-06-21 18:20:04 +10:00
|
|
|
{
|
|
|
|
double up=0, idle=0;
|
2021-09-28 00:00:00 -05:00
|
|
|
locale_t tmplocale;
|
2015-06-21 18:20:04 +10:00
|
|
|
FILE *fp;
|
2021-09-28 00:00:00 -05:00
|
|
|
int rc;
|
2015-06-21 18:20:04 +10:00
|
|
|
|
|
|
|
if ((fp = fopen(UPTIME_FILE, "r")) == NULL)
|
2015-06-29 22:09:59 +10:00
|
|
|
return -errno;
|
|
|
|
|
2021-09-28 00:00:00 -05:00
|
|
|
tmplocale = newlocale(LC_NUMERIC_MASK, "C", (locale_t)0);
|
|
|
|
uselocale(tmplocale);
|
|
|
|
rc = fscanf(fp, "%lf %lf", &up, &idle);
|
2015-06-21 18:20:04 +10:00
|
|
|
fclose(fp);
|
2021-09-28 00:00:00 -05:00
|
|
|
uselocale(LC_GLOBAL_LOCALE);
|
|
|
|
freelocale(tmplocale);
|
|
|
|
|
2015-06-21 18:20:04 +10:00
|
|
|
if (uptime_secs)
|
2015-06-29 22:09:59 +10:00
|
|
|
*uptime_secs = up;
|
2015-06-21 18:20:04 +10:00
|
|
|
if (idle_secs)
|
2015-06-29 22:09:59 +10:00
|
|
|
*idle_secs = idle;
|
2021-09-28 00:00:00 -05:00
|
|
|
|
|
|
|
if (rc < 2)
|
|
|
|
return -ERANGE;
|
2016-05-01 16:50:25 +10:00
|
|
|
return 0;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2015-06-29 11:16:52 -05:00
|
|
|
* procps_uptime_sprint:
|
2015-06-21 18:20:04 +10:00
|
|
|
*
|
|
|
|
* Print current time in nice format
|
|
|
|
*
|
|
|
|
* Returns a statically allocated upbuf or NULL on error
|
|
|
|
*/
|
2015-06-29 22:09:59 +10:00
|
|
|
PROCPS_EXPORT char *procps_uptime_sprint(void)
|
2015-06-21 18:20:04 +10:00
|
|
|
{
|
|
|
|
int upminutes, uphours, updays, users;
|
|
|
|
int pos;
|
|
|
|
time_t realseconds;
|
2021-09-28 00:00:00 -05:00
|
|
|
struct tm realtime;
|
2015-06-21 18:20:04 +10:00
|
|
|
double uptime_secs, idle_secs;
|
|
|
|
double av1, av5, av15;
|
|
|
|
|
2015-06-23 22:22:50 +10:00
|
|
|
upbuf[0] = '\0';
|
2015-06-21 18:20:04 +10:00
|
|
|
if (time(&realseconds) < 0)
|
2015-06-29 22:09:59 +10:00
|
|
|
return upbuf;
|
2021-09-28 00:00:00 -05:00
|
|
|
localtime_r(&realseconds, &realtime);
|
|
|
|
|
2015-06-29 22:09:59 +10:00
|
|
|
if (procps_uptime(&uptime_secs, &idle_secs) < 0)
|
|
|
|
return upbuf;
|
2015-06-21 18:20:04 +10:00
|
|
|
|
|
|
|
updays = ((int) uptime_secs / (60*60*24));
|
2015-06-24 22:16:16 +10:00
|
|
|
uphours = ((int) uptime_secs / (60*60)) % 24;
|
2015-06-21 18:20:04 +10:00
|
|
|
upminutes = ((int) uptime_secs / (60)) % 60;
|
|
|
|
|
2015-09-04 00:00:00 -05:00
|
|
|
pos = sprintf(upbuf, " %02d:%02d:%02d up ",
|
2021-09-28 00:00:00 -05:00
|
|
|
realtime.tm_hour, realtime.tm_min, realtime.tm_sec);
|
2015-09-04 00:00:00 -05:00
|
|
|
|
|
|
|
if (updays)
|
2015-09-07 18:38:39 +10:00
|
|
|
pos += sprintf(upbuf + pos, "%d %s, ", updays, (updays > 1) ? "days" : "day");
|
2015-09-04 00:00:00 -05:00
|
|
|
|
2015-06-21 18:20:04 +10:00
|
|
|
if (uphours)
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(upbuf + pos, "%2d:%02d, ", uphours, upminutes);
|
2015-06-21 18:20:04 +10:00
|
|
|
else
|
2015-09-04 00:00:00 -05:00
|
|
|
pos += sprintf(upbuf + pos, "%d min, ", upminutes);
|
2015-06-21 18:20:04 +10:00
|
|
|
|
|
|
|
users = count_users();
|
2015-07-01 21:47:30 +10:00
|
|
|
procps_loadavg(&av1, &av5, &av15);
|
2015-06-21 18:20:04 +10:00
|
|
|
|
2015-09-04 00:00:00 -05:00
|
|
|
pos += sprintf(upbuf + pos, "%2d %s, load average: %.2f, %.2f, %.2f",
|
|
|
|
users, users > 1 ? "users" : "user",
|
2015-06-29 22:09:59 +10:00
|
|
|
av1, av5, av15);
|
2015-06-21 18:20:04 +10:00
|
|
|
|
2015-06-23 22:22:50 +10:00
|
|
|
return upbuf;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
2015-06-29 22:09:59 +10:00
|
|
|
/*
|
2015-06-29 11:16:52 -05:00
|
|
|
* procps_uptime_sprint_short:
|
2015-06-29 22:09:59 +10:00
|
|
|
*
|
|
|
|
* Print current time in nice format
|
|
|
|
*
|
|
|
|
* Returns a statically allocated buffer or NULL on error
|
|
|
|
*/
|
|
|
|
PROCPS_EXPORT char *procps_uptime_sprint_short(void)
|
2015-06-21 18:20:04 +10:00
|
|
|
{
|
2021-11-02 17:23:37 +11:00
|
|
|
int updecades, upyears, upweeks, updays, uphours, upminutes = 0;
|
2015-06-21 18:20:04 +10:00
|
|
|
int pos = 3;
|
|
|
|
int comma = 0;
|
|
|
|
double uptime_secs, idle_secs;
|
|
|
|
|
2015-06-23 22:22:50 +10:00
|
|
|
shortbuf[0] = '\0';
|
2015-06-29 22:09:59 +10:00
|
|
|
if (procps_uptime(&uptime_secs, &idle_secs) < 0)
|
|
|
|
return shortbuf;
|
2015-06-21 18:20:04 +10:00
|
|
|
|
2021-10-14 19:37:42 +11:00
|
|
|
if (uptime_secs>60*60*24*365*10) {
|
|
|
|
updecades = (int) uptime_secs / (60*60*24*365*10);
|
|
|
|
uptime_secs -= updecades*60*60*24*365*10;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
updecades = 0;
|
|
|
|
}
|
|
|
|
if (uptime_secs>60*60*24*365) {
|
|
|
|
upyears = (int) uptime_secs / (60*60*24*365);
|
|
|
|
uptime_secs -= upyears*60*60*24*365;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
upyears = 0;
|
|
|
|
}
|
|
|
|
if (uptime_secs>60*60*24*7) {
|
|
|
|
upweeks = (int) uptime_secs / (60*60*24*7);
|
|
|
|
uptime_secs -= upweeks*60*60*24*7;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
upweeks = 0;
|
|
|
|
}
|
|
|
|
if (uptime_secs>60*60*24) {
|
|
|
|
updays = (int) uptime_secs / (60*60*24);
|
|
|
|
uptime_secs -= updays*60*60*24;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
updays = 0;
|
|
|
|
}
|
|
|
|
if (uptime_secs>60*60) {
|
|
|
|
uphours = (int) uptime_secs / (60*60);
|
|
|
|
uptime_secs -= uphours*60*60;
|
|
|
|
}
|
|
|
|
if (uptime_secs>60) {
|
|
|
|
upminutes = (int) uptime_secs / 60;
|
|
|
|
uptime_secs -= upminutes*60;
|
|
|
|
}
|
|
|
|
/*updecades = (int) uptime_secs / (60*60*24*365*10);
|
2015-06-21 18:20:04 +10:00
|
|
|
upyears = ((int) uptime_secs / (60*60*24*365)) % 10;
|
|
|
|
upweeks = ((int) uptime_secs / (60*60*24*7)) % 52;
|
|
|
|
updays = ((int) uptime_secs / (60*60*24)) % 7;
|
2015-09-04 00:00:00 -05:00
|
|
|
uphours = ((int) uptime_secs / (60*60)) % 24;
|
2015-06-21 18:20:04 +10:00
|
|
|
upminutes = ((int) uptime_secs / (60)) % 60;
|
2021-10-14 19:37:42 +11:00
|
|
|
*/
|
2015-06-21 18:20:04 +10:00
|
|
|
strcat(shortbuf, "up ");
|
|
|
|
|
|
|
|
if (updecades) {
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(shortbuf + pos, "%d %s",
|
|
|
|
updecades, updecades > 1 ? "decades" : "decade");
|
2015-09-04 00:00:00 -05:00
|
|
|
comma += 1;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (upyears) {
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(shortbuf + pos, "%s%d %s",
|
|
|
|
comma > 0 ? ", " : "", upyears,
|
|
|
|
upyears > 1 ? "years" : "year");
|
|
|
|
comma += 1;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (upweeks) {
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(shortbuf + pos, "%s%d %s",
|
|
|
|
comma > 0 ? ", " : "", upweeks,
|
|
|
|
upweeks > 1 ? "weeks" : "week");
|
|
|
|
comma += 1;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (updays) {
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(shortbuf + pos, "%s%d %s",
|
|
|
|
comma > 0 ? ", " : "", updays,
|
|
|
|
updays > 1 ? "days" : "day");
|
|
|
|
comma += 1;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (uphours) {
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(shortbuf + pos, "%s%d %s",
|
|
|
|
comma > 0 ? ", " : "", uphours,
|
|
|
|
uphours > 1 ? "hours" : "hour");
|
|
|
|
comma += 1;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
|
2017-09-25 08:36:51 +10:00
|
|
|
if (upminutes || (!upminutes && uptime_secs < 60)) {
|
2015-06-29 22:09:59 +10:00
|
|
|
pos += sprintf(shortbuf + pos, "%s%d %s",
|
2017-09-25 08:36:51 +10:00
|
|
|
comma > 0 ? ", " : "", upminutes,
|
|
|
|
upminutes != 1 ? "minutes" : "minute");
|
2015-06-29 22:09:59 +10:00
|
|
|
comma += 1;
|
2015-06-21 18:20:04 +10:00
|
|
|
}
|
|
|
|
return shortbuf;
|
|
|
|
}
|
|
|
|
|