lockf64.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. /* Copyright (C) 1994, 1996, 1997, 1998, 2000 Free Software Foundation, Inc.
  2. This file is part of the GNU C Library.
  3. The GNU C Library is free software; you can redistribute it and/or
  4. modify it under the terms of the GNU Library General Public License as
  5. published by the Free Software Foundation; either version 2 of the
  6. License, or (at your option) any later version.
  7. The GNU C Library is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  10. Library General Public License for more details.
  11. You should have received a copy of the GNU Library General Public
  12. License along with the GNU C Library; see the file COPYING.LIB. If not,
  13. write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  14. Boston, MA 02111-1307, USA. */
  15. #include <_lfs_64.h>
  16. #include <sys/types.h>
  17. #include <unistd.h>
  18. #include <fcntl.h>
  19. #include <errno.h>
  20. #include <string.h>
  21. #include <sys/syscall.h>
  22. #ifdef __NR_fcntl64
  23. #define flock flock64
  24. #define fcntl fcntl64
  25. #undef F_GETLK
  26. #define F_GETLK F_GETLK64
  27. #undef F_SETLK
  28. #define F_SETLK F_SETLK64
  29. #else
  30. #endif
  31. /* lockf is a simplified interface to fcntl's locking facilities. */
  32. int lockf64 (int fd, int cmd, off64_t len64)
  33. {
  34. struct flock fl;
  35. off_t len = (off_t) len64;
  36. if (len64 != (off64_t) len)
  37. {
  38. /* We can't represent the length. */
  39. __set_errno(EOVERFLOW);
  40. return -1;
  41. }
  42. memset((char *) &fl, '\0', sizeof (fl));
  43. /* lockf is always relative to the current file position. */
  44. fl.l_whence = SEEK_CUR;
  45. fl.l_start = 0;
  46. fl.l_len = len;
  47. switch (cmd)
  48. {
  49. case F_TEST:
  50. /* Test the lock: return 0 if FD is unlocked or locked by this process;
  51. return -1, set errno to EACCES, if another process holds the lock. */
  52. fl.l_type = F_RDLCK;
  53. if (fcntl (fd, F_GETLK, &fl) < 0)
  54. return -1;
  55. if (fl.l_type == F_UNLCK || fl.l_pid == getpid ())
  56. return 0;
  57. __set_errno(EACCES);
  58. return -1;
  59. case F_ULOCK:
  60. fl.l_type = F_UNLCK;
  61. cmd = F_SETLK;
  62. break;
  63. case F_LOCK:
  64. fl.l_type = F_WRLCK;
  65. cmd = F_SETLKW;
  66. break;
  67. case F_TLOCK:
  68. fl.l_type = F_WRLCK;
  69. cmd = F_SETLK;
  70. break;
  71. default:
  72. __set_errno(EINVAL);
  73. return -1;
  74. }
  75. return fcntl(fd, cmd, &fl);
  76. }
  77. libc_hidden_def(lockf64)