Home | History | Annotate | Line # | Download | only in who
utmpentry.c revision 1.15
      1 /*	$NetBSD: utmpentry.c,v 1.15 2008/07/13 20:07:48 dholland Exp $	*/
      2 
      3 /*-
      4  * Copyright (c) 2002 The NetBSD Foundation, Inc.
      5  * All rights reserved.
      6  *
      7  * This code is derived from software contributed to The NetBSD Foundation
      8  * by Christos Zoulas.
      9  *
     10  * Redistribution and use in source and binary forms, with or without
     11  * modification, are permitted provided that the following conditions
     12  * are met:
     13  * 1. Redistributions of source code must retain the above copyright
     14  *    notice, this list of conditions and the following disclaimer.
     15  * 2. Redistributions in binary form must reproduce the above copyright
     16  *    notice, this list of conditions and the following disclaimer in the
     17  *    documentation and/or other materials provided with the distribution.
     18  *
     19  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     20  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     21  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     22  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     23  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     24  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     25  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     26  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     27  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     28  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     29  * POSSIBILITY OF SUCH DAMAGE.
     30  */
     31 
     32 #include <sys/cdefs.h>
     33 #ifndef lint
     34 __RCSID("$NetBSD: utmpentry.c,v 1.15 2008/07/13 20:07:48 dholland Exp $");
     35 #endif
     36 
     37 #include <sys/stat.h>
     38 
     39 #include <time.h>
     40 #include <string.h>
     41 #include <err.h>
     42 #include <stdlib.h>
     43 
     44 #ifdef SUPPORT_UTMP
     45 #include <utmp.h>
     46 #endif
     47 #ifdef SUPPORT_UTMPX
     48 #include <utmpx.h>
     49 #endif
     50 
     51 #include "utmpentry.h"
     52 
     53 
     54 /* Fail the compile if x is not true, by constructing an illegal type. */
     55 #define COMPILE_ASSERT(x) ((void)sizeof(struct { unsigned : ((x) ? 1 : -1); }))
     56 
     57 
     58 #ifdef SUPPORT_UTMP
     59 static void getentry(struct utmpentry *, struct utmp *);
     60 static struct timespec utmptime = {0, 0};
     61 #endif
     62 #ifdef SUPPORT_UTMPX
     63 static void getentryx(struct utmpentry *, struct utmpx *);
     64 static struct timespec utmpxtime = {0, 0};
     65 #endif
     66 #if defined(SUPPORT_UTMPX) || defined(SUPPORT_UTMP)
     67 static int setup(const char *);
     68 static void adjust_size(struct utmpentry *e);
     69 #endif
     70 
     71 int maxname = 8, maxline = 8, maxhost = 16;
     72 int etype = 1 << USER_PROCESS;
     73 static int numutmp = 0;
     74 static struct utmpentry *ehead;
     75 
     76 #if defined(SUPPORT_UTMPX) || defined(SUPPORT_UTMP)
     77 static void
     78 adjust_size(struct utmpentry *e)
     79 {
     80 	int max;
     81 
     82 	if ((max = strlen(e->name)) > maxname)
     83 		maxname = max;
     84 	if ((max = strlen(e->line)) > maxline)
     85 		maxline = max;
     86 	if ((max = strlen(e->host)) > maxhost)
     87 		maxhost = max;
     88 }
     89 
     90 static int
     91 setup(const char *fname)
     92 {
     93 	int what = 3;
     94 	struct stat st;
     95 	const char *sfname;
     96 
     97 	if (fname == NULL) {
     98 #ifdef SUPPORT_UTMPX
     99 		setutxent();
    100 #endif
    101 #ifdef SUPPORT_UTMP
    102 		setutent();
    103 #endif
    104 	} else {
    105 		size_t len = strlen(fname);
    106 		if (len == 0)
    107 			errx(1, "Filename cannot be 0 length.");
    108 		what = fname[len - 1] == 'x' ? 1 : 2;
    109 		if (what == 1) {
    110 #ifdef SUPPORT_UTMPX
    111 			if (utmpxname(fname) == 0)
    112 				warnx("Cannot set utmpx file to `%s'",
    113 				    fname);
    114 #else
    115 			warnx("utmpx support not compiled in");
    116 #endif
    117 		} else {
    118 #ifdef SUPPORT_UTMP
    119 			if (utmpname(fname) == 0)
    120 				warnx("Cannot set utmp file to `%s'",
    121 				    fname);
    122 #else
    123 			warnx("utmp support not compiled in");
    124 #endif
    125 		}
    126 	}
    127 #ifdef SUPPORT_UTMPX
    128 	if (what & 1) {
    129 		sfname = fname ? fname : _PATH_UTMPX;
    130 		if (stat(sfname, &st) == -1) {
    131 			warn("Cannot stat `%s'", sfname);
    132 			what &= ~1;
    133 		} else {
    134 			if (timespeccmp(&st.st_mtimespec, &utmpxtime, >))
    135 			    utmpxtime = st.st_mtimespec;
    136 			else
    137 			    what &= ~1;
    138 		}
    139 	}
    140 #endif
    141 #ifdef SUPPORT_UTMP
    142 	if (what & 2) {
    143 		sfname = fname ? fname : _PATH_UTMP;
    144 		if (stat(sfname, &st) == -1) {
    145 			warn("Cannot stat `%s'", sfname);
    146 			what &= ~2;
    147 		} else {
    148 			if (timespeccmp(&st.st_mtimespec, &utmptime, >))
    149 				utmptime = st.st_mtimespec;
    150 			else
    151 				what &= ~2;
    152 		}
    153 	}
    154 #endif
    155 	return what;
    156 }
    157 #endif
    158 
    159 void
    160 endutentries(void)
    161 {
    162 	struct utmpentry *ep;
    163 
    164 #ifdef SUPPORT_UTMP
    165 	timespecclear(&utmptime);
    166 #endif
    167 #ifdef SUPPORT_UTMPX
    168 	timespecclear(&utmpxtime);
    169 #endif
    170 	ep = ehead;
    171 	while (ep) {
    172 		struct utmpentry *sep = ep;
    173 		ep = ep->next;
    174 		free(sep);
    175 	}
    176 	ehead = NULL;
    177 	numutmp = 0;
    178 }
    179 
    180 int
    181 getutentries(const char *fname, struct utmpentry **epp)
    182 {
    183 #ifdef SUPPORT_UTMPX
    184 	struct utmpx *utx;
    185 #endif
    186 #ifdef SUPPORT_UTMP
    187 	struct utmp *ut;
    188 #endif
    189 #if defined(SUPPORT_UTMP) || defined(SUPPORT_UTMPX)
    190 	struct utmpentry *ep;
    191 	int what = setup(fname);
    192 	struct utmpentry **nextp = &ehead;
    193 	switch (what) {
    194 	case 0:
    195 		/* No updates */
    196 		*epp = ehead;
    197 		return numutmp;
    198 	default:
    199 		/* Need to re-scan */
    200 		ehead = NULL;
    201 		numutmp = 0;
    202 	}
    203 #endif
    204 
    205 #ifdef SUPPORT_UTMPX
    206 	while ((what & 1) && (utx = getutxent()) != NULL) {
    207 		if (fname == NULL && ((1 << utx->ut_type) & etype) == 0)
    208 			continue;
    209 		if ((ep = calloc(1, sizeof(struct utmpentry))) == NULL) {
    210 			warn(NULL);
    211 			return 0;
    212 		}
    213 		getentryx(ep, utx);
    214 		*nextp = ep;
    215 		nextp = &(ep->next);
    216 	}
    217 #endif
    218 
    219 #ifdef SUPPORT_UTMP
    220 	if ((etype & (1 << USER_PROCESS)) != 0) {
    221 		while ((what & 2) && (ut = getutent()) != NULL) {
    222 			if (fname == NULL && (*ut->ut_name == '\0' ||
    223 			    *ut->ut_line == '\0'))
    224 				continue;
    225 			/* Don't process entries that we have utmpx for */
    226 			for (ep = ehead; ep != NULL; ep = ep->next) {
    227 				if (strncmp(ep->line, ut->ut_line,
    228 				    sizeof(ut->ut_line)) == 0)
    229 					break;
    230 			}
    231 			if (ep != NULL)
    232 				continue;
    233 			if ((ep = calloc(1, sizeof(*ep))) == NULL) {
    234 				warn(NULL);
    235 				return 0;
    236 			}
    237 			getentry(ep, ut);
    238 			*nextp = ep;
    239 			nextp = &(ep->next);
    240 		}
    241 	}
    242 #endif
    243 	numutmp = 0;
    244 #if defined(SUPPORT_UTMP) && defined(SUPPORT_UTMPX)
    245 	if (ehead != NULL) {
    246 		struct utmpentry *from = ehead, *save;
    247 
    248 		ehead = NULL;
    249 		while (from != NULL) {
    250 			for (nextp = &ehead;
    251 			    (*nextp) && strcmp(from->line, (*nextp)->line) > 0;
    252 			    nextp = &(*nextp)->next)
    253 				continue;
    254 			save = from;
    255 			from = from->next;
    256 			save->next = *nextp;
    257 			*nextp = save;
    258 			numutmp++;
    259 		}
    260 	}
    261 	*epp = ehead;
    262 	return numutmp;
    263 #else
    264 	*epp = NULL;
    265 	return 0;
    266 #endif
    267 }
    268 
    269 #ifdef SUPPORT_UTMP
    270 static void
    271 getentry(struct utmpentry *e, struct utmp *up)
    272 {
    273 	COMPILE_ASSERT(sizeof(e->name) > sizeof(up->ut_name));
    274 	COMPILE_ASSERT(sizeof(e->line) > sizeof(up->ut_line));
    275 	COMPILE_ASSERT(sizeof(e->host) > sizeof(up->ut_host));
    276 
    277 	/*
    278 	 * e has just been calloc'd. We don't need to clear it or
    279 	 * append null-terminators, because its length is strictly
    280 	 * greater than the source string. Use strncpy to _read_
    281 	 * up->ut_* because they may not be terminated. For this
    282 	 * reason we use the size of the _source_ as the length
    283 	 * argument.
    284 	 */
    285 	(void)strncpy(e->name, up->ut_name, sizeof(up->ut_name));
    286 	(void)strncpy(e->line, up->ut_line, sizeof(up->ut_line));
    287 	(void)strncpy(e->host, up->ut_host, sizeof(up->ut_host));
    288 
    289 	e->tv.tv_sec = up->ut_time;
    290 	e->tv.tv_usec = 0;
    291 	e->pid = 0;
    292 	e->term = 0;
    293 	e->exit = 0;
    294 	e->sess = 0;
    295 	e->type = USER_PROCESS;
    296 	adjust_size(e);
    297 }
    298 #endif
    299 
    300 #ifdef SUPPORT_UTMPX
    301 static void
    302 getentryx(struct utmpentry *e, struct utmpx *up)
    303 {
    304 	COMPILE_ASSERT(sizeof(e->name) > sizeof(up->ut_name));
    305 	COMPILE_ASSERT(sizeof(e->line) > sizeof(up->ut_line));
    306 	COMPILE_ASSERT(sizeof(e->host) > sizeof(up->ut_host));
    307 
    308 	/*
    309 	 * e has just been calloc'd. We don't need to clear it or
    310 	 * append null-terminators, because its length is strictly
    311 	 * greater than the source string. Use strncpy to _read_
    312 	 * up->ut_* because they may not be terminated. For this
    313 	 * reason we use the size of the _source_ as the length
    314 	 * argument.
    315 	 */
    316 	(void)strncpy(e->name, up->ut_name, sizeof(up->ut_name));
    317 	(void)strncpy(e->line, up->ut_line, sizeof(up->ut_line));
    318 	(void)strncpy(e->host, up->ut_host, sizeof(up->ut_host));
    319 
    320 	e->tv = up->ut_tv;
    321 	e->pid = up->ut_pid;
    322 	e->term = up->ut_exit.e_termination;
    323 	e->exit = up->ut_exit.e_exit;
    324 	e->sess = up->ut_session;
    325 	e->type = up->ut_type;
    326 	adjust_size(e);
    327 }
    328 #endif
    329