New module 'msvc-nothrow'. Makes _get_osfhandle safe on MSVC 9.
[gnulib.git] / lib / lseek.c
1 /* An lseek() function that detects pipes.
2    Copyright (C) 2007, 2009-2011 Free Software Foundation, Inc.
3
4    This program is free software; you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation; either version 2, or (at your option)
7    any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License along
15    with this program; if not, write to the Free Software Foundation,
16    Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  */
17
18 #include <config.h>
19
20 /* Specification.  */
21 #include <unistd.h>
22
23 #if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
24 /* Windows platforms.  */
25 /* Get GetFileType.  */
26 # include <windows.h>
27 /* Get _get_osfhandle.  */
28 # include "msvc-nothrow.h"
29 #else
30 # include <sys/stat.h>
31 #endif
32 #include <errno.h>
33
34 #undef lseek
35
36 off_t
37 rpl_lseek (int fd, off_t offset, int whence)
38 {
39 #if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
40   /* mingw lseek mistakenly succeeds on pipes, sockets, and terminals.  */
41   HANDLE h = (HANDLE) _get_osfhandle (fd);
42   if (h == INVALID_HANDLE_VALUE)
43     {
44       errno = EBADF;
45       return -1;
46     }
47   if (GetFileType (h) != FILE_TYPE_DISK)
48     {
49       errno = ESPIPE;
50       return -1;
51     }
52 #else
53   /* BeOS lseek mistakenly succeeds on pipes...  */
54   struct stat statbuf;
55   if (fstat (fd, &statbuf) < 0)
56     return -1;
57   if (!S_ISREG (statbuf.st_mode))
58     {
59       errno = ESPIPE;
60       return -1;
61     }
62 #endif
63   return lseek (fd, offset, whence);
64 }