Nso.cs 3.9 KB

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