MemoryStream.h 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. /*
  2. * Copyright (c) 2021, kleines Filmröllchen <filmroellchen@serenityos.org>.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/Error.h>
  8. #include <AK/NonnullOwnPtr.h>
  9. #include <AK/OwnPtr.h>
  10. #include <AK/Span.h>
  11. #include <AK/TypedTransfer.h>
  12. #include <LibCore/Stream.h>
  13. namespace Core::Stream {
  14. /// A stream class that allows for reading/writing on a preallocated memory area
  15. /// using a single read/write head.
  16. class FixedMemoryStream final : public SeekableStream {
  17. public:
  18. static ErrorOr<NonnullOwnPtr<FixedMemoryStream>> construct(Bytes bytes);
  19. static ErrorOr<NonnullOwnPtr<FixedMemoryStream>> construct(ReadonlyBytes bytes);
  20. virtual bool is_eof() const override;
  21. virtual bool is_open() const override;
  22. virtual void close() override;
  23. virtual ErrorOr<void> truncate(off_t) override;
  24. virtual ErrorOr<Bytes> read(Bytes bytes) override;
  25. virtual ErrorOr<size_t> seek(i64 offset, SeekMode seek_mode = SeekMode::SetPosition) override;
  26. virtual ErrorOr<size_t> write(ReadonlyBytes bytes) override;
  27. virtual ErrorOr<void> write_entire_buffer(ReadonlyBytes bytes) override;
  28. Bytes bytes();
  29. ReadonlyBytes bytes() const;
  30. size_t offset() const;
  31. size_t remaining() const;
  32. private:
  33. explicit FixedMemoryStream(Bytes bytes);
  34. explicit FixedMemoryStream(ReadonlyBytes bytes);
  35. Bytes m_bytes;
  36. size_t m_offset { 0 };
  37. bool m_writing_enabled { true };
  38. };
  39. /// A stream class that allows for writing to an automatically allocating memory area
  40. /// and reading back the written data afterwards.
  41. class AllocatingMemoryStream final : public Stream {
  42. public:
  43. virtual ErrorOr<Bytes> read(Bytes) override;
  44. virtual ErrorOr<size_t> write(ReadonlyBytes) override;
  45. virtual ErrorOr<void> discard(size_t) override;
  46. virtual bool is_eof() const override;
  47. virtual bool is_open() const override;
  48. virtual void close() override;
  49. size_t used_buffer_size() const;
  50. ErrorOr<Optional<size_t>> offset_of(ReadonlyBytes needle) const;
  51. private:
  52. // Note: We set the inline buffer capacity to zero to make moving chunks as efficient as possible.
  53. using Chunk = AK::Detail::ByteBuffer<0>;
  54. static constexpr size_t chunk_size = 4096;
  55. ErrorOr<ReadonlyBytes> next_read_range();
  56. ErrorOr<Bytes> next_write_range();
  57. void cleanup_unused_chunks();
  58. Vector<Chunk> m_chunks;
  59. size_t m_read_offset = 0;
  60. size_t m_write_offset = 0;
  61. };
  62. }