+/*
+ * Copyright (c) 1987 Regents of the University of California.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms are permitted
+ * provided that the above copyright notice and this paragraph are
+ * duplicated in all such forms and that any documentation,
+ * advertising materials, and other materials related to such
+ * distribution and use acknowledge that the software was developed
+ * by the University of California, Berkeley. The name of the
+ * University may not be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
+ * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
+ */
+
#ifndef lint
-static char *sccsid = "@(#)last.c 4.8 (Berkeley) %G%";
-#endif
+char copyright[] =
+"@(#) Copyright (c) 1987 Regents of the University of California.\n\
+ All rights reserved.\n";
+#endif /* not lint */
+
+#ifndef lint
+static char sccsid[] = "@(#)last.c 5.13 (Berkeley) %G%";
+#endif /* not lint */
/*
* last
*/
-#include <sys/types.h>
-#include <stdio.h>
-#include <signal.h>
+#include <sys/param.h>
#include <sys/stat.h>
+#include <sys/file.h>
+#include <signal.h>
+#include <time.h>
#include <utmp.h>
+#include <stdio.h>
-#define NMAX sizeof(buf[0].ut_name)
-#define LMAX sizeof(buf[0].ut_line)
-#define HMAX sizeof(buf[0].ut_host)
-#define SECDAY (24*60*60)
-
-#define lineq(a,b) (!strncmp(a,b,LMAX))
-#define nameq(a,b) (!strncmp(a,b,NMAX))
-#define hosteq(a,b) (!strncmp(a,b,HMAX))
+#define SECDAY (24*60*60) /* seconds in a day */
+#define NO 0 /* false/no */
+#define YES 1 /* true/yes */
-#define MAXTTYS 256
+static struct utmp buf[1024]; /* utmp read buffer */
-char **argv;
-int argc;
-int nameargs;
+typedef struct arg {
+ char *name; /* argument */
+#define HOST_TYPE -2
+#define TTY_TYPE -3
+#define USER_TYPE -4
+ int type; /* type of arg */
+ struct arg *next; /* linked list pointer */
+} ARG;
+ARG *arglist; /* head of linked list */
-struct utmp buf[128];
-char ttnames[MAXTTYS][LMAX+1];
-long logouts[MAXTTYS];
+typedef struct ttytab {
+ long logout; /* log out time */
+ char tty[UT_LINESIZE + 1]; /* terminal name */
+ struct ttytab *next; /* linked list pointer */
+} TTY;
+TTY *ttylist; /* head of linked list */
-char *ctime(), *strspl();
-int onintr();
+static long currentout, /* current logout value */
+ maxrec; /* records to display */
+static char *file = "/usr/adm/wtmp"; /* wtmp file */
-main(ac, av)
- char **av;
+main(argc, argv)
+ int argc;
+ char **argv;
{
- register int i, k;
- int bl, wtmp;
- char *ct;
- register struct utmp *bp;
- long otime;
- struct stat stb;
- int print;
- char * crmsg = (char *)0;
- long crtime;
- long outrec = 0;
- long maxrec = 0x7fffffffL;
-
- time(&buf[0].ut_time);
- ac--, av++;
- nameargs = argc = ac;
- argv = av;
- for (i = 0; i < argc; i++) {
- if (argv[i][0] == '-' &&
- argv[i][1] >= '0' && argv[i][1] <= '9') {
- maxrec = atoi(argv[i]+1);
- nameargs--;
- continue;
+ extern int optind;
+ extern char *optarg;
+ int ch;
+ long atol();
+ char *ttyconv();
+
+ while ((ch = getopt(argc, argv, "0123456789f:h:t:")) != EOF)
+ switch((char)ch) {
+ case '0': case '1': case '2': case '3': case '4':
+ case '5': case '6': case '7': case '8': case '9':
+ /*
+ * kludge: last was originally designed to take
+ * a number after a dash.
+ */
+ if (!maxrec)
+ maxrec = atol(argv[optind - 1] + 1);
+ break;
+ case 'f':
+ file = optarg;
+ break;
+ case 'h':
+ hostconv(optarg);
+ addarg(HOST_TYPE, optarg);
+ break;
+ case 't':
+ addarg(TTY_TYPE, ttyconv(optarg));
+ break;
+ case '?':
+ default:
+ fputs("usage: last [-#] [-f file] [-t tty] [-h hostname] [user ...]\n", stderr);
+ exit(1);
}
- if (strlen(argv[i])>2)
- continue;
- if (!strcmp(argv[i], "~"))
- continue;
- if (!strcmp(argv[i], "ftp"))
- continue;
- if (getpwnam(argv[i]))
- continue;
- argv[i] = strspl("tty", argv[i]);
+ for (argv += optind; *argv; ++argv) {
+#define COMPATIBILITY
+#ifdef COMPATIBILITY
+ /* code to allow "last p5" to work */
+ addarg(TTY_TYPE, ttyconv(*argv));
+#endif
+ addarg(USER_TYPE, *argv);
}
- wtmp = open("/usr/adm/wtmp", 0);
- if (wtmp < 0) {
- perror("/usr/adm/wtmp");
+ wtmp();
+ exit(0);
+}
+
+/*
+ * wtmp --
+ * read through the wtmp file
+ */
+static
+wtmp()
+{
+ register struct utmp *bp; /* current structure */
+ register TTY *T; /* tty list entry */
+ struct stat stb; /* stat of file for size */
+ long bl, delta, /* time difference */
+ lseek(), time();
+ int bytes, wfd,
+ onintr();
+ char *ct, *crmsg,
+ *asctime(), *ctime(), *strcpy();
+ TTY *addtty();
+
+ if ((wfd = open(file, O_RDONLY, 0)) < 0 || fstat(wfd, &stb) == -1) {
+ perror(file);
exit(1);
}
- fstat(wtmp, &stb);
- bl = (stb.st_size + sizeof (buf)-1) / sizeof (buf);
- if (signal(SIGINT, SIG_IGN) != SIG_IGN) {
- signal(SIGINT, onintr);
- signal(SIGQUIT, onintr);
- }
- for (bl--; bl >= 0; bl--) {
- lseek(wtmp, bl * sizeof (buf), 0);
- bp = &buf[read(wtmp, buf, sizeof (buf)) / sizeof(buf[0]) - 1];
- for ( ; bp >= buf; bp--) {
- print = want(bp);
- if (print) {
- ct = ctime(&bp->ut_time);
- printf("%-*.*s %-*.*s %-*.*s %10.10s %5.5s ",
- NMAX, NMAX, bp->ut_name,
- LMAX, LMAX, bp->ut_line,
- HMAX, HMAX, bp->ut_host,
- ct, 11+ct);
+ bl = (stb.st_size + sizeof(buf) - 1) / sizeof(buf);
+
+ (void)time(&buf[0].ut_time);
+ (void)signal(SIGINT, onintr);
+ (void)signal(SIGQUIT, onintr);
+
+ while (--bl >= 0) {
+ if (lseek(wfd, (long)(bl * sizeof(buf)), L_SET) == -1 ||
+ (bytes = read(wfd, (char *)buf, sizeof(buf))) == -1) {
+ fprintf(stderr, "last: %s: ", file);
+ perror((char *)NULL);
+ exit(1);
+ }
+ for (bp = &buf[bytes / sizeof(buf[0]) - 1]; bp >= buf; --bp) {
+ /*
+ * if the terminal line is '~', the machine stopped.
+ * see utmp(5) for more info.
+ */
+ if (bp->ut_line[0] == '~' && !bp->ut_line[1]) {
+ /* everybody just logged out */
+ for (T = ttylist; T; T = T->next)
+ T->logout = -bp->ut_time;
+ currentout = -bp->ut_time;
+ crmsg = strncmp(bp->ut_name, "shutdown",
+ UT_NAMESIZE) ? "crash" : "shutdown";
+ if (want(bp, NO)) {
+ ct = ctime(&bp->ut_time);
+ printf("%-*.*s %-*.*s %-*.*s %10.10s %5.5s \n", UT_NAMESIZE, UT_NAMESIZE, bp->ut_name, UT_LINESIZE, UT_LINESIZE, bp->ut_line, UT_HOSTSIZE, UT_HOSTSIZE, bp->ut_host, ct, ct + 11);
+ if (maxrec && !--maxrec)
+ return;
+ }
+ continue;
}
- for (i = 0; i < MAXTTYS; i++) {
- if (ttnames[i][0] == 0) {
- strncpy(ttnames[i], bp->ut_line,
- sizeof(bp->ut_line));
- otime = logouts[i];
- logouts[i] = bp->ut_time;
- break;
+ /*
+ * if the line is '{' or '|', date got set; see
+ * utmp(5) for more info.
+ */
+ if ((bp->ut_line[0] == '{' || bp->ut_line[0] == '|')
+ && !bp->ut_line[1]) {
+ if (want(bp, NO)) {
+ ct = ctime(&bp->ut_time);
+ printf("%-*.*s %-*.*s %-*.*s %10.10s %5.5s \n", UT_NAMESIZE, UT_NAMESIZE, bp->ut_name, UT_LINESIZE, UT_LINESIZE, bp->ut_line, UT_HOSTSIZE, UT_HOSTSIZE, bp->ut_host, ct, ct + 11);
+ if (maxrec && !--maxrec)
+ return;
}
- if (lineq(ttnames[i], bp->ut_line)) {
- otime = logouts[i];
- logouts[i] = bp->ut_time;
+ continue;
+ }
+ /* find associated tty */
+ for (T = ttylist;; T = T->next) {
+ if (!T) {
+ /* add new one */
+ T = addtty(bp->ut_line);
break;
}
+ if (!strncmp(T->tty, bp->ut_line, UT_LINESIZE))
+ break;
}
- if (print) {
- if (lineq(bp->ut_line, "~"))
- printf("\n");
- else if (otime == 0)
- printf(" still logged in\n");
+ if (bp->ut_name[0] && want(bp, YES)) {
+ ct = ctime(&bp->ut_time);
+ printf("%-*.*s %-*.*s %-*.*s %10.10s %5.5s ", UT_NAMESIZE, UT_NAMESIZE, bp->ut_name, UT_LINESIZE, UT_LINESIZE, bp->ut_line, UT_HOSTSIZE, UT_HOSTSIZE, bp->ut_host, ct, ct + 11);
+ if (!T->logout)
+ puts(" still logged in");
else {
- long delta;
- if (otime < 0) {
- otime = -otime;
+ if (T->logout < 0) {
+ T->logout = -T->logout;
printf("- %s", crmsg);
- } else
- printf("- %5.5s",
- ctime(&otime)+11);
- delta = otime - bp->ut_time;
+ }
+ else
+ printf("- %5.5s", ctime(&T->logout)+11);
+ delta = T->logout - bp->ut_time;
if (delta < SECDAY)
- printf(" (%5.5s)\n",
- asctime(gmtime(&delta))+11);
+ printf(" (%5.5s)\n", asctime(gmtime(&delta))+11);
else
- printf(" (%ld+%5.5s)\n",
- delta / SECDAY,
- asctime(gmtime(&delta))+11);
+ printf(" (%ld+%5.5s)\n", delta / SECDAY, asctime(gmtime(&delta))+11);
}
- fflush(stdout);
- if (++outrec >= maxrec)
- exit(0);
- }
- if (lineq(bp->ut_line, "~")) {
- for (i = 0; i < MAXTTYS; i++)
- logouts[i] = -bp->ut_time;
- if (nameq(bp->ut_name, "shutdown"))
- crmsg = "down ";
- else
- crmsg = "crash";
+ if (maxrec != -1 && !--maxrec)
+ return;
}
+ T->logout = bp->ut_time;
}
}
ct = ctime(&buf[0].ut_time);
printf("\nwtmp begins %10.10s %5.5s \n", ct, ct + 11);
- exit(0);
}
-onintr(signo)
- int signo;
+/*
+ * want --
+ * see if want this entry
+ */
+static
+want(bp, check)
+ register struct utmp *bp;
+ int check;
{
- char *ct;
+ register ARG *step;
- if (signo == SIGQUIT)
- signal(SIGQUIT, onintr);
- ct = ctime(&buf[0].ut_time);
- printf("\ninterrupted %10.10s %5.5s \n", ct, ct + 11);
- if (signo == SIGINT)
+ if (check)
+ /*
+ * when uucp and ftp log in over a network, the entry in
+ * the utmp file is the name plus their process id. See
+ * etc/ftpd.c and usr.bin/uucp/uucpd.c for more information.
+ */
+ if (!strncmp(bp->ut_line, "ftp", sizeof("ftp") - 1))
+ bp->ut_line[3] = '\0';
+ else if (!strncmp(bp->ut_line, "uucp", sizeof("uucp") - 1))
+ bp->ut_line[4] = '\0';
+ if (!arglist)
+ return(YES);
+
+ for (step = arglist; step; step = step->next)
+ switch(step->type) {
+ case HOST_TYPE:
+ if (!strncasecmp(step->name, bp->ut_host, UT_HOSTSIZE))
+ return(YES);
+ break;
+ case TTY_TYPE:
+ if (!strncmp(step->name, bp->ut_line, UT_LINESIZE))
+ return(YES);
+ break;
+ case USER_TYPE:
+ if (!strncmp(step->name, bp->ut_name, UT_NAMESIZE))
+ return(YES);
+ break;
+ }
+ return(NO);
+}
+
+/*
+ * addarg --
+ * add an entry to a linked list of arguments
+ */
+static
+addarg(type, arg)
+ int type;
+ char *arg;
+{
+ register ARG *cur;
+ char *malloc();
+
+ if (!(cur = (ARG *)malloc((u_int)sizeof(ARG)))) {
+ fputs("last: malloc failure.\n", stderr);
exit(1);
+ }
+ cur->next = arglist;
+ cur->type = type;
+ cur->name = arg;
+ arglist = cur;
+}
+
+/*
+ * addtty --
+ * add an entry to a linked list of ttys
+ */
+static TTY *
+addtty(ttyname)
+ char *ttyname;
+{
+ register TTY *cur;
+ char *malloc();
+
+ if (!(cur = (TTY *)malloc((u_int)sizeof(TTY)))) {
+ fputs("last: malloc failure.\n", stderr);
+ exit(1);
+ }
+ cur->next = ttylist;
+ cur->logout = currentout;
+ bcopy(ttyname, cur->tty, UT_LINESIZE);
+ return(ttylist = cur);
+}
+
+/*
+ * hostconv --
+ * convert the hostname to search pattern; if the supplied host name
+ * has a domain attached that is the same as the current domain, rip
+ * off the domain suffix since that's what login(1) does.
+ */
+static
+hostconv(arg)
+ char *arg;
+{
+ static int first = 1;
+ static char *hostdot,
+ name[MAXHOSTNAMELEN];
+ char *argdot, *index();
+
+ if (!(argdot = index(arg, '.')))
+ return;
+ if (first) {
+ first = 0;
+ if (gethostname(name, sizeof(name))) {
+ perror("last: gethostname");
+ exit(1);
+ }
+ hostdot = index(name, '.');
+ }
+ if (hostdot && !strcasecmp(hostdot, argdot))
+ *argdot = '\0';
}
-want(bp)
- struct utmp *bp;
+/*
+ * ttyconv --
+ * convert tty to correct name.
+ */
+static char *
+ttyconv(arg)
+ char *arg;
{
- register char **av;
- register int ac;
-
- if (bp->ut_line[0] == '~' && bp->ut_name[0] == '\0')
- strcpy(bp->ut_name, "reboot"); /* bandaid */
- if (strncmp(bp->ut_line, "ftp", 3) == 0)
- bp->ut_line[3] = '\0';
- if (bp->ut_name[0] == 0)
- return (0);
- if (nameargs == 0)
- return (1);
- av = argv;
- for (ac = 0; ac < argc; ac++, av++) {
- if (av[0][0] == '-')
- continue;
- if (nameq(*av, bp->ut_name) || lineq(*av, bp->ut_line))
- return (1);
+ char *mval,
+ *malloc(), *strcpy();
+
+ /*
+ * kludge -- we assume that all tty's end with
+ * a two character suffix.
+ */
+ if (strlen(arg) == 2) {
+ /* either 6 for "ttyxx" or 8 for "console" */
+ if (!(mval = malloc((u_int)8))) {
+ fputs("last: malloc failure.\n", stderr);
+ exit(1);
+ }
+ if (!strcmp(arg, "co"))
+ (void)strcpy(mval, "console");
+ else {
+ (void)strcpy(mval, "tty");
+ (void)strcpy(mval + 3, arg);
+ }
+ return(mval);
}
- return (0);
+ if (!strncmp(arg, "/dev/", sizeof("/dev/") - 1))
+ return(arg + 5);
+ return(arg);
}
-char *
-strspl(left, right)
- char *left, *right;
+/*
+ * onintr --
+ * on interrupt, we inform the user how far we've gotten
+ */
+static
+onintr(signo)
+ int signo;
{
- char *res = (char *)malloc(strlen(left)+strlen(right)+1);
+ char *ct, *ctime();
- strcpy(res, left);
- strcat(res, right);
- return (res);
+ ct = ctime(&buf[0].ut_time);
+ printf("\ninterrupted %10.10s %5.5s \n", ct, ct + 11);
+ if (signo == SIGINT)
+ exit(1);
+ (void)fflush(stdout); /* fix required for rsh */
}