1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-05-28 11:15:10 +00:00

AK: Lower the requirements for InputStream::eof and rename it.

Consider the following snippet:

    void foo(InputStream& stream) {
        if(!stream.eof()) {
            u8 byte;
            stream >> byte;
        }
    }

There is a very subtle bug in this snippet, for some input streams eof()
might return false even if no more data can be read. In this case an
error flag would be set on the stream.

Until now I've always ensured that this is not the case, but this made
the implementation of eof() unnecessarily complicated.
InputFileStream::eof had to keep a ByteBuffer around just to make this
possible. That meant a ton of unnecessary copies just to get a reliable
eof().

In most cases it isn't actually necessary to have a reliable eof()
implementation.

In most other cases a reliable eof() is avaliable anyways because in
some cases like InputMemoryStream it is very easy to implement.
This commit is contained in:
asynts 2020-09-13 12:24:17 +02:00 committed by Andreas Kling
parent 8a21c528ad
commit 96edcbc27c
12 changed files with 61 additions and 89 deletions

View file

@ -68,7 +68,7 @@ GzipDecompressor::~GzipDecompressor()
// FIXME: Again, there are surely a ton of bugs because the code doesn't check for read errors.
size_t GzipDecompressor::read(Bytes bytes)
{
if (has_any_error())
if (has_any_error() || m_eof)
return 0;
if (m_current_member.has_value()) {
@ -99,12 +99,14 @@ size_t GzipDecompressor::read(Bytes bytes)
return nread;
} else {
if (m_input_stream.eof())
return 0;
BlockHeader header;
m_input_stream >> Bytes { &header, sizeof(header) };
if (m_input_stream.handle_any_error()) {
m_eof = true;
return 0;
}
if (!header.valid_magic_number() || !header.supported_by_implementation()) {
set_fatal_error();
return 0;
@ -147,7 +149,7 @@ bool GzipDecompressor::discard_or_error(size_t count)
size_t ndiscarded = 0;
while (ndiscarded < count) {
if (eof()) {
if (unreliable_eof()) {
set_fatal_error();
return false;
}
@ -165,7 +167,7 @@ Optional<ByteBuffer> GzipDecompressor::decompress_all(ReadonlyBytes bytes)
OutputMemoryStream output_stream;
u8 buffer[4096];
while (!gzip_stream.has_any_error() && !gzip_stream.eof()) {
while (!gzip_stream.has_any_error() && !gzip_stream.unreliable_eof()) {
const auto nread = gzip_stream.read({ buffer, sizeof(buffer) });
output_stream.write_or_error({ buffer, nread });
}
@ -176,15 +178,6 @@ Optional<ByteBuffer> GzipDecompressor::decompress_all(ReadonlyBytes bytes)
return output_stream.copy_into_contiguous_buffer();
}
bool GzipDecompressor::eof() const
{
if (m_current_member.has_value()) {
// FIXME: There is an ugly edge case where we read the whole deflate block
// but haven't read CRC32 and ISIZE.
return current_member().m_stream.eof() && m_input_stream.eof();
} else {
return m_input_stream.eof();
}
}
bool GzipDecompressor::unreliable_eof() const { return m_eof; }
}