2022-01-14 01:14:23 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, kleines Filmröllchen <filmroellchen@serenityos.org>.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <AK/OwnPtr.h>
|
2023-01-25 20:19:05 +01:00
|
|
|
#include <AK/Stream.h>
|
|
|
|
#include <AK/Vector.h>
|
2022-01-14 01:14:23 +01:00
|
|
|
|
2023-01-25 20:19:05 +01:00
|
|
|
namespace AK {
|
2022-01-14 01:14:23 +01:00
|
|
|
|
2022-12-07 15:47:44 +01:00
|
|
|
/// A stream class that allows for reading/writing on a preallocated memory area
|
|
|
|
/// using a single read/write head.
|
|
|
|
class FixedMemoryStream final : public SeekableStream {
|
2022-01-14 01:14:23 +01:00
|
|
|
public:
|
2022-12-07 16:22:14 +01:00
|
|
|
static ErrorOr<NonnullOwnPtr<FixedMemoryStream>> construct(Bytes bytes);
|
|
|
|
static ErrorOr<NonnullOwnPtr<FixedMemoryStream>> construct(ReadonlyBytes bytes);
|
2022-01-14 01:14:23 +01:00
|
|
|
|
2022-12-07 16:22:14 +01:00
|
|
|
virtual bool is_eof() const override;
|
|
|
|
virtual bool is_open() const override;
|
|
|
|
virtual void close() override;
|
2023-01-29 13:49:42 +01:00
|
|
|
virtual ErrorOr<void> truncate(size_t) override;
|
2022-12-07 16:22:14 +01:00
|
|
|
virtual ErrorOr<Bytes> read(Bytes bytes) override;
|
2022-11-30 10:03:19 +01:00
|
|
|
|
2023-01-17 14:52:46 +01:00
|
|
|
virtual ErrorOr<size_t> seek(i64 offset, SeekMode seek_mode = SeekMode::SetPosition) override;
|
2022-01-14 01:14:23 +01:00
|
|
|
|
2022-12-07 16:22:14 +01:00
|
|
|
virtual ErrorOr<size_t> write(ReadonlyBytes bytes) override;
|
|
|
|
virtual ErrorOr<void> write_entire_buffer(ReadonlyBytes bytes) override;
|
2022-01-14 01:14:23 +01:00
|
|
|
|
2022-12-07 16:22:14 +01:00
|
|
|
Bytes bytes();
|
|
|
|
ReadonlyBytes bytes() const;
|
|
|
|
size_t offset() const;
|
|
|
|
size_t remaining() const;
|
2022-11-30 10:03:19 +01:00
|
|
|
|
2022-01-14 01:14:23 +01:00
|
|
|
private:
|
2022-12-07 16:22:14 +01:00
|
|
|
explicit FixedMemoryStream(Bytes bytes);
|
|
|
|
explicit FixedMemoryStream(ReadonlyBytes bytes);
|
|
|
|
|
2022-01-14 01:14:23 +01:00
|
|
|
Bytes m_bytes;
|
|
|
|
size_t m_offset { 0 };
|
2022-11-30 10:03:19 +01:00
|
|
|
bool m_writing_enabled { true };
|
2022-01-14 01:14:23 +01:00
|
|
|
};
|
|
|
|
|
2022-12-07 17:43:31 +01:00
|
|
|
/// A stream class that allows for writing to an automatically allocating memory area
|
|
|
|
/// and reading back the written data afterwards.
|
2023-01-25 20:19:05 +01:00
|
|
|
class AllocatingMemoryStream final : public Stream {
|
2022-12-07 17:43:31 +01:00
|
|
|
public:
|
|
|
|
virtual ErrorOr<Bytes> read(Bytes) override;
|
|
|
|
virtual ErrorOr<size_t> write(ReadonlyBytes) override;
|
|
|
|
virtual ErrorOr<void> discard(size_t) override;
|
|
|
|
virtual bool is_eof() const override;
|
|
|
|
virtual bool is_open() const override;
|
|
|
|
virtual void close() override;
|
|
|
|
|
|
|
|
size_t used_buffer_size() const;
|
|
|
|
|
2023-01-13 13:41:14 +01:00
|
|
|
ErrorOr<Optional<size_t>> offset_of(ReadonlyBytes needle) const;
|
|
|
|
|
2022-12-07 17:43:31 +01:00
|
|
|
private:
|
|
|
|
// Note: We set the inline buffer capacity to zero to make moving chunks as efficient as possible.
|
|
|
|
using Chunk = AK::Detail::ByteBuffer<0>;
|
|
|
|
static constexpr size_t chunk_size = 4096;
|
|
|
|
|
|
|
|
ErrorOr<ReadonlyBytes> next_read_range();
|
|
|
|
ErrorOr<Bytes> next_write_range();
|
|
|
|
void cleanup_unused_chunks();
|
|
|
|
|
|
|
|
Vector<Chunk> m_chunks;
|
|
|
|
size_t m_read_offset = 0;
|
|
|
|
size_t m_write_offset = 0;
|
|
|
|
};
|
|
|
|
|
2022-01-14 01:14:23 +01:00
|
|
|
}
|
2023-01-25 20:19:05 +01:00
|
|
|
|
|
|
|
#if USING_AK_GLOBALLY
|
|
|
|
using AK::AllocatingMemoryStream;
|
|
|
|
using AK::FixedMemoryStream;
|
|
|
|
#endif
|