X-Git-Url: https://pd.if.org/git/?p=pdclib.old;a=blobdiff_plain;f=functions%2Fstdio%2Ffclose.c;h=37349c06881b98291e8a824c30b6d3c808ba3c2a;hp=f672fb0abb9cadd16ddc32f6b5a2020400a431d6;hb=35d48051496adf237dcd22ff1324e93b6f351284;hpb=c8f799d852e3698468a78954d82588e841cc0b70 diff --git a/functions/stdio/fclose.c b/functions/stdio/fclose.c index f672fb0..37349c0 100644 --- a/functions/stdio/fclose.c +++ b/functions/stdio/fclose.c @@ -1,43 +1,112 @@ -/* ---------------------------------------------------------------------------- - * $Id$ - * ---------------------------------------------------------------------------- - * Public Domain C Library - http://pdclib.sourceforge.net - * This code is Public Domain. Use, modify, and redistribute at will. - * --------------------------------------------------------------------------*/ - -int fclose( FILE * stream ) { /* TODO */ }; - -/* PDPC code - unreviewed. -Read the note in fopen.c. -int fclose(FILE *stream) -{ -#ifdef __OS2__ - APIRET rc; -#endif +/* $Id$ */ - fflush(stream); -#ifdef __OS2__ - rc = DosClose(stream->hfile); -#endif -#ifdef __MSDOS__ - __close(stream->hfile); -#endif -#ifdef __MVS__ - __aclose(stream->hfile); -#endif - __userFiles[stream->intFno] = NULL; - if (!stream->theirBuffer) - { - free(stream->intBuffer); - } - free(stream); -#ifdef __OS2__ - if (rc != 0) +/* fclose( FILE * ) + + This file is part of the Public Domain C Library (PDCLib). + Permission is granted to use, modify, and / or redistribute at will. +*/ + +#include +#include +#include + +#ifndef REGTEST +#include <_PDCLIB_io.h> +#include + +extern FILE * _PDCLIB_filelist; + +int fclose( FILE * stream ) +{ + FILE * current = _PDCLIB_filelist; + FILE * previous = NULL; + /* Checking that the FILE handle is actually one we had opened before. */ + while ( current != NULL ) { - errno = rc; - return (EOF); + if ( stream == current ) + { + /* Flush buffer */ + if ( stream->status & _PDCLIB_FWRITE ) + { + if ( _PDCLIB_flushbuffer( stream ) == EOF ) + { + /* Flush failed, errno already set */ + return EOF; + } + } + + /* Release mutex*/ + mtx_destroy( &stream->lock ); + + /* Close handle */ + stream->ops->close(stream->handle); + + /* Remove stream from list */ + if ( previous != NULL ) + { + previous->next = stream->next; + } + else + { + _PDCLIB_filelist = stream->next; + } + /* Delete tmpfile() */ + if ( stream->status & _PDCLIB_DELONCLOSE ) + { + remove( stream->filename ); + } + /* Free user buffer (SetVBuf allocated) */ + if ( stream->status & _PDCLIB_FREEBUFFER ) + { + free( stream->buffer ); + } + /* Free stream */ + if ( ! ( stream->status & _PDCLIB_STATIC ) ) + { + free( stream ); + } + return 0; + } + previous = current; + current = current->next; } + + errno = EINVAL; + return -1; +} + +#endif + +#ifdef TEST +#include <_PDCLIB_test.h> + +int main( void ) +{ +#ifndef REGTEST + FILE * file1; + FILE * file2; + remove( testfile1 ); + remove( testfile2 ); + TESTCASE( _PDCLIB_filelist == stdin ); + TESTCASE( ( file1 = fopen( testfile1, "w" ) ) != NULL ); + TESTCASE( _PDCLIB_filelist == file1 ); + TESTCASE( ( file2 = fopen( testfile2, "w" ) ) != NULL ); + TESTCASE( _PDCLIB_filelist == file2 ); + TESTCASE( fclose( file2 ) == 0 ); + TESTCASE( _PDCLIB_filelist == file1 ); + TESTCASE( ( file2 = fopen( testfile2, "w" ) ) != NULL ); + TESTCASE( _PDCLIB_filelist == file2 ); + TESTCASE( fclose( file1 ) == 0 ); + TESTCASE( _PDCLIB_filelist == file2 ); + TESTCASE( fclose( file2 ) == 0 ); + TESTCASE( _PDCLIB_filelist == stdin ); + TESTCASE( remove( testfile1 ) == 0 ); + TESTCASE( remove( testfile2 ) == 0 ); +#else + puts( " NOTEST fclose() test driver is PDCLib-specific." ); #endif - return (0); + return TEST_RESULTS; } -*/ + +#endif +