Nso.cs 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. using Ryujinx.Core.Loaders.Compression;
  2. using System;
  3. using System.Collections.ObjectModel;
  4. using System.IO;
  5. namespace Ryujinx.Core.Loaders.Executables
  6. {
  7. class Nso : IExecutable
  8. {
  9. private byte[] m_Text;
  10. private byte[] m_RO;
  11. private byte[] m_Data;
  12. public ReadOnlyCollection<byte> Text => Array.AsReadOnly(m_Text);
  13. public ReadOnlyCollection<byte> RO => Array.AsReadOnly(m_RO);
  14. public ReadOnlyCollection<byte> Data => Array.AsReadOnly(m_Data);
  15. public int Mod0Offset { get; private set; }
  16. public int TextOffset { get; private set; }
  17. public int ROOffset { get; private set; }
  18. public int DataOffset { get; private set; }
  19. public int BssSize { get; private set; }
  20. public Extensions Extension { get; private set; }
  21. [Flags]
  22. private enum NsoFlags
  23. {
  24. IsTextCompressed = 1 << 0,
  25. IsROCompressed = 1 << 1,
  26. IsDataCompressed = 1 << 2,
  27. HasTextHash = 1 << 3,
  28. HasROHash = 1 << 4,
  29. HasDataHash = 1 << 5
  30. }
  31. public Nso(Stream Input)
  32. {
  33. BinaryReader Reader = new BinaryReader(Input);
  34. Input.Seek(0, SeekOrigin.Begin);
  35. int NsoMagic = Reader.ReadInt32();
  36. int Version = Reader.ReadInt32();
  37. int Reserved = Reader.ReadInt32();
  38. int FlagsMsk = Reader.ReadInt32();
  39. int TextOffset = Reader.ReadInt32();
  40. int TextMemOffset = Reader.ReadInt32();
  41. int TextDecSize = Reader.ReadInt32();
  42. int ModNameOffset = Reader.ReadInt32();
  43. int ROOffset = Reader.ReadInt32();
  44. int ROMemOffset = Reader.ReadInt32();
  45. int RODecSize = Reader.ReadInt32();
  46. int ModNameSize = Reader.ReadInt32();
  47. int DataOffset = Reader.ReadInt32();
  48. int DataMemOffset = Reader.ReadInt32();
  49. int DataDecSize = Reader.ReadInt32();
  50. int BssSize = Reader.ReadInt32();
  51. byte[] BuildId = Reader.ReadBytes(0x20);
  52. int TextSize = Reader.ReadInt32();
  53. int ROSize = Reader.ReadInt32();
  54. int DataSize = Reader.ReadInt32();
  55. Input.Seek(0x24, SeekOrigin.Current);
  56. int DynStrOffset = Reader.ReadInt32();
  57. int DynStrSize = Reader.ReadInt32();
  58. int DynSymOffset = Reader.ReadInt32();
  59. int DynSymSize = Reader.ReadInt32();
  60. byte[] TextHash = Reader.ReadBytes(0x20);
  61. byte[] ROHash = Reader.ReadBytes(0x20);
  62. byte[] DataHash = Reader.ReadBytes(0x20);
  63. NsoFlags Flags = (NsoFlags)FlagsMsk;
  64. this.TextOffset = TextMemOffset;
  65. this.ROOffset = ROMemOffset;
  66. this.DataOffset = DataMemOffset;
  67. this.BssSize = BssSize;
  68. this.Extension = Extensions.NSO;
  69. //Text segment
  70. Input.Seek(TextOffset, SeekOrigin.Begin);
  71. m_Text = Reader.ReadBytes(TextSize);
  72. if (Flags.HasFlag(NsoFlags.IsTextCompressed) || true)
  73. {
  74. m_Text = Lz4.Decompress(m_Text, TextDecSize);
  75. }
  76. //Read-only data segment
  77. Input.Seek(ROOffset, SeekOrigin.Begin);
  78. m_RO = Reader.ReadBytes(ROSize);
  79. if (Flags.HasFlag(NsoFlags.IsROCompressed) || true)
  80. {
  81. m_RO = Lz4.Decompress(m_RO, RODecSize);
  82. }
  83. //Data segment
  84. Input.Seek(DataOffset, SeekOrigin.Begin);
  85. m_Data = Reader.ReadBytes(DataSize);
  86. if (Flags.HasFlag(NsoFlags.IsDataCompressed) || true)
  87. {
  88. m_Data = Lz4.Decompress(m_Data, DataDecSize);
  89. }
  90. using (MemoryStream Text = new MemoryStream(m_Text))
  91. {
  92. BinaryReader TextReader = new BinaryReader(Text);
  93. Text.Seek(4, SeekOrigin.Begin);
  94. Mod0Offset = TextReader.ReadInt32();
  95. }
  96. }
  97. }
  98. }