19b50d902SRodney W. Grimes /* 29b50d902SRodney W. Grimes * Copyright (c) 1989, 1993 39b50d902SRodney W. Grimes * The Regents of the University of California. All rights reserved. 49b50d902SRodney W. Grimes * 59b50d902SRodney W. Grimes * This code is derived from software contributed to Berkeley by 69b50d902SRodney W. Grimes * Tony Nardo of the Johns Hopkins University/Applied Physics Lab. 79b50d902SRodney W. Grimes * 89b50d902SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 99b50d902SRodney W. Grimes * modification, are permitted provided that the following conditions 109b50d902SRodney W. Grimes * are met: 119b50d902SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 129b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 139b50d902SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 149b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 159b50d902SRodney W. Grimes * documentation and/or other materials provided with the distribution. 16*fbbd9655SWarner Losh * 3. Neither the name of the University nor the names of its contributors 179b50d902SRodney W. Grimes * may be used to endorse or promote products derived from this software 189b50d902SRodney W. Grimes * without specific prior written permission. 199b50d902SRodney W. Grimes * 209b50d902SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 219b50d902SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 229b50d902SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 239b50d902SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 249b50d902SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 259b50d902SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 269b50d902SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 279b50d902SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 289b50d902SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 299b50d902SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 309b50d902SRodney W. Grimes * SUCH DAMAGE. 319b50d902SRodney W. Grimes */ 329b50d902SRodney W. Grimes 3386641d8fSPaul Traina /* 3486641d8fSPaul Traina * Luke Mewburn <lm@rmit.edu.au> added the following on 940622: 3586641d8fSPaul Traina * - mail status ("No Mail", "Mail read:...", or "New Mail ..., 3686641d8fSPaul Traina * Unread since ...".) 3786641d8fSPaul Traina * - 4 digit phone extensions (3210 is printed as x3210.) 3886641d8fSPaul Traina * - host/office toggling in short format with -h & -o. 3986641d8fSPaul Traina * - short day names (`Tue' printed instead of `Jun 21' if the 4086641d8fSPaul Traina * login time is < 6 days. 4186641d8fSPaul Traina */ 4286641d8fSPaul Traina 439b50d902SRodney W. Grimes #ifndef lint 441e832bf8SRuslan Ermilov static const char copyright[] = 459b50d902SRodney W. Grimes "@(#) Copyright (c) 1989, 1993\n\ 469b50d902SRodney W. Grimes The Regents of the University of California. All rights reserved.\n"; 479b50d902SRodney W. Grimes #endif /* not lint */ 489b50d902SRodney W. Grimes 499f5b04e9SDavid Malone #if 0 509b50d902SRodney W. Grimes #ifndef lint 519f5b04e9SDavid Malone static char sccsid[] = "@(#)finger.c 8.5 (Berkeley) 5/4/95"; 524c86f3adSPhilippe Charnier #endif 539f5b04e9SDavid Malone #endif 549f5b04e9SDavid Malone 559f5b04e9SDavid Malone #include <sys/cdefs.h> 569f5b04e9SDavid Malone __FBSDID("$FreeBSD$"); 579b50d902SRodney W. Grimes 589b50d902SRodney W. Grimes /* 599b50d902SRodney W. Grimes * Finger prints out information about users. It is not portable since 609b50d902SRodney W. Grimes * certain fields (e.g. the full user name, office, and phone numbers) are 619b50d902SRodney W. Grimes * extracted from the gecos field of the passwd file which other UNIXes 629b50d902SRodney W. Grimes * may not have or may use for other things. 639b50d902SRodney W. Grimes * 649b50d902SRodney W. Grimes * There are currently two output formats; the short format is one line 659b50d902SRodney W. Grimes * per user and displays login name, tty, login time, real name, idle time, 6686641d8fSPaul Traina * and either remote host information (default) or office location/phone 6786641d8fSPaul Traina * number, depending on if -h or -o is used respectively. 6886641d8fSPaul Traina * The long format gives the same information (in a more legible format) as 6986641d8fSPaul Traina * well as home directory, shell, mail info, and .plan/.project files. 709b50d902SRodney W. Grimes */ 719b50d902SRodney W. Grimes 723daa8471SHajimu UMEMOTO #include <sys/types.h> 733daa8471SHajimu UMEMOTO #include <sys/socket.h> 74df3f5d9dSPeter Wemm #include <db.h> 75df3f5d9dSPeter Wemm #include <err.h> 76df3f5d9dSPeter Wemm #include <pwd.h> 779b50d902SRodney W. Grimes #include <stdio.h> 789b50d902SRodney W. Grimes #include <stdlib.h> 799b50d902SRodney W. Grimes #include <string.h> 80df3f5d9dSPeter Wemm #include <time.h> 81df3f5d9dSPeter Wemm #include <unistd.h> 82550dcc59SEd Schouten #include <utmpx.h> 83dc675fc2SAndrey A. Chernov #include <locale.h> 84df3f5d9dSPeter Wemm 859b50d902SRodney W. Grimes #include "finger.h" 86098c6e87SBrian Somers #include "pathnames.h" 879b50d902SRodney W. Grimes 889b50d902SRodney W. Grimes DB *db; 899b50d902SRodney W. Grimes time_t now; 90bf70beceSEd Schouten static int kflag, mflag, sflag; 91bf70beceSEd Schouten int entries, gflag, lflag, pplan, oflag; 921acb5329SMaxim Sobolev sa_family_t family = PF_UNSPEC; 93f4d292b7SAndrey A. Chernov int d_first = -1; 949b50d902SRodney W. Grimes char tbuf[1024]; 95d2e4ea2aSDiomidis Spinellis int invoker_root = 0; 969b50d902SRodney W. Grimes 97f1bb2cd2SWarner Losh static void loginlist(void); 98f1bb2cd2SWarner Losh static int option(int, char **); 99f1bb2cd2SWarner Losh static void usage(void); 100f1bb2cd2SWarner Losh static void userlist(int, char **); 1019b50d902SRodney W. Grimes 1024e030ba6SMark Murray static int 103f4ac32deSDavid Malone option(int argc, char **argv) 1049b50d902SRodney W. Grimes { 1059b50d902SRodney W. Grimes int ch; 1069b50d902SRodney W. Grimes 10759be6088SPaul Traina optind = 1; /* reset getopt */ 10886641d8fSPaul Traina 109b49339d7SDag-Erling Smørgrav while ((ch = getopt(argc, argv, "46gklmpsho")) != -1) 1109b50d902SRodney W. Grimes switch(ch) { 1113daa8471SHajimu UMEMOTO case '4': 1123daa8471SHajimu UMEMOTO family = AF_INET; 1133daa8471SHajimu UMEMOTO break; 1143daa8471SHajimu UMEMOTO case '6': 1153daa8471SHajimu UMEMOTO family = AF_INET6; 1163daa8471SHajimu UMEMOTO break; 1170830bd13SJonathan Mini case 'g': 1180830bd13SJonathan Mini gflag = 1; 1190830bd13SJonathan Mini break; 120ad468669SJuli Mallett case 'k': 121ad468669SJuli Mallett kflag = 1; /* keep going without utmp */ 122ad468669SJuli Mallett break; 1239b50d902SRodney W. Grimes case 'l': 1249b50d902SRodney W. Grimes lflag = 1; /* long format */ 1259b50d902SRodney W. Grimes break; 1269b50d902SRodney W. Grimes case 'm': 1279b50d902SRodney W. Grimes mflag = 1; /* force exact match of names */ 1289b50d902SRodney W. Grimes break; 1299b50d902SRodney W. Grimes case 'p': 1309b50d902SRodney W. Grimes pplan = 1; /* don't show .plan/.project */ 1319b50d902SRodney W. Grimes break; 1329b50d902SRodney W. Grimes case 's': 1339b50d902SRodney W. Grimes sflag = 1; /* short format */ 1349b50d902SRodney W. Grimes break; 13586641d8fSPaul Traina case 'h': 13686641d8fSPaul Traina oflag = 0; /* remote host info */ 13786641d8fSPaul Traina break; 13886641d8fSPaul Traina case 'o': 13986641d8fSPaul Traina oflag = 1; /* office info */ 14086641d8fSPaul Traina break; 1419b50d902SRodney W. Grimes case '?': 1429b50d902SRodney W. Grimes default: 143b14d8277SPhilippe Charnier usage(); 1449b50d902SRodney W. Grimes } 14559be6088SPaul Traina 14659be6088SPaul Traina return optind; 14759be6088SPaul Traina } 14859be6088SPaul Traina 149b14d8277SPhilippe Charnier static void 150f4ac32deSDavid Malone usage(void) 151b14d8277SPhilippe Charnier { 152f682f10cSRuslan Ermilov (void)fprintf(stderr, 153b49339d7SDag-Erling Smørgrav "usage: finger [-46gklmpsho] [user ...] [user@host ...]\n"); 154b14d8277SPhilippe Charnier exit(1); 155b14d8277SPhilippe Charnier } 156b14d8277SPhilippe Charnier 157b14d8277SPhilippe Charnier int 158f4ac32deSDavid Malone main(int argc, char **argv) 15959be6088SPaul Traina { 160b14d8277SPhilippe Charnier int envargc, argcnt; 16159be6088SPaul Traina char *envargv[3]; 162dd5288f3SDavid E. O'Brien struct passwd *pw; 1634e030ba6SMark Murray static char myname[] = "finger"; 164dd5288f3SDavid E. O'Brien 165dd5288f3SDavid E. O'Brien if (getuid() == 0 || geteuid() == 0) { 166d2e4ea2aSDiomidis Spinellis invoker_root = 1; 167dd5288f3SDavid E. O'Brien if ((pw = getpwnam(UNPRIV_NAME)) && pw->pw_uid > 0) { 1686e46c1ccSSimon L. B. Nielsen if (setgid(pw->pw_gid) != 0) 1696e46c1ccSSimon L. B. Nielsen err(1, "setgid()"); 1706e46c1ccSSimon L. B. Nielsen if (setuid(pw->pw_uid) != 0) 1716e46c1ccSSimon L. B. Nielsen err(1, "setuid()"); 172dd5288f3SDavid E. O'Brien } else { 1736e46c1ccSSimon L. B. Nielsen if (setgid(UNPRIV_UGID) != 0) 1746e46c1ccSSimon L. B. Nielsen err(1, "setgid()"); 1756e46c1ccSSimon L. B. Nielsen if (setuid(UNPRIV_UGID) != 0) 1766e46c1ccSSimon L. B. Nielsen err(1, "setuid()"); 177dd5288f3SDavid E. O'Brien } 178dd5288f3SDavid E. O'Brien } 17959be6088SPaul Traina 180d1b2ad1aSAndrey A. Chernov (void) setlocale(LC_ALL, ""); 181dc675fc2SAndrey A. Chernov 18259be6088SPaul Traina /* remove this line to get remote host */ 18359be6088SPaul Traina oflag = 1; /* default to old "office" behavior */ 18459be6088SPaul Traina 18559be6088SPaul Traina /* 18659be6088SPaul Traina * Process environment variables followed by command line arguments. 18759be6088SPaul Traina */ 18859be6088SPaul Traina if ((envargv[1] = getenv("FINGER"))) { 18959be6088SPaul Traina envargc = 2; 1904e030ba6SMark Murray envargv[0] = myname; 19159be6088SPaul Traina envargv[2] = NULL; 19259be6088SPaul Traina (void) option(envargc, envargv); 19359be6088SPaul Traina } 19459be6088SPaul Traina 19559be6088SPaul Traina argcnt = option(argc, argv); 19659be6088SPaul Traina argc -= argcnt; 19759be6088SPaul Traina argv += argcnt; 1989b50d902SRodney W. Grimes 1999b50d902SRodney W. Grimes (void)time(&now); 2009b50d902SRodney W. Grimes setpassent(1); 2019b50d902SRodney W. Grimes if (!*argv) { 2029b50d902SRodney W. Grimes /* 2039b50d902SRodney W. Grimes * Assign explicit "small" format if no names given and -l 2049b50d902SRodney W. Grimes * not selected. Force the -s BEFORE we get names so proper 2059b50d902SRodney W. Grimes * screening will be done. 2069b50d902SRodney W. Grimes */ 2079b50d902SRodney W. Grimes if (!lflag) 2089b50d902SRodney W. Grimes sflag = 1; /* if -l not explicit, force -s */ 2099b50d902SRodney W. Grimes loginlist(); 2109b50d902SRodney W. Grimes if (entries == 0) 2119b50d902SRodney W. Grimes (void)printf("No one logged on.\n"); 2129b50d902SRodney W. Grimes } else { 2139b50d902SRodney W. Grimes userlist(argc, argv); 2149b50d902SRodney W. Grimes /* 2159b50d902SRodney W. Grimes * Assign explicit "large" format if names given and -s not 2169b50d902SRodney W. Grimes * explicitly stated. Force the -l AFTER we get names so any 2179b50d902SRodney W. Grimes * remote finger attempts specified won't be mishandled. 2189b50d902SRodney W. Grimes */ 2199b50d902SRodney W. Grimes if (!sflag) 2209b50d902SRodney W. Grimes lflag = 1; /* if -s not explicit, force -l */ 2219b50d902SRodney W. Grimes } 2229ef5c48bSBill Fumerola if (entries) { 2239b50d902SRodney W. Grimes if (lflag) 2249b50d902SRodney W. Grimes lflag_print(); 2259b50d902SRodney W. Grimes else 2269b50d902SRodney W. Grimes sflag_print(); 2279ef5c48bSBill Fumerola } 228df3f5d9dSPeter Wemm return (0); 2299b50d902SRodney W. Grimes } 2309b50d902SRodney W. Grimes 2319b50d902SRodney W. Grimes static void 232f4ac32deSDavid Malone loginlist(void) 2339b50d902SRodney W. Grimes { 2344e030ba6SMark Murray PERSON *pn; 2359b50d902SRodney W. Grimes DBT data, key; 2369b50d902SRodney W. Grimes struct passwd *pw; 23770ad88f3SEd Schouten struct utmpx *user; 2384e030ba6SMark Murray int r, sflag1; 2399b50d902SRodney W. Grimes 240ad468669SJuli Mallett if (kflag) 241ad468669SJuli Mallett errx(1, "can't list logins without reading utmp"); 242ad468669SJuli Mallett 24370ad88f3SEd Schouten setutxent(); 24470ad88f3SEd Schouten while ((user = getutxent()) != NULL) { 24570ad88f3SEd Schouten if (user->ut_type != USER_PROCESS) 2469b50d902SRodney W. Grimes continue; 24770ad88f3SEd Schouten if ((pn = find_person(user->ut_user)) == NULL) { 24870ad88f3SEd Schouten if ((pw = getpwnam(user->ut_user)) == NULL) 2499b50d902SRodney W. Grimes continue; 2508829c73eSJordan K. Hubbard if (hide(pw)) 2518829c73eSJordan K. Hubbard continue; 2529b50d902SRodney W. Grimes pn = enter_person(pw); 2539b50d902SRodney W. Grimes } 25470ad88f3SEd Schouten enter_where(user, pn); 2559b50d902SRodney W. Grimes } 25670ad88f3SEd Schouten endutxent(); 2579b50d902SRodney W. Grimes if (db && lflag) 2584e030ba6SMark Murray for (sflag1 = R_FIRST;; sflag1 = R_NEXT) { 259df3f5d9dSPeter Wemm PERSON *tmp; 260df3f5d9dSPeter Wemm 2614e030ba6SMark Murray r = (*db->seq)(db, &key, &data, sflag1); 2629b50d902SRodney W. Grimes if (r == -1) 263df3f5d9dSPeter Wemm err(1, "db seq"); 2649b50d902SRodney W. Grimes if (r == 1) 2659b50d902SRodney W. Grimes break; 266df3f5d9dSPeter Wemm memmove(&tmp, data.data, sizeof tmp); 267df3f5d9dSPeter Wemm enter_lastlog(tmp); 2689b50d902SRodney W. Grimes } 2699b50d902SRodney W. Grimes } 2709b50d902SRodney W. Grimes 2719b50d902SRodney W. Grimes static void 272f4ac32deSDavid Malone userlist(int argc, char **argv) 2739b50d902SRodney W. Grimes { 2744e030ba6SMark Murray PERSON *pn; 2759b50d902SRodney W. Grimes DBT data, key; 27670ad88f3SEd Schouten struct utmpx *user; 2779b50d902SRodney W. Grimes struct passwd *pw; 2784e030ba6SMark Murray int r, sflag1, *used, *ip; 2799b50d902SRodney W. Grimes char **ap, **nargv, **np, **p; 280098c6e87SBrian Somers FILE *conf_fp; 281098c6e87SBrian Somers char conf_alias[LINE_MAX]; 282098c6e87SBrian Somers char *conf_realname; 283098c6e87SBrian Somers int conf_length; 2849b50d902SRodney W. Grimes 2859b50d902SRodney W. Grimes if ((nargv = malloc((argc+1) * sizeof(char *))) == NULL || 2869b50d902SRodney W. Grimes (used = calloc(argc, sizeof(int))) == NULL) 287df3f5d9dSPeter Wemm err(1, NULL); 2889b50d902SRodney W. Grimes 2899b50d902SRodney W. Grimes /* Pull out all network requests. */ 2909b50d902SRodney W. Grimes for (ap = p = argv, np = nargv; *p; ++p) 291b3608ae1SEd Schouten if (strchr(*p, '@')) 2929b50d902SRodney W. Grimes *np++ = *p; 2939b50d902SRodney W. Grimes else 2949b50d902SRodney W. Grimes *ap++ = *p; 2959b50d902SRodney W. Grimes 2969b50d902SRodney W. Grimes *np++ = NULL; 2979b50d902SRodney W. Grimes *ap++ = NULL; 2989b50d902SRodney W. Grimes 2999b50d902SRodney W. Grimes if (!*argv) 3009b50d902SRodney W. Grimes goto net; 3019b50d902SRodney W. Grimes 3029b50d902SRodney W. Grimes /* 3037ce8924aSBrian Somers * Mark any arguments beginning with '/' as invalid so that we 304487ac9acSUlrich Spörlein * don't accidentally confuse them with expansions from finger.conf 3057ce8924aSBrian Somers */ 3067ce8924aSBrian Somers for (p = argv, ip = used; *p; ++p, ++ip) 3077ce8924aSBrian Somers if (**p == '/') { 3087ce8924aSBrian Somers *ip = 1; 3097ce8924aSBrian Somers warnx("%s: no such user", *p); 3107ce8924aSBrian Somers } 3117ce8924aSBrian Somers 3127ce8924aSBrian Somers /* 313098c6e87SBrian Somers * Traverse the finger alias configuration file of the form 314098c6e87SBrian Somers * alias:(user|alias), ignoring comment lines beginning '#'. 315098c6e87SBrian Somers */ 316098c6e87SBrian Somers if ((conf_fp = fopen(_PATH_FINGERCONF, "r")) != NULL) { 317098c6e87SBrian Somers while(fgets(conf_alias, sizeof(conf_alias), conf_fp) != NULL) { 318098c6e87SBrian Somers conf_length = strlen(conf_alias); 319098c6e87SBrian Somers if (*conf_alias == '#' || conf_alias[--conf_length] != '\n') 320098c6e87SBrian Somers continue; 321098c6e87SBrian Somers conf_alias[conf_length] = '\0'; /* Remove trailing LF */ 322098c6e87SBrian Somers if ((conf_realname = strchr(conf_alias, ':')) == NULL) 323098c6e87SBrian Somers continue; 324098c6e87SBrian Somers *conf_realname = '\0'; /* Replace : with NUL */ 325098c6e87SBrian Somers for (p = argv; *p; ++p) { 326f342fec6SBruce Evans if (strcmp(*p, conf_alias) == 0) { 327098c6e87SBrian Somers if ((*p = strdup(conf_realname+1)) == NULL) { 328098c6e87SBrian Somers err(1, NULL); 329098c6e87SBrian Somers } 330098c6e87SBrian Somers } 331098c6e87SBrian Somers } 332098c6e87SBrian Somers } 333098c6e87SBrian Somers (void)fclose(conf_fp); 334098c6e87SBrian Somers } 335098c6e87SBrian Somers 336098c6e87SBrian Somers /* 3379b50d902SRodney W. Grimes * Traverse the list of possible login names and check the login name 338040864acSBrian Somers * and real name against the name specified by the user. If the name 339040864acSBrian Somers * begins with a '/', try to read the file of that name instead of 340040864acSBrian Somers * gathering the traditional finger information. 3419b50d902SRodney W. Grimes */ 3429b50d902SRodney W. Grimes if (mflag) 3430272ef9bSRuslan Ermilov for (p = argv, ip = used; *p; ++p, ++ip) { 3440272ef9bSRuslan Ermilov if (**p != '/' || *ip == 1 || !show_text("", *p, "")) { 345df3f5d9dSPeter Wemm if (((pw = getpwnam(*p)) != NULL) && !hide(pw)) 3469b50d902SRodney W. Grimes enter_person(pw); 3470272ef9bSRuslan Ermilov else if (!*ip) 348b14d8277SPhilippe Charnier warnx("%s: no such user", *p); 349040864acSBrian Somers } 350040864acSBrian Somers } 3519b50d902SRodney W. Grimes else { 3521b57e365SBrian Somers while ((pw = getpwent()) != NULL) { 3539b50d902SRodney W. Grimes for (p = argv, ip = used; *p; ++p, ++ip) 354040864acSBrian Somers if (**p == '/' && *ip != 1 3551b57e365SBrian Somers && show_text("", *p, "")) 356040864acSBrian Somers *ip = 1; 3571b57e365SBrian Somers else if (match(pw, *p) && !hide(pw)) { 3589b50d902SRodney W. Grimes enter_person(pw); 3599b50d902SRodney W. Grimes *ip = 1; 3609b50d902SRodney W. Grimes } 3618829c73eSJordan K. Hubbard } 3629b50d902SRodney W. Grimes for (p = argv, ip = used; *p; ++p, ++ip) 3639b50d902SRodney W. Grimes if (!*ip) 364b14d8277SPhilippe Charnier warnx("%s: no such user", *p); 3659b50d902SRodney W. Grimes } 3669b50d902SRodney W. Grimes 3679b50d902SRodney W. Grimes /* Handle network requests. */ 368a5af661eSPaul Traina net: for (p = nargv; *p;) { 3699b50d902SRodney W. Grimes netfinger(*p++); 370a5af661eSPaul Traina if (*p || entries) 371a5af661eSPaul Traina printf("\n"); 372a5af661eSPaul Traina } 3739b50d902SRodney W. Grimes 37413cf266eSXin LI free(used); 3759b50d902SRodney W. Grimes if (entries == 0) 3769b50d902SRodney W. Grimes return; 3779b50d902SRodney W. Grimes 378ad468669SJuli Mallett if (kflag) 379ad468669SJuli Mallett return; 380ad468669SJuli Mallett 3819b50d902SRodney W. Grimes /* 3829b50d902SRodney W. Grimes * Scan thru the list of users currently logged in, saving 3839b50d902SRodney W. Grimes * appropriate data whenever a match occurs. 3849b50d902SRodney W. Grimes */ 38570ad88f3SEd Schouten setutxent(); 38670ad88f3SEd Schouten while ((user = getutxent()) != NULL) { 38770ad88f3SEd Schouten if (user->ut_type != USER_PROCESS) 3889b50d902SRodney W. Grimes continue; 38970ad88f3SEd Schouten if ((pn = find_person(user->ut_user)) == NULL) 3909b50d902SRodney W. Grimes continue; 39170ad88f3SEd Schouten enter_where(user, pn); 3929b50d902SRodney W. Grimes } 39370ad88f3SEd Schouten endutxent(); 3949b50d902SRodney W. Grimes if (db) 3954e030ba6SMark Murray for (sflag1 = R_FIRST;; sflag1 = R_NEXT) { 396df3f5d9dSPeter Wemm PERSON *tmp; 397df3f5d9dSPeter Wemm 3984e030ba6SMark Murray r = (*db->seq)(db, &key, &data, sflag1); 3999b50d902SRodney W. Grimes if (r == -1) 400df3f5d9dSPeter Wemm err(1, "db seq"); 4019b50d902SRodney W. Grimes if (r == 1) 4029b50d902SRodney W. Grimes break; 403df3f5d9dSPeter Wemm memmove(&tmp, data.data, sizeof tmp); 404df3f5d9dSPeter Wemm enter_lastlog(tmp); 4059b50d902SRodney W. Grimes } 4069b50d902SRodney W. Grimes } 407