Free a malloc()ed ungetc buffer.
[gnulib.git] / lib / fpurge.c
1 /* Flushing buffers of a FILE 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 "fpurge.h"
22
23 int
24 fpurge (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->_IO_read_end = fp->_IO_read_ptr;
31   fp->_IO_write_ptr = fp->_IO_write_base;
32   /* Avoid memory leak when there is an active ungetc buffer.  */
33   if (fp->_IO_save_base != NULL)
34     {
35       free (fp->_IO_save_base);
36       fp->_IO_save_base = NULL;
37     }
38   return 0;
39 #elif defined __sferror             /* FreeBSD, NetBSD, OpenBSD, MacOS X, Cygwin */
40   fp->_p = fp->_bf._base;
41   fp->_r = 0;
42   fp->_w = ((fp->_flags & (__SLBF | __SNBF)) == 0 /* fully buffered? */
43             ? fp->_bf._size
44             : 0);
45   /* Avoid memory leak when there is an active ungetc buffer.  */
46 # if defined __NetBSD__ || defined __OpenBSD__ /* NetBSD, OpenBSD */
47    /* See <http://cvsweb.netbsd.org/bsdweb.cgi/src/lib/libc/stdio/fileext.h?rev=HEAD&content-type=text/x-cvsweb-markup>
48       and <http://www.openbsd.org/cgi-bin/cvsweb/src/lib/libc/stdio/fileext.h?rev=HEAD&content-type=text/x-cvsweb-markup> */
49 #  define fp_ub ((struct { struct __sbuf _ub; } *) fp->_ext._base)->_ub
50 # else                                         /* FreeBSD, MacOS X, Cygwin */
51 #  define fp_ub fp->_ub
52 # endif
53   if (fp_ub._base != NULL)
54     {
55       if (fp_ub._base != fp->_ubuf)
56         free (fp_ub._base);
57       fp_ub._base = NULL;
58     }
59   return 0;
60 #elif defined _IOERR                /* AIX, HP-UX, IRIX, OSF/1, Solaris, mingw */
61   fp->_ptr = fp->_base;
62   if (fp->_ptr != NULL)
63     fp->_cnt = 0;
64   return 0;
65 #else
66  #error "Please port gnulib fpurge.c to your platform! Look at the definitions of fflush, setvbuf and ungetc on your system, then report this to bug-gnulib."
67 #endif
68 }