LZBSDecompressor.cpp 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475
  1. /* Copyright (C) Teemu Suutari */
  2. #include "LZBSDecompressor.hpp"
  3. #include "InputStream.hpp"
  4. #include "OutputStream.hpp"
  5. #include "common/Common.hpp"
  6. namespace ancient::internal
  7. {
  8. bool LZBSDecompressor::detectHeaderXPK(uint32_t hdr) noexcept
  9. {
  10. return hdr==FourCC("LZBS");
  11. }
  12. std::shared_ptr<XPKDecompressor> LZBSDecompressor::create(uint32_t hdr,uint32_t recursionLevel,const Buffer &packedData,std::shared_ptr<XPKDecompressor::State> &state,bool verify)
  13. {
  14. return std::make_shared<LZBSDecompressor>(hdr,recursionLevel,packedData,state,verify);
  15. }
  16. LZBSDecompressor::LZBSDecompressor(uint32_t hdr,uint32_t recursionLevel,const Buffer &packedData,std::shared_ptr<XPKDecompressor::State> &state,bool verify) :
  17. XPKDecompressor(recursionLevel),
  18. _packedData(packedData)
  19. {
  20. if (!detectHeaderXPK(hdr) || _packedData.size()<1) throw Decompressor::InvalidFormatError();
  21. }
  22. LZBSDecompressor::~LZBSDecompressor()
  23. {
  24. // nothing needed
  25. }
  26. const std::string &LZBSDecompressor::getSubName() const noexcept
  27. {
  28. static std::string name="XPK-LZBS: LZBS CyberYAFA compressor";
  29. return name;
  30. }
  31. void LZBSDecompressor::decompressImpl(Buffer &rawData,const Buffer &previousData,bool verify)
  32. {
  33. ForwardInputStream inputStream(_packedData,1,_packedData.size());
  34. MSBBitReader<ForwardInputStream> bitReader(inputStream);
  35. auto readBits=[&](uint32_t count)->uint32_t
  36. {
  37. return rotateBits(bitReader.readBits8(count),count);
  38. };
  39. ForwardOutputStream outputStream(rawData,0,rawData.size());
  40. uint32_t bits=0,maxBits=uint32_t(_packedData[0]);
  41. while (!outputStream.eof())
  42. {
  43. if (!readBits(1))
  44. {
  45. outputStream.writeByte(readBits(8));
  46. } else {
  47. uint32_t count=readBits(8)+2;
  48. if (count==2)
  49. {
  50. count=readBits(12);
  51. if (!count) throw Decompressor::DecompressionError();
  52. for (uint32_t i=0;i<count;i++)
  53. outputStream.writeByte(readBits(8));
  54. } else {
  55. while (outputStream.getOffset()>=(1ULL<<bits) && bits<maxBits) bits++;
  56. uint32_t distance=readBits(bits);
  57. outputStream.copy(distance,count);
  58. }
  59. }
  60. }
  61. }
  62. }