]> pd.if.org Git - pdclib/blob - functions/stdio/fread.c
PDCLIB-16: Add _unlocked variations of all I/O routines; move work into these versions
[pdclib] / functions / stdio / fread.c
1 /* $Id$ */
2
3 /* fwrite( void *, size_t, size_t, FILE * )
4
5    This file is part of the Public Domain C Library (PDCLib).
6    Permission is granted to use, modify, and / or redistribute at will.
7 */
8
9 #include <stdio.h>
10
11 #ifndef REGTEST
12
13 #include <_PDCLIB_glue.h>
14
15 #include <stdbool.h>
16 #include <string.h>
17
18 size_t fread_unlocked( void * _PDCLIB_restrict ptr, 
19                        size_t size, size_t nmemb, 
20                        struct _PDCLIB_file_t * _PDCLIB_restrict stream )
21 {
22     if ( _PDCLIB_prepread( stream ) == EOF )
23     {
24         return 0;
25     }
26     char * dest = (char *)ptr;
27     size_t nmemb_i;
28     for ( nmemb_i = 0; nmemb_i < nmemb; ++nmemb_i )
29     {
30         for ( size_t size_i = 0; size_i < size; ++size_i )
31         {
32             if ( stream->bufidx == stream->bufend )
33             {
34                 if ( _PDCLIB_fillbuffer( stream ) == EOF )
35                 {
36                     /* Could not read requested data */
37                     return nmemb_i;
38                 }
39             }
40             dest[ nmemb_i * size + size_i ] = stream->buffer[ stream->bufidx++ ];
41         }
42     }
43     return nmemb_i;
44 }
45
46 size_t fread( void * _PDCLIB_restrict ptr, 
47               size_t size, size_t nmemb, 
48               struct _PDCLIB_file_t * _PDCLIB_restrict stream )
49 {
50     flockfile( stream );
51     size_t r = fread_unlocked( ptr, size, nmemb, stream );
52     funlockfile( stream );
53     return r;
54 }
55
56 #endif
57
58 #ifdef TEST
59 #include <_PDCLIB_test.h>
60
61 int main( void )
62 {
63     FILE * fh;
64     char const * message = "Testing fwrite()...\n";
65     char buffer[21];
66     buffer[20] = 'x';
67     TESTCASE( ( fh = tmpfile() ) != NULL );
68     /* fwrite() / readback */
69     TESTCASE( fwrite( message, 1, 20, fh ) == 20 );
70     rewind( fh );
71     TESTCASE( fread( buffer, 1, 20, fh ) == 20 );
72     TESTCASE( memcmp( buffer, message, 20 ) == 0 );
73     TESTCASE( buffer[20] == 'x' );
74     /* same, different nmemb / size settings */
75     rewind( fh );
76     TESTCASE( memset( buffer, '\0', 20 ) == buffer );
77     TESTCASE( fwrite( message, 5, 4, fh ) == 4 );
78     rewind( fh );
79     TESTCASE( fread( buffer, 5, 4, fh ) == 4 );
80     TESTCASE( memcmp( buffer, message, 20 ) == 0 );
81     TESTCASE( buffer[20] == 'x' );
82     /* same... */
83     rewind( fh );
84     TESTCASE( memset( buffer, '\0', 20 ) == buffer );
85     TESTCASE( fwrite( message, 20, 1, fh ) == 1 );
86     rewind( fh );
87     TESTCASE( fread( buffer, 20, 1, fh ) == 1 );
88     TESTCASE( memcmp( buffer, message, 20 ) == 0 );
89     TESTCASE( buffer[20] == 'x' );
90     /* Done. */
91     TESTCASE( fclose( fh ) == 0 );
92     return TEST_RESULTS;
93 }
94
95 #endif
96