[comp.sys.att] SYSINFO utility for information on UNIX pc SLK lines

lenny@icus.islp.ny.us (Lenny Tropiano) (10/16/89)

Since for some reason a lot of people have the older version, and don't
have this version, my release 2.1, I'm reposting it to unix-pc.sources.
It hasn't been posted here in a long time, so it isn't too bad.  This
version has the loadavgd (daemon) to report load avg results.  Now
if someone coulld EXPLAIN to me the significance of the three numbers,
other than the larger the number the more loaded the machine.  Note,
I didn't write loadavgd, I just incorporated the ipc-shared memory
segment.

-Lenny
--

#! /bin/sh
# This is a shell archive.  Remove anything before this line, then unpack
# it by saving it into a file and typing "sh file".  To overwrite existing
# files, type "sh file -c".  You can also feed this as standard input via
# unshar, or by typing "sh <file", e.g..  If this archive is complete, you
# will see the following message at the end:
#		"End of shell archive."
# Contents:  README README2.1 Makefile sysinfo.c loadavgd.c
# Wrapped by lenny@icus on Mon Oct 16 08:35:18 1989
PATH=/bin:/usr/bin:/usr/ucb ; export PATH
if test -f README -a "${1}" != "-c" ; then 
  echo shar: Will not over-write existing file \"README\"
else
echo shar: Extracting \"README\" \(1549 characters\)
sed "s/^X//" >README <<'END_OF_README'
XI wrote this little ditty just to utilize the two lines on the bottom of
Xthe screen where the soft labels for the function keys are normally displayed.
XThis program will only work if you login on the console (obviously) and
Xyou aren't using the "user agent" or something that needs those two lines.
X
XI give thanks to Scott Hazen Mueller for his contribution of "newmgr.c"
Xthat initially gave me the idea for displaying the file system information.
X
XAfter unpacking the shar file, "su" and become root.  Then type:
X
X	# make install
X
XThis will copy the program into /usr/lbin (as defined by the Makefile) but
Xyou might just want to have this for your own "bin" directory, because
Xit is only useful if you run on a windowed device.
X
XThe mail file is only read for the number of messages if the modification
Xtime changes since the last read (saving on disk I/O).  This program also
Xruns with a nice value of 5, and sleeps for 15 seconds between passes.
X(#defined by NICE and SLEEP respectively)
X
XYou might want to place the following lines in your /etc/localprofile
Xor .profile to activate sysinfo apon login:
X
XTTY=`tty`
Xif [ "`expr $TTY : '/dev/w' `" != "0" ]
Xthen 
X	sysinfo
Xfi
X
XPlease report any bugs, suggestions and problems to me:
X
X| Lenny Tropiano            ICUS Software Systems      [w] +1 (516) 589-7930 |
X| lenny@icus.islp.ny.us     Telex; 154232428 ICUS      [h] +1 (516) 968-8576 |
X| {ames,pacbell,decuac,hombre,sbcs,attctc}!icus!lenny     attmail!icus!lenny |
X+------- ICUS Software Systems -- PO Box 1;  Islip Terrace, NY  11752 -------+
END_OF_README
if test 1549 -ne `wc -c <README`; then
    echo shar: \"README\" unpacked with wrong size!
fi
# end of overwriting check
fi
if test -f README2.1 -a "${1}" != "-c" ; then 
  echo shar: Will not over-write existing file \"README2.1\"
else
echo shar: Extracting \"README2.1\" \(2230 characters\)
sed "s/^X//" >README2.1 <<'END_OF_README2.1'
XSomething changed from previous version of sysinfo.
X
X	1)  added memory computation so you can see you memory slowly
X	    fading away...
X
X	2)  Used a loadavgd routine that created BSD-type load average
X	    shared memory segment:
X
XIPC status from /dev/kmem as of Mon Jun 27 19:11:36 1988
XT     ID     KEY        MODE       OWNER    GROUP
XMessage Queues:
XShared Memory:
Xm      0 0x610205de --rw-r--r--     root     root
XSemaphores:
X
X	    That can be read by this program and updated so you can
X	    see how your machine is loaded down.
X
XBug fixes:
X
X	1)  Clearing out the mail message counter when you deleted ALL
X	    your incoming mail.
X
XUpdates:
X
XRelease 2.1:
X
XDavid H. Brierley
XHome: dave@galaxia.Newport.RI.US   {rayssd,xanth,lazlo,jclyde}!galaxia!dave
XWork: dhb@rayssd.ray.com           {sun,decuac,gatech,necntc,ukma}!rayssd!dhb
X
XThe following is a diff listing for a minor change that I made to the version
X2.0 release of sysinfo.  It is used in conjunction with the mode that makes
Xsysinfo follow the active window and what it does is look to see if the label
Xlines are blank before attempting to use them.  I like having sysinfo follow
Xthe active window because I do a lot of window type stuff using windy and
Xsome other software that I have developed, but every once in a while I will
Xbring up the UA or some other program that wants to write into the label lines.
XA few seconds later, the labels are overwritten by sysinfo.
X
XThe basic changes I made were to enclose the calls that rewrote the label
Xlines inside an if statement that calls a new routine to check if it is ok
Xto write on the labels and then, of course, I included the new routine.  The
Xroutine says it is ok to write if the label lines are completely blank or if
Xthey look like they were previously written by sysinfo.  The changes are simple
Xenough that you should have no problems incorporating them (assuming that you
Xwant to) into the source even if you have made massive changes since the last
Xtime you posted it.
X
XI was thinking of posting this patch to unix-pc.sources but I thought I would
Xlet you take a look at it first.  If you want me to go ahead and post it let
Xme know, otherwise I will wait for it to be incorporated into the next release.
X
END_OF_README2.1
if test 2230 -ne `wc -c <README2.1`; then
    echo shar: \"README2.1\" unpacked with wrong size!
fi
# end of overwriting check
fi
if test -f Makefile -a "${1}" != "-c" ; then 
  echo shar: Will not over-write existing file \"Makefile\"
else
echo shar: Extracting \"Makefile\" \(961 characters\)
sed "s/^X//" >Makefile <<'END_OF_Makefile'
X#
X# Makefile to compile sysinfo.c  (System Info)  Version 2.0
X# By Lenny Tropiano  
X# (c)1987,1988 ICUS Software Systems  UUCP:  ...icus!lenny
X#
XCFLAGS=-v -O
XLDFLAGS=-s
XLIBS=/lib/crt0s.o /lib/shlib.ifile
XDEST=/usr/lbin/
XDAEMONS=/etc/daemons/
X#
XWINDOW= -DCHGWIND # Change sysinfo to current window always
X#
Xsysinfo.o:
X	$(CC) $(CFLAGS) $(WINDOW) -c sysinfo.c
X#
Xall:	sysinfo loadavgd
X
Xsysinfo:  sysinfo.o
X	@echo "Loading ..."
X	$(LD) $(LDFLAGS) -o sysinfo sysinfo.o $(LIBS) 
X
Xloadavgd:  loadavgd.o 
X	@echo "Loading ..."
X	$(LD) $(LDFLAGS) -o loadavgd loadavgd.o -lm $(LIBS) 
X#
X# You need to be superuser to do this
X#
X$(DEST):
X	mkdir $(DEST)
X$(DAEMONS):
X	mkdir $(DAEMONS)
X
Xinstall: all $(DEST) $(DAEMONS)
X	cp sysinfo $(DEST)
X	chown root $(DEST)sysinfo
X	chgrp bin  $(DEST)sysinfo
X	chmod 4755 $(DEST)sysinfo
X	cp loadavgd $(DAEMONS)
X	chown root $(DAEMONS)loadavgd
X	chgrp bin  $(DAEMONS)loadavgd
X	chmod 4750 $(DAEMONS)loadavgd
X	nohup $(DAEMONS)loadavgd> /dev/null 2>&1 &
END_OF_Makefile
if test 961 -ne `wc -c <Makefile`; then
    echo shar: \"Makefile\" unpacked with wrong size!
fi
# end of overwriting check
fi
if test -f sysinfo.c -a "${1}" != "-c" ; then 
  echo shar: Will not over-write existing file \"sysinfo.c\"
else
echo shar: Extracting \"sysinfo.c\" \(9977 characters\)
sed "s/^X//" >sysinfo.c <<'END_OF_sysinfo.c'
X/************************************************************************\
X**                                                                      **
X** Program name: sysinfo.c (System Info)                                **
X** Programmer:   Lenny Tropiano            UUCP: ...icus!lenny          **
X** Organization: ICUS Software Systems     (c)1988, 1989                **
X** Date:         January 23, 1988                                       **
X** Version 2.0:  June 27, 1988                                          **
X** Version 2.1:  Februrary 19, 1989 (Revision Level 4)                  **
X**                                                                      **
X** Revision Level: 1  [Updated to switch to current window]             **
X** Revision Level: 2  [Updated to show load average in the window]      **
X** Revision Level: 3  [Updated to available main memory in the window]  **
X** Revision Level: 4  [Updated by David H. Brierley to check the the    **
X**                     current status of the SLK lines for existing     **
X**                     information and not display the sysinfo lines    **
X**                     over those SLK's if in CHGWIND #define'd, good   **
X**                     for use with the UA and Telephone Manager F-key  **
X**                     SLK definitions]  -- dave@galaxia.Newport.RI.US  **
X**                                                                      **
X**************************************************************************
X**                                                                      **
X** Credits:      The idea of displaying the file system information     **
X**               came from Scott Hazen Mueller's  newmgr, the replace-  **
X**               ment to the smgr.                                      **
X**                                                                      **
X**************************************************************************
X**                                                                      **
X** Program use:  Program is run as a info process from your .profile    **
X**               This program the file system and displays the          **
X**               pertinent information on the bottom of the screen      **
X**               where the software labels would normally be displayed. **
X**               The program also displays the uptime.                  **
X**                                                                      **
X**************************************************************************
X** Permission is granted to distribute this program by any means as     **
X** long as credit is given for my work.     I would also like to see    **
X** any modifications or enhancements to this program.                   **
X\************************************************************************/
X
X#include <stdio.h>
X#include <fcntl.h>
X#include <errno.h>
X#include <signal.h>
X#include <sys/types.h>
X#include <sys/stat.h>
X#include <sys/window.h>
X#include <sys/filsys.h>
X#include <sys/param.h>
X#include <sys/ipc.h>
X#include <sys/shm.h>
X#include <nlist.h>
X#include <utmp.h>
X#include <setjmp.h>
X
X#ifndef ctob			/* For System V */
X#include <sys/sysmacros.h>
X#endif
X
Xstruct nlist unixsym[] = {
X#define X_FREEMEM	0
X	{ "freemem" },
X	{ NULL }
X};
X
Xstatic  char *LINE_1 = "%-20.20s %-25.25s   up  %2d day%1.1s %2d hour%1.1s %2d minute%1.1s";
Xstatic	char *LINE_2 = "Filesystem: %6ld free  %6ld total %2d%% available %5d i-nodes  %4.2fMB memory";
X
X#define UNIX	"/unix"
X#define KMEM	"/dev/kmem"
X#define	MINUTE	60L
X#define	HOUR	(60L * 60L)
X#define	DAY	(24L * 60L * 60L)
X#define	SLEEP	15			/* Sleep time (interval between)*/
X#define	NICE	5			/* Niceness value		*/
X#define MEGABYTE 1048576.0
X                             
X#define	FILESYS	"/dev/rfp002"
X#ifndef TRUE
X#define TRUE	1
X#endif
X
Xint	wd;				/* window descriptor		*/
Xint 	kmem;				/* kmem file descriptor		*/
Xjmp_buf	jmpbuf;				/* save the current environ	*/
Xchar	*progname;			/* program name			*/
Xchar	mailfile[30];			/* mailbox file name            */
Xstruct  utdata utd;			/* Window data structure        */
Xstruct  filsys fs;			/* Filesystem superblock struct */
Xtime_t	boottime;			/* system boot time in sec      */
Xtime_t	mailtime = 0L;			/* mail modification time       */
Xint	msgs = 0;
Xint	shm;
Xunsigned long	days = 0L;
Xunsigned long	hrs  = 0L;
Xunsigned long	mins = 0L;
X
X/************************************************************************/
X
Xmain(argc,argv)
Xint	argc;
Xchar	*argv[];
X{
X	int	terminate();
X	char	*getenv();
X	struct	utmp	*utent, *getutent();
X
X	if (fork() != 0) 		/* detach process-info */
X		exit(0);
X		
X	nice(NICE);			/* Be a little nice	*/
X	signal (SIGINT, SIG_IGN);
X	signal (SIGQUIT, SIG_IGN);
X 	signal (SIGHUP, terminate);
X 	signal (SIGTERM, terminate);
X
X    	progname = *argv;
X	sprintf(mailfile,"%s",getenv("MAIL"));
X
X	setutent();
X	while ((utent = getutent()) != (struct utmp *)NULL) {
X		if (utent->ut_type == BOOT_TIME) {
X			boottime = utent->ut_time;
X			break;
X		}
X	}
X	endutent();
X
X	shm = shmget(ftok("/unix",'a'),12,0);
X	/*
X	 * Look up addresses of variables.
X	 */
X	if (nlist(UNIX, unixsym) < 0) {
X		fprintf(stderr, "%s: no namelist.\n", UNIX);
X		exit(1);
X	}
X	/*
X	 * Open kernel memory.
X	 */
X	if ((kmem = open(KMEM, 0)) < 0) {
X		perror(KMEM);
X		exit(1);
X	}
X
X	info_process();
X	terminate();
X
X}
X
Xinfo_process()
X{
X	int	refrsh();
X	long	memory();
X	float	freemem;
X	char	mailbuffer[30], loadbuf[30];
X
X#ifdef CHGWIND
X 	signal (SIGWIND, refrsh);  /* trap window change signal */
X#else
X	open_current_window();	   /* must initialize if not using -DCHGWIND */
X#endif 
X
X	while (TRUE) {
X
X#ifdef CHGWIND
X		setjmp(jmpbuf);
X		open_current_window();
X#endif
X
X		if (slktest(wd) == 0) {
X			filestatus();
X			uptime();
X			mailcheck(mailbuffer);
X			loadaverage(loadbuf);
X
X			utd.ut_num = WTXTSLK1;
X			sprintf(utd.ut_text,LINE_1,
X				mailbuffer, loadbuf,
X				days,(days == 1) ? " " : "s",
X				hrs, (hrs  == 1) ? " " : "s",
X				mins,(mins == 1) ? " " : "s"
X			);
X			ioctl(wd, WIOCSETTEXT, &utd);
X
X			utd.ut_num = WTXTSLK2;
X			freemem    = memory() / MEGABYTE;
X			sprintf(utd.ut_text,LINE_2,
X			   fs.s_tfree * 2, fs.s_fsize * 2, 
X			   (int)((((float)fs.s_tfree / (float)fs.s_fsize) + 0.005) 
X				   * 100.0), fs.s_tinode, freemem
X			);
X			ioctl(wd, WIOCSETTEXT, &utd);
X		}
X#ifdef CHGWIND
X		close(wd);
X#endif
X		sleep(SLEEP);
X	}
X}
X
Xint refrsh()
X{
X	int	window, 
X		dummy, 
X		refrsh();
X	char	windev[9];
X
X	signal(SIGWIND,SIG_IGN);
X	window = ioctl(0, WIOCGPREV, dummy);
X	sprintf(windev,"/dev/w%d",window);
X	if ((wd = open(windev,O_RDWR|O_NDELAY)) == -1) 
X		wd = 0;
X	utd.ut_num = WTXTSLK1;		/* clear the label area     */
X	utd.ut_text[0] = '\0';
X        ioctl(wd, WIOCSETTEXT, &utd);
X	utd.ut_num = WTXTSLK2;
X        ioctl(wd, WIOCSETTEXT, &utd);
X	close(wd);			/* close previous window    */
X	longjmp(jmpbuf, 0);		/* return back to main loop */
X}
X
Xint open_current_window()
X{
X	int	window, 
X		dummy, 
X		refrsh();
X	char	windev[9];
X
X	window = ioctl(0, WIOCGCURR, dummy);
X	sprintf(windev,"/dev/w%d",window);
X	if ((wd = open(windev,O_RDWR|O_NDELAY)) == -1) 
X		wd = 0;
X}
X
Xint terminate()
X{
X	utd.ut_num = WTXTSLK1;		/* clear the label area */
X	utd.ut_text[0] = '\0';
X        ioctl(0, WIOCSETTEXT, &utd);
X	utd.ut_num = WTXTSLK2;
X        ioctl(0, WIOCSETTEXT, &utd);
X#ifndef CHGWIND
X	close(wd);
X#endif
X	close(kmem);
X
X	exit(0);
X}
X
Xfilestatus()
X{
X	int	fd;
X
X	if ((fd = open(FILESYS, O_RDONLY)) == -1) {
X		fprintf(stderr,"%s: cannot open %s for read\n",
X			progname, FILESYS);
X		terminate();
X	}
X	
X	if (lseek(fd, 512, 0) == -1) {
X		fprintf(stderr,"%s: cannot lseek to superblock\n", progname);
X		terminate();
X	}
X
X	if (read(fd, &fs, sizeof(struct filsys)) == -1) {
X		fprintf(stderr,"%s: cannot read the superblock\n", progname);
X		terminate();
X	}
X
X	close(fd);
X
X}
X
Xuptime()
X{
X	time_t   curtime, bootsec;
X
X	time(&curtime);
X	bootsec = curtime - boottime;
X
X	days = bootsec / DAY;
X	bootsec -= DAY * days;
X	hrs = bootsec / HOUR;
X	bootsec -= HOUR * hrs;
X	mins = bootsec / MINUTE;
X	bootsec -= MINUTE * mins;
X}
X
Xmailcheck(buf)
Xchar *buf;
X{
X	FILE	*fp;
X	char	buffer[BUFSIZ];
X	struct	stat	statbuf;
X
X	if (access(mailfile,4) == 0) {
X		if (stat(mailfile,&statbuf) != -1) {
X		    if (statbuf.st_ctime > mailtime) {
X			msgs = 0;
X			if ((fp = fopen(mailfile,"r")) != NULL) {
X			   while (fgets(buffer, BUFSIZ, fp) != NULL) 
X				if (strncmp(buffer,"From ",5) == 0)
X					msgs++;
X			}
X			fclose(fp);
X			mailtime = statbuf.st_ctime;
X		    }
X		}
X	} else
X		msgs = 0;
X	if (msgs > 0) 
X		sprintf(buf,"%d mail message%s",msgs,
X			(msgs == 1) ? "" : "s");
X	else
X		buf[0] = '\0';
X}
X
Xloadaverage(lbuf)
Xchar *lbuf;
X{
X	double	*shmseg;
X
X	shmseg = (double *)shmat(shm, (char *) 0, SHM_RDONLY);
X	sprintf(lbuf,"Load avg:  %4.2f %4.2f %4.2f", 
X		shmseg[0], shmseg[1], shmseg[2]);
X	shmdt(shmseg);
X}
X
Xlong memory()
X{
X	int mem;
X
X	/*
X	 * Read variables.
X	 */
X	lseek(kmem, (long) unixsym[X_FREEMEM].n_value, 0);
X	read(kmem, (char *) &mem, sizeof(int));
X
X	return(ctob(mem));
X}
X
Xslktest (wd)
Xint wd;
X{
X	/*
X	 * check the SLK lines for the existence of anything currently there
X	 * if found two blank SLK lines or some remaining sysinfo information,
X	 * denoted by "Filesystem" in the second line, then return 0, else
X	 * return -1
X	 * Developed by David H. Brierley (dave@galaxia.Newport.RI.US)
X	 */
X
X	int rvalue;
X	int n;
X
X	rvalue = 0;
X	utd.ut_num = WTXTSLK1;
X	ioctl (wd, WIOCGETTEXT, &utd);
X	if (utd.ut_text[0] == '\0') 
X		++rvalue;
X	else 
X		for (n = 0; n < WTXTLEN; ++n) 
X			if ((utd.ut_text[n] != '\0') && 
X			    (utd.ut_text[n] != ' ')) 
X				break;
X
X	if (n == WTXTLEN) 
X	    ++rvalue;
X
X	utd.ut_num = WTXTSLK2;
X	ioctl (wd, WIOCGETTEXT, &utd);
X	if (utd.ut_text[0] == '\0') 
X		++rvalue;
X	else 
X		for (n = 0; n < WTXTLEN; ++n) 
X			if ((utd.ut_text[n] != '\0') && 
X			    (utd.ut_text[n] != ' ')) 
X				break;
X
X	if (n == WTXTLEN) 
X		++rvalue;
X
X	if (rvalue == 2) 	/* two blank lines */
X		return (0);
X
X				/* found the beginning of LINE_2 */
X
X    	if (strncmp (utd.ut_text, LINE_2, 12) == 0) 
X		return (0);
X
X	/* not blank, return -1 */
X	return (-1);
X
X}
END_OF_sysinfo.c
if test 9977 -ne `wc -c <sysinfo.c`; then
    echo shar: \"sysinfo.c\" unpacked with wrong size!
fi
# end of overwriting check
fi
if test -f loadavgd.c -a "${1}" != "-c" ; then 
  echo shar: Will not over-write existing file \"loadavgd.c\"
else
echo shar: Extracting \"loadavgd.c\" \(10503 characters\)
sed "s/^X//" >loadavgd.c <<'END_OF_loadavgd.c'
X/*
X * renamed loadavgd.c by Lenny Tropiano (lenny@icus.UUCP ICUS Software Systems)
X *
X *	ldavg.c -- compute load averages for System V
X *	Phil Budne @ Boston U / DSG
X *
X *	Forges BSD 4.2 rwhod packets containing system load averages
X *	(#ifdef RWHOD for this, else a shm segment is used, ftok("/unix", 'a'))
X */
X
X# include <sys/types.h>			/* system types */
X# include <sys/sysinfo.h>		/* sysinfo structure */
X# include <sys/utsname.h>		/* for uname(2) */
X# include <sys/stat.h>			/* for stat(2) */
X# include <sys/param.h>			/* for HZ */
X# include <stdio.h>
X# include <nlist.h>
X# include <time.h>
X# include <math.h>
X# include <utmp.h>
X# include <fcntl.h>
X#ifdef RWHOD
X# include "rwhod.h"			/* (from BSD) */
X#else  SHM
X# include <sys/ipc.h>
X# include <sys/shm.h>
X#endif RWHOD
X#ifdef UNIXPC
X/*
X * Open a window on the status line.  While SMGR owns columns 33-74, it doesn't
X * seem to use columns 56-74, which by some strange chance is just enough for
X * our window.
X * (Whoops!  Can you say "icon"?  We use the spot reserved for the Phone
X * Manager instead... your choice.  Compile with -DUNIXPC to get windows;
X * the generic-System V code will work on the UNIX PC if you don't want them.)
X */
X# include <sys/window.h>
X# include <status.h>
X#endif UNIXPC
X
X/* # define DEBUG /**/
X#ifdef RWHOD
X# define UDP 1
X# define DSK 1
X# define PMUL 100
X
X# if UDP
X# include "netdb.h"
Xunsigned short port = 513;
Xunsigned long ipaddr;
X# endif
X#endif RWHOD
X
Xextern struct utmp *getutent();
X
X# define UNIX "/unix"
X# define KMEM "/dev/kmem"
X
Xstruct nlist nlsym[] = {
X# define NL_SYSINFO 0
X        { "sysinfo" },			/* 0 */
X#ifdef RWHOD
X# define NL_LBOLT 1
X	{ "lbolt" },			/* 1 */
X#endif RWHOD
X        { 0 }
X};
X
X#ifdef RWHOD
Xstruct whod proto;
Xstruct utsname utsn;
Xchar whopacket[100];
X#else  SHM
Xkey_t aven_key;
Xint aven_shm;
Xdouble *aven_seg;
X#endif RWHOD
Xint fd, memfd;
X#ifdef UNIXPC
Xstruct uwdata avenwin = {
X	0,	0,	288,	12,	NBORDER,	0,	0,	0
X};
Xextern char *sys_errlist[];
Xextern int errno;
X#endif
X
Xchar *unixsys = UNIX;
Xchar *kmem = KMEM;
Xchar *argv0;
X
Xmain(argc, argv)
Xint  argc;
Xchar *argv[];
X{
X	switch (fork()) {
X	case -1:
X		perror("fork");
X		exit(1);
X	case 0:
X		break;
X	default:
X		exit(0);
X	}
X	argv0 = argv[0];
X#ifdef RWHOD
X	uname(&utsn);			/* get system names */
X#endif RWHOD
X	setpgrp();			/* create own pgrp */
X#ifdef UNIXPC
X	if (freopen("/dev/window", "r+", stdout) == (FILE *) 0) {
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Can't open load average window:\n%s", argv0, sys_errlist[errno]);
X		exit(1);
X	}
X	ioctl(1, WIOCSETD, &avenwin);
X	ioctl(0, WIOCSELECT);
X	fclose(stdin);
X	fclose(stderr);
X#endif UNIXPC
X	init_nlist();			/* get name values, open kmem */
X	init_packet();			/* initialize packet prototype */
X	doit();				/* never returns */
X} /* main */
X
Xinit_nlist() {
X	nlist(unixsys, nlsym);		/* get system values */
X
X        if(nlsym[NL_SYSINFO].n_value == 0) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Cannot locate `sysinfo' structure", argv0);
X#else
X                fprintf(stderr, "%s: can't find sysinf structure\n", argv0);
X#endif
X                exit(1);
X        } /* no value */
X
X	if ((memfd = open(kmem, O_RDONLY)) < 0) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Cannot open kernel memory image:\n%s", argv0, sys_errlist[errno]);
X#else
X		fprintf(stderr, "%s: no mem\n", argv0);
X#endif UNIXPC
X		exit(1);
X	} /* could not open kmem */
X
X} /* init_nlist */
X
X# define PERIOD 5			/* sample period (in seconds) */
X# define INTERVAL1 60			/* average interval 1 (in seconds) */
X# define INTERVAL2 (5*60)		/* average interval 2 (in seconds) */
X# define INTERVAL3 (15*60)		/* average interval 3 (in seconds) */
X# define PACKINTERVAL 30		/* interval for make_packet */
X
Xdoit() {
X	struct sysinfo sinf;
X	int packt = 0;
X	long occ, que, nocc, nque, n, c;
X	double avg1, avg2, avg3, new;
X	double exp1, exp2, exp3;
X
X	exp1 = exp( - ((double) PERIOD) / INTERVAL1 );
X	exp2 = exp( - ((double) PERIOD) / INTERVAL2 );
X	exp3 = exp( - ((double) PERIOD) / INTERVAL3 );
X
X	getsysinf(&sinf);		/* prime the pump */
X	occ = sinf.runocc;		/* number of samples */
X	que = sinf.runque;		/* run queue summation */
X
X	avg1 = avg2 = avg3 = ((double) que) / occ;
X
X	for( ; ; ) {
X		if( --packt < 0 ) {
X#ifdef RWHOD
X			make_packet((int) (avg1 * PMUL),
X				    (int) (avg2 * PMUL),
X				    (int) (avg3 * PMUL));
X#else  SHM
X			make_packet(avg1, avg2, avg3);
X#endif RWHOD
X			packt = PACKINTERVAL / PERIOD;
X		} /* packet time */
X
X/*		printf("runque: %ld  runocc: %ld\n", que, occ ); /**/
X
X		sleep(PERIOD);
X		getsysinf(&sinf);	/* get new info */
X		nocc = sinf.runocc;
X		nque = sinf.runque;
X
X		n = nocc - occ;		/* get number of times updated */
X		if( n <= 0 ) continue;
X		c = nque - que - n;	/* get number of runners w/o us */
X		if( c < 0 ) c = 0;	/* mumble! */
X
X		new = ((double) c ) / n; /* new instantaneous avg */
X
X		/************************************************/
X		/*   The following formula is used to achieve   */
X		/*   exponential decay of old measurements:	*/
X		/*	avgN = avgN * expN  +  new * (1 - expN)	*/
X		/*						*/
X		/*   However, the factorized forms below	*/
X		/*   require fewer floating point operations.	*/
X		/************************************************/
X
X		avg1 = ((avg1 - new) * exp1) + new;
X		avg2 = ((avg2 - new) * exp2) + new;
X		avg3 = ((avg3 - new) * exp3) + new;
X
X		occ = nocc;
X		que = nque;
X
X	} /* for ever */
X} /* doit */
X
Xgetsysinf(s)
Xstruct sysinfo *s;
X{
X	l_lseek(memfd, (long)nlsym[NL_SYSINFO].n_value, 0);
X	r_read(memfd, (char *)s, sizeof(struct sysinfo));
X}
X
X/* lseek with error checking */
Xl_lseek(fd, offset, whence)
Xint fd, whence;
Xlong	offset;
X{
X	if (lseek(fd, offset, whence) == -1) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Error during seek in kernel memory image:\n%s", argv0, sys_errlist[errno]);
X#else
X		fprintf(stderr, "%s: error on lseek\n", argv0);
X#endif UNIXPC
X		exit(1);
X	}
X}
X
X/* read with error checking */
Xr_read (fd, buf, nbytes)
Xint	fd, nbytes;
Xchar	*buf;
X{
X	if (read(fd, buf, nbytes) != nbytes) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Error during read from kernel memory image:\n%s", argv0, sys_errlist[errno]);
X#else
X		fprintf(stderr, "%s: error on read\n", argv0);
X#endif UNIXPC
X		exit(1);
X	}
X}
X
Xinit_packet() {
X#ifdef RWHOD
X	time_t boothz;
X# if UDP
X	struct hostent *he;
X
X	he = gethostbyname( "localnet" );
X	if( he == NULL || he->h_addr == 0 ) {
X		fprintf(stderr, "no address: localnet\n");
X		exit( 1 );
X	}
X	ipaddr = he->h_addr;
X# endif
X# if DSK
X	sprintf(whopacket, "/usr/spool/rwho/whod.%s", utsn.nodename);
X# endif
X	memset(&proto, '\0', sizeof proto);	/* clear proto packet */
X
X	strncat(proto.wd_hostname, utsn.nodename, 9); /* at most 9, add null */
X	proto.wd_vers = WHODVERSION;
X	proto.wd_type = WHODTYPE_STATUS;
X
X	l_lseek(memfd, (long)nlsym[NL_LBOLT].n_value, 0);
X	r_read(memfd, (char *)&boothz, sizeof( boothz ) );
X	proto.wd_boottime = time(0) - (boothz / HZ);
X#else  SHM
X	if ((aven_key = ftok(UNIX, 'a')) == (key_t) -1) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Can't find shared memory keyfile:\n%s: %s", argv0, UNIX, sys_errlist[errno]);
X#else
X		perror(UNIX);
X#endif UNIXPC
X		exit(1);
X	}
X	if ((aven_shm = shmget(aven_key, 3 * sizeof (double), IPC_CREAT|IPC_EXCL|0644)) < 0) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Can't create shared memory segment:\n%s", argv0, sys_errlist[errno]);
X#else
X		perror("shmget");
X#endif UNIXPC
X		exit(1);
X	}
X	if ((int) (aven_seg = (double *) shmat(aven_shm, (char *) 0, 0)) == -1) {
X#ifdef UNIXPC
X		eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Can't attach shared memory segment:\n%s", argv0, sys_errlist[errno]);
X#else
X		perror("shmat");
X#endif UNIXPC
X		if (shmctl(aven_shm, IPC_RMID, (struct shmid_ds *) 0) < 0)
X#ifdef UNIXPC
X			eprintf(ST_OTHER, ST_DISPLAY, (char *) 0, "%s: Can't remove shared memory segment:\n%s", argv0, sys_errlist[errno]);
X#else
X			perror("shmctl(IPC_RMID)");
X#endif UNIXPC
X		exit(1);
X	}
X#endif RWHOD
X	
X} /* init_packet */
X
Xmake_packet(iavg1, iavg2, iavg3)
X#ifdef RWHOD
Xlong iavg1, iavg2, iavg3;
X#else  SHM
Xdouble iavg1, iavg2, iavg3;
X#endif RWHOD
X{
X#ifdef RWHOD
X	static struct whod packet;	/* local packet copy */
X	register struct whoent *wep;	/* pointer to packet whoent */
X	register struct utmp *utp;	/* return from getutent */
X	int whof, cc;			/* output file, char count */
X
X	packet = proto;			/* copy proto packet */
X	time(&packet.wd_sendtime);
X	time(&packet.wd_recvtime);	/* forge this !! */
X	packet.wd_loadav[0] = iavg1;
X	packet.wd_loadav[1] = iavg2;
X	packet.wd_loadav[2] = iavg3;
X
X	setutent();			/* open utmp file */
X	wep = &packet.wd_we[0];		/* get pointer to first user in pkt */
X
X	while( (utp = getutent()) != NULL ) {
X	    if( (utp->ut_type == USER_PROCESS) && utp->ut_user[0]) {
X		strncpy(wep->we_utmp.out_line, utp->ut_id, 4);
X		wep->we_utmp.out_line[4] = '\0';
X
X		strncpy(wep->we_utmp.out_name, utp->ut_user, 8);
X
X		wep->we_utmp.out_time = utp->ut_time;
X
X		wep->we_idle = idletime(utp);
X		wep++;			/* bump packet pointer */
X	    } /* user process */
X	} /* while */
X	endutent();
X
X# if DSK
X	whof = creat(whopacket, 0644);	/* open packt file */
X	if( whof >= 0 ) {
X	    cc = (char *)wep - (char *)&packet;
X	    if( write(whof, (char *)&packet, cc) != cc )
X	    	perror("write failed");
X	    close(whof);
X	} /* file opened */
X	else perror(whopacket);
X# endif
X# if UDP
X	cc = (char *)wep - (char *)&packet;
X	udpsend( (char *)&packet, cc, ipaddr, port, port, 1);
X# endif
X# ifdef DEBUG
X	fprintf(stderr, "wrote packet (%d)\n", cc);
X	fflush(stderr);
X# endif
X#else  SHM
X	aven_seg[0] = iavg1;
X	aven_seg[1] = iavg2;
X	aven_seg[2] = iavg3;
X#ifdef UNIXPC
X	printf("\rLoad average: %4.2f %4.2f %4.2f", iavg1, iavg2, iavg3);
X	fflush(stdout);
X#endif UNIXPC
X#endif RWHOD
X} /* make_packet */
X
X#ifdef RWHOD
Xidletime(up)
Xstruct utmp *up;
X{
X    register int i;
X    register char *cp, *dp;
X    char ttyname[10];
X    struct stat buf;
X    time_t now;
X
X    cp = "/dev/";
X    dp = ttyname;
X    
X    while( *cp != '\0' )		/* copy "/dev/" */
X        *dp++ = *cp++;
X
X    cp = up->ut_line;			/* get line name */
X    if( *cp == 's' )			/* starts with an 's'? (sxtnnn) */
X        *dp++ = 'v';			/* insert a 'v' */
X
X    for( i = 0; i < 8; i++ )		/* copy line name */
X        if( (*dp++ = *cp++) == '\0' ) break;	/* or until null */
X
X    if( stat(ttyname, &buf) != 0 )	/* get file status */
X        return( 0 );
X
X    time(&now);				/* get current time */
X    i = now - buf.st_atime;		/* get differnce from last acces */
X    return( i );			/* return idle time */
X} /* idletime */
X#endif RWHOD
END_OF_loadavgd.c
if test 10503 -ne `wc -c <loadavgd.c`; then
    echo shar: \"loadavgd.c\" unpacked with wrong size!
fi
# end of overwriting check
fi
echo shar: End of shell archive.
exit 0
-- 
| Lenny Tropiano            ICUS Software Systems      [w] +1 (516) 589-7930 |
| lenny@icus.islp.ny.us     Telex; 154232428 ICUS      [h] +1 (516) 968-8576 |
| {ames,pacbell,decuac,hombre,sbcs,attctc}!icus!lenny     attmail!icus!lenny |
+------- ICUS Software Systems -- PO Box 1;  Islip Terrace, NY  11752 -------+

brant@manta.pha.pa.us (Brant Cheikes) (10/17/89)

Lenny Tropiano writes:
>This version has the loadavgd (daemon) to report load avg results.  Now
>if someone could EXPLAIN to me the significance of the three numbers,
>other than the larger the number the more loaded the machine.

First, thanks to Lenny & the other folks who contributed to sysinfo.
It's a very handy little utility (BTW, I've replaced all previous
versions of sysinfo on osu-cis with the new version, and included
binaries to boot).

As far as the explanation goes, if the numbers have the same meaning
as in BSD, then they reflect the average over the last minute, 5
minutes, and 15 minutes, of processes eligible to run (in queue).
The one-minute figure gives you the strongest sense of dynamic system
load.  When the number climbs, more processes are contending for the
CPU, and thus turnaround time increases (so the system seems to get
slower).
-- 
Brant Cheikes
University of Pennsylvania, Department of Computer and Information Science
brant@manta.pha.pa.us, brant@linc.cis.upenn.edu, bpa!manta!brant