xref: /freebsd/lib/libc/gen/getutxent.c (revision 559a218c9b257775fb249b67945fe4a05b7a6b9f)
1a627ac61SEd Schouten /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3d915a14eSPedro F. Giffuni  *
4a627ac61SEd Schouten  * Copyright (c) 2010 Ed Schouten <ed@FreeBSD.org>
5a627ac61SEd Schouten  * All rights reserved.
6a627ac61SEd Schouten  *
7a627ac61SEd Schouten  * Redistribution and use in source and binary forms, with or without
8a627ac61SEd Schouten  * modification, are permitted provided that the following conditions
9a627ac61SEd Schouten  * are met:
10a627ac61SEd Schouten  * 1. Redistributions of source code must retain the above copyright
11a627ac61SEd Schouten  *    notice, this list of conditions and the following disclaimer.
12a627ac61SEd Schouten  * 2. Redistributions in binary form must reproduce the above copyright
13a627ac61SEd Schouten  *    notice, this list of conditions and the following disclaimer in the
14a627ac61SEd Schouten  *    documentation and/or other materials provided with the distribution.
15a627ac61SEd Schouten  *
16a627ac61SEd Schouten  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17a627ac61SEd Schouten  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18a627ac61SEd Schouten  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19a627ac61SEd Schouten  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20a627ac61SEd Schouten  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21a627ac61SEd Schouten  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22a627ac61SEd Schouten  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23a627ac61SEd Schouten  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24a627ac61SEd Schouten  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25a627ac61SEd Schouten  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26a627ac61SEd Schouten  * SUCH DAMAGE.
27a627ac61SEd Schouten  */
28a627ac61SEd Schouten 
29a627ac61SEd Schouten #include "namespace.h"
30a627ac61SEd Schouten #include <sys/endian.h>
31a627ac61SEd Schouten #include <sys/param.h>
32a627ac61SEd Schouten #include <sys/stat.h>
33a627ac61SEd Schouten #include <errno.h>
34a627ac61SEd Schouten #include <stdio.h>
35a627ac61SEd Schouten #include <string.h>
36a627ac61SEd Schouten #include <utmpx.h>
37a627ac61SEd Schouten #include "utxdb.h"
38a627ac61SEd Schouten #include "un-namespace.h"
39a627ac61SEd Schouten 
40126b6df9SEd Schouten static _Thread_local FILE *uf = NULL;
41126b6df9SEd Schouten static _Thread_local int udb;
42a627ac61SEd Schouten 
43a627ac61SEd Schouten int
setutxdb(int db,const char * file)44a627ac61SEd Schouten setutxdb(int db, const char *file)
45a627ac61SEd Schouten {
46a627ac61SEd Schouten 	struct stat sb;
47a627ac61SEd Schouten 
48a627ac61SEd Schouten 	switch (db) {
49a627ac61SEd Schouten 	case UTXDB_ACTIVE:
50a627ac61SEd Schouten 		if (file == NULL)
51a627ac61SEd Schouten 			file = _PATH_UTX_ACTIVE;
52a627ac61SEd Schouten 		break;
53a627ac61SEd Schouten 	case UTXDB_LASTLOGIN:
54a627ac61SEd Schouten 		if (file == NULL)
55a627ac61SEd Schouten 			file = _PATH_UTX_LASTLOGIN;
56a627ac61SEd Schouten 		break;
57a627ac61SEd Schouten 	case UTXDB_LOG:
58a627ac61SEd Schouten 		if (file == NULL)
59a627ac61SEd Schouten 			file = _PATH_UTX_LOG;
60a627ac61SEd Schouten 		break;
61a627ac61SEd Schouten 	default:
62a627ac61SEd Schouten 		errno = EINVAL;
63a627ac61SEd Schouten 		return (-1);
64a627ac61SEd Schouten 	}
65a627ac61SEd Schouten 
66a627ac61SEd Schouten 	if (uf != NULL)
67a627ac61SEd Schouten 		fclose(uf);
681084b38bSJilles Tjoelker 	uf = fopen(file, "re");
69a627ac61SEd Schouten 	if (uf == NULL)
70a627ac61SEd Schouten 		return (-1);
71a627ac61SEd Schouten 
72a8f77c1fSEd Schouten 	if (db != UTXDB_LOG) {
73a627ac61SEd Schouten 		/* Safety check: never use broken files. */
74a8f77c1fSEd Schouten 		if (_fstat(fileno(uf), &sb) != -1 &&
75a627ac61SEd Schouten 		    sb.st_size % sizeof(struct futx) != 0) {
76a627ac61SEd Schouten 			fclose(uf);
77a627ac61SEd Schouten 			uf = NULL;
78a627ac61SEd Schouten 			errno = EFTYPE;
79a627ac61SEd Schouten 			return (-1);
80a627ac61SEd Schouten 		}
81a8f77c1fSEd Schouten 		/* Prevent reading of partial records. */
82a8f77c1fSEd Schouten 		(void)setvbuf(uf, NULL, _IOFBF,
83a8f77c1fSEd Schouten 		    rounddown(BUFSIZ, sizeof(struct futx)));
84a8f77c1fSEd Schouten 	}
85a627ac61SEd Schouten 
86a627ac61SEd Schouten 	udb = db;
87a627ac61SEd Schouten 	return (0);
88a627ac61SEd Schouten }
89a627ac61SEd Schouten 
90a627ac61SEd Schouten void
setutxent(void)91a627ac61SEd Schouten setutxent(void)
92a627ac61SEd Schouten {
93a627ac61SEd Schouten 
94a627ac61SEd Schouten 	setutxdb(UTXDB_ACTIVE, NULL);
95a627ac61SEd Schouten }
96a627ac61SEd Schouten 
97a627ac61SEd Schouten void
endutxent(void)98a627ac61SEd Schouten endutxent(void)
99a627ac61SEd Schouten {
100a627ac61SEd Schouten 
101a627ac61SEd Schouten 	if (uf != NULL) {
102a627ac61SEd Schouten 		fclose(uf);
103a627ac61SEd Schouten 		uf = NULL;
104a627ac61SEd Schouten 	}
105a627ac61SEd Schouten }
106a627ac61SEd Schouten 
10798c63a48SEd Schouten static int
getfutxent(struct futx * fu)10898c63a48SEd Schouten getfutxent(struct futx *fu)
109a627ac61SEd Schouten {
110a627ac61SEd Schouten 
111a627ac61SEd Schouten 	if (uf == NULL)
112a627ac61SEd Schouten 		setutxent();
113a627ac61SEd Schouten 	if (uf == NULL)
11498c63a48SEd Schouten 		return (-1);
115a627ac61SEd Schouten 
116a627ac61SEd Schouten 	if (udb == UTXDB_LOG) {
117a627ac61SEd Schouten 		uint16_t len;
118a627ac61SEd Schouten 
119a1739165SGleb Smirnoff retry:
1201ae6a21dSEd Schouten 		if (fread(&len, sizeof(len), 1, uf) != 1)
12198c63a48SEd Schouten 			return (-1);
122a627ac61SEd Schouten 		len = be16toh(len);
123a1739165SGleb Smirnoff 		if (len == 0) {
124a1739165SGleb Smirnoff 			/*
125a1739165SGleb Smirnoff 			 * XXX: Though zero-size records are valid in theory,
126a1739165SGleb Smirnoff 			 * they can never occur in practice. Zero-size records
127a1739165SGleb Smirnoff 			 * indicate file corruption. Seek one byte forward, to
128a1739165SGleb Smirnoff 			 * see if we can find a record there.
129a1739165SGleb Smirnoff 			 */
130a1739165SGleb Smirnoff 			ungetc('\0', uf);
131a1739165SGleb Smirnoff 			goto retry;
132a1739165SGleb Smirnoff 		}
13398c63a48SEd Schouten 		if (len > sizeof *fu) {
134a627ac61SEd Schouten 			/* Forward compatibility. */
1351ae6a21dSEd Schouten 			if (fread(fu, sizeof(*fu), 1, uf) != 1)
13698c63a48SEd Schouten 				return (-1);
1371ae6a21dSEd Schouten 			fseek(uf, len - sizeof(*fu), SEEK_CUR);
138a627ac61SEd Schouten 		} else {
139a627ac61SEd Schouten 			/* Partial record. */
1401ae6a21dSEd Schouten 			memset(fu, 0, sizeof(*fu));
14198c63a48SEd Schouten 			if (fread(fu, len, 1, uf) != 1)
14298c63a48SEd Schouten 				return (-1);
143a627ac61SEd Schouten 		}
144a627ac61SEd Schouten 	} else {
1451ae6a21dSEd Schouten 		if (fread(fu, sizeof(*fu), 1, uf) != 1)
14698c63a48SEd Schouten 			return (-1);
147a627ac61SEd Schouten 	}
14898c63a48SEd Schouten 	return (0);
149a627ac61SEd Schouten }
150a627ac61SEd Schouten 
151a627ac61SEd Schouten struct utmpx *
getutxent(void)152a627ac61SEd Schouten getutxent(void)
153a627ac61SEd Schouten {
15498c63a48SEd Schouten 	struct futx fu;
155a627ac61SEd Schouten 
15698c63a48SEd Schouten 	if (getfutxent(&fu) != 0)
157a627ac61SEd Schouten 		return (NULL);
15898c63a48SEd Schouten 	return (futx_to_utx(&fu));
159a627ac61SEd Schouten }
160a627ac61SEd Schouten 
161a627ac61SEd Schouten struct utmpx *
getutxid(const struct utmpx * id)162a627ac61SEd Schouten getutxid(const struct utmpx *id)
163a627ac61SEd Schouten {
16498c63a48SEd Schouten 	struct futx fu;
165a627ac61SEd Schouten 
166a627ac61SEd Schouten 	for (;;) {
16798c63a48SEd Schouten 		if (getfutxent(&fu) != 0)
168a627ac61SEd Schouten 			return (NULL);
169a627ac61SEd Schouten 
17098c63a48SEd Schouten 		switch (fu.fu_type) {
171a627ac61SEd Schouten 		case USER_PROCESS:
172a627ac61SEd Schouten 		case INIT_PROCESS:
173a627ac61SEd Schouten 		case LOGIN_PROCESS:
174a627ac61SEd Schouten 		case DEAD_PROCESS:
175a627ac61SEd Schouten 			switch (id->ut_type) {
176a627ac61SEd Schouten 			case USER_PROCESS:
177a627ac61SEd Schouten 			case INIT_PROCESS:
178a627ac61SEd Schouten 			case LOGIN_PROCESS:
179a627ac61SEd Schouten 			case DEAD_PROCESS:
18098c63a48SEd Schouten 				if (memcmp(fu.fu_id, id->ut_id,
1811ae6a21dSEd Schouten 				    MIN(sizeof(fu.fu_id), sizeof(id->ut_id))) ==
1821ae6a21dSEd Schouten 				    0)
183a627ac61SEd Schouten 					goto found;
184a627ac61SEd Schouten 			}
185e35a88d3SEd Schouten 			break;
18698c63a48SEd Schouten 		default:
18798c63a48SEd Schouten 			if (fu.fu_type == id->ut_type)
18898c63a48SEd Schouten 				goto found;
18998c63a48SEd Schouten 			break;
190a627ac61SEd Schouten 		}
191a627ac61SEd Schouten 	}
192a627ac61SEd Schouten 
193a627ac61SEd Schouten found:
19498c63a48SEd Schouten 	return (futx_to_utx(&fu));
195a627ac61SEd Schouten }
196a627ac61SEd Schouten 
197a627ac61SEd Schouten struct utmpx *
getutxline(const struct utmpx * line)198a627ac61SEd Schouten getutxline(const struct utmpx *line)
199a627ac61SEd Schouten {
20098c63a48SEd Schouten 	struct futx fu;
201a627ac61SEd Schouten 
202a627ac61SEd Schouten 	for (;;) {
20398c63a48SEd Schouten 		if (getfutxent(&fu) != 0)
204a627ac61SEd Schouten 			return (NULL);
205a627ac61SEd Schouten 
20698c63a48SEd Schouten 		switch (fu.fu_type) {
207a627ac61SEd Schouten 		case USER_PROCESS:
208a627ac61SEd Schouten 		case LOGIN_PROCESS:
20998c63a48SEd Schouten 			if (strncmp(fu.fu_line, line->ut_line,
2101ae6a21dSEd Schouten 			    MIN(sizeof(fu.fu_line), sizeof(line->ut_line))) ==
2111ae6a21dSEd Schouten 			    0)
212a627ac61SEd Schouten 				goto found;
21398c63a48SEd Schouten 			break;
214a627ac61SEd Schouten 		}
215a627ac61SEd Schouten 	}
216a627ac61SEd Schouten 
217a627ac61SEd Schouten found:
21898c63a48SEd Schouten 	return (futx_to_utx(&fu));
219a627ac61SEd Schouten }
220a627ac61SEd Schouten 
221a627ac61SEd Schouten struct utmpx *
getutxuser(const char * user)222a627ac61SEd Schouten getutxuser(const char *user)
223a627ac61SEd Schouten {
22498c63a48SEd Schouten 	struct futx fu;
225a627ac61SEd Schouten 
226a627ac61SEd Schouten 	for (;;) {
22798c63a48SEd Schouten 		if (getfutxent(&fu) != 0)
228a627ac61SEd Schouten 			return (NULL);
229a627ac61SEd Schouten 
23098c63a48SEd Schouten 		switch (fu.fu_type) {
231a627ac61SEd Schouten 		case USER_PROCESS:
2321ae6a21dSEd Schouten 			if (strncmp(fu.fu_user, user, sizeof(fu.fu_user)) == 0)
233a627ac61SEd Schouten 				goto found;
23498c63a48SEd Schouten 			break;
235a627ac61SEd Schouten 		}
236a627ac61SEd Schouten 	}
237a627ac61SEd Schouten 
238a627ac61SEd Schouten found:
23998c63a48SEd Schouten 	return (futx_to_utx(&fu));
240a627ac61SEd Schouten }
241