chiark / gitweb /
eglibc (2.11.3-4+deb6u3) squeeze-lts; urgency=medium
[eglibc.git] / ports / sysdeps / unix / sysv / linux / mips / pwrite64.c
1 /* Copyright (C) 1997, 1998, 2000, 2002, 2003, 2004
2    Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4    Contributed by Ralf Baechle <ralf@gnu.org>, 1998.
5
6    The GNU C Library is free software; you can redistribute it and/or
7    modify it under the terms of the GNU Lesser General Public
8    License as published by the Free Software Foundation; either
9    version 2.1 of the License, or (at your option) any later version.
10
11    The GNU C Library is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14    Lesser General Public License for more details.
15
16    You should have received a copy of the GNU Lesser General Public
17    License along with the GNU C Library; if not, write to the Free
18    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
19    02111-1307 USA.  */
20
21 #include <errno.h>
22 #ifndef NO_SGIDEFS_H
23 #include <sgidefs.h>
24 #endif
25 #include <unistd.h>
26 #include <endian.h>
27
28 #include <sysdep-cancel.h>
29 #include <sys/syscall.h>
30 #include <bp-checks.h>
31
32 #include <kernel-features.h>
33
34 #ifdef __NR_pwrite64            /* Newer kernels renamed but it's the same.  */
35 # ifdef __NR_pwrite
36 #  error "__NR_pwrite and __NR_pwrite64 both defined???"
37 # endif
38 # define __NR_pwrite __NR_pwrite64
39 #endif
40
41 #if defined __NR_pwrite || __ASSUME_PWRITE_SYSCALL > 0
42
43 # if __ASSUME_PWRITE_SYSCALL == 0
44 static ssize_t __emulate_pwrite64 (int fd, const void *buf, size_t count,
45                                    off64_t offset) internal_function;
46 # endif
47
48 ssize_t
49 __libc_pwrite64 (fd, buf, count, offset)
50      int fd;
51      const void *buf;
52      size_t count;
53      off64_t offset;
54 {
55   ssize_t result;
56
57   if (SINGLE_THREAD_P)
58     {
59      /* First try the syscall.  */
60 #if _MIPS_SIM == _ABIN32 || _MIPS_SIM == _ABI64
61       result = INLINE_SYSCALL (pwrite, 4, fd, CHECK_N (buf, count), count,
62                                offset);
63 #else
64      result = INLINE_SYSCALL (pwrite, 6, fd, CHECK_N (buf, count), count, 0,
65                               __LONG_LONG_PAIR ((off_t) (offset >> 32),
66                              (off_t) (offset & 0xffffffff)));
67 #endif
68 # if __ASSUME_PWRITE_SYSCALL == 0
69      if (result == -1 && errno == ENOSYS)
70      /* No system call available.  Use the emulation.  */
71      result = __emulate_pwrite64 (fd, buf, count, offset);
72 # endif
73
74      return result;
75     }
76
77   int oldtype = LIBC_CANCEL_ASYNC ();
78
79   /* First try the syscall.  */
80 #if _MIPS_SIM == _ABIN32 || _MIPS_SIM == _ABI64
81   result = INLINE_SYSCALL (pwrite, 4, fd, CHECK_N (buf, count), count, offset);
82 #else
83   result = INLINE_SYSCALL (pwrite, 6, fd, CHECK_N (buf, count), count, 0,
84                            __LONG_LONG_PAIR ((off_t) (offset >> 32),
85                                              (off_t) (offset & 0xffffffff)));
86 #endif
87 # if __ASSUME_PWRITE_SYSCALL == 0
88   if (result == -1 && errno == ENOSYS)
89     /* No system call available.  Use the emulation.  */
90     result = __emulate_pwrite64 (fd, buf, count, offset);
91 # endif
92
93   LIBC_CANCEL_RESET (oldtype);
94
95   return result;
96 }
97
98 weak_alias (__libc_pwrite64, __pwrite64)
99 libc_hidden_weak (__pwrite64)
100 weak_alias (__libc_pwrite64, pwrite64)
101
102 # define __libc_pwrite64(fd, buf, count, offset) \
103      static internal_function __emulate_pwrite64 (fd, buf, count, offset)
104 #endif
105
106 #if __ASSUME_PWRITE_SYSCALL == 0
107 # include <sysdeps/posix/pwrite64.c>
108 #endif