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