mirror of
https://github.com/reactos/reactos.git
synced 2025-06-23 12:00:17 +00:00

Imported from https://www.nuget.org/packages/Microsoft.Windows.SDK.CRTSource/10.0.22621.3 License: MIT
264 lines
7.6 KiB
C++
264 lines
7.6 KiB
C++
//
|
|
// fflush.cpp
|
|
//
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//
|
|
// Defines fflush() and related functions, which flush stdio streams.
|
|
//
|
|
#include <corecrt_internal_stdio.h>
|
|
#include <corecrt_internal_ptd_propagation.h>
|
|
|
|
|
|
|
|
static bool __cdecl is_stream_allocated(long const stream_flags) throw()
|
|
{
|
|
return (stream_flags & _IOALLOCATED) != 0;
|
|
}
|
|
|
|
static bool __cdecl is_stream_flushable(long const stream_flags) throw()
|
|
{
|
|
if ((stream_flags & (_IOREAD | _IOWRITE)) != _IOWRITE)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if ((stream_flags & (_IOBUFFER_CRT | _IOBUFFER_USER)) == 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool __cdecl is_stream_flushable_or_commitable(long const stream_flags) throw()
|
|
{
|
|
if (is_stream_flushable(stream_flags))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (stream_flags & _IOCOMMIT)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Returns true if the common_flush_all function should attempt to flush the
|
|
// stream; otherwise returns false. This function returns false for streams
|
|
// that are not in use (not allocated) and for streams for which the flush
|
|
// operation will be a no-op.
|
|
//
|
|
// In the case where this function determines that the flush would be a no-op,
|
|
// it increments the flushed_stream_count. This allows common_flush_all to
|
|
// keep track of the number of streams that it would have flushed.
|
|
static bool __cdecl common_flush_all_should_try_to_flush_stream(
|
|
_In_ __crt_stdio_stream const stream,
|
|
_Inout_ int* const flushed_stream_count
|
|
) throw()
|
|
{
|
|
if (!stream.valid())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
long const stream_flags = stream.get_flags();
|
|
if (!is_stream_allocated(stream_flags))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (!is_stream_flushable_or_commitable(stream_flags))
|
|
{
|
|
++*flushed_stream_count;
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
// Internal implementation of the "flush all" functionality. If the
|
|
// flush_read_mode_streams argument is false, only write mode streams are
|
|
// flushed and the return value is zero on success, EOF on failure.
|
|
//
|
|
// If the flush_read_mode_streams argument is true, this function flushes
|
|
// all streams regardless of mode and returns the number of streams that it
|
|
// flushed.
|
|
//
|
|
// Note that in both cases, if we can determine that a call to fflush for a
|
|
// particular stream would be a no-op, then we will not call fflush for that
|
|
// stream. This allows us to avoid acquiring the stream lock unnecessarily,
|
|
// which can help to avoid deadlock-like lock contention.
|
|
//
|
|
// Notably, by doing this, we can avoid attempting to acquire the stream lock
|
|
// for most read mode streams. Attempting to acquire the stream lock for a
|
|
// read mode stream can be problematic beause another thread may hold the lock
|
|
// and be blocked on an I/O operation (e.g., a call to fread on stdin may block
|
|
// until the user types input into the console).
|
|
static int __cdecl common_flush_all(bool const flush_read_mode_streams) throw()
|
|
{
|
|
int count = 0;
|
|
int error = 0;
|
|
|
|
__acrt_lock_and_call(__acrt_stdio_index_lock, [&]
|
|
{
|
|
__crt_stdio_stream_data** const first_file = __piob;
|
|
__crt_stdio_stream_data** const last_file = first_file + _nstream;
|
|
|
|
for (__crt_stdio_stream_data** it = first_file; it != last_file; ++it)
|
|
{
|
|
__crt_stdio_stream const stream(*it);
|
|
|
|
// Before we acquire the stream lock, check to see if flushing the
|
|
// stream would be a no-op. If it would be, then skip this stream.
|
|
if (!common_flush_all_should_try_to_flush_stream(stream, &count))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
__acrt_lock_stream_and_call(stream.public_stream(), [&]
|
|
{
|
|
// Re-verify the state of the stream. Another thread may have
|
|
// closed the stream, reopened it into a different mode, or
|
|
// otherwise altered the state of the stream such that this
|
|
// flush would be a no-op.
|
|
if (!common_flush_all_should_try_to_flush_stream(stream, &count))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!flush_read_mode_streams && !stream.has_all_of(_IOWRITE))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (_fflush_nolock(stream.public_stream()) != EOF)
|
|
{
|
|
++count;
|
|
}
|
|
else
|
|
{
|
|
error = EOF;
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
return flush_read_mode_streams ? count : error;
|
|
}
|
|
|
|
|
|
|
|
// Flushes the buffer of the given stream. If the file is open for writing and
|
|
// is buffered, the buffer is flushed. On success, returns 0. On failure (e.g.
|
|
// if there is an error writing the buffer), returns EOF and sets errno.
|
|
extern "C" int __cdecl fflush(FILE* const public_stream)
|
|
{
|
|
__crt_stdio_stream const stream(public_stream);
|
|
|
|
// If the stream is null, flush all the streams:
|
|
if (!stream.valid())
|
|
{
|
|
return common_flush_all(false);
|
|
}
|
|
|
|
// Before acquiring the stream lock, inspect the stream to see if the flush
|
|
// is a no-op. If it will be a no-op then we can return without attempting
|
|
// to acquire the lock (this can help prevent locking conflicts; see the
|
|
// common_flush_all implementation for more information).
|
|
if (!is_stream_flushable_or_commitable(stream.get_flags()))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
return __acrt_lock_stream_and_call(stream.public_stream(), [&]
|
|
{
|
|
return _fflush_nolock(stream.public_stream());
|
|
});
|
|
}
|
|
|
|
|
|
|
|
static int __cdecl _fflush_nolock_internal(FILE* const public_stream, __crt_cached_ptd_host& ptd)
|
|
{
|
|
__crt_stdio_stream const stream(public_stream);
|
|
|
|
// If the stream is null, flush all the streams.
|
|
if (!stream.valid())
|
|
{
|
|
return common_flush_all(false);
|
|
}
|
|
|
|
if (__acrt_stdio_flush_nolock(stream.public_stream(), ptd) != 0)
|
|
{
|
|
// If the flush fails, do not attempt to commit:
|
|
return EOF;
|
|
}
|
|
|
|
// Perform the lowio commit to ensure data is written to disk:
|
|
if (stream.has_all_of(_IOCOMMIT))
|
|
{
|
|
if (_commit(_fileno(public_stream)))
|
|
{
|
|
return EOF;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
extern "C" int __cdecl _fflush_nolock(FILE* const public_stream)
|
|
{
|
|
__crt_cached_ptd_host ptd;
|
|
return _fflush_nolock_internal(public_stream, ptd);
|
|
}
|
|
|
|
// Flushes the buffer of the given stream. If the file is open for writing and
|
|
// is buffered, the buffer is flushed. On success, returns 0. On failure (e.g.
|
|
// if there is an error writing the buffer), returns EOF and sets errno.
|
|
extern "C" int __cdecl __acrt_stdio_flush_nolock(FILE* const public_stream, __crt_cached_ptd_host& ptd)
|
|
{
|
|
__crt_stdio_stream const stream(public_stream);
|
|
|
|
if (!is_stream_flushable(stream.get_flags()))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int const bytes_to_write = static_cast<int>(stream->_ptr - stream->_base);
|
|
|
|
__acrt_stdio_reset_buffer(stream);
|
|
|
|
if (bytes_to_write <= 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int const bytes_written = _write_internal(_fileno(stream.public_stream()), stream->_base, bytes_to_write, ptd);
|
|
if (bytes_to_write != bytes_written)
|
|
{
|
|
stream.set_flags(_IOERROR);
|
|
return EOF;
|
|
}
|
|
|
|
// If this is a read/write file, clear _IOWRITE so that the next operation can
|
|
// be a read:
|
|
if (stream.has_all_of(_IOUPDATE))
|
|
{
|
|
stream.unset_flags(_IOWRITE);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
// Flushes the buffers for all output streams and clears all input buffers.
|
|
// Returns the number of open streams.
|
|
extern "C" int __cdecl _flushall()
|
|
{
|
|
return common_flush_all(true);
|
|
}
|