2002-02-01 22:47:29 +00:00
|
|
|
/*
|
|
|
|
* tload.c - terminal version of xload
|
|
|
|
*
|
|
|
|
* Options:
|
|
|
|
* -s initial scaling exponent (default = 6)
|
|
|
|
* -d delay
|
|
|
|
*
|
|
|
|
* Copyright (c) 1992 Branko Lankester
|
|
|
|
* /proc changes by David Engel (david@ods.com)
|
|
|
|
* Made a little more efficient by Michael K. Johnson (johnsonm@sunsite.unc.edu)
|
2012-03-02 13:29:36 +01: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
|
2002-02-01 22:47:29 +00:00
|
|
|
*/
|
2012-03-02 13:29:36 +01:00
|
|
|
|
2016-04-16 17:03:57 +10:00
|
|
|
#include <proc/procps.h>
|
2011-10-09 12:25:19 +02:00
|
|
|
#include "c.h"
|
2012-03-23 13:32:24 +01:00
|
|
|
#include "fileutils.h"
|
2011-10-09 12:25:19 +02:00
|
|
|
#include "nls.h"
|
2011-12-18 14:29:19 +01:00
|
|
|
#include "strutils.h"
|
2011-10-09 21:29:26 +02:00
|
|
|
#include "xalloc.h"
|
2011-10-09 12:25:19 +02:00
|
|
|
|
2011-06-05 13:19:35 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <getopt.h>
|
2002-02-01 22:47:29 +00:00
|
|
|
#include <setjmp.h>
|
|
|
|
#include <signal.h>
|
2011-06-05 13:19:35 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2002-02-01 22:47:29 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <sys/ioctl.h>
|
2011-06-05 13:19:35 +02:00
|
|
|
#include <termios.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <limits.h>
|
2002-02-01 22:47:29 +00:00
|
|
|
|
|
|
|
static char *screen;
|
|
|
|
|
|
|
|
static int nrows = 25;
|
|
|
|
static int ncols = 80;
|
|
|
|
static int scr_size;
|
|
|
|
static int fd = STDOUT_FILENO;
|
2011-12-18 14:29:19 +01:00
|
|
|
static unsigned int dly = 5;
|
2002-02-01 22:47:29 +00:00
|
|
|
static jmp_buf jb;
|
|
|
|
|
2011-06-05 13:19:35 +02:00
|
|
|
static void alrm(int signo __attribute__ ((__unused__)))
|
2002-02-01 22:47:29 +00:00
|
|
|
{
|
2011-06-05 13:19:35 +02:00
|
|
|
signal(SIGALRM, alrm);
|
|
|
|
alarm(dly);
|
2002-02-01 22:47:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void setsize(int i)
|
|
|
|
{
|
2011-06-05 13:19:35 +02:00
|
|
|
struct winsize win;
|
|
|
|
|
|
|
|
signal(SIGWINCH, setsize);
|
|
|
|
if (ioctl(fd, TIOCGWINSZ, &win) != -1) {
|
|
|
|
if (win.ws_col > 0)
|
|
|
|
ncols = win.ws_col;
|
|
|
|
if (win.ws_row > 0)
|
|
|
|
nrows = win.ws_row;
|
|
|
|
}
|
|
|
|
if (ncols < 2 || ncols >= INT_MAX)
|
|
|
|
xerrx(EXIT_FAILURE, _("screen too small or too large"));
|
|
|
|
if (nrows < 2 || nrows >= INT_MAX / ncols)
|
|
|
|
xerrx(EXIT_FAILURE, _("screen too small or too large"));
|
2011-06-05 13:19:35 +02:00
|
|
|
scr_size = nrows * ncols;
|
2017-02-19 15:12:42 +00:00
|
|
|
if (scr_size < 2)
|
|
|
|
xerrx(EXIT_FAILURE, _("screen too small"));
|
2011-06-05 13:19:35 +02:00
|
|
|
if (screen == NULL)
|
2011-10-09 21:29:26 +02:00
|
|
|
screen = (char *)xmalloc(scr_size);
|
2011-06-05 13:19:35 +02:00
|
|
|
else
|
2011-10-09 21:29:26 +02:00
|
|
|
screen = (char *)xrealloc(screen, scr_size);
|
2011-06-05 13:19:35 +02:00
|
|
|
|
|
|
|
memset(screen, ' ', scr_size - 1);
|
|
|
|
*(screen + scr_size - 2) = '\0';
|
|
|
|
if (i)
|
|
|
|
longjmp(jb, 1);
|
2002-02-01 22:47:29 +00:00
|
|
|
}
|
|
|
|
|
2011-10-09 12:25:19 +02:00
|
|
|
static void __attribute__ ((__noreturn__)) usage(FILE * out)
|
2002-02-01 22:47:29 +00:00
|
|
|
{
|
2011-10-09 12:25:19 +02:00
|
|
|
fputs(USAGE_HEADER, out);
|
2011-12-17 18:32:47 +01:00
|
|
|
fprintf(out,
|
|
|
|
_(" %s [options] [tty]\n"), program_invocation_short_name);
|
2011-10-09 12:25:19 +02:00
|
|
|
fputs(USAGE_OPTIONS, out);
|
2013-10-11 10:07:10 +11:00
|
|
|
fputs(_(" -d, --delay <secs> update delay in seconds\n"), out);
|
|
|
|
fputs(_(" -s, --scale <num> vertical scale\n"), out);
|
2011-10-09 12:25:19 +02:00
|
|
|
fputs(USAGE_SEPARATOR, out);
|
|
|
|
fputs(USAGE_HELP, out);
|
|
|
|
fputs(USAGE_VERSION, out);
|
|
|
|
fprintf(out, USAGE_MAN_TAIL("tload(1)"));
|
2011-06-05 13:19:35 +02:00
|
|
|
|
|
|
|
exit(out == stderr ? EXIT_FAILURE : EXIT_SUCCESS);
|
|
|
|
}
|
2002-02-01 22:47:29 +00:00
|
|
|
|
2011-06-05 13:19:35 +02:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int lines, row, col = 0;
|
|
|
|
int i, opt;
|
|
|
|
double av[3];
|
2011-12-18 15:58:48 +01:00
|
|
|
static double max_scale = 0, scale_fact;
|
2011-12-18 14:29:19 +01:00
|
|
|
long tmpdly;
|
2011-06-05 13:19:35 +02:00
|
|
|
|
|
|
|
static const struct option longopts[] = {
|
|
|
|
{"scale", required_argument, NULL, 's'},
|
|
|
|
{"delay", required_argument, NULL, 'd'},
|
|
|
|
{"help", no_argument, NULL, 'h'},
|
|
|
|
{"version", no_argument, NULL, 'V'},
|
|
|
|
{NULL, 0, NULL, 0}
|
|
|
|
};
|
2013-02-20 18:31:48 +01:00
|
|
|
#ifdef HAVE_PROGRAM_INVOCATION_NAME
|
2012-01-03 18:48:43 +11:00
|
|
|
program_invocation_name = program_invocation_short_name;
|
2013-02-20 18:31:48 +01:00
|
|
|
#endif
|
2011-12-07 13:27:21 +01:00
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
bindtextdomain(PACKAGE, LOCALEDIR);
|
|
|
|
textdomain(PACKAGE);
|
2012-03-23 13:32:24 +01:00
|
|
|
atexit(close_stdout);
|
2011-12-07 13:27:21 +01:00
|
|
|
|
|
|
|
while ((opt =
|
|
|
|
getopt_long(argc, argv, "s:d:Vh", longopts, NULL)) != -1)
|
2011-06-05 13:19:35 +02:00
|
|
|
switch (opt) {
|
|
|
|
case 's':
|
2011-12-18 15:58:48 +01:00
|
|
|
max_scale = strtod_or_err(optarg, _("failed to parse argument"));
|
|
|
|
if (max_scale < 0)
|
2012-01-03 18:48:43 +11:00
|
|
|
xerrx(EXIT_FAILURE, _("scale cannot be negative"));
|
2011-06-05 13:19:35 +02:00
|
|
|
break;
|
|
|
|
case 'd':
|
2011-12-18 14:29:19 +01:00
|
|
|
tmpdly = strtol_or_err(optarg, _("failed to parse argument"));
|
|
|
|
if (tmpdly < 1)
|
2012-01-03 18:48:43 +11:00
|
|
|
xerrx(EXIT_FAILURE, _("delay must be positive integer"));
|
2011-12-18 14:29:19 +01:00
|
|
|
else if (UINT_MAX < tmpdly)
|
2012-01-03 18:48:43 +11:00
|
|
|
xerrx(EXIT_FAILURE, _("too large delay value"));
|
2011-12-18 14:29:19 +01:00
|
|
|
dly = tmpdly;
|
2011-06-05 13:19:35 +02:00
|
|
|
break;
|
|
|
|
case 'V':
|
2011-10-09 12:25:19 +02:00
|
|
|
printf(PROCPS_NG_VERSION);
|
|
|
|
return EXIT_SUCCESS;
|
2011-06-05 13:19:35 +02:00
|
|
|
break;
|
|
|
|
case 'h':
|
|
|
|
usage(stdout);
|
|
|
|
default:
|
|
|
|
usage(stderr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc > optind)
|
|
|
|
if ((fd = open(argv[optind], O_WRONLY)) == -1)
|
2012-01-03 18:48:43 +11:00
|
|
|
xerr(EXIT_FAILURE, _("can not open tty"));
|
2011-06-05 13:19:35 +02:00
|
|
|
|
|
|
|
setsize(0);
|
|
|
|
|
2011-12-18 15:58:48 +01:00
|
|
|
if (max_scale == 0)
|
2011-06-05 13:19:35 +02:00
|
|
|
max_scale = nrows;
|
|
|
|
|
|
|
|
scale_fact = max_scale;
|
|
|
|
|
|
|
|
setjmp(jb);
|
|
|
|
col = 0;
|
|
|
|
alrm(0);
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
|
|
|
|
if (scale_fact < max_scale)
|
|
|
|
scale_fact *= 2.0; /* help it drift back up. */
|
|
|
|
|
2015-07-01 21:47:30 +10:00
|
|
|
procps_loadavg(&av[0], &av[1], &av[2]);
|
2011-06-05 13:19:35 +02:00
|
|
|
|
2014-06-06 17:31:02 -07:00
|
|
|
do {
|
2012-01-21 23:29:02 +01:00
|
|
|
lines = av[0] * scale_fact;
|
|
|
|
row = nrows - 1;
|
|
|
|
|
|
|
|
while (0 <= --lines) {
|
|
|
|
*(screen + row * ncols + col) = '*';
|
|
|
|
if (--row < 0) {
|
|
|
|
scale_fact /= 2.0;
|
|
|
|
break;
|
|
|
|
}
|
2011-06-05 13:19:35 +02:00
|
|
|
}
|
2014-06-06 17:31:02 -07:00
|
|
|
} while (0 <= lines);
|
2012-01-21 23:29:02 +01:00
|
|
|
|
2011-06-05 13:19:35 +02:00
|
|
|
while (row >= 0)
|
|
|
|
*(screen + row-- * ncols + col) = ' ';
|
|
|
|
|
|
|
|
for (i = 1;; ++i) {
|
|
|
|
char *p;
|
|
|
|
row = nrows - (i * scale_fact);
|
|
|
|
if (row < 0 || row >= nrows)
|
2011-06-05 13:19:35 +02:00
|
|
|
break;
|
|
|
|
if (*(p = screen + row * ncols + col) == ' ')
|
|
|
|
*p = '-';
|
|
|
|
else
|
|
|
|
*p = '=';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++col == ncols) {
|
|
|
|
--col;
|
|
|
|
memmove(screen, screen + 1, scr_size - 1);
|
|
|
|
|
|
|
|
for (row = nrows - 2; row >= 0; --row)
|
|
|
|
*(screen + row * ncols + col) = ' ';
|
|
|
|
}
|
|
|
|
i = snprintf(screen, scr_size, " %.2f, %.2f, %.2f", av[0], av[1], av[2]);
|
|
|
|
if (i > 0 && i < scr_size)
|
2011-06-05 13:19:35 +02:00
|
|
|
screen[i] = ' ';
|
|
|
|
|
2012-01-21 16:24:33 +01:00
|
|
|
if (write(fd, "\033[H", 3) < 0)
|
|
|
|
xerr(EXIT_FAILURE, _("writing to tty failed"));
|
|
|
|
if (write(fd, screen, scr_size - 1) < 0)
|
|
|
|
xerr(EXIT_FAILURE, _("writing to tty failed"));
|
2011-06-05 13:19:35 +02:00
|
|
|
pause();
|
2002-02-01 22:47:29 +00:00
|
|
|
}
|
|
|
|
}
|