1
0

LIN1Decompressor.cpp 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. /* Copyright (C) Teemu Suutari */
  2. #include "LIN1Decompressor.hpp"
  3. #include "InputStream.hpp"
  4. #include "OutputStream.hpp"
  5. #include "common/Common.hpp"
  6. namespace ancient::internal
  7. {
  8. bool LIN1Decompressor::detectHeaderXPK(uint32_t hdr) noexcept
  9. {
  10. return hdr==FourCC("LIN1") || hdr==FourCC("LIN3");
  11. }
  12. std::shared_ptr<XPKDecompressor> LIN1Decompressor::create(uint32_t hdr,uint32_t recursionLevel,const Buffer &packedData,std::shared_ptr<XPKDecompressor::State> &state,bool verify)
  13. {
  14. return std::make_shared<LIN1Decompressor>(hdr,recursionLevel,packedData,state,verify);
  15. }
  16. LIN1Decompressor::LIN1Decompressor(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)) throw Decompressor::InvalidFormatError();
  21. _ver=(hdr==FourCC("LIN1"))?1:3;
  22. if (packedData.size()<5) throw Decompressor::InvalidFormatError();
  23. uint32_t tmp=packedData.readBE32(0);
  24. if (tmp) throw Decompressor::InvalidFormatError(); // password set
  25. }
  26. LIN1Decompressor::~LIN1Decompressor()
  27. {
  28. // nothing needed
  29. }
  30. const std::string &LIN1Decompressor::getSubName() const noexcept
  31. {
  32. static std::string name1="XPK-LIN1: LIN1 LINO packer";
  33. static std::string name3="XPK-LIN3: LIN3 LINO packer";
  34. return (_ver==1)?name1:name3;
  35. }
  36. void LIN1Decompressor::decompressImpl(Buffer &rawData,const Buffer &previousData,bool verify)
  37. {
  38. ForwardInputStream inputStream(_packedData,5,_packedData.size());
  39. MSBBitReader<ForwardInputStream> bitReader(inputStream);
  40. auto readBits=[&](uint32_t count)->uint32_t
  41. {
  42. return bitReader.readBits8(count);
  43. };
  44. auto readByte=[&]()->uint8_t
  45. {
  46. return inputStream.readByte();
  47. };
  48. size_t rawSize=rawData.size();
  49. ForwardOutputStream outputStream(rawData,0,rawSize);
  50. while (!outputStream.eof())
  51. {
  52. if (!readBits(1))
  53. {
  54. outputStream.writeByte(readByte()^0x55);
  55. } else {
  56. uint32_t count=3;
  57. if (readBits(1))
  58. {
  59. count=readBits(2);
  60. if (count==3)
  61. {
  62. count=readBits(3);
  63. if (count==7)
  64. {
  65. count=readBits(4);
  66. if (count==15)
  67. {
  68. count=readByte();
  69. if (count==0xff) throw Decompressor::DecompressionError();
  70. count+=3;
  71. } else count+=14;
  72. } else count+=7;
  73. } else count+=4;
  74. }
  75. uint32_t distance = 0;
  76. switch (readBits(2))
  77. {
  78. case 0:
  79. distance=readByte()+1;
  80. break;
  81. case 1:
  82. distance=uint32_t(readBits(2))<<8;
  83. distance|=readByte();
  84. distance+=0x101;
  85. break;
  86. case 2:
  87. distance=uint32_t(readBits(4))<<8;
  88. distance|=readByte();
  89. distance+=0x501;
  90. break;
  91. case 3:
  92. distance=uint32_t(readBits(6))<<8;
  93. distance|=readByte();
  94. distance+=0x1501;
  95. break;
  96. }
  97. // buggy compressors
  98. count=std::min(count,uint32_t(rawSize-outputStream.getOffset()));
  99. if (!count) throw Decompressor::DecompressionError();
  100. outputStream.copy(distance,count);
  101. }
  102. }
  103. }
  104. }