2022-01-14 03:14:23 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, kleines Filmröllchen <filmroellchen@serenityos.org>.
|
2023-10-06 19:03:54 +03:00
|
|
|
* Copyright (c) 2023, Sam Atkins <atkinssj@serenityos.org>.
|
2022-01-14 03:14:23 +03:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2023-10-06 19:03:54 +03:00
|
|
|
#include <AK/Error.h>
|
2022-01-14 03:14:23 +03:00
|
|
|
#include <AK/OwnPtr.h>
|
2023-01-25 22:19:05 +03:00
|
|
|
#include <AK/Stream.h>
|
|
|
|
#include <AK/Vector.h>
|
2022-01-14 03:14:23 +03:00
|
|
|
|
2023-01-25 22:19:05 +03:00
|
|
|
namespace AK {
|
2022-01-14 03:14:23 +03:00
|
|
|
|
2022-12-07 17:47:44 +03:00
|
|
|
/// A stream class that allows for reading/writing on a preallocated memory area
|
|
|
|
/// using a single read/write head.
|
2023-09-19 13:01:04 +03:00
|
|
|
class FixedMemoryStream : public SeekableStream {
|
2022-01-14 03:14:23 +03:00
|
|
|
public:
|
2023-11-05 23:31:17 +03:00
|
|
|
enum class Mode {
|
|
|
|
ReadOnly,
|
|
|
|
ReadWrite,
|
|
|
|
};
|
|
|
|
|
|
|
|
explicit FixedMemoryStream(Bytes bytes, Mode mode = Mode::ReadWrite);
|
2023-01-30 13:05:43 +03:00
|
|
|
explicit FixedMemoryStream(ReadonlyBytes bytes);
|
2022-01-14 03:14:23 +03:00
|
|
|
|
2022-12-07 18:22:14 +03:00
|
|
|
virtual bool is_eof() const override;
|
|
|
|
virtual bool is_open() const override;
|
|
|
|
virtual void close() override;
|
2023-01-29 15:49:42 +03:00
|
|
|
virtual ErrorOr<void> truncate(size_t) override;
|
2023-02-25 00:38:01 +03:00
|
|
|
virtual ErrorOr<Bytes> read_some(Bytes bytes) override;
|
2023-06-06 20:52:56 +03:00
|
|
|
virtual ErrorOr<void> read_until_filled(Bytes bytes) override;
|
2022-11-30 12:03:19 +03:00
|
|
|
|
2023-01-17 16:52:46 +03:00
|
|
|
virtual ErrorOr<size_t> seek(i64 offset, SeekMode seek_mode = SeekMode::SetPosition) override;
|
2022-01-14 03:14:23 +03:00
|
|
|
|
2023-02-25 00:38:01 +03:00
|
|
|
virtual ErrorOr<size_t> write_some(ReadonlyBytes bytes) override;
|
2023-03-01 17:37:45 +03:00
|
|
|
virtual ErrorOr<void> write_until_depleted(ReadonlyBytes bytes) override;
|
2022-01-14 03:14:23 +03:00
|
|
|
|
2022-12-07 18:22:14 +03:00
|
|
|
size_t offset() const;
|
|
|
|
size_t remaining() const;
|
2022-11-30 12:03:19 +03:00
|
|
|
|
2023-10-06 19:03:54 +03:00
|
|
|
/// Read a value, but referring to the stream's underlying data instead of copying it.
|
|
|
|
/// Of course, only use this if you know the lifetime of the data will exceed the value's.
|
|
|
|
// FIXME: Would be nicer to be able to return T& but Variant (and thus ErrorOr) can't hold references.
|
|
|
|
template<typename T>
|
|
|
|
requires(Traits<T>::is_trivially_serializable())
|
|
|
|
ErrorOr<T*> read_in_place()
|
|
|
|
{
|
|
|
|
if constexpr (!IsConst<T>) {
|
|
|
|
if (!m_writing_enabled)
|
|
|
|
return Error::from_string_view_or_print_error_and_return_errno("Tried to obtain a non-const reference from a read-only FixedMemoryStream"sv, EINVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
T* value = reinterpret_cast<T*>(m_bytes.offset_pointer(m_offset));
|
|
|
|
TRY(discard(sizeof(T)));
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read a span of values, referring to the stream's underlying data instead of copying it.
|
|
|
|
/// Of course, only use this if you know the lifetime of the data will exceed the span's.
|
|
|
|
template<typename T>
|
|
|
|
requires(Traits<T>::is_trivially_serializable())
|
|
|
|
ErrorOr<Span<T>> read_in_place(size_t count)
|
|
|
|
{
|
|
|
|
if constexpr (!IsConst<T>) {
|
|
|
|
if (!m_writing_enabled)
|
|
|
|
return Error::from_string_view_or_print_error_and_return_errno("Tried to obtain a non-const span from a read-only FixedMemoryStream"sv, EINVAL);
|
|
|
|
}
|
|
|
|
|
2023-11-01 17:56:32 +03:00
|
|
|
Span<T> span { reinterpret_cast<T*>(m_bytes.offset_pointer(m_offset)), count };
|
2023-10-06 19:03:54 +03:00
|
|
|
TRY(discard(sizeof(T) * count));
|
|
|
|
return span;
|
|
|
|
}
|
|
|
|
|
2022-01-14 03:14:23 +03:00
|
|
|
private:
|
|
|
|
Bytes m_bytes;
|
|
|
|
size_t m_offset { 0 };
|
2022-11-30 12:03:19 +03:00
|
|
|
bool m_writing_enabled { true };
|
2022-01-14 03:14:23 +03:00
|
|
|
};
|
|
|
|
|
2022-12-07 19:43:31 +03:00
|
|
|
/// A stream class that allows for writing to an automatically allocating memory area
|
|
|
|
/// and reading back the written data afterwards.
|
2023-01-25 22:19:05 +03:00
|
|
|
class AllocatingMemoryStream final : public Stream {
|
2022-12-07 19:43:31 +03:00
|
|
|
public:
|
2023-05-29 12:00:25 +03:00
|
|
|
static constexpr size_t CHUNK_SIZE = 4096;
|
|
|
|
|
2023-02-25 00:38:01 +03:00
|
|
|
virtual ErrorOr<Bytes> read_some(Bytes) override;
|
|
|
|
virtual ErrorOr<size_t> write_some(ReadonlyBytes) override;
|
2022-12-07 19:43:31 +03:00
|
|
|
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 15:41:14 +03:00
|
|
|
ErrorOr<Optional<size_t>> offset_of(ReadonlyBytes needle) const;
|
|
|
|
|
2022-12-07 19:43:31 +03: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>;
|
|
|
|
|
|
|
|
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 03:14:23 +03:00
|
|
|
}
|
2023-01-25 22:19:05 +03:00
|
|
|
|
|
|
|
#if USING_AK_GLOBALLY
|
|
|
|
using AK::AllocatingMemoryStream;
|
|
|
|
using AK::FixedMemoryStream;
|
|
|
|
#endif
|