Source code of Windows XP (NT5)
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
|
|
/***
*fclose.c - close a file * * Copyright (c) 1985-2001, Microsoft Corporation. All rights reserved. * *Purpose: * defines fclose() - close an open file * *Revision History: * 09-02-83 RN initial version * 05-14-87 SKS error return from fflush must not be clobbered * 08-10-87 JCR Added code to support P_tmpdir with or without trailing '\' * 11-01-87 JCR Multi-thread support * 12-11-87 JCR Added "_LOAD_DS" to declaration * 01-13-88 JCR Removed unnecessary calls to mthread fileno/feof/ferror * 05-31-88 PHG Merged DLL and normal versions * 06-14-88 JCR Use near pointer to reference _iob[] entries * 08-24-88 GJF Don't use FP_OFF() macro for the 386 * 08-17-89 GJF Clean up, now specific to OS/2 2.0 (i.e., 386 flat * model). Also fixed copyright and indents. * 02-15-90 GJF Fixed copyright * 03-16-90 GJF Made calling type _CALLTYPE1, added #include * <cruntime.h> and removed #include <register.h>. * 05-29-90 SBM Use _flush, not [_]fflush[_lk] * 07-25-90 SBM Replaced <assertm.h> by <assert.h> * 10-02-90 GJF New-style function declarator. * 01-21-91 GJF ANSI naming. * 03-11-92 GJF For Win32, revised temporary file cleanup. * 03-25-92 DJM POSIX support. * 08-26-92 GJF Include unistd.h for POSIX build. * 01-13-93 GJF Don't need/want to remove() tmp files on Windows NT * (file is removed by the OS when handle is closed). * 04-06-93 SKS Replace _CRTAPI* with __cdecl * 09-06-94 CFW Remove Cruiser support. * 09-06-94 CFW Replace MTHREAD with _MT. * 01-10-95 CFW Debug CRT allocs. * 02-06-94 CFW assert -> _ASSERTE. * 02-17-95 GJF Merged in Mac version. Removed some useless #ifdef-s. * 03-07-95 GJF _[un]lock_str macros now take FILE * arg. * 05-12-95 CFW Paranoia: set _tmpfname field to NULL. * 02-25-98 GJF Exception-safe locking. * 05-17-99 PML Remove all Macintosh support. * *******************************************************************************/
#include <cruntime.h>
#ifdef _POSIX_
#include <unistd.h>
#endif
#include <stdio.h>
#include <file2.h>
#include <string.h>
#include <io.h>
#include <stdlib.h>
#include <internal.h>
#include <mtdll.h>
#include <dbgint.h>
/***
*int fclose(stream) - close a stream * *Purpose: * Flushes and closes a stream and frees any buffer associated * with that stream, unless it was set with setbuf. * *Entry: * FILE *stream - stream to close * *Exit: * returns 0 if OK, EOF if fails (can't _flush, not a FILE, not open, etc.) * closes the file -- affecting FILE structure * *Exceptions: * *******************************************************************************/
#ifdef _MT /* multi-thread; define both fclose and _fclose_lk */
int __cdecl fclose ( FILE *stream ) { int result = EOF;
_ASSERTE(stream != NULL);
/* If stream is a string, simply clear flag and return EOF */ if (stream->_flag & _IOSTRG) stream->_flag = 0; /* IS THIS REALLY NEEDED ??? */
/* Stream is a real file. */ else { _lock_str(stream); __try { result = _fclose_lk(stream); } __finally { _unlock_str(stream); } }
return(result); }
/***
*int _fclose_lk() - close a stream (lock already held) * *Purpose: * Core fclose() routine; assumes caller has stream lock held. * * [See fclose() above for more information.] * *Entry: * *Exit: * *Exceptions: * *******************************************************************************/
int __cdecl _fclose_lk ( FILE *str ) { REG1 FILE *stream; REG2 int result = EOF;
/* Init near stream pointer */ stream = str;
#else /* non multi-thread; just define fclose() */
int __cdecl fclose ( FILE *str ) { REG1 FILE *stream; REG2 int result = EOF;
/* Init near stream pointer */ stream = str;
if (stream->_flag & _IOSTRG) { stream->_flag = 0; return(EOF); }
#endif
_ASSERTE(str != NULL);
if (inuse(stream)) {
/* Stream is in use:
(1) flush stream (2) free the buffer (3) close the file (4) delete the file if temporary */
result = _flush(stream); _freebuf(stream);
#ifdef _POSIX_
if (close(fileno(stream)) <0) #else
if (_close(_fileno(stream)) < 0) #endif
result = EOF;
else if ( stream->_tmpfname != NULL ) { /*
* temporary file (i.e., one created by tmpfile() * call). delete, if necessary (don't have to on * Windows NT because it was done by the system when * the handle was closed). also, free up the heap * block holding the pathname. */ #ifdef _POSIX_
if ( unlink(stream->_tmpfname) ) result = EOF; #endif
_free_crt(stream->_tmpfname); stream->_tmpfname = NULL; }
}
stream->_flag = 0; return(result); }
|