#include "stdio_impl.h" #include #include #include #include #include #include #include "libc.h" FILE *__fdopen(int fd, const char *mode) { FILE *f; struct winsize wsz; pthread_mutex_t filelockinit = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP; /* Check for valid initial mode character */ if (!strchr("rwa", *mode)) { errno = EINVAL; return 0; } /* Allocate FILE+buffer or fail */ if (!(f=malloc(sizeof *f + UNGET + BUFSIZ + sizeof(pthread_mutex_t)))) return 0; /* Zero-fill only the struct, not the buffer */ memset(f, 0, sizeof *f); /* Impose mode restrictions */ if (!strchr(mode, '+')) f->flags = (*mode == 'r') ? F_NOWR : F_NORD; /* Apply close-on-exec flag */ if (strchr(mode, 'e')) fcntl(fd, F_SETFD, FD_CLOEXEC); /* Set append mode on fd if opened for append */ if (*mode == 'a') { int flags = fcntl(fd, F_GETFL); if (!(flags & O_APPEND)) fcntl(fd, F_SETFL, flags | O_APPEND); f->flags |= F_APP; } f->fd = fd; f->buf = (unsigned char *)f + sizeof *f + UNGET; f->buf_size = BUFSIZ; f->lock = (pthread_mutex_t *)((unsigned char *)f + sizeof *f + UNGET + BUFSIZ); memcpy(f->lock, &filelockinit, sizeof(pthread_mutex_t)); /* Activate line buffered mode for terminals */ f->lbf = EOF; if (!(f->flags & F_NOWR) && !ioctl(fd, TIOCGWINSZ, &wsz)) f->lbf = '\n'; /* Initialize op ptrs. No problem if some are unneeded. */ f->read = __stdio_read; f->write = __stdio_write; f->seek = __stdio_seek; f->close = __stdio_close; /* Add new FILE to open file list */ return __ofl_add(f); } weak_alias(__fdopen, fdopen);