1 1.36 dholland /* $NetBSD: last.c,v 1.36 2012/03/15 03:04:05 dholland Exp $ */ 2 1.5 jtc 3 1.1 cgd /* 4 1.5 jtc * Copyright (c) 1987, 1993, 1994 5 1.5 jtc * The Regents of the University of California. All rights reserved. 6 1.1 cgd * 7 1.1 cgd * Redistribution and use in source and binary forms, with or without 8 1.1 cgd * modification, are permitted provided that the following conditions 9 1.1 cgd * are met: 10 1.1 cgd * 1. Redistributions of source code must retain the above copyright 11 1.1 cgd * notice, this list of conditions and the following disclaimer. 12 1.1 cgd * 2. Redistributions in binary form must reproduce the above copyright 13 1.1 cgd * notice, this list of conditions and the following disclaimer in the 14 1.1 cgd * documentation and/or other materials provided with the distribution. 15 1.21 agc * 3. Neither the name of the University nor the names of its contributors 16 1.1 cgd * may be used to endorse or promote products derived from this software 17 1.1 cgd * without specific prior written permission. 18 1.1 cgd * 19 1.1 cgd * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 20 1.1 cgd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 21 1.1 cgd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 22 1.1 cgd * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 23 1.1 cgd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 24 1.1 cgd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 25 1.1 cgd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 26 1.1 cgd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 27 1.1 cgd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 28 1.1 cgd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 29 1.1 cgd * SUCH DAMAGE. 30 1.1 cgd */ 31 1.1 cgd 32 1.8 kleink #include <sys/cdefs.h> 33 1.1 cgd #ifndef lint 34 1.32 lukem __COPYRIGHT("@(#) Copyright (c) 1987, 1993, 1994\ 35 1.32 lukem The Regents of the University of California. All rights reserved."); 36 1.1 cgd #endif /* not lint */ 37 1.1 cgd 38 1.1 cgd #ifndef lint 39 1.5 jtc #if 0 40 1.5 jtc static char sccsid[] = "@(#)last.c 8.2 (Berkeley) 4/2/94"; 41 1.5 jtc #endif 42 1.36 dholland __RCSID("$NetBSD: last.c,v 1.36 2012/03/15 03:04:05 dholland Exp $"); 43 1.1 cgd #endif /* not lint */ 44 1.1 cgd 45 1.1 cgd #include <sys/param.h> 46 1.1 cgd #include <sys/stat.h> 47 1.5 jtc 48 1.5 jtc #include <err.h> 49 1.30 cbiere #include <errno.h> 50 1.5 jtc #include <fcntl.h> 51 1.5 jtc #include <paths.h> 52 1.5 jtc #include <stdio.h> 53 1.5 jtc #include <stdlib.h> 54 1.5 jtc #include <string.h> 55 1.1 cgd #include <time.h> 56 1.5 jtc #include <tzfile.h> 57 1.5 jtc #include <unistd.h> 58 1.23 christos #include <arpa/inet.h> 59 1.16 christos #ifdef SUPPORT_UTMPX 60 1.16 christos #include <utmpx.h> 61 1.16 christos #endif 62 1.16 christos #ifdef SUPPORT_UTMP 63 1.1 cgd #include <utmp.h> 64 1.16 christos #endif 65 1.25 christos #include <util.h> 66 1.16 christos 67 1.16 christos #ifndef UT_NAMESIZE 68 1.16 christos #define UT_NAMESIZE 8 69 1.16 christos #define UT_LINESIZE 8 70 1.16 christos #define UT_HOSTSIZE 16 71 1.16 christos #endif 72 1.16 christos #ifndef SIGNATURE 73 1.16 christos #define SIGNATURE -1 74 1.16 christos #endif 75 1.16 christos 76 1.16 christos 77 1.1 cgd 78 1.14 simonb #define NO 0 /* false/no */ 79 1.14 simonb #define YES 1 /* true/yes */ 80 1.1 cgd 81 1.14 simonb #define TBUFLEN 30 /* length of time string buffer */ 82 1.15 simonb #define TFMT "%a %b %d %R" /* strftime format string */ 83 1.15 simonb #define LTFMT "%a %b %d %Y %T" /* strftime long format string */ 84 1.14 simonb #define TFMTS "%R" /* strftime format string - time only */ 85 1.14 simonb #define LTFMTS "%T" /* strftime long format string - " */ 86 1.14 simonb 87 1.14 simonb /* fmttime() flags */ 88 1.14 simonb #define FULLTIME 0x1 /* show year, seconds */ 89 1.14 simonb #define TIMEONLY 0x2 /* show time only, not date */ 90 1.14 simonb #define GMT 0x4 /* show time at GMT, for offsets only */ 91 1.14 simonb 92 1.27 christos #define MAXUTMP 1024 93 1.1 cgd 94 1.1 cgd typedef struct arg { 95 1.33 lukem const char *name; /* argument */ 96 1.1 cgd #define HOST_TYPE -2 97 1.1 cgd #define TTY_TYPE -3 98 1.1 cgd #define USER_TYPE -4 99 1.33 lukem int type; /* type of arg */ 100 1.14 simonb struct arg *next; /* linked list pointer */ 101 1.1 cgd } ARG; 102 1.16 christos static ARG *arglist; /* head of linked list */ 103 1.1 cgd 104 1.1 cgd typedef struct ttytab { 105 1.14 simonb time_t logout; /* log out time */ 106 1.16 christos char tty[128]; /* terminal name */ 107 1.14 simonb struct ttytab *next; /* linked list pointer */ 108 1.1 cgd } TTY; 109 1.16 christos static TTY *ttylist; /* head of linked list */ 110 1.1 cgd 111 1.14 simonb static time_t currentout; /* current logout value */ 112 1.14 simonb static long maxrec; /* records to display */ 113 1.14 simonb static int fulltime = 0; /* Display seconds? */ 114 1.26 christos static int xflag; /* Assume file is wtmpx format */ 115 1.1 cgd 116 1.33 lukem static void addarg(int, const char *); 117 1.16 christos static TTY *addtty(const char *); 118 1.16 christos static void hostconv(char *); 119 1.33 lukem static const char *ttyconv(char *); 120 1.16 christos #ifdef SUPPORT_UTMPX 121 1.23 christos static void wtmpx(const char *, int, int, int, int); 122 1.16 christos #endif 123 1.16 christos #ifdef SUPPORT_UTMP 124 1.23 christos static void wtmp(const char *, int, int, int, int); 125 1.16 christos #endif 126 1.16 christos static char *fmttime(time_t, int); 127 1.35 joerg __dead static void usage(void); 128 1.16 christos 129 1.16 christos static 130 1.16 christos void usage(void) 131 1.16 christos { 132 1.26 christos (void)fprintf(stderr, "usage: %s [-#%s] [-nTx] [-f file]" 133 1.18 wiz " [-H hostsize] [-h host] [-L linesize]\n" 134 1.18 wiz "\t [-N namesize] [-t tty] [user ...]\n", getprogname(), 135 1.18 wiz #ifdef NOTYET_SUPPORT_UTMPX 136 1.16 christos "w" 137 1.16 christos #else 138 1.16 christos "" 139 1.16 christos #endif 140 1.16 christos ); 141 1.30 cbiere exit(EXIT_FAILURE); 142 1.16 christos } 143 1.5 jtc 144 1.5 jtc int 145 1.16 christos main(int argc, char *argv[]) 146 1.1 cgd { 147 1.1 cgd int ch; 148 1.5 jtc char *p; 149 1.33 lukem const char *file = NULL; 150 1.16 christos int namesize = UT_NAMESIZE; 151 1.16 christos int linesize = UT_LINESIZE; 152 1.16 christos int hostsize = UT_HOSTSIZE; 153 1.23 christos int numeric = 0; 154 1.1 cgd 155 1.1 cgd maxrec = -1; 156 1.16 christos 157 1.26 christos while ((ch = getopt(argc, argv, "0123456789f:H:h:L:nN:Tt:x")) != -1) 158 1.5 jtc switch (ch) { 159 1.1 cgd case '0': case '1': case '2': case '3': case '4': 160 1.1 cgd case '5': case '6': case '7': case '8': case '9': 161 1.1 cgd /* 162 1.1 cgd * kludge: last was originally designed to take 163 1.1 cgd * a number after a dash. 164 1.1 cgd */ 165 1.1 cgd if (maxrec == -1) { 166 1.1 cgd p = argv[optind - 1]; 167 1.1 cgd if (p[0] == '-' && p[1] == ch && !p[2]) 168 1.1 cgd maxrec = atol(++p); 169 1.29 christos else if (optind < argc) 170 1.1 cgd maxrec = atol(argv[optind] + 1); 171 1.29 christos else 172 1.29 christos usage(); 173 1.1 cgd if (!maxrec) 174 1.29 christos return 0; 175 1.1 cgd } 176 1.1 cgd break; 177 1.1 cgd case 'f': 178 1.1 cgd file = optarg; 179 1.30 cbiere if ('\0' == file[0]) 180 1.30 cbiere usage(); 181 1.1 cgd break; 182 1.18 wiz case 'H': 183 1.18 wiz hostsize = atoi(optarg); 184 1.30 cbiere if (hostsize < 1) 185 1.30 cbiere usage(); 186 1.18 wiz break; 187 1.1 cgd case 'h': 188 1.1 cgd hostconv(optarg); 189 1.1 cgd addarg(HOST_TYPE, optarg); 190 1.1 cgd break; 191 1.16 christos case 'L': 192 1.16 christos linesize = atoi(optarg); 193 1.30 cbiere if (linesize < 1) 194 1.30 cbiere usage(); 195 1.16 christos break; 196 1.18 wiz case 'N': 197 1.18 wiz namesize = atoi(optarg); 198 1.30 cbiere if (namesize < 1) 199 1.30 cbiere usage(); 200 1.16 christos break; 201 1.23 christos case 'n': 202 1.23 christos numeric = 1; 203 1.23 christos break; 204 1.7 perry case 'T': 205 1.7 perry fulltime = 1; 206 1.18 wiz break; 207 1.18 wiz case 't': 208 1.18 wiz addarg(TTY_TYPE, ttyconv(optarg)); 209 1.7 perry break; 210 1.26 christos case 'x': 211 1.26 christos xflag = 1; 212 1.26 christos break; 213 1.1 cgd case '?': 214 1.1 cgd default: 215 1.16 christos usage(); 216 1.1 cgd } 217 1.1 cgd 218 1.1 cgd if (argc) { 219 1.1 cgd setlinebuf(stdout); 220 1.1 cgd for (argv += optind; *argv; ++argv) { 221 1.1 cgd #define COMPATIBILITY 222 1.1 cgd #ifdef COMPATIBILITY 223 1.1 cgd /* code to allow "last p5" to work */ 224 1.1 cgd addarg(TTY_TYPE, ttyconv(*argv)); 225 1.1 cgd #endif 226 1.1 cgd addarg(USER_TYPE, *argv); 227 1.1 cgd } 228 1.1 cgd } 229 1.16 christos if (file == NULL) { 230 1.16 christos #ifdef SUPPORT_UTMPX 231 1.16 christos if (access(_PATH_WTMPX, R_OK) == 0) 232 1.16 christos file = _PATH_WTMPX; 233 1.16 christos else 234 1.16 christos #endif 235 1.16 christos #ifdef SUPPORT_UTMP 236 1.16 christos if (access(_PATH_WTMP, R_OK) == 0) 237 1.16 christos file = _PATH_WTMP; 238 1.16 christos #endif 239 1.16 christos if (file == NULL) 240 1.16 christos #if defined(SUPPORT_UTMPX) && defined(SUPPORT_UTMP) 241 1.30 cbiere errx(EXIT_FAILURE, "Cannot access `%s' or `%s'", _PATH_WTMPX, 242 1.16 christos _PATH_WTMP); 243 1.16 christos #elif defined(SUPPORT_UTMPX) 244 1.30 cbiere errx(EXIT_FAILURE, "Cannot access `%s'", _PATH_WTMPX); 245 1.16 christos #elif defined(SUPPORT_UTMP) 246 1.30 cbiere errx(EXIT_FAILURE, "Cannot access `%s'", _PATH_WTMP); 247 1.16 christos #else 248 1.30 cbiere errx(EXIT_FAILURE, "No utmp or utmpx support compiled in."); 249 1.16 christos #endif 250 1.16 christos } 251 1.16 christos #if defined(SUPPORT_UTMPX) && defined(SUPPORT_UTMP) 252 1.26 christos if (file[strlen(file) - 1] == 'x' || xflag) 253 1.23 christos wtmpx(file, namesize, linesize, hostsize, numeric); 254 1.16 christos else 255 1.23 christos wtmp(file, namesize, linesize, hostsize, numeric); 256 1.16 christos #elif defined(SUPPORT_UTMPX) 257 1.23 christos wtmpx(file, namesize, linesize, hostsize, numeric); 258 1.16 christos #elif defined(SUPPORT_UTMP) 259 1.23 christos wtmp(file, namesize, linesize, hostsize, numeric); 260 1.16 christos #else 261 1.30 cbiere errx(EXIT_FAILURE, "No utmp or utmpx support compiled in."); 262 1.16 christos #endif 263 1.30 cbiere exit(EXIT_SUCCESS); 264 1.1 cgd } 265 1.1 cgd 266 1.1 cgd 267 1.1 cgd /* 268 1.1 cgd * addarg -- 269 1.1 cgd * add an entry to a linked list of arguments 270 1.1 cgd */ 271 1.16 christos static void 272 1.33 lukem addarg(int type, const char *arg) 273 1.1 cgd { 274 1.5 jtc ARG *cur; 275 1.1 cgd 276 1.30 cbiere if (!(cur = (ARG *)malloc(sizeof(ARG)))) 277 1.30 cbiere err(EXIT_FAILURE, "malloc failure"); 278 1.1 cgd cur->next = arglist; 279 1.1 cgd cur->type = type; 280 1.1 cgd cur->name = arg; 281 1.1 cgd arglist = cur; 282 1.1 cgd } 283 1.1 cgd 284 1.1 cgd /* 285 1.1 cgd * addtty -- 286 1.1 cgd * add an entry to a linked list of ttys 287 1.1 cgd */ 288 1.16 christos static TTY * 289 1.33 lukem addtty(const char *tty) 290 1.1 cgd { 291 1.5 jtc TTY *cur; 292 1.1 cgd 293 1.30 cbiere if (!(cur = (TTY *)malloc(sizeof(TTY)))) 294 1.30 cbiere err(EXIT_FAILURE, "malloc failure"); 295 1.1 cgd cur->next = ttylist; 296 1.1 cgd cur->logout = currentout; 297 1.33 lukem memmove(cur->tty, tty, sizeof(cur->tty)); 298 1.5 jtc return (ttylist = cur); 299 1.1 cgd } 300 1.1 cgd 301 1.1 cgd /* 302 1.1 cgd * hostconv -- 303 1.1 cgd * convert the hostname to search pattern; if the supplied host name 304 1.1 cgd * has a domain attached that is the same as the current domain, rip 305 1.1 cgd * off the domain suffix since that's what login(1) does. 306 1.1 cgd */ 307 1.16 christos static void 308 1.16 christos hostconv(char *arg) 309 1.1 cgd { 310 1.1 cgd static int first = 1; 311 1.10 mrg static char *hostdot, name[MAXHOSTNAMELEN + 1]; 312 1.5 jtc char *argdot; 313 1.1 cgd 314 1.5 jtc if (!(argdot = strchr(arg, '.'))) 315 1.1 cgd return; 316 1.1 cgd if (first) { 317 1.1 cgd first = 0; 318 1.5 jtc if (gethostname(name, sizeof(name))) 319 1.30 cbiere err(EXIT_FAILURE, "gethostname"); 320 1.10 mrg name[sizeof(name) - 1] = '\0'; 321 1.5 jtc hostdot = strchr(name, '.'); 322 1.1 cgd } 323 1.1 cgd if (hostdot && !strcasecmp(hostdot, argdot)) 324 1.1 cgd *argdot = '\0'; 325 1.1 cgd } 326 1.1 cgd 327 1.1 cgd /* 328 1.1 cgd * ttyconv -- 329 1.1 cgd * convert tty to correct name. 330 1.1 cgd */ 331 1.33 lukem static const char * 332 1.16 christos ttyconv(char *arg) 333 1.1 cgd { 334 1.5 jtc char *mval; 335 1.1 cgd 336 1.30 cbiere if (!strcmp(arg, "co")) 337 1.30 cbiere return ("console"); 338 1.1 cgd /* 339 1.1 cgd * kludge -- we assume that all tty's end with 340 1.1 cgd * a two character suffix. 341 1.1 cgd */ 342 1.1 cgd if (strlen(arg) == 2) { 343 1.30 cbiere if (asprintf(&mval, "tty%s", arg) == -1) 344 1.30 cbiere err(EXIT_FAILURE, "malloc failure"); 345 1.5 jtc return (mval); 346 1.1 cgd } 347 1.1 cgd if (!strncmp(arg, _PATH_DEV, sizeof(_PATH_DEV) - 1)) 348 1.30 cbiere return (&arg[sizeof(_PATH_DEV) - 1]); 349 1.5 jtc return (arg); 350 1.1 cgd } 351 1.1 cgd 352 1.1 cgd /* 353 1.14 simonb * fmttime -- 354 1.14 simonb * return pointer to (static) formatted time string. 355 1.14 simonb */ 356 1.16 christos static char * 357 1.16 christos fmttime(time_t t, int flags) 358 1.14 simonb { 359 1.14 simonb struct tm *tm; 360 1.14 simonb static char tbuf[TBUFLEN]; 361 1.14 simonb 362 1.14 simonb tm = (flags & GMT) ? gmtime(&t) : localtime(&t); 363 1.34 dholland if (tm == NULL) { 364 1.34 dholland strcpy(tbuf, "????"); 365 1.34 dholland return tbuf; 366 1.34 dholland } 367 1.14 simonb strftime(tbuf, sizeof(tbuf), 368 1.14 simonb (flags & TIMEONLY) 369 1.14 simonb ? (flags & FULLTIME ? LTFMTS : TFMTS) 370 1.14 simonb : (flags & FULLTIME ? LTFMT : TFMT), 371 1.14 simonb tm); 372 1.14 simonb return (tbuf); 373 1.14 simonb } 374 1.14 simonb 375 1.16 christos #ifdef SUPPORT_UTMP 376 1.16 christos #define TYPE(a) 0 377 1.16 christos #define NAMESIZE UT_NAMESIZE 378 1.16 christos #define LINESIZE UT_LINESIZE 379 1.16 christos #define HOSTSIZE UT_HOSTSIZE 380 1.16 christos #define ut_timefld ut_time 381 1.36 dholland #define HAS_UT_SS 0 382 1.16 christos #include "want.c" 383 1.17 matt #undef TYPE /*(a)*/ 384 1.16 christos #undef NAMESIZE 385 1.16 christos #undef LINESIZE 386 1.16 christos #undef HOSTSIZE 387 1.16 christos #undef ut_timefld 388 1.36 dholland #undef HAS_UT_SS 389 1.16 christos #endif 390 1.1 cgd 391 1.16 christos #ifdef SUPPORT_UTMPX 392 1.16 christos #define utmp utmpx 393 1.16 christos #define want wantx 394 1.16 christos #define wtmp wtmpx 395 1.23 christos #define gethost gethostx 396 1.16 christos #define buf bufx 397 1.16 christos #define onintr onintrx 398 1.16 christos #define TYPE(a) (a)->ut_type 399 1.16 christos #define NAMESIZE UTX_USERSIZE 400 1.16 christos #define LINESIZE UTX_LINESIZE 401 1.16 christos #define HOSTSIZE UTX_HOSTSIZE 402 1.16 christos #define ut_timefld ut_xtime 403 1.36 dholland #define HAS_UT_SS 1 404 1.16 christos #include "want.c" 405 1.16 christos #endif 406