xref: /freebsd/lib/libc/stdio/fflush.c (revision 29ac6bd228d1c75dc4c19105fa149861bff04720)
158f0484fSRodney W. Grimes /*-
258f0484fSRodney W. Grimes  * Copyright (c) 1990, 1993
358f0484fSRodney W. Grimes  *	The Regents of the University of California.  All rights reserved.
458f0484fSRodney W. Grimes  *
558f0484fSRodney W. Grimes  * This code is derived from software contributed to Berkeley by
658f0484fSRodney W. Grimes  * Chris Torek.
758f0484fSRodney W. Grimes  *
858f0484fSRodney W. Grimes  * Redistribution and use in source and binary forms, with or without
958f0484fSRodney W. Grimes  * modification, are permitted provided that the following conditions
1058f0484fSRodney W. Grimes  * are met:
1158f0484fSRodney W. Grimes  * 1. Redistributions of source code must retain the above copyright
1258f0484fSRodney W. Grimes  *    notice, this list of conditions and the following disclaimer.
1358f0484fSRodney W. Grimes  * 2. Redistributions in binary form must reproduce the above copyright
1458f0484fSRodney W. Grimes  *    notice, this list of conditions and the following disclaimer in the
1558f0484fSRodney W. Grimes  *    documentation and/or other materials provided with the distribution.
1658f0484fSRodney W. Grimes  * 3. All advertising materials mentioning features or use of this software
1758f0484fSRodney W. Grimes  *    must display the following acknowledgement:
1858f0484fSRodney W. Grimes  *	This product includes software developed by the University of
1958f0484fSRodney W. Grimes  *	California, Berkeley and its contributors.
2058f0484fSRodney W. Grimes  * 4. Neither the name of the University nor the names of its contributors
2158f0484fSRodney W. Grimes  *    may be used to endorse or promote products derived from this software
2258f0484fSRodney W. Grimes  *    without specific prior written permission.
2358f0484fSRodney W. Grimes  *
2458f0484fSRodney W. Grimes  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2558f0484fSRodney W. Grimes  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2658f0484fSRodney W. Grimes  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2758f0484fSRodney W. Grimes  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2858f0484fSRodney W. Grimes  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2958f0484fSRodney W. Grimes  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
3058f0484fSRodney W. Grimes  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
3158f0484fSRodney W. Grimes  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3258f0484fSRodney W. Grimes  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3358f0484fSRodney W. Grimes  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3458f0484fSRodney W. Grimes  * SUCH DAMAGE.
3558f0484fSRodney W. Grimes  */
3658f0484fSRodney W. Grimes 
3758f0484fSRodney W. Grimes #if defined(LIBC_SCCS) && !defined(lint)
38ce51cf03SJames Raynard #if 0
3958f0484fSRodney W. Grimes static char sccsid[] = "@(#)fflush.c	8.1 (Berkeley) 6/4/93";
40ce51cf03SJames Raynard #endif
41ce51cf03SJames Raynard static const char rcsid[] =
427f3dea24SPeter Wemm   "$FreeBSD$";
4358f0484fSRodney W. Grimes #endif /* LIBC_SCCS and not lint */
4458f0484fSRodney W. Grimes 
45d201fe46SDaniel Eischen #include "namespace.h"
4658f0484fSRodney W. Grimes #include <errno.h>
4758f0484fSRodney W. Grimes #include <stdio.h>
48d201fe46SDaniel Eischen #include "un-namespace.h"
49ec216c26SJohn Birrell #include "libc_private.h"
50d201fe46SDaniel Eischen #include "local.h"
5158f0484fSRodney W. Grimes 
5229ac6bd2SDaniel Eischen static int	sflush_locked(FILE *);
5329ac6bd2SDaniel Eischen 
54d201fe46SDaniel Eischen /*
55d201fe46SDaniel Eischen  * Flush a single file, or (if fp is NULL) all files.
56d201fe46SDaniel Eischen  * MT-safe version
57d201fe46SDaniel Eischen  */
58f70177e7SJulian Elischer int
59d201fe46SDaniel Eischen fflush(FILE *fp)
6058f0484fSRodney W. Grimes {
61f70177e7SJulian Elischer 	int retval;
6258f0484fSRodney W. Grimes 
6358f0484fSRodney W. Grimes 	if (fp == NULL)
6429ac6bd2SDaniel Eischen 		return (_fwalk(sflush_locked));
65ec216c26SJohn Birrell 	FLOCKFILE(fp);
6658f0484fSRodney W. Grimes 	if ((fp->_flags & (__SWR | __SRW)) == 0) {
6758f0484fSRodney W. Grimes 		errno = EBADF;
68f70177e7SJulian Elischer 		retval = EOF;
69d201fe46SDaniel Eischen 	} else
70f70177e7SJulian Elischer 		retval = __sflush(fp);
71ec216c26SJohn Birrell 	FUNLOCKFILE(fp);
72f70177e7SJulian Elischer 	return (retval);
7358f0484fSRodney W. Grimes }
7458f0484fSRodney W. Grimes 
75d201fe46SDaniel Eischen /*
76d201fe46SDaniel Eischen  * Flush a single file, or (if fp is NULL) all files.
77d201fe46SDaniel Eischen  * Non-MT-safe version
78d201fe46SDaniel Eischen  */
79f70177e7SJulian Elischer int
80d201fe46SDaniel Eischen __fflush(FILE *fp)
8158f0484fSRodney W. Grimes {
82d201fe46SDaniel Eischen 	int retval;
83d201fe46SDaniel Eischen 
84d201fe46SDaniel Eischen 	if (fp == NULL)
8529ac6bd2SDaniel Eischen 		return (_fwalk(sflush_locked));
86d201fe46SDaniel Eischen 	if ((fp->_flags & (__SWR | __SRW)) == 0) {
87d201fe46SDaniel Eischen 		errno = EBADF;
88d201fe46SDaniel Eischen 		retval = EOF;
89d201fe46SDaniel Eischen 	} else
90d201fe46SDaniel Eischen 		retval = __sflush(fp);
91d201fe46SDaniel Eischen 	return (retval);
92d201fe46SDaniel Eischen }
93d201fe46SDaniel Eischen 
94d201fe46SDaniel Eischen int
95d201fe46SDaniel Eischen __sflush(FILE *fp)
96d201fe46SDaniel Eischen {
97d201fe46SDaniel Eischen 	unsigned char *p;
98d201fe46SDaniel Eischen 	int n, t;
9958f0484fSRodney W. Grimes 
10058f0484fSRodney W. Grimes 	t = fp->_flags;
10158f0484fSRodney W. Grimes 	if ((t & __SWR) == 0)
10258f0484fSRodney W. Grimes 		return (0);
10358f0484fSRodney W. Grimes 
10458f0484fSRodney W. Grimes 	if ((p = fp->_bf._base) == NULL)
10558f0484fSRodney W. Grimes 		return (0);
10658f0484fSRodney W. Grimes 
10758f0484fSRodney W. Grimes 	n = fp->_p - p;		/* write this much */
10858f0484fSRodney W. Grimes 
10958f0484fSRodney W. Grimes 	/*
11058f0484fSRodney W. Grimes 	 * Set these immediately to avoid problems with longjmp and to allow
11158f0484fSRodney W. Grimes 	 * exchange buffering (via setvbuf) in user write function.
11258f0484fSRodney W. Grimes 	 */
11358f0484fSRodney W. Grimes 	fp->_p = p;
11458f0484fSRodney W. Grimes 	fp->_w = t & (__SLBF|__SNBF) ? 0 : fp->_bf._size;
11558f0484fSRodney W. Grimes 
11658f0484fSRodney W. Grimes 	for (; n > 0; n -= t, p += t) {
11758f0484fSRodney W. Grimes 		t = (*fp->_write)(fp->_cookie, (char *)p, n);
11858f0484fSRodney W. Grimes 		if (t <= 0) {
11958f0484fSRodney W. Grimes 			fp->_flags |= __SERR;
12058f0484fSRodney W. Grimes 			return (EOF);
12158f0484fSRodney W. Grimes 		}
12258f0484fSRodney W. Grimes 	}
12358f0484fSRodney W. Grimes 	return (0);
12458f0484fSRodney W. Grimes }
12529ac6bd2SDaniel Eischen 
12629ac6bd2SDaniel Eischen static int
12729ac6bd2SDaniel Eischen sflush_locked(FILE *fp)
12829ac6bd2SDaniel Eischen {
12929ac6bd2SDaniel Eischen 	int	ret;
13029ac6bd2SDaniel Eischen 
13129ac6bd2SDaniel Eischen 	FLOCKFILE(fp);
13229ac6bd2SDaniel Eischen 	ret = __sflush(fp);
13329ac6bd2SDaniel Eischen 	FUNLOCKFILE(fp);
13429ac6bd2SDaniel Eischen 	return (ret);
13529ac6bd2SDaniel Eischen }
136