BackwardsLz.cs 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. using System;
  2. using System.IO;
  3. namespace Ryujinx.HLE.Loaders.Compression
  4. {
  5. static class BackwardsLz
  6. {
  7. private class BackwardsReader
  8. {
  9. private Stream _baseStream;
  10. public BackwardsReader(Stream baseStream)
  11. {
  12. _baseStream = baseStream;
  13. }
  14. public byte ReadByte()
  15. {
  16. _baseStream.Seek(-1, SeekOrigin.Current);
  17. byte value = (byte)_baseStream.ReadByte();
  18. _baseStream.Seek(-1, SeekOrigin.Current);
  19. return value;
  20. }
  21. public short ReadInt16()
  22. {
  23. return (short)((ReadByte() << 8) | (ReadByte() << 0));
  24. }
  25. public int ReadInt32()
  26. {
  27. return ((ReadByte() << 24) |
  28. (ReadByte() << 16) |
  29. (ReadByte() << 8) |
  30. (ReadByte() << 0));
  31. }
  32. }
  33. public static byte[] Decompress(Stream input, int decompressedLength)
  34. {
  35. long end = input.Position;
  36. BackwardsReader reader = new BackwardsReader(input);
  37. int additionalDecLength = reader.ReadInt32();
  38. int startOffset = reader.ReadInt32();
  39. int compressedLength = reader.ReadInt32();
  40. input.Seek(12 - startOffset, SeekOrigin.Current);
  41. byte[] dec = new byte[decompressedLength];
  42. int decompressedLengthUnpadded = compressedLength + additionalDecLength;
  43. int decompressionStart = decompressedLength - decompressedLengthUnpadded;
  44. int decPos = dec.Length;
  45. byte mask = 0;
  46. byte header = 0;
  47. while (decPos > decompressionStart)
  48. {
  49. if ((mask >>= 1) == 0)
  50. {
  51. header = reader.ReadByte();
  52. mask = 0x80;
  53. }
  54. if ((header & mask) == 0)
  55. {
  56. dec[--decPos] = reader.ReadByte();
  57. }
  58. else
  59. {
  60. ushort pair = (ushort)reader.ReadInt16();
  61. int length = (pair >> 12) + 3;
  62. int position = (pair & 0xfff) + 3;
  63. decPos -= length;
  64. if (length <= position)
  65. {
  66. int srcPos = decPos + position;
  67. Buffer.BlockCopy(dec, srcPos, dec, decPos, length);
  68. }
  69. else
  70. {
  71. for (int offset = 0; offset < length; offset++)
  72. {
  73. dec[decPos + offset] = dec[decPos + position + offset];
  74. }
  75. }
  76. }
  77. }
  78. return dec;
  79. }
  80. }
  81. }