[mod.sources] ttyuse

sources-request@panda.UUCP (10/04/85)

Mod.sources:  Volume 3, Issue 21
Submitted by: harvard!seismo!elsie!ado


: Run this file through sh, not csh.
: It contains a manual page and source code for 'ttyuse';
: you will also need some implementation of 'getopt' on your system.
: To date, 'ttyuse' has only been used on 4.1bsd and 4.2bsd systems.
sed 's/^X//' << 'EOF' > ttyuse.n
X.TH TTYUSE 1
X.SH NAME
Xttyuse \- show terminal usage
X.SH SYNOPSIS
X.B ttyuse
X[
X.B \-l
X] [
X.B \-R
X] [
X.B \-w
Xwtmp ]
X.I time
X.IR "user " ...
X.IR "/dev/* " ...
X.br
X.BR "ttyuse =" "     (to get a `how to use it' message)"
X.SH DESCRIPTION
X.I Ttyuse
Xwrites (to the standard output) a summary of daily terminal usage.
XIf one or more
X.IR user s
Xare named on the command line,
Xthe report is limited to those users.
XIf one or more device names are given,
Xthe report is limited to those devices.
X(All--and only--command line arguments that begin with
X.RB ` /dev/ '
Xare taken to be device names.)
XAnd if a
X.I time
Xargument is given, the report is limited to given date(s); the
Xtime argument may take these forms:
X.nf
X.in +.5i
X.ta \w'm/d-m/d  'u
Xm/d	a month and day (example:  5/15);
Xm/d-d	a month, starting day, and stopping day;
Xm/d-m/d	starting and stopping months and days;
Xm/d-	starting month and day
X	(the current day is used as the ending day)
X.fi
X.in -.5i
XIn the absence of a
X.I time
Xargument,
Xoutput for the current day is produced.
X.PP
XThe leftmost of the 79 columns of output look like this:
X.in +.5i
X.nf
XMon Sep 23 1985
XTT USE 12-1--2--3--4--5--6--7--8--~--10-11-12-1--
X01 5.0                       Snorri              
X02 8.7                         Mark  Mark::::::::
X03 7.7                           MRoot::::::::Mar
X05 0.8                         Root              
X07 12. ado                        ~  Ado:::::::::
X09 0.4                            ~   Ma         
X13 3.9 UU 2U 2U UU22U UU UUUUU UU U2 U UUucUucp::
X  39.3 <   2><   1><   1><   2><   5><   5><   4>
X.in -.5i
X.fi
XThe first line gives the date the report is for.
XThe second line provides headings for the lines that follow.
XIn those following lines, the
X.RB ` TT '
Xcolumns give the
X.IR ps (1)-style
Xabbreviation of the terminal the line of output is about,
Xand the
X.RB ` USE '
Xcolumns give the total hours of use for the terminal on the
Xdate the report is for.
XEach other column represents a twenty-minute time period.
XA capital letter in one of these columns means that a
Xuser whose name begins with that letter logged in during
Xthe time period; the user's name continues in lower case into
Xfollowing columns for as long as they remain logged in
X(followed by colons if they stay logged in for a long time).
XIf more than one login occurs during a time period, the
Xnumber of logins appears in the time period's column;
Xan asterisk appears if there were ten or more logins.
X.PP
XThe final line for a date's report shows the total
Xterminal-hours for the day,
Xand the total number of terminals used during each two-hour period.
X.PP
XReboots are shown by a
X.RB ` ~ '
Xcharacter in the relevant column of output;
Xtime changes are shown by a
X.RB ` | '
Xcharacter in the `time-changed-from' column and a
X.RB ` { '
Xcharacter in the `time-changed-to' column.
XThese characters appear in the headings line and in lines
Xof output for terminals that were idle during the time
Xperiod when the reboot or time change occurred.
X.PP
XThe
X.RB ` USE '
Xcolumns are absent from reports for days with 25 hours
X(Daylight Saving Time transition days).
X.PP
XThese options are available:
X.TP
X.B \-l
XProduce long lines of output (up to 131 columns) where each column
Xrepresents twelve minutes.  The final line of a date's report
Xshows the total number of terminals used during each one-hour
Xperiod.
X.TP
X.B \-R
XInterpret the entire accounting file.
X.TP
X.BI "\-w " wtmp
XUse the file whose name is given by
X.I wtmp
Xas the accounting file,
Xrather than using the default name shown below.
X.SH FILES
X/usr/adm/wtmp
X.SH SEE ALSO
Xps(1)
X.. @(#)ttyuse.n	1.1
EOF
sed 's/^X//' << 'EOF' > ttyuse.c
X/*
X** Still to handle:  do days with no logins or logouts but tty use!
X*/
X
X#include "stdio.h"
X#include "ctype.h"
X#include "utmp.h"
X#include "time.h"
X
X#ifdef OBJECTID
Xstatic char	sccsid[] = "@(#)ttyuse.c	1.1";
X#endif
X
X#ifdef USG
X/* Uncompatible Software Glitch */
X#define index	strchr
X#endif
X
X#ifndef TRUE
X#define TRUE	(1)
X#define FALSE	(0)
X#endif
X
X#ifndef arg4alloc
X#define arg4alloc	unsigned
X#endif
X
Xextern char *		calloc();
Xextern char *		ctime();
Xextern char *		index();
Xextern struct tm *	localtime();
Xextern long		lseek();
Xextern char *		realloc();
Xextern char *		sprintf();
Xextern char *		strcat();
Xextern char *		strcpy();
Xextern long		time();
X
Xstatic long		midnight();
Xstatic long		mdy2t();
X
X#define MAXDAYHOURS	25
X#define HOURSLOTS	3
X#define LONGHOURSLOTS	5
X
X#define MINUTE		60
X#define HOUR		(60*MINUTE)
X
Xstruct entry {
X	struct utmp	u;
X	int		marked;
X	int		daysecs;
X	char		l[MAXDAYHOURS * LONGHOURSLOTS + 1];
X};
X
Xstatic struct entry *	entries;
Xstatic struct entry *	ebeyond;
X
Xstatic int		dayhours;
Xstatic int		dayslots;
Xstatic int		hourslots;
X
X#define NAMESIZE	(sizeof entries[0].u.ut_name)
X#define LINESIZE	(sizeof entries[0].u.ut_line)
X#define namecmp(name1, name2) strncmp(name1, name2, NAMESIZE)
X#define linecmp(line1, line2) strncmp(line1, line2, LINESIZE)
X#define namecpy(to, from) strncpy(to, from, NAMESIZE)
X#define linecpy(to, from) strncpy(to, from, LINESIZE)
X
X#define BOOTCHAR	'~'
X#define OLDTCHAR	'|'
X/*
X** 4.[12]bsd writes '{' to the file, but the documentation says that '}' is
X** the character.  We'll take either.
X**
X** System V does things differently, of course.
X*/
X#define NEWTCHAR	'{'
X#define OTHTCHAR	'}'
X
X#define BOOTLINE(s) (*(s) == BOOTCHAR && (s)[1] == '\0' || \
X			strcmp((s), "reboot") == 0)
X#define OLDTLINE(s) (*(s) == OLDTCHAR && (s)[1] == '\0' || \
X			strncmp((s), "old time", 8) == 0)
X#define NEWTLINE(s) (*(s) == NEWTCHAR && (s)[1] == '\0' || \
X			*(s) == OTHTCHAR && (s)[1] == '\0' || \
X			strncmp((s), "new time", 8) == 0)
X#define TIMELINE(s) (OLDTLINE(s) || NEWTLINE(s))
X#define SPCLLINE(s) (TIMELINE(s) || BOOTLINE(s))
X
Xstatic struct entry *
Xlookup(up, doenter)
Xregister struct utmp *	up;
X{
X	register struct entry *	ep;
X	register char *		cp;
X	register int		i;
X
X	cp = up->ut_line;
X	if (SPCLLINE(cp))
X		return NULL;
X	for (ep = entries; ep < ebeyond; ++ep)
X		if (linecmp(cp, ep->u.ut_line) == 0)
X			return ep;
X	if (!doenter)
X		return NULL;
X	i = ebeyond - entries;
X	if (i == 0)
X		ep = (struct entry *) calloc(1, sizeof *ep);
X	else	ep = (struct entry *) realloc((char *) entries,
X			(arg4alloc) ((i + 1) * sizeof *ep));
X	if (ep == NULL)
X		for ( ; ; )
X			wildrexit("alloc");
X	entries = ep;
X	ebeyond = ep + i + 1;
X	ep = ebeyond - 1;
X	(void) linecpy(ep->u.ut_line, up->ut_line);
X	(void) sprintf(ep->l, "%*s", dayslots, "");
X	ep->u.ut_time = 0;
X	ep->u.ut_name[0] = '\0';
X	ep->marked = FALSE;
X	ep->daysecs = 0;
X	return ep;
X}
X
Xstatic char		headline[MAXDAYHOURS * LONGHOURSLOTS + 1];
X
Xstatic int		lflag;	/* long lines */
Xstatic char *		wname = "/usr/adm/wtmp";
X
Xstatic char *		name;
X
Xstatic char **		argusers;
Xstatic int		cntusers;
X
Xstatic long		starttime;
Xstatic long		stoptime;
X
Xstatic
Xdodate(string)
Xregister char *	string;
X{
X	struct tm	local;
X	int		bmon, bday, byear, emon, eday, eyear, i;
X	long		now;
X	char		c;
X
X	(void) time(&now);
X	local = *localtime(&now);
X	c = '\0';
X	i = sscanf(string, "%d/%d-%d/%d%c", &bmon, &bday, &emon, &eday, &c);
X	if (i != 4 || c != '\0') {
X		c = '\0';
X		i = sscanf(string, "%d/%d-%d%c", &bmon, &bday, &eday, &c);
X		if (i == 3 && c == '\0') {
X			emon = bmon;
X			if (bday > eday)
X				for ( ; ; )
X					wildexit("date");
X		} else {
X			c = '\0';
X			i = sscanf(string, "%d/%d%c", &bmon, &bday, &c);
X			if (i == 2 && c == '\0') {
X				emon = bmon;
X				eday = bday;
X			} else {
X				c = '\0';
X				i = sscanf(string, "%d/%d-%c",
X					&bmon, &bday, &c);
X				if (i == 2 && c == '\0') {
X					emon = local.tm_mon + 1;
X					eday = local.tm_mday;
X				} else for ( ; ; )
X					wildexit("date");
X			}
X		}
X	}
X	byear = eyear = local.tm_year + 1900;
X	if (bmon > emon || bmon == emon && bday > eday)
X		--byear;
X	starttime = mdy2t(bmon, bday, byear);
X	stoptime = mdy2t(emon, eday, eyear);
X}
X
Xstatic char *
Xttyabbr(string)
Xregister char *	string;
X{
X	static char	buf[3];
X
X	if (strncmp(string, "tty", 3) == 0)
X		(void) strncpy(buf, string + 3, 2);
X	else	(void) strncpy(buf, string, 2);
X	buf[2] = '\0';
X	return buf;
X}
X
Xstatic struct utmp *	utmp;
Xstatic struct utmp *	ubeyond;
Xstatic struct utmp * 	unext;
Xstatic struct utmp *	ubase;
X
Xstatic int
Xqtime(up1, up2)
Xchar *	up1;
Xchar *	up2;
X{
X	register long	diff;
X
X	diff = ((struct utmp *) up1)->ut_time - ((struct utmp *) up2)->ut_time;
X	if (diff == 0)
X		return 0;
X	else if (diff > 0)
X		return 1;
X	else	return -1;
X}
X
Xmain(argc, argv)
Xint	argc;
Xchar *	argv[];
X{
X	register FILE *		fp;
X	register struct entry *	ep;
X	register int		i;
X	register long		t;
X	register long		lastm, m;
X	register int		slot;
X	register int		didttys;
X	register long		filesize;
X	static int		Rflag;
X	static int		diddate;
X	char			buf[134];
X	extern int		optind;
X	extern char *		optarg;
X
X	for (name = argv[0]; *name != '\0'; ++name)
X		if (name[0] == '/' && name[1] != '\0' && name[1] != '/')
X			argv[0] = ++name;
X	name = argv[0];
X	if (argc == 2 && strcmp(argv[1], "=") == 0) {
X		(void) printf(
X"%s: usage is %s [-lR] [-w wtmp] [m/d[-[[m/]d]]] [user...] [/dev/*...]\n",
X			name, name);
X		for ( ; ; )
X			exit(1);
X	}
X	while ((i = getopt(argc, argv, "lRw:")) != EOF)
X		switch (i) {
X			case 'l':
X				lflag = TRUE;
X				break;
X			case 'R':
X				Rflag = TRUE;
X				starttime = 0;
X				(void) time(&stoptime);
X				break;
X			case 'w':
X				wname = optarg;
X				break;
X			default:
X				for ( ; ; )
X					wildexit("usage");
X		}
X	hourslots = lflag ? LONGHOURSLOTS : HOURSLOTS;
X	/*
X	** Parse arguments
X	*/
X	i = (argc - optind) + 1;
X	argusers = (char **) calloc((arg4alloc) i, sizeof *argusers);
X	if (argusers == NULL)
X		for ( ; ; )
X			wildrexit("calloc");
X	cntusers = 0;
X	for (i = optind; i < argc; ++i)
X		if (strncmp(argv[i], "/dev/", 5) == 0) {
X			struct utmp	fake;
X
X			(void) linecpy(fake.ut_line, &argv[i][5]);
X			(void) namecpy(fake.ut_name, "fake");
X			if (lookup(&fake, FALSE) != NULL)
X				for ( ; ; )
X					wildrexit("duplicate /dev argument");
X			if (lookup(&fake, TRUE) == NULL) /* "cannot happen */
X				for ( ; ; )
X					wildrexit("lookup");
X		} else if (index(argv[i], '/') == 0)
X			argusers[cntusers++] = argv[i];
X		else if (Rflag)
X			for ( ; ; )
X				wildexit("combination of date and -R");
X		else if (diddate)
X			for ( ; ; )
X				wildexit("multiple dates");
X		else {
X			dodate(argv[i]);
X			diddate = TRUE;
X		}
X	argusers[cntusers] = NULL;
X	didttys = entries != NULL;
X	if (!Rflag && !diddate) {
X		(void) time(&starttime);
X		stoptime = starttime;
X	}
X	starttime = midnight(starttime);
X	stoptime = midnight(stoptime);
X	stoptime = midnight(stoptime + MAXDAYHOURS * HOUR);
X	/*
X	** And now the real work begins.
X	*/
X	if ((fp = fopen(wname, "r")) == NULL)
X		for ( ; ; )
X			wildrexit("opening wtmp");
X	(void) fseek(fp, 0L, 2);
X	filesize = ftell(fp);
X	if (filesize == 0)
X		for ( ; ; )
X			tameexit();
X	if (filesize < 0 || (filesize % sizeof *utmp) != 0)
X		for ( ; ; )
X			wildexit("wtmp size");
X	i = filesize / sizeof *utmp;
X	utmp = (struct utmp *) calloc((arg4alloc) i, sizeof *utmp);
X	if (utmp == NULL)
X		for ( ; ; )
X			wildrexit("calloc");
X	ubeyond = utmp + i;
X	/*
X	** Step 1--read it in.
X	*/
X#ifdef fileno
X	(void) lseek(fileno(fp), 0L, 0);
X	i = read(fileno(fp), (char *) utmp, (int) filesize);
X#endif
X#ifndef fileno
X	(void) fseek(fp, 0L, 0);
X	i = fread((char *) utmp, sizeof *utmp, (int) i, fp) * sizeof *utmp);
X#endif
X	if (i != filesize || ferror(fp) || fclose(fp))
X		for ( ; ; )
X			wildrexit("reading wtmp");
X	/*
X	** Step 2--eliminate records up to and including last reboot before
X	** start time.
X	*/
X	for (unext = utmp; unext<ubeyond && unext->ut_time<starttime; ++unext)
X		if (BOOTLINE(unext->ut_line))
X			utmp = unext + 1;
X	/*
X	** Step 3--eliminate trailing records for times past stoptime.
X	*/
X	while ((ubeyond - 1) > utmp && (ubeyond - 1)->ut_time >= stoptime)
X		--ubeyond;
X	/*
X	** Step 4--eliminate bogus records.
X	*/
X	for (unext = ubase = utmp; unext < ubeyond; ++unext)
X		if (unext->ut_time == 0 || unext->ut_line[0] == '\0')
X			continue;
X		else if (ubase == unext)
X			++ubase;
X		else	*ubase++ = *unext;
X	/*
X	** Step 5--eliminate leading logouts, reboots, and time changes.
X	*/
X	while (utmp < ubeyond && SPCLLINE(utmp->ut_line))
X		++utmp;
X	/*
X	** Step 6--eliminate trailing reboots and time changes.
X	*/
X	while ((ubeyond - 1) > utmp && SPCLLINE((ubeyond - 1)->ut_line))
X		--ubeyond;
X	/*
X	** Step 7--if ttys were specified, eliminate records for other ttys.
X	*/
X	if (didttys)
X		for (ubase = unext = utmp; unext < ubeyond; ++unext)
X			if (SPCLLINE(unext->ut_line) ||
X				lookup(unext, FALSE) != NULL)
X					if (ubase == unext)
X						++ubase;
X					else	*ubase++ = *unext;
X	/*
X	** Final step--sort records between time changes.
X	*/
X	for (ubase = unext = utmp; unext < ubeyond; ++unext)
X		if (TIMELINE(unext->ut_line)) {
X			if (unext != ubase)
X				(void) qsort((char *) ubase, unext - ubase,
X					sizeof *ubase, qtime);
X			ubase = unext + 1;
X		}
X	if (ubeyond > ubase)
X		(void) qsort((char *) ubase, ubeyond - ubase,
X			sizeof *ubase, qtime);
X	if (Rflag)
X		unext = utmp;
X	else {
X		/*
X		** Find first record that's for after the start time.
X		*/
X		for (unext = utmp; unext < ubeyond; ++unext)
X			if (unext->ut_time >= starttime)
X				break;
X		if (unext >= ubeyond)	/* easy enough! */
X			for ( ; ; )
X				tameexit();
X		ubase = unext;
X		/*
X		** Scan back to a reboot or to the first record
X		** (or, if we didttys, until all ttys are accounted for).
X		*/
X		for ( ; unext > utmp; --unext) {
X			if (BOOTLINE(unext->ut_line))
X				break;
X			if ((ep = lookup(unext, !didttys)) == NULL)
X				continue;
X			if (ep->marked)
X				continue;
X			ep->marked = TRUE;
X			(void) namecpy(ep->u.ut_name, unext->ut_name);
X			if (didttys) {
X				for (ep = entries; ep < ebeyond; ++ep)
X					if (!ep->marked)
X						break;
X				if (ep >= ebeyond)
X					break;
X			}
X		}
X		/*
X		** If users were specified, zap the records for anybody else.
X		*/
X		if (cntusers > 0)
X			for (ep = entries; ep < ebeyond; ++ep) {
X				for (i = 0; i < cntusers; ++i)
X					if (namecmp(argusers[i],
X						ep->u.ut_name) == 0)
X							break;
X				if (i >= cntusers)
X					ep->u.ut_name[0] = '\0';
X			}
X		/*
X		** We're ready!
X		*/
X		utmp = unext = ubase;
X	}
X	lastm = starttime - 1;
X	for ( ; unext < ubeyond; ++unext, lastm = m) {
X		t = unext->ut_time;
X		m = midnight(t);
X		if (m != lastm) {
X			if (lastm >= starttime && lastm < stoptime) {
X				for (ep = entries; ep < ebeyond; ++ep) {
X					if (ep->u.ut_name[0] == '\0')
X						continue;
X					ep->daysecs += (lastm + dayhours *
X						HOUR) - ep->u.ut_time;
X					if (ep->daysecs == 0)
X						++(ep->daysecs);
X					colonize(ep->l, dayslots - 1);
X				}
X				dump(lastm);
X			}
X			newday(m, 0L, '\0');
X		}
X		slot = (unext->ut_time - m) / (HOUR / hourslots);
X		if (BOOTLINE(unext->ut_line)) {
X			headline[slot] = BOOTCHAR;
X			for (ep = entries; ep < ebeyond; ++ep) {
X				if (ep->u.ut_name[0] != '\0') {
X					ep->daysecs += unext->ut_time -
X						ep->u.ut_time;
X					if (ep->daysecs == 0)
X						++(ep->daysecs);
X					colonize(ep->l, slot);
X					ep->u.ut_name[0] = '\0';
X					ep->u.ut_time = 0;
X				}
X			}
X			continue;
X		}
X		if (NEWTLINE(unext->ut_line))
X			for ( ; ; )
X				wildexit("new time without old time");
X		if (OLDTLINE(unext->ut_line)) {
X			if (unext == ubeyond || !NEWTLINE((unext + 1)->ut_line))
X				for ( ; ; )
X					wildexit("old time without new time");
X			headline[slot] = OLDTCHAR;
X			for (ep = entries; ep < ebeyond; ++ep) {
X				if (ep->u.ut_name[0] == '\0')
X					ep->l[slot] = OLDTCHAR;
X				else {
X					ep->daysecs += unext->ut_time -
X						ep->u.ut_time;
X					if (ep->daysecs == 0)
X						++(ep->daysecs);
X					colonize(ep->l, slot);
X				}
X			}
X			dump(lastm);
X			++unext;
X			if (unext->ut_line[0] == NEWTCHAR ||
X				unext->ut_line[0] == OTHTCHAR)
X					newday(m, unext->ut_time,
X						unext->ut_line[0]);
X			else	newday(m, unext->ut_time, NEWTCHAR);
X			continue;
X		}
X		if ((ep = lookup(unext, !didttys)) == NULL)
X			continue;
X		/*
X		** If a login is in progress, either a login or
X		** logout terminates it.
X		*/
X		if (ep->u.ut_name[0] != '\0') {
X			ep->daysecs += unext->ut_time - ep->u.ut_time;
X			if (ep->daysecs == 0)
X				++(ep->daysecs);
X			colonize(ep->l, slot);
X			ep->u.ut_name[0] = '\0';
X			ep->u.ut_time = 0;
X		}
X		if (unext->ut_name[0] == '\0')
X			continue;
X		if (cntusers > 0) {
X			for (i = 0; i < cntusers; ++i)
X				if (namecmp(argusers[i], unext->ut_name) == 0)
X					break;
X			if (i == cntusers)
X				continue;
X		}
X		ep->u.ut_time = unext->ut_time;
X		if (isupper(ep->l[slot])) {
X			buf[0] = '2';
X			i = 1;
X		} else {
X			register char *	cp;
X
X			cp = index("23456789**", ep->l[slot]);
X			if (cp != 0) {
X				buf[0] = *(cp + 1);
X				i = 1;
X			} else	i = 0;
X		}
X		fudge(unext->ut_name);
X		(void) namecpy(ep->u.ut_name, unext->ut_name);
X		(void) namecpy(&buf[i], unext->ut_name);
X		/*
X		** Ensure termination.
X		*/
X		buf[NAMESIZE + i] = '\0';
X		if (i == 0 && islower(buf[0]))
X			buf[0] = toupper(buf[0]);
X		i = strlen(buf);
X		if (i > (dayslots - slot))
X			i = dayslots - slot;
X		(void) strncpy(&ep->l[slot], buf, i);
X	}
X	for (ep = entries; ep < ebeyond; ++ep) {
X		if (ep->u.ut_name[0] == '\0')
X			continue;
X		ep->daysecs += t - ep->u.ut_time;
X		if (ep->daysecs == 0)
X			++(ep->daysecs);
X		colonize(ep->l, slot);
X	}
X	dump(lastm);
X	for ( ; ; )
X		tameexit();
X}
X
Xstatic int
Xqline(ep1, ep2)
Xchar *	ep1;
Xchar *	ep2;
X{
X	return linecmp(((struct entry *) ep1)->u.ut_line,
X		((struct entry *) ep2)->u.ut_line);
X}
X
Xstatic int
Xttysused(base, count)
Xregister int	base;
Xregister int	count;
X{
X	register struct entry *	ep;
X	register int		i;
X	register int		result;
X
X	result = 0;
X	for (ep = entries; ep < ebeyond; ++ep)
X		for (i = 0; i < count; ++i) {
X			switch (ep->l[base + i]) {
X				case ' ':
X				case BOOTCHAR:
X				case OLDTCHAR:
X				case NEWTCHAR:
X				case OTHTCHAR:
X					continue;
X			}
X			++result;
X			break;
X		}
X	return result;
X}
X
Xstatic
Xdump(m)
Xlong	m;	/* midnight */
X{
X	register struct entry *	ep;
X	register char *		cp;
X	register int		i, j, k, lines;
X	register int		width, prec;
X	register double		d;
X	struct tm		tm;
X	long			grandtot;
X	char			buf[20];
X
X	tm = *localtime(&m);
X	grandtot = 0;
X	for (ep = entries; ep < ebeyond; ++ep)
X		grandtot += ep->daysecs;
X	if (grandtot == 0)
X		return;
X	(void) qsort((char *) entries, ebeyond - entries,
X		sizeof entries[0], qline);
X	lines = 0;
X	width = (lflag ? 131 : 79) - dayslots;
X	prec = (width >= 7) ? width : 3;
X	for (ep = entries; ep < ebeyond; ++ep) {
X		if (ep->daysecs == 0)
X			continue;
X		if (lines == 0) {
X			cp = ctime(&m);
X			(void) printf("%.11s%s", cp, cp + 20);
X			(void) printf("%-*.*s", width, prec, "TT USE");
X			(void) puts(headline);
X		}
X		d = ep->daysecs / (double) HOUR;
X		(void) sprintf(buf, "%2.2s%4.1f", ttyabbr(ep->u.ut_line), d);
X		if (buf[2] != ' ')
X			(void) sprintf(buf, "%2.2s%4.0f",
X				ttyabbr(ep->u.ut_line), d);
X		(void) printf("%-*.*s", width, prec, buf);
X		for (i = 0; i < dayslots; ++i)
X			if (headline[i] == BOOTCHAR && ep->l[i] == ' ')
X				ep->l[i] = BOOTCHAR;
X		cp = ep->l + dayslots;
X		while (cp > ep->l && *(cp - 1) == ' ')
X			--cp;
X		(void) printf("%.*s\n", cp - ep->l, ep->l);
X		++lines;
X	}
X	if (lines <= 1)
X		return;
X	if (prec > 6)
X		(void) sprintf(buf, "%6.1f", grandtot / (double) HOUR);
X	else	buf[0] = '\0';
X	(void) printf("%-*.*s", width, prec, buf);
X	for (i = 0; i < dayslots; i += j) {
X		j = lflag ? hourslots : (2 * hourslots);
X		if (i == 0 && (dayslots % j) != 0)
X			j += dayslots % j;
X		if ((k = ttysused(i, j)) == 0)
X			(void) printf("%*s", j, "");
X		else	(void) printf("<%*d>", j - 2, k);
X	}
X	(void) printf("\n");
X}
X
Xstatic
Xcolonize(line, slot)
Xregister char *	line;
Xregister int	slot;
X{
X	register int	i;
X
X	for (i = slot + 1; i < dayslots && line[slot] != ' '; ++i)
X		line[i] = ' ';
X	while (slot >= 0 && line[slot] == ' ') {
X		line[slot] = ':';
X		--slot;
X	}
X}
X
Xstatic
Xwildrexit(string)
Xchar *	string;
X{
X	(void) fprintf(stderr, "%s: wild result from %s\n", name, string);
X	for ( ; ; )
X		exit(1);
X}
X
Xstatic
Xwildexit(string)
Xchar *	string;
X{
X	(void) fprintf(stderr, "%s: wild %s\n", name, string);
X	for ( ; ; )
X		exit(1);
X}
X
Xstatic
Xtameexit()
X{
X	for ( ; ; )
X		exit(0);
X}
X
Xstatic
Xfudge(buf)
Xregister char *	buf;
X{
X	register int	i;
X
X	for (i = 0; i < NAMESIZE; ++i)
X		switch (buf[i]) {
X			case '\0':
X				return;
X			case BOOTCHAR:
X			case OLDTCHAR:
X			case NEWTCHAR:
X			case OTHTCHAR:
X			case ':':
X			case '*':
X			case ' ':
X			case '2': case '3': case '4': case '5':
X			case '6': case '7': case '8': case '9':
X				buf[i] = '?';
X				break;
X			default:
X				if (!isascii(buf[i]))
X					buf[i] = '?';
X				else if (isupper(buf[i]))
X					buf[i] = tolower(buf[i]);
X				else if (!isprint(buf[i]))
X					buf[i] = '?';
X				break;
X		}
X}
X
Xstatic long
Xmdy2t(m, d, y)
X{
X	register struct tm *	tmp;
X	register int		diff;
X	register int		lastdiff;
X	long			guess;
X
X	if (y < 1969 || m <= 0 || m > 12 || d <= 0 || d >= 32)
X		for ( ; ; )
X			wildrexit("date");
X	guess = d + (m - 1) * (365.25 / 12.) + (y - 1970) * 365.25;
X	guess = guess * HOUR * 24;
X	y -= 1900;
X	--m;
X	lastdiff = 0;
X	for ( ; ; ) {
X		tmp = localtime(&guess);
X		if ((diff = tmp->tm_year - y) == 0 &&
X			(diff = tmp->tm_mon - m) == 0)
X				diff = tmp->tm_mday - d;
X		if (diff == 0)
X			return midnight(guess);
X		if (lastdiff != 0)
X			if ((diff > 0) != (lastdiff > 0))
X				for ( ; ; )
X					wildrexit("date");
X		if (diff > 0)
X			guess -= 12 * HOUR;
X		else	guess += 12 * HOUR;
X		lastdiff = diff;
X	}
X}
X
Xstatic long
Xmidnight(t)
Xlong	t;
X{
X	register int	diff, lastdiff;
X	struct tm 	tm;
X	struct tm	mtm;
X	long		m;
X
X	tm = *localtime(&t);
X	if (tm.tm_hour == 0 && tm.tm_min == 0 && tm.tm_sec == 0)
X		return t;
X	m = t;
X	m -= tm.tm_hour * HOUR;
X	m -= tm.tm_min * MINUTE;
X	m -= tm.tm_sec;
X	lastdiff = 0;
X	for ( ; ; ) {
X		mtm = *localtime(&m);
X		if ((diff = mtm.tm_year - tm.tm_year) == 0 &&
X			(diff = mtm.tm_mon - tm.tm_mon) == 0 &&
X			(diff = mtm.tm_mday - tm.tm_mday) == 0 &&
X			(diff = mtm.tm_hour) == 0)
X				if (mtm.tm_min == 0 && mtm.tm_sec == 0)
X					return m;
X				else	for ( ; ; )
X						wildrexit("finding midnight");
X		if (lastdiff != 0)
X			if ((diff > 0) != (lastdiff > 0))
X				for ( ; ; )
X					wildrexit("finding midnight");
X		if (diff > 0)
X			m -= HOUR;
X		else	m += HOUR;
X		lastdiff = diff;
X	}
X}
X
Xstatic
Xnewday(m, t, headchar)
Xlong	m, t;
X{
X	register struct entry *	ep;
X	register int		i, j, slot;
X	long			t2;
X	char			buf[20];
X
X	dayhours = (midnight(m + MAXDAYHOURS * HOUR) - m) / HOUR;
X	if (dayhours <= 0 || dayhours > MAXDAYHOURS)
X		for ( ; ; )
X			wildexit("number of hours in a day");
X	dayslots = hourslots * dayhours;
X	headline[0] = '\0';
X	for (i = 0; i < dayhours; ++i) {
X		if (dayhours == 24)
X			j = i;
X		else {
X			t2 = m + i * HOUR;
X			j = localtime(&t2)->tm_hour;
X		}
X		if ((j %= 12) == 0)
X			j = 12;
X		if (j == 12 && lflag)
X			if (i == 0)
X				(void) strcpy(buf, "Mid.-");
X			else	(void) strcpy(buf, "Noon-");
X		else	(void) sprintf(buf, "%d-----", j);
X		buf[hourslots] = '\0';
X		(void) strcat(headline, buf);
X	}
X	if (t == 0)
X		slot = 0;
X	else {
X		slot = (t - m) / (HOUR / hourslots);
X		headline[slot] = headchar;
X	}
X	for (ep = entries; ep < ebeyond; ++ep) {
X		(void) sprintf(ep->l, "%*s", dayslots, "");
X		ep->daysecs = 0;
X		if (ep->u.ut_name[0] == '\0') {
X			ep->u.ut_time = 0;
X			if (t != 0)
X				ep->l[slot] = headchar;
X		} else {
X			ep->u.ut_time = (t == 0) ? m : t;
X			for (i = 0; i < NAMESIZE; ++i) {
X				if (ep->u.ut_name[i] == '\0')
X					break;
X				if ((slot + i) >= dayslots)
X					break;
X				ep->l[slot + i] = ep->u.ut_name[i];
X			}
X		}
X	}
X}
EOF
exit