New module 'fseterr'.
[pspp] / lib / fseterr.c
1 /* Set the error indicator of a stream.
2    Copyright (C) 2007 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 "fseterr.h"
22
23 void
24 fseterr (FILE *fp)
25 {
26   /* Most systems provide FILE as a struct and the necessary bitmask in
27      <stdio.h>, because they need it for implementing getc() and putc() as
28      fast macros.  */
29 #if defined _IO_ferror_unlocked     /* GNU libc, BeOS */
30   fp->_flags |= _IO_ERR_SEEN;
31 #elif defined __sferror             /* FreeBSD, NetBSD, OpenBSD, MacOS X, Cygwin */
32   fp->_flags |= __SERR;
33 #elif defined _IOERR                /* AIX, HP-UX, IRIX, OSF/1, Solaris, mingw */
34   fp->_flag |= _IOERR;
35 #else                               /* unknown  */
36   /* Portable fallback, based on an idea by Rich Felker.
37      Wow! 6 system calls for something that is just a bit operation!  */
38   int fd;
39   int fd2;
40
41   fflush (fp);
42   fd = fileno (fp);
43   fd2 = dup (fd);
44   if (fd2 >= 0)
45     {
46       close (fd);
47       fputc ('\0', fp); /* This should set the error indicator.  */
48       fflush (fp);      /* Or this.  */
49       if (dup2 (fd2, fd) < 0)
50         /* Whee... we botched the stream and now cannot restore it!  */
51         abort ();
52       close (fd2);
53     }
54 #endif
55 }