/* utent.c */ /* Let it be known that this is very possibly the worst standard ever. HP-UX does one thing, someone else does another, linux another... If anyone actually has the standard, please send it to me. Note that because of the way this stupid stupid standard works, you have to call endutent() to close the file even if you've not called setutent -- getutid and family use the same file descriptor. Modified by Erik Andersen for uClibc... */ #include #include #include #include #include #include #include #include #include #ifdef __UCLIBC_HAS_THREADS_NATIVE__ #include #endif /* Experimentally off - libc_hidden_proto(strcmp) */ /* Experimentally off - libc_hidden_proto(strdup) */ /* Experimentally off - libc_hidden_proto(strncmp) */ libc_hidden_proto(read) libc_hidden_proto(write) libc_hidden_proto(open) libc_hidden_proto(fcntl) libc_hidden_proto(close) libc_hidden_proto(lseek) #include __UCLIBC_MUTEX_STATIC(utmplock, PTHREAD_MUTEX_INITIALIZER); /* Some global crap */ static int static_fd = -1; static struct utmp static_utmp; static const char default_file_name[] = _PATH_UTMP; static const char *static_ut_name = (const char *) default_file_name; /* This function must be called with the LOCK held */ static void __setutent(void) { int ret; if (static_fd == -1) { #ifdef __UCLIBC_HAS_THREADS_NATIVE__ if ((static_fd = open_not_cancel_2(static_ut_name, O_RDWR)) < 0) { if ((static_fd = open_not_cancel_2(static_ut_name, O_RDONLY)) < 0) { #else if ((static_fd = open(static_ut_name, O_RDWR)) < 0) { if ((static_fd = open(static_ut_name, O_RDONLY)) < 0) { #endif goto bummer; } } /* Make sure the file will be closed on exec() */ #ifdef __UCLIBC_HAS_THREADS_NATIVE__ ret = fcntl_not_cancel(static_fd, F_GETFD, 0); #else ret = fcntl(static_fd, F_GETFD, 0); #endif if (ret >= 0) { #ifdef __UCLIBC_HAS_THREADS_NATIVE__ ret = fcntl_not_cancel(static_fd, F_SETFD, ret | FD_CLOEXEC); #else ret = fcntl(static_fd, F_SETFD, ret | FD_CLOEXEC); #endif } if (ret < 0) { bummer: static_fd = -1; #ifdef __UCLIBC_HAS_THREADS_NATIVE__ close_not_cancel_no_status(static_fd); #else close(static_fd); #endif return; } } lseek(static_fd, 0, SEEK_SET); return; } libc_hidden_proto(setutent) void setutent(void) { __UCLIBC_MUTEX_LOCK(utmplock); __setutent(); __UCLIBC_MUTEX_UNLOCK(utmplock); } libc_hidden_def(setutent) /* This function must be called with the LOCK held */ static struct utmp *__getutent(int utmp_fd) { struct utmp *ret = NULL; if (utmp_fd == -1) { __setutent(); } if (utmp_fd == -1) { return NULL; } #ifdef __UCLIBC_HAS_THREADS_NATIVE__ if (read_not_cancel(utmp_fd, (char *) &static_utmp, sizeof(struct utmp)) == sizeof(struct utmp)) #else if (read(utmp_fd, (char *) &static_utmp, sizeof(struct utmp)) == sizeof(struct utmp)) #endif { ret = &static_utmp; } return ret; } void endutent(void) { __UCLIBC_MUTEX_LOCK(utmplock); if (static_fd != -1) #ifdef __UCLIBC_HAS_THREADS_NATIVE__ close_not_cancel_no_status(static_fd); #else close(static_fd); #endif static_fd = -1; __UCLIBC_MUTEX_UNLOCK(utmplock); } struct utmp *getutent(void) { struct utmp *ret = NULL; __UCLIBC_MUTEX_LOCK(utmplock); ret = __getutent(static_fd); __UCLIBC_MUTEX_UNLOCK(utmplock); return ret; } /* This function must be called with the LOCK held */ static struct utmp *__getutid(const struct utmp *utmp_entry) { struct utmp *lutmp; while ((lutmp = __getutent(static_fd)) != NULL) { if ( (utmp_entry->ut_type == RUN_LVL || utmp_entry->ut_type == BOOT_TIME || utmp_entry->ut_type == NEW_TIME || utmp_entry->ut_type == OLD_TIME) && lutmp->ut_type == utmp_entry->ut_type) { return lutmp; } if ( (utmp_entry->ut_type == INIT_PROCESS || utmp_entry->ut_type == DEAD_PROCESS || utmp_entry->ut_type == LOGIN_PROCESS || utmp_entry->ut_type == USER_PROCESS) && !strncmp(lutmp->ut_id, utmp_entry->ut_id, sizeof(lutmp->ut_id))) { return lutmp; } } return NULL; } libc_hidden_proto(getutid) struct utmp *getutid(const struct utmp *utmp_entry) { struct utmp *ret = NULL; __UCLIBC_MUTEX_LOCK(utmplock); ret = __getutid(utmp_entry); __UCLIBC_MUTEX_UNLOCK(utmplock); return ret; } libc_hidden_def(getutid) struct utmp *getutline(const struct utmp *utmp_entry) { struct utmp *lutmp = NULL; __UCLIBC_MUTEX_LOCK(utmplock); while ((lutmp = __getutent(static_fd)) != NULL) { if ((lutmp->ut_type == USER_PROCESS || lutmp->ut_type == LOGIN_PROCESS) && !strcmp(lutmp->ut_line, utmp_entry->ut_line)) { break; } } __UCLIBC_MUTEX_UNLOCK(utmplock); return lutmp; } struct utmp *pututline (const struct utmp *utmp_entry) { __UCLIBC_MUTEX_LOCK(utmplock); /* Ignore the return value. That way, if they've already positioned the file pointer where they want it, everything will work out. */ lseek(static_fd, (off_t) - sizeof(struct utmp), SEEK_CUR); if (__getutid(utmp_entry) != NULL) lseek(static_fd, (off_t) - sizeof(struct utmp), SEEK_CUR); else lseek(static_fd, (off_t) 0, SEEK_END); #ifdef __UCLIBC_HAS_THREADS_NATIVE__ if (write(static_fd, utmp_entry, sizeof(struct utmp)) != sizeof(struct utmp)) #else if (write_not_cancel(static_fd, utmp_entry, sizeof(struct utmp)) != sizeof(struct utmp)) #endif utmp_entry = NULL; __UCLIBC_MUTEX_UNLOCK(utmplock); return (struct utmp *)utmp_entry; } int utmpname (const char *new_ut_name) { __UCLIBC_MUTEX_LOCK(utmplock); if (new_ut_name != NULL) { if (static_ut_name != default_file_name) free((char *)static_ut_name); static_ut_name = strdup(new_ut_name); if (static_ut_name == NULL) { /* We should probably whine about out-of-memory * errors here... Instead just reset to the default */ static_ut_name = default_file_name; } } if (static_fd != -1) #ifdef __UCLIBC_HAS_THREADS_NATIVE__ close_not_cancel_no_status(static_fd); #else close(static_fd); #endif static_fd = -1; __UCLIBC_MUTEX_UNLOCK(utmplock); return 0; }