Updated to fedora-glibc-20041124T0741
[kopensolaris-gnu/glibc.git] / sysdeps / unix / sysv / linux / mips / pread.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 Ulrich Drepper <drepper@cygnus.com>, 1997.
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 <assert.h>
22 #include <errno.h>
23 #include <sgidefs.h>
24 #include <unistd.h>
25 #include <endian.h>
26
27 #include <sysdep-cancel.h>
28 #include <sys/syscall.h>
29 #include <bp-checks.h>
30
31 #include <kernel-features.h>
32 #ifndef NO_SGIDEFS_H
33 #include <sgidefs.h>
34 #endif
35
36 #ifdef __NR_pread64             /* Newer kernels renamed but it's the same.  */
37 # ifdef __NR_pread
38 #  error "__NR_pread and __NR_pread64 both defined???"
39 # endif
40 # define __NR_pread __NR_pread64
41 #endif
42
43 #if defined __NR_pread || __ASSUME_PREAD_SYSCALL > 0
44
45 # if __ASSUME_PREAD_SYSCALL == 0
46 static ssize_t __emulate_pread (int fd, void *buf, size_t count,
47                                 off_t offset) internal_function;
48 # endif
49
50 ssize_t
51 __libc_pread (fd, buf, count, offset)
52      int fd;
53      void *buf;
54      size_t count;
55      off_t offset;
56 {
57   ssize_t result;
58
59 #if _MIPS_SIM != _ABI64
60   assert (sizeof (offset) == 4);
61 #endif
62
63   if (SINGLE_THREAD_P)
64     {
65       /* First try the syscall.  */
66 #if _MIPS_SIM == _ABIN32 || _MIPS_SIM == _ABI64
67       result = INLINE_SYSCALL (pread, 4, fd, CHECK_N (buf, count), count,
68                                offset);
69 #else
70       result = INLINE_SYSCALL (pread, 6, fd, CHECK_N (buf, count), count, 0,
71                                __LONG_LONG_PAIR (offset >> 31, offset));
72 #endif
73 # if __ASSUME_PREAD_SYSCALL == 0
74       if (result == -1 && errno == ENOSYS)
75         /* No system call available.  Use the emulation.  */
76         result = __emulate_pread (fd, buf, count, offset);
77 # endif
78       return result;
79     }
80
81   int oldtype = LIBC_CANCEL_ASYNC ();
82
83   /* First try the syscall.  */
84 #if _MIPS_SIM == _ABIN32 || _MIPS_SIM == _ABI64
85   result = INLINE_SYSCALL (pread, 4, fd, CHECK_N (buf, count), count, offset);
86 #else
87   result = INLINE_SYSCALL (pread, 6, fd, CHECK_N (buf, count), count, 0,
88                            __LONG_LONG_PAIR (offset >> 31, offset));
89 #endif
90 # if __ASSUME_PREAD_SYSCALL == 0
91   if (result == -1 && errno == ENOSYS)
92     /* No system call available.  Use the emulation.  */
93     result = __emulate_pread (fd, buf, count, offset);
94 # endif
95
96   LIBC_CANCEL_RESET (oldtype);
97
98   return result;
99 }
100
101 strong_alias (__libc_pread, __pread)
102 weak_alias (__libc_pread, pread)
103
104 # define __libc_pread(fd, buf, count, offset) \
105      static internal_function __emulate_pread (fd, buf, count, offset)
106 #endif
107
108 #if __ASSUME_PREAD_SYSCALL == 0
109 # include <sysdeps/posix/pread.c>
110 #endif