mirror of
https://github.com/reactos/reactos.git
synced 2025-06-01 15:38:37 +00:00
255 lines
9.3 KiB
C++
255 lines
9.3 KiB
C++
//
|
|
// fread.cpp
|
|
//
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//
|
|
// Defines fread() and related functions, which read unformatted data from a
|
|
// stdio stream.
|
|
//
|
|
#include <corecrt_internal_stdio.h>
|
|
|
|
#ifdef _DEBUG
|
|
#define _BUFFER_FILL_PATTERN _SECURECRT_FILL_BUFFER_PATTERN
|
|
#else
|
|
#define _BUFFER_FILL_PATTERN 0
|
|
#endif
|
|
|
|
|
|
|
|
// Reads data from a stream into the result buffer. The function reads elements
|
|
// of size 'element_size' until it has read 'element_count' elements, until the
|
|
// buffer is full, or until EOF is reached.
|
|
//
|
|
// Returns the number of "whole" elements that were read into the buffer. This
|
|
// may be fewer than the requested number of elements if an error occurs or if
|
|
// EOF is encountered. In this case, ferror() or feof() should be used to
|
|
// distinguish between the two conditions.
|
|
//
|
|
// If the result buffer becomes full before the requested number of elements are
|
|
// read, the buffer is zero-filled, zero is returned, and errno is set to ERANGE.
|
|
extern "C" size_t __cdecl fread_s(
|
|
void* const buffer,
|
|
size_t const buffer_size,
|
|
size_t const element_size,
|
|
size_t const element_count,
|
|
FILE* const stream
|
|
)
|
|
{
|
|
if (element_size == 0 || element_count == 0)
|
|
return 0;
|
|
|
|
// The rest of the argument validation is done in the _nolock function. Here
|
|
// we only need to validate that the stream is non-null before we lock it.
|
|
if (stream == nullptr)
|
|
{
|
|
if (buffer_size != _CRT_UNBOUNDED_BUFFER_SIZE)
|
|
memset(buffer, _BUFFER_FILL_PATTERN, buffer_size);
|
|
|
|
_VALIDATE_RETURN(stream != nullptr, EINVAL, 0);
|
|
}
|
|
|
|
size_t return_value = 0;
|
|
|
|
_lock_file(stream);
|
|
__try
|
|
{
|
|
return_value = _fread_nolock_s(buffer, buffer_size, element_size, element_count, stream);
|
|
}
|
|
__finally
|
|
{
|
|
_unlock_file(stream);
|
|
}
|
|
__endtry
|
|
|
|
return return_value;
|
|
}
|
|
|
|
|
|
|
|
extern "C" size_t __cdecl _fread_nolock_s(
|
|
void* const buffer,
|
|
size_t const buffer_size,
|
|
size_t const element_size,
|
|
size_t const element_count,
|
|
FILE* const public_stream
|
|
)
|
|
{
|
|
__crt_stdio_stream const stream(public_stream);
|
|
|
|
if (element_size == 0 || element_count == 0)
|
|
return 0;
|
|
|
|
_VALIDATE_RETURN(buffer != nullptr, EINVAL, 0);
|
|
if (!stream.valid() || element_count > (SIZE_MAX / element_size))
|
|
{
|
|
if (buffer_size != _CRT_UNBOUNDED_BUFFER_SIZE)
|
|
memset(buffer, _BUFFER_FILL_PATTERN, buffer_size);
|
|
|
|
_VALIDATE_RETURN(stream.valid(), EINVAL, 0);
|
|
_VALIDATE_RETURN(element_count <= (SIZE_MAX / element_size), EINVAL, 0);
|
|
}
|
|
|
|
// Figure out how big the buffer is; if the stream doesn't currently have a
|
|
// buffer, we assume that we'll get one with the usual internal buffer size:
|
|
unsigned stream_buffer_size = stream.has_any_buffer()
|
|
? stream->_bufsiz
|
|
: _INTERNAL_BUFSIZ;
|
|
|
|
// The total number of bytes to be read into the buffer:
|
|
size_t const total_bytes = element_size * element_count;
|
|
|
|
char* data = static_cast<char*>(buffer);
|
|
|
|
// Read blocks of data from the stream until we have read the requested
|
|
// number of elements or we fill the buffer.
|
|
size_t remaining_bytes = total_bytes;
|
|
size_t remaining_buffer = buffer_size;
|
|
while (remaining_bytes != 0)
|
|
{
|
|
// If the stream is buffered and has characters, copy them into the
|
|
// result buffer:
|
|
if (stream.has_any_buffer() && stream->_cnt != 0)
|
|
{
|
|
if(stream->_cnt < 0)
|
|
{
|
|
_ASSERTE(("Inconsistent Stream Count. Flush between consecutive read and write", stream->_cnt >= 0));
|
|
stream.set_flags(_IOERROR);
|
|
return (total_bytes - remaining_bytes) / element_size;
|
|
}
|
|
|
|
unsigned const bytes_to_read = remaining_bytes < static_cast<size_t>(stream->_cnt)
|
|
? static_cast<unsigned>(remaining_bytes)
|
|
: static_cast<unsigned>(stream->_cnt);
|
|
|
|
if (bytes_to_read > remaining_buffer)
|
|
{
|
|
if (buffer_size != _CRT_UNBOUNDED_BUFFER_SIZE)
|
|
memset(buffer, _BUFFER_FILL_PATTERN, buffer_size);
|
|
|
|
_VALIDATE_RETURN(("buffer too small", 0), ERANGE, 0)
|
|
}
|
|
|
|
memcpy_s(data, remaining_buffer, stream->_ptr, bytes_to_read);
|
|
|
|
// Update the stream and local tracking variables to account for the
|
|
// read. Note that the number of bytes actually read is always equal
|
|
// to the number of bytes that we expected to read, because the data
|
|
// was already buffered in the stream.
|
|
remaining_bytes -= bytes_to_read;
|
|
stream->_cnt -= bytes_to_read;
|
|
stream->_ptr += bytes_to_read;
|
|
data += bytes_to_read;
|
|
remaining_buffer -= bytes_to_read;
|
|
}
|
|
// There is no data remaining in the stream buffer to be read, and we
|
|
// need to read more data than will fit in the buffer (or we need to read
|
|
// at least enough data to fill the buffer completely):
|
|
else if (remaining_bytes >= stream_buffer_size)
|
|
{
|
|
// We can read at most INT_MAX bytes at a time. This is a hard limit
|
|
// of the lowio _read() function.
|
|
unsigned const maximum_bytes_to_read = remaining_bytes > INT_MAX
|
|
? static_cast<unsigned>(INT_MAX)
|
|
: static_cast<unsigned>(remaining_bytes);
|
|
|
|
// If the stream has a buffer, we want to read the largest chunk that
|
|
// is a multiple of the buffer size, to keep the stream buffer state
|
|
// consistent. If the stream is not buffered, we can read the maximum
|
|
// number of bytes that we can:
|
|
unsigned const bytes_to_read = stream_buffer_size != 0
|
|
? static_cast<unsigned>(maximum_bytes_to_read - maximum_bytes_to_read % stream_buffer_size)
|
|
: maximum_bytes_to_read;
|
|
|
|
if (bytes_to_read > remaining_buffer)
|
|
{
|
|
if (buffer_size != _CRT_UNBOUNDED_BUFFER_SIZE)
|
|
memset(buffer, _BUFFER_FILL_PATTERN, buffer_size);
|
|
|
|
_VALIDATE_RETURN(("buffer too small", 0), ERANGE, 0)
|
|
}
|
|
|
|
// We are about to read data directly from the underlying file
|
|
// descriptor, bypassing the stream buffer. We reset the stream
|
|
// buffer state to ensure that future seeks do not incorrectly
|
|
// assume that the buffer contents are valid.
|
|
__acrt_stdio_reset_buffer(stream);
|
|
|
|
// Do the read. Note that if the stream is open in text mode, the
|
|
// bytes_read may not be the same as the bytes_to_read, due to
|
|
// newline translation.
|
|
int const bytes_read = _read_nolock(_fileno(stream.public_stream()), data, bytes_to_read);
|
|
if (bytes_read == 0)
|
|
{
|
|
// We encountered EOF:
|
|
stream.set_flags(_IOEOF);
|
|
return (total_bytes - remaining_bytes) / element_size;
|
|
}
|
|
else if (bytes_read < 0)
|
|
{
|
|
// The _read failed:
|
|
stream.set_flags(_IOERROR);
|
|
return (total_bytes - remaining_bytes) / element_size;
|
|
}
|
|
|
|
// Update the iteration state to reflect the read:
|
|
remaining_bytes -= bytes_read;
|
|
data += bytes_read;
|
|
remaining_buffer -= bytes_read;
|
|
}
|
|
// Otherwise, the stream does not have a buffer, or the stream buffer
|
|
// is full and there is insufficient space to do a direct read, so use
|
|
// __acrt_stdio_refill_and_read_narrow_nolock:
|
|
else
|
|
{
|
|
int const c = __acrt_stdio_refill_and_read_narrow_nolock(stream.public_stream());
|
|
if (c == EOF)
|
|
return (total_bytes - remaining_bytes) / element_size;
|
|
|
|
// If we have filled the result buffer before we have read the
|
|
// requested number of elements or reached EOF, it is an error:
|
|
if (remaining_buffer == 0)
|
|
{
|
|
if (buffer_size != _CRT_UNBOUNDED_BUFFER_SIZE)
|
|
memset(buffer, _BUFFER_FILL_PATTERN, buffer_size);
|
|
|
|
_VALIDATE_RETURN(("buffer too small", 0), ERANGE, 0)
|
|
}
|
|
|
|
*data++ = static_cast<char>(c);
|
|
--remaining_bytes;
|
|
--remaining_buffer;
|
|
|
|
stream_buffer_size = stream->_bufsiz;
|
|
}
|
|
}
|
|
|
|
return element_count; // Success!
|
|
}
|
|
|
|
|
|
|
|
extern "C" size_t __cdecl fread(
|
|
void* const buffer,
|
|
size_t const element_size,
|
|
size_t const element_count,
|
|
FILE* const stream
|
|
)
|
|
{
|
|
// Assume there is enough space in the destination buffer
|
|
#pragma warning(suppress:__WARNING_POTENTIAL_BUFFER_OVERFLOW_HIGH_PRIORITY) // 26015 - fread is unsafe
|
|
return fread_s(buffer, _CRT_UNBOUNDED_BUFFER_SIZE, element_size, element_count, stream);
|
|
}
|
|
|
|
|
|
|
|
extern "C" size_t __cdecl _fread_nolock(
|
|
void* const buffer,
|
|
size_t const element_size,
|
|
size_t const element_count,
|
|
FILE* const stream
|
|
)
|
|
{
|
|
// Assume there is enough space in the destination buffer
|
|
#pragma warning(suppress:__WARNING_POTENTIAL_BUFFER_OVERFLOW_HIGH_PRIORITY) // 26015 - _fread_nolock is unsafe
|
|
return _fread_nolock_s(buffer, _CRT_UNBOUNDED_BUFFER_SIZE, element_size, element_count, stream);
|
|
}
|