/* * Copyright (c) 1988 The Regents of the University of California. * All rights reserved. * * Redistribution and use in source and binary forms are permitted * provided that the above copyright notice and this paragraph are * duplicated in all such forms and that any documentation, * advertising materials, and other materials related to such * distribution and use acknowledge that the software was developed * by the University of California, Berkeley. The name of the * University may not be used to endorse or promote products derived * from this software without specific prior written permission. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE. */ #include #include #include #include #include #include #include #ifdef NEED_SYS_FCNTL_H #include #endif #ifdef HAVE_UNISTD_H #include #endif #ifndef UTMP_FILE #define UTMP_FILE "/etc/utmp" #endif #ifndef WTMP_FILE #ifdef SYSV #define WTMPFILE "/etc/wtmp" #else #define WTMP_FILE "/usr/adm/wtmp" #endif #endif #ifndef EMPTY /* linux has UT_UNKNOWN but not EMPTY */ #define EMPTY UT_UNKNOWN #endif void login(ut) struct utmp *ut; { register int fd; struct utmp utmp; int tty; #if defined(HAVE_GETUTENT) && !defined(NO_UT_PID) if (!ut->ut_pid) ut->ut_pid = getppid(); ut->ut_type = USER_PROCESS; (void) strncpy(ut->ut_id, ut->ut_line, sizeof(ut->ut_id)); (void) setutent(); (void) memset((char *)&utmp, 0, sizeof(utmp)); (void) strncpy(utmp.ut_id, ut->ut_id, sizeof(utmp.ut_id)); utmp.ut_type = DEAD_PROCESS; (void) getutid(&utmp); (void) pututline(ut); (void) endutent(); #else tty = ttyslot(); if (tty > 0 && (fd = open(UTMP_FILE, O_WRONLY, 0)) >= 0) { (void)lseek(fd, (off_t)(tty * sizeof(struct utmp)), SEEK_SET); (void)write(fd, (char *)ut, sizeof(struct utmp)); (void)close(fd); } #endif if ((fd = open(WTMP_FILE, O_WRONLY|O_APPEND, 0)) >= 0) { (void)write(fd, (char *)ut, sizeof(struct utmp)); (void)close(fd); } } logout(line) register char *line; { register FILE *fp; struct utmp ut; int rval; if (!(fp = fopen(UTMP_FILE, "r+"))) return(0); rval = 1; while (fread((char *)&ut, sizeof(struct utmp), 1, fp) == 1) { if (!ut.ut_name[0] || strncmp(ut.ut_line, line, sizeof(ut.ut_line))) continue; memset(ut.ut_name,0, sizeof(ut.ut_name)); #ifndef NO_UT_HOST memset(ut.ut_host,0, sizeof(ut.ut_host)); #endif (void)time(&ut.ut_time); #if defined(HAVE_GETUTENT) && !defined(NO_UT_PID) memset(ut.ut_id, 0, sizeof(ut.ut_id)); ut.ut_pid = 0; #ifndef NO_UT_EXIT ut.ut_exit.e_exit = 0; #endif ut.ut_type = EMPTY; #endif (void)fseek(fp, (off_t)-sizeof(struct utmp), SEEK_CUR); (void)fwrite((char *)&ut, sizeof(struct utmp), 1, fp); (void)fseek(fp, (off_t)0, SEEK_CUR); rval = 0; } (void)fclose(fp); return(rval); } static int fd = -1; #ifndef SYSV logwtmp(line, name, host, keep_open) #else logwtmp(line, name, host, keep_open, logingin) #endif char *line, *name, *host; int keep_open; #ifdef SYSV int logingin; #endif { struct utmp ut; struct stat buf; time_t time(); char *strncpy(); if (fd < 0 && (fd = open(WTMP_FILE, O_WRONLY|O_APPEND, 0)) < 0) return; if (!fstat(fd, &buf)) { (void)strncpy(ut.ut_line, line, sizeof(ut.ut_line)); (void)strncpy(ut.ut_name, name, sizeof(ut.ut_name)); #ifndef NO_UT_HOST (void)strncpy(ut.ut_host, host, sizeof(ut.ut_host)); #endif #ifdef SYSV (void)strncpy(ut.ut_id, (char *)ut.ut_line + 4, sizeof(ut.ut_id)); ut.ut_type = logingin ? USER_PROCESS : DEAD_PROCESS; ut.ut_pid = getpid(); #endif (void)time(&ut.ut_time); #if defined(HAVE_GETUTENT) && !defined(NO_UT_PID) if (*name) { if (!ut.ut_pid) ut.ut_pid = getpid(); ut.ut_type = USER_PROCESS; } else { ut.ut_type = EMPTY; } #endif if (write(fd, (char *)&ut, sizeof(struct utmp)) != sizeof(struct utmp)) (void)ftruncate(fd, buf.st_size); } if ( !keep_open) (void)close(fd); }