Gzip.h 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. /*
  2. * Copyright (c) 2020-2022, the SerenityOS developers.
  3. * Copyright (c) 2021, Idan Horowitz <idan.horowitz@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/NonnullOwnPtr.h>
  9. #include <AK/OwnPtr.h>
  10. #include <AK/Stream.h>
  11. #include <LibCompress/Deflate.h>
  12. #include <LibCrypto/Checksum/CRC32.h>
  13. namespace Compress {
  14. constexpr u8 gzip_magic_1 = 0x1f;
  15. constexpr u8 gzip_magic_2 = 0x8b;
  16. struct [[gnu::packed]] BlockHeader {
  17. u8 identification_1;
  18. u8 identification_2;
  19. u8 compression_method;
  20. u8 flags;
  21. LittleEndian<u32> modification_time;
  22. u8 extra_flags;
  23. u8 operating_system;
  24. bool valid_magic_number() const;
  25. bool supported_by_implementation() const;
  26. };
  27. struct Flags {
  28. static constexpr u8 FTEXT = 1 << 0;
  29. static constexpr u8 FHCRC = 1 << 1;
  30. static constexpr u8 FEXTRA = 1 << 2;
  31. static constexpr u8 FNAME = 1 << 3;
  32. static constexpr u8 FCOMMENT = 1 << 4;
  33. static constexpr u8 MAX = FTEXT | FHCRC | FEXTRA | FNAME | FCOMMENT;
  34. };
  35. class GzipDecompressor final : public Stream {
  36. public:
  37. GzipDecompressor(MaybeOwned<Stream>);
  38. ~GzipDecompressor();
  39. virtual ErrorOr<Bytes> read_some(Bytes) override;
  40. virtual ErrorOr<size_t> write_some(ReadonlyBytes) override;
  41. virtual bool is_eof() const override;
  42. virtual bool is_open() const override { return true; }
  43. virtual void close() override {};
  44. static ErrorOr<ByteBuffer> decompress_all(ReadonlyBytes);
  45. static ErrorOr<void> decompress_file(StringView input_file, NonnullOwnPtr<Stream> output_stream);
  46. static ErrorOr<Optional<String>> describe_header(ReadonlyBytes);
  47. static bool is_likely_compressed(ReadonlyBytes bytes);
  48. private:
  49. class Member {
  50. public:
  51. static ErrorOr<NonnullOwnPtr<Member>> construct(BlockHeader header, LittleEndianInputBitStream&);
  52. BlockHeader m_header;
  53. NonnullOwnPtr<DeflateDecompressor> m_stream;
  54. Crypto::Checksum::CRC32 m_checksum;
  55. size_t m_nread { 0 };
  56. private:
  57. Member(BlockHeader, NonnullOwnPtr<DeflateDecompressor>);
  58. };
  59. Member const& current_member() const { return *m_current_member; }
  60. Member& current_member() { return *m_current_member; }
  61. NonnullOwnPtr<LittleEndianInputBitStream> m_input_stream;
  62. u8 m_partial_header[sizeof(BlockHeader)];
  63. size_t m_partial_header_offset { 0 };
  64. OwnPtr<Member> m_current_member {};
  65. bool m_eof { false };
  66. };
  67. class GzipCompressor final : public Stream {
  68. public:
  69. GzipCompressor(MaybeOwned<Stream>);
  70. virtual ErrorOr<Bytes> read_some(Bytes) override;
  71. virtual ErrorOr<size_t> write_some(ReadonlyBytes) override;
  72. virtual bool is_eof() const override;
  73. virtual bool is_open() const override;
  74. virtual void close() override;
  75. static ErrorOr<ByteBuffer> compress_all(ReadonlyBytes bytes);
  76. static ErrorOr<void> compress_file(StringView input_file, NonnullOwnPtr<Stream> output_stream);
  77. private:
  78. MaybeOwned<Stream> m_output_stream;
  79. };
  80. }