if ( ( rc->status = filemode( mode ) ) == 0 ) goto fail; /* invalid mode */
rc->handle = _PDCLIB_open( filename, rc->status );
if ( rc->handle == _PDCLIB_NOHANDLE ) goto fail; /* OS open() failed */
-
/* Adding to list of open files */
rc->next = _PDCLIB_filelist;
_PDCLIB_filelist = rc;
/* Setting buffer, and mark as internal. TODO: Check for unbuffered? */
if ( ( rc->buffer = malloc( BUFSIZ ) ) == NULL ) goto fail;
- rc->status |= _PDCLIB_LIBBUFFER;
+ rc->status |= ( _PDCLIB_LIBBUFFER | _PDCLIB_VIRGINSTR );
/* TODO: Setting mbstate */
return rc;
fail:
#ifndef REGTEST
-int setvbuf( FILE * _PDCLIB_restrict stream, char * _PDCLIB_restrict buf, int mode, size_t size )
+int setvbuf( struct _PDCLIB_file_t * _PDCLIB_restrict stream, char * _PDCLIB_restrict buf, int mode, size_t size )
{
- /* TODO: Only allowed on a "virgin" stream; add check. */
- if ( ( stream->status & ( _IOFBF | _IOLBF | _IONBF ) ) /* Only allowed on "virgin" stream */
- || ( ( mode != _IOFBF ) && ( mode != _IOLBF ) && ( mode != _IONBF ) ) /* invalid mode */
- || ( ( buf == NULL ) && ( ( buf = malloc( size ) ) == NULL ) ) /* no memory available */
- )
+ /* Only allowed on "virgin" streams (i.e., before first I/O occurs), and
+ a valid value for mode.
+ */
+ if ( ( ! stream->status & _PDCLIB_VIRGINSTR ) ||
+ ( ( mode != _IOFBF ) && ( mode != _IOLBF ) && ( mode != _IONBF ) ) )
{
return -1;
}
+ /* If a buffer is provided by user... */
+ if ( buf != NULL )
+ {
+ /* ...do not free it in library functions like fclose(), freopen(). */
+ stream->status &= ~_PDCLIB_LIBBUFFER;
+ }
+ /* If no buffer is provided by user, but required... */
+ else if ( mode != _IONBF )
+ {
+ /* Since setvbuf() may be called (successfully) on a stream only once,
+ the stream's buffer at this point should *always* be that allocated
+ by fopen(), but better make sure.
+ */
+ if ( ! ( stream->status & _PDCLIB_LIBBUFFER ) )
+ {
+ return -1;
+ }
+ /* Drop old buffer, allocate new one of requested size (unless that is
+ equal to BUFSIZ, in which case we can use the one already allocated
+ by fopen().)
+ */
+ if ( size != BUFSIZ )
+ {
+ if ( ( buf = malloc( size ) ) == NULL )
+ {
+ return -1;
+ }
+ free( stream->buffer );
+ }
+ }
+ /* Applying new settings to stream. */
+ stream->status &= ~( _IOFBF | _IOLBF | _IONBF );
stream->status |= mode;
- stream->buffer = buf;
+ stream->buffer = buf;
stream->bufsize = size;
+ stream->status &= ~_PDCLIB_VIRGINSTR;
return 0;
}
int main( void )
{
+#ifndef REGTEST
TESTCASE( NO_TESTDRIVER );
+#else
+ puts( " NOTEST setvbuf() test driver is PDCLib-specific." );
+#endif
return TEST_RESULTS;
}
/* Various <stdio.h> internals */
/* -------------------------------------------------------------------------- */
-/* Flags for representing mode (see fopen()). */
-#define _PDCLIB_FREAD 1u
-#define _PDCLIB_FWRITE 2u
-#define _PDCLIB_FAPPEND 4u
-#define _PDCLIB_FRW 8u
-#define _PDCLIB_FBIN 16u
+/* Flags for representing mode (see fopen()). Note these must fit the same
+ status field as the _IO?BF flags in <stdio.h> and the internal flags below.
+*/
+#define _PDCLIB_FREAD 8u
+#define _PDCLIB_FWRITE 16u
+#define _PDCLIB_FAPPEND 32u
+#define _PDCLIB_FRW 64u
+#define _PDCLIB_FBIN 128u
/* Internal flags, made to fit the same status field as the flags above. */
-#define _PDCLIB_WROTELAST 32u
-#define _PDCLIB_LIBBUFFER 64u
+#define _PDCLIB_WROTELAST 256u
+#define _PDCLIB_LIBBUFFER 512u
+#define _PDCLIB_VIRGINSTR 1024u
struct _PDCLIB_file_t
{