__stdio_read.c 738 B

123456789101112131415161718192021222324252627282930313233
  1. #include "stdio_impl.h"
  2. #include <sys/uio.h>
  3. #include <pthread.h>
  4. static void cleanup(void *p)
  5. {
  6. FILE *f = p;
  7. if (!f->lockcount) __unlockfile(f);
  8. }
  9. size_t __stdio_read(FILE *f, unsigned char *buf, size_t len)
  10. {
  11. struct iovec iov[2] = {
  12. { .iov_base = buf, .iov_len = len - !!f->buf_size },
  13. { .iov_base = f->buf, .iov_len = f->buf_size }
  14. };
  15. ssize_t cnt;
  16. pthread_cleanup_push(cleanup, f);
  17. cnt = syscall_cp(SYS_readv, f->fd, iov, 2);
  18. pthread_cleanup_pop(0);
  19. if (cnt <= 0) {
  20. f->flags |= F_EOF ^ ((F_ERR^F_EOF) & cnt);
  21. f->rpos = f->rend = 0;
  22. return cnt;
  23. }
  24. if (cnt <= iov[0].iov_len) return cnt;
  25. cnt -= iov[0].iov_len;
  26. f->rpos = f->buf;
  27. f->rend = f->buf + cnt;
  28. if (f->buf_size) buf[len-1] = *f->rpos++;
  29. return len;
  30. }