lseek.c 1.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061
  1. /* vi: set sw=4 ts=4: */
  2. /*
  3. * lseek() for uClibc
  4. *
  5. * Copyright (C) 2000-2006 Erik Andersen <andersen@uclibc.org>
  6. *
  7. * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
  8. */
  9. #include <sys/syscall.h>
  10. #include <unistd.h>
  11. #include <cancel.h>
  12. #ifdef __NR_lseek
  13. # define __NR___lseek_nocancel __NR_lseek
  14. _syscall3(off_t, __NC(lseek), int, fd, off_t, offset, int, whence)
  15. /* Use lseek64 if __NR_lseek is not defined but UCLIBC_HAS_LFS is enabled */
  16. #elif !defined __NR_lseek && defined __NR_llseek
  17. #include <endian.h>
  18. off_t __NC(lseek)(int fd, off_t offset, int whence)
  19. {
  20. #if defined __UCLIBC_HAS_LFS__
  21. return lseek64(fd, offset, whence);
  22. #elif __WORDSIZE == 32
  23. __off64_t result;
  24. __off_t high = 0;
  25. return INLINE_SYSCALL(llseek, 5, fd, high, offset, &result, whence) ?: result;
  26. #endif
  27. /* No need to handle __WORDSIZE == 64 as such a kernel won't define __NR_llseek */
  28. }
  29. #else
  30. # include <errno.h>
  31. off_t __NC(lseek)(int fd, off_t offset attribute_unused, int whence)
  32. {
  33. if (fd < 0) {
  34. __set_errno(EBADF);
  35. return -1;
  36. }
  37. switch(whence) {
  38. case SEEK_SET:
  39. case SEEK_CUR:
  40. case SEEK_END:
  41. break;
  42. default:
  43. __set_errno(EINVAL);
  44. return -1;
  45. }
  46. __set_errno(ENOSYS);
  47. return -1;
  48. }
  49. #endif
  50. CANCELLABLE_SYSCALL(off_t, lseek, (int fd, off_t offset, int whence), (fd, offset, whence))
  51. lt_libc_hidden(lseek)
  52. #if defined __UCLIBC_HAS_LFS__ && (__WORDSIZE == 64 || (!defined __NR__llseek && !defined __NR_llseek))
  53. strong_alias_untyped(__NC(lseek),__NC(lseek64))
  54. strong_alias_untyped(lseek,lseek64)
  55. lt_strong_alias(lseek64)
  56. lt_libc_hidden(lseek64)
  57. #endif