Merge from vendor branch BSDTAR:
[dragonfly.git] / contrib / lukemftpd / src / logutmp.c
1 /*
2  * Portions Copyright (c) 1988, 1993
3  *      The Regents of the University of California.  All rights reserved.
4  * Portions Copyright (c) 1996, Jason Downs.  All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  * 3. All advertising materials mentioning features or use of this software
15  *    must display the following acknowledgement:
16  *      This product includes software developed by the University of
17  *      California, Berkeley and its contributors.
18  * 4. Neither the name of the University nor the names of its contributors
19  *    may be used to endorse or promote products derived from this software
20  *    without specific prior written permission.
21  *
22  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32  * SUCH DAMAGE.
33  */
34
35 #include "lukemftpd.h"
36
37 typedef struct utmp UTMP;
38
39 static int fd = -1;
40 static int topslot = -1;
41
42 /*
43  * Special versions of login()/logout() which hold the utmp file open,
44  * for use with ftpd.
45  */
46
47 void
48 login(const UTMP *ut)
49 {
50         UTMP ubuf;
51
52         /*
53          * First, loop through /etc/ttys, if needed, to initialize the
54          * top of the tty slots, since ftpd has no tty.
55          */
56         if (topslot < 0) {
57                 topslot = 0;
58                 while (getttyent() != (struct ttyent *)NULL)
59                         topslot++;
60         }
61         if ((topslot < 0) || ((fd < 0)
62             && (fd = open(_PATH_UTMP, O_RDWR|O_CREAT, 0644)) < 0))
63                 return;
64
65         /*
66          * Now find a slot that's not in use...
67          */
68         (void)lseek(fd, (off_t)(topslot * sizeof(UTMP)), SEEK_SET);
69
70         while (1) {
71                 if (read(fd, &ubuf, sizeof(UTMP)) == sizeof(UTMP)) {
72                         if (!ubuf.ut_name[0]) {
73                                 (void)lseek(fd, -(off_t)sizeof(UTMP), SEEK_CUR);
74                                 break;
75                         }
76                         topslot++;
77                 } else {
78                         (void)lseek(fd, (off_t)(topslot * sizeof(UTMP)),
79                             SEEK_SET);
80                         break;
81                 }
82         }
83
84         (void)write(fd, ut, sizeof(UTMP));
85 }
86
87 int
88 logout(const char *line)
89 {
90         UTMP ut;
91         int rval;
92
93         rval = 0;
94         if (fd < 0)
95                 return(rval);
96
97         (void)lseek(fd, 0, SEEK_SET);
98
99         while (read(fd, &ut, sizeof(UTMP)) == sizeof(UTMP)) {
100                 if (!ut.ut_name[0]
101                     || strncmp(ut.ut_line, line, UT_LINESIZE))
102                         continue;
103                 memset(ut.ut_name, 0, UT_NAMESIZE);
104                 memset(ut.ut_host, 0, UT_HOSTSIZE);
105                 (void)time(&ut.ut_time);
106                 (void)lseek(fd, -(off_t)sizeof(UTMP), SEEK_CUR);
107                 (void)write(fd, &ut, sizeof(UTMP));
108                 rval = 1;
109         }
110         return(rval);
111 }