📄 last.c
字号:
/* * Berkeley last for Linux. Currently maintained by poe@daimi.aau.dk at * ftp://ftp.daimi.aau.dk/pub/linux/poe/admutil* * * 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. */ /* 1999-02-22 Arkadiusz Mi秌iewicz <misiek@pld.ORG.PL> * - added Native Language Support */ /* 2001-02-14 Marek Zelem <marek@fornax.sk> * - using mmap() on Linux - great speed improvement *//* * last */#include <sys/param.h>#include <sys/stat.h>#include <sys/file.h>#include <sys/types.h>#include <sys/mman.h>#include <signal.h>#include <string.h>#include <time.h>#include <utmp.h>#include <stdio.h>#include <getopt.h>#include <stdlib.h>#include <unistd.h>#include <sys/socket.h>#include <netinet/in.h>#include <arpa/inet.h>#include "pathnames.h"#include "nls.h"#define SECDAY (24*60*60) /* seconds in a day */#define NO 0 /* false/no */#define YES 1 /* true/yes */static struct utmp utmpbuf;#define HMAX (int)sizeof(utmpbuf.ut_host) /* size of utmp host field */#define LMAX (int)sizeof(utmpbuf.ut_line) /* size of utmp tty field */#define NMAX (int)sizeof(utmpbuf.ut_name) /* size of utmp name field */#ifndef MIN#define MIN(a,b) (((a) < (b)) ? (a) : (b))#endif/* maximum sizes used for printing *//* probably we want a two-pass version that computes the right length */int hmax = MIN(HMAX, 16);int lmax = MIN(LMAX, 8);int nmax = MIN(NMAX, 16);typedef struct arg { char *name; /* argument */#define HOST_TYPE -2#define TTY_TYPE -3#define USER_TYPE -4#define INET_TYPE -5 int type; /* type of arg */ struct arg *next; /* linked list pointer */} ARG;ARG *arglist; /* head of linked list */typedef struct ttytab { long logout; /* log out time */ char tty[LMAX + 1]; /* terminal name */ struct ttytab *next; /* linked list pointer */} TTY;TTY *ttylist; /* head of linked list */static long currentout, /* current logout value */ maxrec; /* records to display */static char *file = _PATH_WTMP; /* wtmp file */static int doyear = 0; /* output year in dates */static int dolong = 0; /* print also ip-addr */static void wtmp(void);static void addarg(int, char *);static void hostconv(char *);static void onintr(int);static int want(struct utmp *, int);TTY *addtty(char *);static char *ttyconv(char *);intmain(int argc, char **argv) { extern int optind; extern char *optarg; int ch; setlocale(LC_ALL, ""); bindtextdomain(PACKAGE, LOCALEDIR); textdomain(PACKAGE); while ((ch = getopt(argc, argv, "0123456789yli:f:h:t:")) != -1) 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 'y': doyear = 1; break; case 'l': dolong = 1; break; case 'i': addarg(INET_TYPE, optarg); break; case '?': default: fputs(_("usage: last [-#] [-f file] [-t tty] [-h hostname] [user ...]\n"), stderr); exit(1); } 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(); exit(0);}/* * print_partial_line -- * print the first part of each output line according to specified format */static voidprint_partial_line(struct utmp *bp) { char *ct; ct = ctime(&bp->ut_time); printf("%-*.*s %-*.*s ", nmax, nmax, bp->ut_name, lmax, lmax, bp->ut_line); if (dolong) { if (bp->ut_addr) { struct in_addr foo; foo.s_addr = bp->ut_addr; printf("%-*.*s ", hmax, hmax, inet_ntoa(foo)); } else { printf("%-*.*s ", hmax, hmax, ""); } } else { printf("%-*.*s ", hmax, hmax, bp->ut_host); } if (doyear) { printf("%10.10s %4.4s %5.5s ", ct, ct + 20, ct + 11); } else { printf("%10.10s %5.5s ", ct, ct + 11); }}/* * wtmp -- * read through the wtmp file */static voidwtmp(void) { register struct utmp *bp; /* current structure */ register TTY *T; /* tty list entry */ long delta; /* time difference */ char *crmsg = NULL; char *ct = NULL;#if USE_GETUTENT struct utmp **utmplist = NULL; int listlen = 0;#else int fd; struct utmp *utl; struct stat st; int utl_len;#endif int listnr = 0; int i; utmpname(file); (void)time(&utmpbuf.ut_time); (void)signal(SIGINT, onintr); (void)signal(SIGQUIT, onintr);#if USE_GETUTENT setutent(); while((bp = getutent())) { if(listnr >= listlen) { listlen += 10; listlen *= 2; /* avoid quadratic behaviour */ utmplist = realloc(utmplist, sizeof(bp) * listlen); } utmplist[listnr] = malloc(sizeof(*bp)); memcpy(utmplist[listnr++], bp, sizeof(*bp)); } endutent();#else if ((fd = open(file,O_RDONLY)) < 0) exit(1); fstat(fd, &st); utl_len = st.st_size; utl = mmap(NULL, utl_len, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FILE, fd, 0); if (utl == NULL) exit(1); listnr = utl_len/sizeof(struct utmp);#endif if(listnr) #if USE_GETUTENT ct = ctime(&utmplist[0]->ut_time);#else ct = ctime(&utl[0].ut_time);#endif for(i = listnr - 1; i >= 0; i--) {#if USE_GETUTENT bp = utmplist[i];#else bp = utl+i;#endif /* * if the terminal line is '~', the machine stopped. * see utmp(5) for more info. */ if (!strncmp(bp->ut_line, "~", LMAX)) { /* * utmp(5) also mentions that the user * name should be 'shutdown' or 'reboot'. * Not checking the name causes e.g. runlevel * changes to be displayed as 'crash'. -thaele */ if (!strncmp(bp->ut_user, "reboot", NMAX) || !strncmp(bp->ut_user, "shutdown", NMAX)) { /* 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", NMAX) ? "crash" : "down "); if (!bp->ut_name[0]) (void)strcpy(bp->ut_name, "reboot"); if (want(bp, NO)) { ct = ctime(&bp->ut_time); if(bp->ut_type != LOGIN_PROCESS) { print_partial_line(bp); putchar('\n'); } if (maxrec && !--maxrec) return; } 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, LMAX)) break; } if (bp->ut_name[0] && bp->ut_type != LOGIN_PROCESS && bp->ut_type != DEAD_PROCESS && want(bp, YES)) { print_partial_line(bp); if (!T->logout) puts(_(" still logged in")); else { if (T->logout < 0) { T->logout = -T->logout; printf("- %s", crmsg); } 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); else printf(" (%ld+%5.5s)\n", delta / SECDAY, asctime(gmtime(&delta))+11); } if (maxrec != -1 && !--maxrec) return; } T->logout = bp->ut_time; utmpbuf.ut_time = bp->ut_time;#if USE_GETUTENT free(bp); } if(utmplist) free(utmplist);#else } munmap(utl,utl_len); close(fd);#endif if(ct) printf(_("\nwtmp begins %s"), ct); /* ct already ends in \n */}/* * want -- * see if want this entry */static intwant(struct utmp *bp, int check) { register ARG *step; 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 (!strncmp(step->name, bp->ut_host, HMAX)) return(YES); break; case TTY_TYPE: if (!strncmp(step->name, bp->ut_line, LMAX)) return(YES); break; case USER_TYPE: if (!strncmp(step->name, bp->ut_name, NMAX)) return(YES); break; case INET_TYPE: if (bp->ut_addr == inet_addr(step->name)) return(YES); break; } return(NO);}/* * addarg -- * add an entry to a linked list of arguments */static voidaddarg(int type, char *arg) { register ARG *cur; if (!(cur = (ARG *)malloc((unsigned 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 */TTY *addtty(char *ttyname) { register TTY *cur; if (!(cur = (TTY *)malloc((unsigned int)sizeof(TTY)))) { fputs(_("last: malloc failure.\n"), stderr); exit(1); } cur->next = ttylist; cur->logout = currentout; memcpy(cur->tty, ttyname, LMAX); 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 voidhostconv(char *arg) { static int first = 1; static char *hostdot, name[MAXHOSTNAMELEN]; char *argdot; if (!(argdot = strchr(arg, '.'))) return; if (first) { first = 0; if (gethostname(name, sizeof(name))) { perror(_("last: gethostname")); exit(1); } hostdot = strchr(name, '.'); } if (hostdot && !strcmp(hostdot, argdot)) *argdot = '\0';}/* * ttyconv -- * convert tty to correct name. */static char *ttyconv(char *arg) { char *mval; /* * 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((unsigned 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); } if (!strncmp(arg, "/dev/", sizeof("/dev/") - 1)) return(arg + 5); return(arg);}/* * onintr -- * on interrupt, we inform the user how far we've gotten */static voidonintr(int signo) { char *ct; ct = ctime(&utmpbuf.ut_time); printf(_("\ninterrupted %10.10s %5.5s \n"), ct, ct + 11); if (signo == SIGINT) exit(1); (void)fflush(stdout); /* fix required for rsh */}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -