xref: /freebsd/lib/libc/stdio/fclose.c (revision 46ffdf3bbd65941a229f48a1c441b7ee3f348435)
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  * 4. Neither the name of the University nor the names of its contributors
1758f0484fSRodney W. Grimes  *    may be used to endorse or promote products derived from this software
1858f0484fSRodney W. Grimes  *    without specific prior written permission.
1958f0484fSRodney W. Grimes  *
2058f0484fSRodney W. Grimes  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2158f0484fSRodney W. Grimes  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2258f0484fSRodney W. Grimes  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2358f0484fSRodney W. Grimes  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2458f0484fSRodney W. Grimes  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2558f0484fSRodney W. Grimes  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2658f0484fSRodney W. Grimes  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2758f0484fSRodney W. Grimes  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2858f0484fSRodney W. Grimes  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2958f0484fSRodney W. Grimes  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3058f0484fSRodney W. Grimes  * SUCH DAMAGE.
3158f0484fSRodney W. Grimes  */
3258f0484fSRodney W. Grimes 
3358f0484fSRodney W. Grimes #if defined(LIBC_SCCS) && !defined(lint)
3458f0484fSRodney W. Grimes static char sccsid[] = "@(#)fclose.c	8.1 (Berkeley) 6/4/93";
3558f0484fSRodney W. Grimes #endif /* LIBC_SCCS and not lint */
36333fc21eSDavid E. O'Brien #include <sys/cdefs.h>
37333fc21eSDavid E. O'Brien __FBSDID("$FreeBSD$");
3858f0484fSRodney W. Grimes 
39d201fe46SDaniel Eischen #include "namespace.h"
4058f0484fSRodney W. Grimes #include <errno.h>
4158f0484fSRodney W. Grimes #include <stdio.h>
4258f0484fSRodney W. Grimes #include <stdlib.h>
43d201fe46SDaniel Eischen #include "un-namespace.h"
44*46ffdf3bSKonstantin Belousov #include <spinlock.h>
45ec216c26SJohn Birrell #include "libc_private.h"
46d201fe46SDaniel Eischen #include "local.h"
4758f0484fSRodney W. Grimes 
48f70177e7SJulian Elischer int
4929ac6bd2SDaniel Eischen fclose(FILE *fp)
5058f0484fSRodney W. Grimes {
51d201fe46SDaniel Eischen 	int r;
5258f0484fSRodney W. Grimes 
5358f0484fSRodney W. Grimes 	if (fp->_flags == 0) {	/* not open! */
5458f0484fSRodney W. Grimes 		errno = EBADF;
5558f0484fSRodney W. Grimes 		return (EOF);
5658f0484fSRodney W. Grimes 	}
57ec216c26SJohn Birrell 	FLOCKFILE(fp);
5858f0484fSRodney W. Grimes 	r = fp->_flags & __SWR ? __sflush(fp) : 0;
5958f0484fSRodney W. Grimes 	if (fp->_close != NULL && (*fp->_close)(fp->_cookie) < 0)
6058f0484fSRodney W. Grimes 		r = EOF;
6158f0484fSRodney W. Grimes 	if (fp->_flags & __SMBF)
6258f0484fSRodney W. Grimes 		free((char *)fp->_bf._base);
6358f0484fSRodney W. Grimes 	if (HASUB(fp))
6458f0484fSRodney W. Grimes 		FREEUB(fp);
6558f0484fSRodney W. Grimes 	if (HASLB(fp))
6658f0484fSRodney W. Grimes 		FREELB(fp);
67f70177e7SJulian Elischer 	fp->_file = -1;
6858f0484fSRodney W. Grimes 	fp->_r = fp->_w = 0;	/* Mess up if reaccessed. */
69*46ffdf3bSKonstantin Belousov 
70*46ffdf3bSKonstantin Belousov 	/*
71*46ffdf3bSKonstantin Belousov 	 * Lock the spinlock used to protect __sglue list walk in
72*46ffdf3bSKonstantin Belousov 	 * __sfp().  The __sfp() uses fp->_flags == 0 test as an
73*46ffdf3bSKonstantin Belousov 	 * indication of the unused FILE.
74*46ffdf3bSKonstantin Belousov 	 *
75*46ffdf3bSKonstantin Belousov 	 * Taking the lock prevents possible compiler or processor
76*46ffdf3bSKonstantin Belousov 	 * reordering of the writes performed before the final _flags
77*46ffdf3bSKonstantin Belousov 	 * cleanup, making sure that we are done with the FILE before
78*46ffdf3bSKonstantin Belousov 	 * it is considered available.
79*46ffdf3bSKonstantin Belousov 	 */
80*46ffdf3bSKonstantin Belousov 	STDIO_THREAD_LOCK();
81dcd26325SDmitrij Tejblum 	fp->_flags = 0;		/* Release this FILE for reuse. */
82*46ffdf3bSKonstantin Belousov 	STDIO_THREAD_UNLOCK();
8329ac6bd2SDaniel Eischen 	FUNLOCKFILE(fp);
8458f0484fSRodney W. Grimes 	return (r);
8558f0484fSRodney W. Grimes }
86