posix_fadvise64.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  1. /* vi: set sw=4 ts=4: */
  2. /*
  3. * posix_fadvise64() for uClibc
  4. * http://www.opengroup.org/onlinepubs/009695399/functions/posix_fadvise.html
  5. *
  6. * Copyright (C) 2000-2006 Erik Andersen <andersen@uclibc.org>
  7. *
  8. * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
  9. */
  10. #include <features.h>
  11. #include <unistd.h>
  12. #include <errno.h>
  13. #include <endian.h>
  14. #include <stdint.h>
  15. #include <sys/types.h>
  16. #include <sys/syscall.h>
  17. #include <fcntl.h>
  18. #ifdef __UCLIBC_HAS_LFS__
  19. #ifdef __NR_fadvise64_64
  20. /* 64 bit implementation is cake ... or more like pie ... */
  21. #if __WORDSIZE == 64
  22. #define __NR_posix_fadvise64 __NR_fadvise64_64
  23. #ifdef INTERNAL_SYSCALL
  24. int posix_fadvise64(int fd, __off64_t offset, __off64_t len, int advice)
  25. {
  26. if (len != (off_t) len)
  27. return EOVERFLOW;
  28. INTERNAL_SYSCALL_DECL (err);
  29. int ret = INTERNAL_SYSCALL (posix_fadvise64, err, 5, fd,
  30. __LONG_LONG_PAIR ((long) (offset >> 32),
  31. (long) offset),
  32. (off_t) len, advice);
  33. if (!INTERNAL_SYSCALL_ERROR_P (ret, err))
  34. return 0;
  35. return INTERNAL_SYSCALL_ERRNO (ret, err);
  36. }
  37. #else
  38. _syscall4(int, posix_fadvise64, int, fd, __off64_t, offset,
  39. __off64_t, len, int, advice);
  40. #endif
  41. /* 32 bit implementation is kind of a pita */
  42. #elif __WORDSIZE == 32
  43. #ifdef _syscall6 /* workaround until everyone has _syscall6() */
  44. #define __NR___syscall_fadvise64_64 __NR_fadvise64_64
  45. static __inline__ _syscall6(int, __syscall_fadvise64_64, int, fd,
  46. unsigned long, high_offset, unsigned long, low_offset,
  47. unsigned long, high_len, unsigned long, low_len,
  48. int, advice);
  49. int posix_fadvise64(int fd, __off64_t offset, __off64_t len, int advice)
  50. {
  51. return (__syscall_fadvise64_64(fd,
  52. __LONG_LONG_PAIR(offset >> 32, offset & 0xffffffff),
  53. __LONG_LONG_PAIR(len >> 32, len & 0xffffffff),
  54. advice));
  55. }
  56. #else
  57. #warning _syscall6 has not been defined for your machine :(
  58. #endif /* _syscall6 */
  59. #else
  60. #error your machine is neither 32 bit or 64 bit ... it must be magical
  61. #endif
  62. #elif !defined __NR_fadvise64
  63. /* This is declared as a strong alias in posix_fadvise.c if __NR_fadvise64
  64. * is defined.
  65. */
  66. int posix_fadvise64(int fd, __off64_t offset, __off64_t len, int advice)
  67. {
  68. #warning This is not correct as far as SUSv3 is concerned.
  69. return ENOSYS;
  70. }
  71. #endif /* __NR_fadvise64_64 */
  72. #endif /* __UCLIBC_HAS_LFS__ */