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. this.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. }