| /* Copyright (C) 2004       Manuel Novoa III    <mjn3@codepoet.org> | 
 |  * | 
 |  * GNU Library General Public License (LGPL) version 2 or later. | 
 |  * | 
 |  * Dedicated to Toni.  See uClibc/DEDICATION.mjn3 for details. | 
 |  */ | 
 |  | 
 | #include "_stdio.h" | 
 |  | 
 |  | 
 | #ifndef __DO_LARGEFILE | 
 | # define FILEDES_ARG    (-1) | 
 | #endif | 
 |  | 
 | FILE *freopen(const char * __restrict filename, const char * __restrict mode, | 
 | 			  register FILE * __restrict stream) | 
 | { | 
 | 	/* | 
 | 	 * ANSI/ISO allow (implementation-defined) change of mode for an | 
 | 	 * existing file if filename is NULL.  It doesn't look like Linux | 
 | 	 * supports this, so we don't here. | 
 | 	 * | 
 | 	 * NOTE: Whether or not the stream is free'd on failure is unclear | 
 | 	 *       w.r.t. ANSI/ISO.  This implementation chooses to NOT free | 
 | 	 *       the stream and associated buffer if they were dynamically | 
 | 	 *       allocated. | 
 | 	 * NOTE: Previous versions of uClibc did free dynamic storage. | 
 | 	 * | 
 | 	 * TODO: Apparently linux allows setting append mode.  Implement? | 
 | 	 */ | 
 | 	unsigned short dynmode; | 
 | 	register FILE *fp; | 
 | 	__STDIO_AUTO_THREADLOCK_VAR; | 
 |  | 
 | 	__STDIO_AUTO_THREADLOCK(stream); | 
 |  | 
 | 	__STDIO_STREAM_VALIDATE(stream); | 
 |  | 
 | 	__STDIO_OPENLIST_INC_USE;	/* Do not remove the file from the list. */ | 
 |  | 
 | 	/* First, flush and close, but don't deallocate, the stream. */ | 
 | 	/* This also removes the stream for the open file list. */ | 
 | 	dynmode = (stream->__modeflags & (__FLAG_FREEBUF|__FLAG_FREEFILE)); | 
 |  | 
 | 	stream->__modeflags &= ~(__FLAG_FREEBUF|__FLAG_FREEFILE); | 
 |  | 
 | 	/* Only call fclose on the stream if it is not already closed. */ | 
 | 	if ((stream->__modeflags & (__FLAG_READONLY|__FLAG_WRITEONLY)) | 
 | 		!= (__FLAG_READONLY|__FLAG_WRITEONLY) | 
 | 		) { | 
 | 		fclose(stream);			/* Failures are ignored. */ | 
 | 		/* NOTE: fclose always does __STDIO_OPENLIST_INC_DEL_CNT.  But we don't | 
 | 		 * want to remove this FILE from the open list, even if the freopen fails. | 
 | 		 * Consider the case of a failed freopen() on stdin.  You probably still | 
 | 		 * want to be able to call freopen() again.  Similarly for other "malloc'd" | 
 | 		 * streams. */ | 
 | 		__STDIO_OPENLIST_DEC_DEL_CNT; | 
 | 	} | 
 |  | 
 | 	fp = _stdio_fopen(((intptr_t) filename), mode, stream, FILEDES_ARG); | 
 | 	if (!fp) { | 
 | 		/* Don't remove stream from the open file list and (potentially) free it. | 
 | 		 * See _stdio_openlist_dec_use() in fflush.c. */ | 
 | 		stream->__modeflags = __FLAG_READONLY|__FLAG_WRITEONLY|__FLAG_FAILED_FREOPEN; | 
 | 	} | 
 |  | 
 | 	/* Reset the allocation flags. */ | 
 | 	stream->__modeflags |= dynmode; | 
 |  | 
 | 	__STDIO_OPENLIST_DEC_USE; | 
 |  | 
 | 	__STDIO_AUTO_THREADUNLOCK(stream); | 
 |  | 
 | 	return fp; | 
 | } |