2000-03-05 08:16:03 +00:00
|
|
|
/* vi: set sw=4 ts=4: */
|
|
|
|
/*
|
|
|
|
* Mini uptime implementation for busybox
|
|
|
|
*
|
2004-03-15 08:29:22 +00:00
|
|
|
* Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
|
2000-03-05 08:16:03 +00:00
|
|
|
*
|
2010-08-16 20:14:46 +02:00
|
|
|
* Licensed under GPLv2, see file LICENSE in this source tree.
|
2000-03-05 08:16:03 +00:00
|
|
|
*/
|
|
|
|
|
2011-08-09 04:09:17 +02:00
|
|
|
/* 2011 Pere Orga <gotrunks@gmail.com>
|
|
|
|
*
|
|
|
|
* Added FEATURE_UPTIME_UTMP_SUPPORT flag.
|
2000-03-05 08:16:03 +00:00
|
|
|
*/
|
2011-08-09 04:09:17 +02:00
|
|
|
//config:config UPTIME
|
2018-12-28 03:20:17 +01:00
|
|
|
//config: bool "uptime (3.7 kb)"
|
2011-08-09 04:09:17 +02:00
|
|
|
//config: default y
|
|
|
|
//config: select PLATFORM_LINUX #sysinfo()
|
|
|
|
//config: help
|
2017-07-21 09:50:55 +02:00
|
|
|
//config: uptime gives a one line display of the current time, how long
|
|
|
|
//config: the system has been running, how many users are currently logged
|
|
|
|
//config: on, and the system load averages for the past 1, 5, and 15 minutes.
|
2011-08-09 04:09:17 +02:00
|
|
|
//config:
|
|
|
|
//config:config FEATURE_UPTIME_UTMP_SUPPORT
|
2017-01-10 14:58:54 +01:00
|
|
|
//config: bool "Show the number of users"
|
2011-08-09 04:09:17 +02:00
|
|
|
//config: default y
|
|
|
|
//config: depends on UPTIME && FEATURE_UTMP
|
|
|
|
//config: help
|
2017-07-21 09:50:55 +02:00
|
|
|
//config: Display the number of users currently logged on.
|
2011-08-09 04:09:17 +02:00
|
|
|
|
2017-08-07 16:47:34 +02:00
|
|
|
//applet:IF_UPTIME(APPLET_NOEXEC(uptime, uptime, BB_DIR_USR_BIN, BB_SUID_DROP, uptime))
|
2016-11-23 06:23:44 +01:00
|
|
|
|
|
|
|
//kbuild:lib-$(CONFIG_UPTIME) += uptime.o
|
|
|
|
|
2011-04-11 03:29:49 +02:00
|
|
|
//usage:#define uptime_trivial_usage
|
|
|
|
//usage: ""
|
|
|
|
//usage:#define uptime_full_usage "\n\n"
|
|
|
|
//usage: "Display the time since the last boot"
|
|
|
|
//usage:
|
|
|
|
//usage:#define uptime_example_usage
|
|
|
|
//usage: "$ uptime\n"
|
|
|
|
//usage: " 1:55pm up 2:30, load average: 0.09, 0.04, 0.00\n"
|
|
|
|
|
2007-05-26 19:00:18 +00:00
|
|
|
#include "libbb.h"
|
2011-07-26 13:42:12 +02:00
|
|
|
#ifdef __linux__
|
|
|
|
# include <sys/sysinfo.h>
|
|
|
|
#endif
|
|
|
|
|
2005-08-27 18:18:06 +00:00
|
|
|
#ifndef FSHIFT
|
|
|
|
# define FSHIFT 16 /* nr of bits of precision */
|
|
|
|
#endif
|
2011-08-10 13:00:04 +02:00
|
|
|
#define FIXED_1 (1 << FSHIFT) /* 1.0 as fixed-point */
|
|
|
|
#define LOAD_INT(x) (unsigned)((x) >> FSHIFT)
|
|
|
|
#define LOAD_FRAC(x) LOAD_INT(((x) & (FIXED_1 - 1)) * 100)
|
2000-03-05 08:16:03 +00:00
|
|
|
|
2007-10-11 10:05:36 +00:00
|
|
|
int uptime_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
2008-07-05 09:18:54 +00:00
|
|
|
int uptime_main(int argc UNUSED_PARAM, char **argv UNUSED_PARAM)
|
2000-03-05 08:16:03 +00:00
|
|
|
{
|
2011-08-10 13:00:04 +02:00
|
|
|
unsigned updays, uphours, upminutes;
|
2018-08-03 19:53:49 +02:00
|
|
|
unsigned opts;
|
2000-03-05 08:16:03 +00:00
|
|
|
struct sysinfo info;
|
|
|
|
struct tm *current_time;
|
|
|
|
time_t current_secs;
|
|
|
|
|
2018-08-03 19:53:49 +02:00
|
|
|
opts = getopt32(argv, "s");
|
|
|
|
|
2000-03-05 08:16:03 +00:00
|
|
|
time(¤t_secs);
|
2018-08-03 19:53:49 +02:00
|
|
|
sysinfo(&info);
|
|
|
|
|
|
|
|
if (opts) // -s
|
|
|
|
current_secs -= info.uptime;
|
|
|
|
|
2000-03-05 08:16:03 +00:00
|
|
|
current_time = localtime(¤t_secs);
|
|
|
|
|
2018-08-03 19:53:49 +02:00
|
|
|
if (opts) { // -s
|
|
|
|
printf("%04u-%02u-%02u %02u:%02u:%02u\n",
|
|
|
|
current_time->tm_year + 1900, current_time->tm_mon + 1, current_time->tm_mday,
|
|
|
|
current_time->tm_hour, current_time->tm_min, current_time->tm_sec
|
|
|
|
);
|
|
|
|
/* The above way of calculating boot time is wobbly,
|
|
|
|
* info.uptime has only 1 second precision, which makes
|
|
|
|
* "uptime -s" wander +- one second.
|
|
|
|
* /proc/uptime may be better, it has 0.01s precision.
|
|
|
|
*/
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
2000-03-05 08:16:03 +00:00
|
|
|
|
2011-08-10 13:00:04 +02:00
|
|
|
printf(" %02u:%02u:%02u up ",
|
2018-08-03 19:53:49 +02:00
|
|
|
current_time->tm_hour, current_time->tm_min, current_time->tm_sec
|
|
|
|
);
|
2011-08-10 13:00:04 +02:00
|
|
|
updays = (unsigned) info.uptime / (unsigned)(60*60*24);
|
2018-08-03 19:53:49 +02:00
|
|
|
if (updays != 0)
|
2011-08-10 13:00:04 +02:00
|
|
|
printf("%u day%s, ", updays, (updays != 1) ? "s" : "");
|
|
|
|
upminutes = (unsigned) info.uptime / (unsigned)60;
|
|
|
|
uphours = (upminutes / (unsigned)60) % (unsigned)24;
|
2000-03-05 08:16:03 +00:00
|
|
|
upminutes %= 60;
|
2018-08-03 19:53:49 +02:00
|
|
|
if (uphours != 0)
|
2011-08-10 13:00:04 +02:00
|
|
|
printf("%2u:%02u", uphours, upminutes);
|
2000-03-05 08:16:03 +00:00
|
|
|
else
|
2011-08-10 13:00:04 +02:00
|
|
|
printf("%u min", upminutes);
|
2000-03-05 08:16:03 +00:00
|
|
|
|
2011-08-09 04:09:17 +02:00
|
|
|
#if ENABLE_FEATURE_UPTIME_UTMP_SUPPORT
|
2011-08-10 13:00:04 +02:00
|
|
|
{
|
2015-04-02 23:03:46 +02:00
|
|
|
struct utmpx *ut;
|
2011-08-10 13:00:04 +02:00
|
|
|
unsigned users = 0;
|
2015-04-02 23:03:46 +02:00
|
|
|
while ((ut = getutxent()) != NULL) {
|
|
|
|
if ((ut->ut_type == USER_PROCESS) && (ut->ut_user[0] != '\0'))
|
2011-08-10 13:00:04 +02:00
|
|
|
users++;
|
|
|
|
}
|
|
|
|
printf(", %u users", users);
|
2011-08-09 04:09:17 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-08-10 13:00:04 +02:00
|
|
|
printf(", load average: %u.%02u, %u.%02u, %u.%02u\n",
|
2004-03-15 08:29:22 +00:00
|
|
|
LOAD_INT(info.loads[0]), LOAD_FRAC(info.loads[0]),
|
|
|
|
LOAD_INT(info.loads[1]), LOAD_FRAC(info.loads[1]),
|
2000-03-05 08:16:03 +00:00
|
|
|
LOAD_INT(info.loads[2]), LOAD_FRAC(info.loads[2]));
|
|
|
|
|
2000-12-01 02:55:13 +00:00
|
|
|
return EXIT_SUCCESS;
|
2000-03-05 08:16:03 +00:00
|
|
|
}
|