Whamcloud - gitweb
LU-2673 procfs: call lprocfs_free_xxx_stats() later
[fs/lustre-release.git] / libsysio / src / lseek.c
1 /*
2  *    This Cplant(TM) source code is the property of Sandia National
3  *    Laboratories.
4  *
5  *    This Cplant(TM) source code is copyrighted by Sandia National
6  *    Laboratories.
7  *
8  *    The redistribution of this Cplant(TM) source code is subject to the
9  *    terms of the GNU Lesser General Public License
10  *    (see cit/LGPL or http://www.gnu.org/licenses/lgpl.html)
11  *
12  *    Cplant(TM) Copyright 1998-2005 Sandia Corporation. 
13  *    Under the terms of Contract DE-AC04-94AL85000, there is a non-exclusive
14  *    license for use of this work by or on behalf of the US Government.
15  *    Export of this program may require a license from the United States
16  *    Government.
17  */
18
19 /*
20  * This library is free software; you can redistribute it and/or
21  * modify it under the terms of the GNU Lesser General Public
22  * License as published by the Free Software Foundation; either
23  * version 2.1 of the License, or (at your option) any later version.
24  * 
25  * This library is distributed in the hope that it will be useful,
26  * but WITHOUT ANY WARRANTY; without even the implied warranty of
27  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
28  * Lesser General Public License for more details.
29  * 
30  * You should have received a copy of the GNU Lesser General Public
31  * License along with this library; if not, write to the Free Software
32  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
33  *
34  * Questions or comments about this library should be sent to:
35  *
36  * Lee Ward
37  * Sandia National Laboratories, New Mexico
38  * P.O. Box 5800
39  * Albuquerque, NM 87185-1110
40  *
41  * lee@sandia.gov
42  */
43
44 #include <errno.h>
45 #include <unistd.h>
46 #include <assert.h>
47 #include <sys/types.h>
48 #include <sys/stat.h>
49 #include <fcntl.h>
50 #include <sys/queue.h>
51
52 #include "sysio.h"
53 #include "inode.h"
54 #include "file.h"
55
56 #include "sysio-symbols.h"
57
58 _SYSIO_OFF_T
59 _sysio_lseek_prepare(struct file *fil,
60                      _SYSIO_OFF_T offset,
61                      int whence,
62                      _SYSIO_OFF_T max)
63 {
64         _SYSIO_OFF_T off, pos;
65         struct intnl_stat stbuf;
66
67         off = -1;
68         switch (whence) {
69         
70         case SEEK_SET:
71                 off = 0;
72                 break;
73         case SEEK_CUR:
74                 off = fil->f_pos;
75                 break;
76         case SEEK_END:
77                 {
78                         int     err;
79
80                         /*
81                          * Don't blindly trust the attributes
82                          * in the inode record for this. Give the
83                          * driver a chance to refresh them.
84                          */
85                         err =
86                             (*fil->f_ino->i_ops.inop_getattr)(NULL,
87                                                               fil->f_ino,
88                                                               &stbuf);
89                         if (err)
90                                 return err;
91         
92                 }
93                 off = stbuf.st_size;
94                 break;
95         default:
96                 return -EINVAL;
97         }
98         pos = off + offset;
99         if ((offset < 0 && -offset > off) || (offset > 0 && pos <= off))
100                 return -EINVAL;
101         if (pos >= max)
102                 return -EOVERFLOW;
103         return pos;
104 }
105
106 static _SYSIO_OFF_T
107 _sysio_lseek(struct file *fil,
108              _SYSIO_OFF_T offset,
109              int whence,
110              _SYSIO_OFF_T max)
111 {
112         _SYSIO_OFF_T pos;
113
114         pos = _sysio_lseek_prepare(fil, offset, whence, max);
115         if (pos < 0)
116                 return pos;
117         pos = (fil->f_ino->i_ops.inop_pos)(fil->f_ino, pos);
118         if (pos < 0)
119                 return pos;
120         fil->f_pos = pos;
121         return pos;
122 }
123
124 #ifdef _LARGEFILE64_SOURCE
125 #undef lseek64
126
127 extern off64_t
128 SYSIO_INTERFACE_NAME(lseek64)(int fd, off64_t offset, int whence)
129 {
130         struct file *fil;
131         off64_t off;
132         SYSIO_INTERFACE_DISPLAY_BLOCK;
133
134         SYSIO_INTERFACE_ENTER;
135         fil = _sysio_fd_find(fd);
136         if (!fil)
137                 SYSIO_INTERFACE_RETURN((off64_t )-1, -EBADF);
138         off = _sysio_lseek(fil, offset, whence, _SEEK_MAX(fil));
139         SYSIO_INTERFACE_RETURN(off < 0 ? (off64_t )-1 : off,
140                                off < 0 ? (int )off : 0);
141
142 }
143 #ifdef __GLIBC__
144 #undef __lseek64
145 sysio_sym_weak_alias(SYSIO_INTERFACE_NAME(lseek64),
146                      PREPEND(__, SYSIO_INTERFACE_NAME(lseek64)))
147 #endif
148 #ifdef REDSTORM
149 #undef __libc_lseek64
150 sysio_sym_weak_alias(SYSIO_INTERFACE_NAME(lseek64),
151                      PREPEND(__, SYSIO_INTERFACE_NAME(libc_lseek64)))
152 #endif
153 #endif
154
155 #undef lseek
156
157 extern off_t
158 SYSIO_INTERFACE_NAME(lseek)(int fd, off_t offset, int whence)
159 {
160         struct file *fil;
161         _SYSIO_OFF_T off;
162         off_t   rtn;
163         SYSIO_INTERFACE_DISPLAY_BLOCK;
164
165         SYSIO_INTERFACE_ENTER;
166         fil = _sysio_fd_find(fd);
167         if (!fil)
168                 SYSIO_INTERFACE_RETURN((off_t )-1, -EBADF);
169         off = _sysio_lseek(fil, offset, whence, LONG_MAX);
170         if (off < 0)
171                 SYSIO_INTERFACE_RETURN((off_t )-1, (int )off);
172         rtn = (off_t )off;
173         assert(rtn == off);
174         SYSIO_INTERFACE_RETURN(rtn, 0);
175 }
176
177 #ifdef __GLIBC__
178 #undef __lseek
179 sysio_sym_weak_alias(SYSIO_INTERFACE_NAME(lseek),
180                      PREPEND(__, SYSIO_INTERFACE_NAME(lseek)))
181 #endif
182
183 #ifdef __linux__
184 #undef llseek
185 int
186 SYSIO_INTERFACE_NAME(llseek)(unsigned int fd __IS_UNUSED,
187        unsigned long offset_high __IS_UNUSED,
188        unsigned long offset_low __IS_UNUSED,
189        loff_t *result __IS_UNUSED,
190        unsigned int whence __IS_UNUSED)
191 {
192         struct file *fil;
193         loff_t  off;
194         SYSIO_INTERFACE_DISPLAY_BLOCK;
195
196         /*
197          * This is just plain goofy.
198          */
199         SYSIO_INTERFACE_ENTER;
200         fil = _sysio_fd_find(fd);
201         if (!fil)
202                 SYSIO_INTERFACE_RETURN(-1, -EBADF);
203 #ifndef _LARGEFILE64_SOURCE
204         if (offset_high) {
205                 /*
206                  * We are using 32-bit internals. This just isn't
207                  * going to work.
208                  */
209                 SYSIO_INTERFACE_RETURN(-1, -EOVERFLOW);
210         }
211 #else
212         off = offset_high;
213         off <<= 32;
214         off |= offset_low;
215 #endif
216         off = _sysio_lseek(fil, off, whence, _SEEK_MAX(fil));
217         if (off < 0)
218                 SYSIO_INTERFACE_RETURN((off_t )-1, (int )off);
219         *result = off;
220         SYSIO_INTERFACE_RETURN(0, 0);
221 }
222
223 #undef __llseek
224 sysio_sym_weak_alias(SYSIO_INTERFACE_NAME(llseek), 
225                      PREPEND(__, SYSIO_INTERFACE_NAME(llseek)))
226 #endif