1 /*
2  * Copyright (c) 1990 The Regents of the University of California.
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms are permitted
6  * provided that the above copyright notice and this paragraph are
7  * duplicated in all such forms and that any documentation,
8  * and/or other materials related to such
9  * distribution and use acknowledge that the software was developed
10  * by the University of California, Berkeley.  The name of the
11  * University may not be used to endorse or promote products derived
12  * from this software without specific prior written permission.
13  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
14  * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
15  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
16  */
17 
18 /*
19 FUNCTION
20 <<fseeko64>>---set file position for large file
21 
22 INDEX
23 	fseeko64
24 INDEX
25 	_fseeko64_r
26 
27 SYNOPSIS
28 	#include <stdio.h>
29 	int fseeko64(FILE *<[fp]>, _off64_t <[offset]>, int <[whence]>);
30 	int _fseeko64_r (struct _reent *<[ptr]>, FILE *<[fp]>,
31                          _off64_t <[offset]>, int <[whence]>);
32 
33 DESCRIPTION
34 Objects of type <<FILE>> can have a ``position'' that records how much
35 of the file your program has already read.  Many of the <<stdio>> functions
36 depend on this position, and many change it as a side effect.
37 
38 You can use <<fseeko64>> to set the position for the file identified by
39 <[fp]> that was opened via <<fopen64>>.  The value of <[offset]> determines
40 the new position, in one of three ways selected by the value of <[whence]>
41 (defined as macros in `<<stdio.h>>'):
42 
43 <<SEEK_SET>>---<[offset]> is the absolute file position (an offset
44 from the beginning of the file) desired.  <[offset]> must be positive.
45 
46 <<SEEK_CUR>>---<[offset]> is relative to the current file position.
47 <[offset]> can meaningfully be either positive or negative.
48 
49 <<SEEK_END>>---<[offset]> is relative to the current end of file.
50 <[offset]> can meaningfully be either positive (to increase the size
51 of the file) or negative.
52 
53 See <<ftello64>> to determine the current file position.
54 
55 RETURNS
56 <<fseeko64>> returns <<0>> when successful.  On failure, the
57 result is <<EOF>>.  The reason for failure is indicated in <<errno>>:
58 either <<ESPIPE>> (the stream identified by <[fp]> doesn't support
59 repositioning or wasn't opened via <<fopen64>>) or <<EINVAL>>
60 (invalid file position).
61 
62 PORTABILITY
63 <<fseeko64>> is a glibc extension.
64 
65 Supporting OS subroutines required: <<close>>, <<fstat64>>, <<isatty>>,
66 <<lseek64>>, <<read>>, <<sbrk>>, <<write>>.
67 */
68 
69 #define _GNU_SOURCE
70 #include <stdio.h>
71 #include <time.h>
72 #include <fcntl.h>
73 #include <stdlib.h>
74 #include <errno.h>
75 #include <sys/types.h>
76 #include <sys/stat.h>
77 #include "../stdio/local.h"
78 
79 #define	POS_ERR	(-(_fpos64_t)1)
80 
81 #ifdef __LARGE64_FILES
82 
83 /*
84  * Seek the given file to the given offset.
85  * `Whence' must be one of the three SEEK_* macros.
86  */
87 
88 _off64_t
fseeko64(register FILE * fp,_off64_t offset,int whence)89 fseeko64 (
90      register FILE *fp,
91      _off64_t offset,
92      int whence)
93 {
94   _fpos64_t (*seekfn) (void *, _fpos64_t, int);
95   _fpos64_t curoff;
96 #ifdef _FSEEK_OPTIMIZATION
97   _fpos64_t target;
98   size_t n;
99 #endif
100 
101   int havepos;
102 
103   /* Only do 64-bit seek on large file.  */
104   if (!(fp->_flags & __SL64))
105     {
106       if ((_off_t) offset != offset)
107 	{
108 	  _REENT_ERRNO(ptr) = EOVERFLOW;
109 	  return EOF;
110 	}
111       return (_off64_t) fseeko (fp, offset, whence);
112     }
113 
114   /* Make sure stdio is set up.  */
115 
116   CHECK_INIT (ptr, fp);
117 
118   _newlib_flockfile_start (fp);
119 
120   curoff = fp->_offset;
121 
122   /* If we've been doing some writing, and we're in append mode
123      then we don't really know where the filepos is.  */
124 
125   if (fp->_flags & __SAPP && fp->_flags & __SWR)
126     {
127       /* So flush the buffer and seek to the end.  */
128       fflush (fp);
129     }
130 
131   /* Have to be able to seek.  */
132 
133   if ((seekfn = fp->_seek64) == NULL)
134     {
135       _REENT_ERRNO(ptr) = ESPIPE;	/* ??? */
136       _newlib_flockfile_exit(fp);
137       return EOF;
138     }
139 
140   /*
141    * Change any SEEK_CUR to SEEK_SET, and check `whence' argument.
142    * After this, whence is either SEEK_SET or SEEK_END.
143    */
144 
145   switch (whence)
146     {
147     case SEEK_CUR:
148       /*
149        * In order to seek relative to the current stream offset,
150        * we have to first find the current stream offset a la
151        * ftell (see ftell for details).
152        */
153       fflush (fp);   /* may adjust seek offset on append stream */
154       if (fp->_flags & __SOFF)
155 	curoff = fp->_offset;
156       else
157 	{
158 	  curoff = seekfn (fp->_cookie, (_fpos64_t) 0, SEEK_CUR);
159 	  if (curoff == -1L)
160 	    {
161 	      _newlib_flockfile_exit(fp);
162 	      return EOF;
163 	    }
164 	}
165       if (fp->_flags & __SRD)
166 	{
167 	  curoff -= fp->_r;
168 	  if (HASUB (fp))
169 	    curoff -= fp->_ur;
170 	}
171       else if (fp->_flags & __SWR && fp->_p != NULL)
172 	curoff += fp->_p - fp->_bf._base;
173 
174       offset += curoff;
175       whence = SEEK_SET;
176       havepos = 1;
177       break;
178 
179     case SEEK_SET:
180     case SEEK_END:
181       havepos = 0;
182       break;
183 
184     default:
185       _REENT_ERRNO(ptr) = EINVAL;
186       _newlib_flockfile_exit(fp);
187       return (EOF);
188     }
189 
190   (void) havepos;
191   /*
192    * Can only optimise if:
193    *	reading (and not reading-and-writing);
194    *	not unbuffered; and
195    *	this is a `regular' Unix file (and hence seekfn==__sseek).
196    * We must check __NBF first, because it is possible to have __NBF
197    * and __SOPT both set.
198    */
199 
200   if (fp->_bf._base == NULL)
201     _smakebuf (fp);
202 
203 #ifdef _FSEEK_OPTIMIZATION
204   if (fp->_flags & (__SWR | __SRW | __SNBF | __SNPT))
205     goto dumb;
206   if ((fp->_flags & __SOPT) == 0)
207     {
208       struct stat64 st;
209       if (seekfn != __sseek64
210 	  || fp->_file < 0
211 	  || _fstat64 (fp->_file, &st)
212 	  || (st.st_mode & S_IFMT) != S_IFREG)
213 	{
214 	  fp->_flags |= __SNPT;
215 	  goto dumb;
216 	}
217 #ifdef	HAVE_BLKSIZE
218       fp->_blksize = st.st_blksize;
219 #else
220       fp->_blksize = 1024;
221 #endif
222       fp->_flags |= __SOPT;
223     }
224 
225   /*
226    * We are reading; we can try to optimise.
227    * Figure out where we are going and where we are now.
228    */
229 
230   if (whence == SEEK_SET)
231     target = offset;
232   else
233     {
234       struct stat64 st;
235       if (_fstat64 (fp->_file, &st))
236 	goto dumb;
237       target = st.st_size + offset;
238     }
239 
240   if (!havepos)
241     {
242       if (fp->_flags & __SOFF)
243 	curoff = fp->_offset;
244       else
245 	{
246 	  curoff = seekfn (fp->_cookie, (_fpos64_t)0, SEEK_CUR);
247 	  if (curoff == POS_ERR)
248 	    goto dumb;
249 	}
250       curoff -= fp->_r;
251       if (HASUB (fp))
252 	curoff -= fp->_ur;
253     }
254 
255   /*
256    * Compute the number of bytes in the input buffer (pretending
257    * that any ungetc() input has been discarded).  Adjust current
258    * offset backwards by this count so that it represents the
259    * file offset for the first byte in the current input buffer.
260    */
261 
262   if (HASUB (fp))
263     {
264       curoff += fp->_r;       /* kill off ungetc */
265       n = fp->_up - fp->_bf._base;
266       curoff -= n;
267       n += fp->_ur;
268     }
269   else
270     {
271       n = fp->_p - fp->_bf._base;
272       curoff -= n;
273       n += fp->_r;
274     }
275 
276   /*
277    * If the target offset is within the current buffer,
278    * simply adjust the pointers, clear EOF, undo ungetc(),
279    * and return.
280    */
281 
282   if (target >= curoff && (size_t) target < (size_t) curoff + n)
283     {
284       register int o = target - curoff;
285 
286       fp->_p = fp->_bf._base + o;
287       fp->_r = n - o;
288       if (HASUB (fp))
289 	FREEUB (ptr, fp);
290       fp->_flags &= ~__SEOF;
291       _newlib_flockfile_exit(fp);
292       return 0;
293     }
294 
295   /*
296    * The place we want to get to is not within the current buffer,
297    * but we can still be kind to the kernel copyout mechanism.
298    * By aligning the file offset to a block boundary, we can let
299    * the kernel use the VM hardware to map pages instead of
300    * copying bytes laboriously.  Using a block boundary also
301    * ensures that we only read one block, rather than two.
302    */
303 
304   curoff = target & ~((_fpos64_t)(fp->_blksize - 1));
305   if (seekfn (fp->_cookie, curoff, SEEK_SET) == POS_ERR)
306     goto dumb;
307   fp->_r = 0;
308   fp->_p = fp->_bf._base;
309   if (HASUB (fp))
310     FREEUB (ptr, fp);
311   fp->_flags &= ~__SEOF;
312   n = target - curoff;
313   if (n)
314     {
315       if (_srefill (fp) || fp->_r < (int) n)
316 	goto dumb;
317       fp->_p += n;
318       fp->_r -= n;
319     }
320   _newlib_flockfile_exit(fp);
321   return 0;
322 
323   /*
324    * We get here if we cannot optimise the seek ... just
325    * do it.  Allow the seek function to change fp->_bf._base.
326    */
327 dumb:
328 #endif
329 
330   if (fflush (fp)
331       || seekfn (fp->_cookie, offset, whence) == POS_ERR)
332     {
333       _newlib_flockfile_exit(fp);
334       return EOF;
335     }
336   /* success: clear EOF indicator and discard ungetc() data */
337   if (HASUB (fp))
338     FREEUB (ptr, fp);
339   fp->_p = fp->_bf._base;
340   fp->_r = 0;
341   /* fp->_w = 0; *//* unnecessary (I think...) */
342   fp->_flags &= ~__SEOF;
343   _newlib_flockfile_end (fp);
344   return 0;
345 }
346 
347 #endif /* __LARGE64_FILES */
348