/* * Copyright (c) 2021, kleines Filmröllchen . * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include namespace Core::Stream { class MemoryStream final : public SeekableStream { public: static ErrorOr> construct(Bytes bytes) { return adopt_nonnull_own_or_enomem(new (nothrow) MemoryStream(bytes)); } virtual bool is_eof() const override { return m_offset >= m_bytes.size(); } virtual bool is_open() const override { return true; } // FIXME: It doesn't make sense to close an memory stream. Therefore, we don't do anything here. Is that fine? virtual void close() override { } // FIXME: It doesn't make sense to truncate a memory stream. Therefore, we don't do anything here. Is that fine? virtual ErrorOr truncate(off_t) override { return Error::from_errno(ENOTSUP); } virtual ErrorOr read(Bytes bytes) override { auto to_read = min(remaining(), bytes.size()); if (to_read == 0) return Bytes {}; m_bytes.slice(m_offset, to_read).copy_to(bytes); m_offset += to_read; return bytes.trim(to_read); } virtual ErrorOr seek(i64 offset, SeekMode seek_mode = SeekMode::SetPosition) override { switch (seek_mode) { case SeekMode::SetPosition: if (offset >= static_cast(m_bytes.size())) return Error::from_string_literal("Offset past the end of the stream memory"sv); m_offset = offset; break; case SeekMode::FromCurrentPosition: if (offset + static_cast(m_offset) >= static_cast(m_bytes.size())) return Error::from_string_literal("Offset past the end of the stream memory"sv); m_offset += offset; break; case SeekMode::FromEndPosition: if (offset >= static_cast(m_bytes.size())) return Error::from_string_literal("Offset past the start of the stream memory"sv); m_offset = m_bytes.size() - offset; break; } return static_cast(m_offset); } virtual ErrorOr write(ReadonlyBytes bytes) override { // FIXME: Can this not error? auto const nwritten = bytes.copy_trimmed_to(m_bytes.slice(m_offset)); m_offset += nwritten; return nwritten; } virtual bool write_or_error(ReadonlyBytes bytes) override { if (remaining() < bytes.size()) return false; MUST(write(bytes)); return true; } Bytes bytes() { return m_bytes; } ReadonlyBytes bytes() const { return m_bytes; } size_t offset() const { return m_offset; } size_t remaining() const { return m_bytes.size() - m_offset; } protected: explicit MemoryStream(Bytes bytes) : m_bytes(bytes) { } private: Bytes m_bytes; size_t m_offset { 0 }; }; }