fix copy/paste error in popen changes that broke signals
[musl] / src / stdio / __stdio_read.c
index d9bb326..c99ca9a 100644 (file)
@@ -1,6 +1,36 @@
 #include "stdio_impl.h"
+#include <pthread.h>
+
+static void cleanup(void *p)
+{
+       FILE *f = p;
+       if (!f->lockcount) __unlockfile(f);
+}
 
 size_t __stdio_read(FILE *f, unsigned char *buf, size_t len)
 {
-       return syscall(SYS_read, f->fd, buf, len);
+       struct iovec iov[2] = {
+               { .iov_base = buf, .iov_len = len - !!f->buf_size },
+               { .iov_base = f->buf, .iov_len = f->buf_size }
+       };
+       ssize_t cnt;
+
+       if (libc.main_thread) {
+               pthread_cleanup_push(cleanup, f);
+               cnt = syscall_cp(SYS_readv, f->fd, iov, 2);
+               pthread_cleanup_pop(0);
+       } else {
+               cnt = syscall(SYS_readv, f->fd, iov, 2);
+       }
+       if (cnt <= 0) {
+               f->flags |= F_EOF ^ ((F_ERR^F_EOF) & cnt);
+               f->rpos = f->rend = 0;
+               return cnt;
+       }
+       if (cnt <= iov[0].iov_len) return cnt;
+       cnt -= iov[0].iov_len;
+       f->rpos = f->buf;
+       f->rend = f->buf + cnt;
+       if (f->buf_size) buf[len-1] = *f->rpos++;
+       return len;
 }