blob: d2fcc70d582f730c43799e76c2c51f29f40abead [file] [log] [blame]
/* 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"
#ifdef __DO_UNLOCKED
size_t fread_unlocked(void * __restrict ptr, size_t size, size_t nmemb,
FILE * __restrict stream)
{
__STDIO_STREAM_VALIDATE(stream);
assert(stream->__filedes >= -1);
/* Note: If nmbem * size > SIZE_MAX then there is an application
* bug since no array can be larger than SIZE_MAX in size. */
if ((__STDIO_STREAM_IS_NARROW_READING(stream)
|| !__STDIO_STREAM_TRANS_TO_READ(stream, __FLAG_NARROW))
&& size && nmemb
) {
if (nmemb <= (SIZE_MAX / size)) {
unsigned char *buffer = (unsigned char *) ptr;
size_t todo, bytes, avail;
todo = bytes = size * nmemb;
/* Check for ungots... */
while (stream->__modeflags & __FLAG_UNGOT) {
*buffer++ = stream->__ungot[(stream->__modeflags--) & 1];
stream->__ungot[1] = 0;
if (!--todo) {
goto DONE;
}
}
#ifdef __STDIO_BUFFERS
/* Next check for available buffered... */
if ((avail = stream->__bufread - stream->__bufpos) > 0) {
if (avail > todo) {
avail = todo;
}
memcpy(buffer, stream->__bufpos, avail);
buffer += avail;
stream->__bufpos += avail;
if (!(todo -= avail)) {
goto DONE;
}
}
/* We need to read from the host environment, so we must
* flush all line buffered streams if the stream is not
* fully buffered. */
if (!__STDIO_STREAM_IS_FBF(stream)) {
__STDIO_FLUSH_LBF_STREAMS;
}
#endif
#ifdef __UCLIBC_MJN3_ONLY__
#warning CONSIDER: should we refill and read from the buffer sometimes?
#endif
while ((avail = __stdio_READ(stream, buffer, todo)) > 0) {
buffer += avail;
if (!(todo -= avail)) {
break;
}
}
DONE:
__STDIO_STREAM_VALIDATE(stream);
return (bytes - todo) / size;
}
__STDIO_STREAM_SET_ERROR(stream);
__set_errno(EINVAL);
}
__STDIO_STREAM_VALIDATE(stream);
return 0;
}
libc_hidden_def(fread_unlocked)
#ifndef __UCLIBC_HAS_THREADS__
strong_alias(fread_unlocked,fread)
libc_hidden_def(fread)
#endif
#elif defined __UCLIBC_HAS_THREADS__
size_t fread(void * __restrict ptr, size_t size, size_t nmemb,
register FILE * __restrict stream)
{
size_t retval;
__STDIO_AUTO_THREADLOCK_VAR;
__STDIO_AUTO_THREADLOCK(stream);
retval = fread_unlocked(ptr, size, nmemb, stream);
__STDIO_AUTO_THREADUNLOCK(stream);
return retval;
}
libc_hidden_def(fread)
#endif