forked from Mirror/Ryujinx
d8f2497f15
* Fix issues with compressed NSO without a section Avoid trying to decompress something that doesn't exist * don't use the lengths of the buffer directly
109 lines
No EOL
3.5 KiB
C#
109 lines
No EOL
3.5 KiB
C#
using Ryujinx.HLE.Loaders.Compression;
|
|
using System;
|
|
using System.IO;
|
|
|
|
namespace Ryujinx.HLE.Loaders.Executables
|
|
{
|
|
class NxStaticObject : IExecutable
|
|
{
|
|
public byte[] Text { get; private set; }
|
|
public byte[] Ro { get; private set; }
|
|
public byte[] Data { get; private set; }
|
|
|
|
public int TextOffset { get; private set; }
|
|
public int RoOffset { get; private set; }
|
|
public int DataOffset { get; private set; }
|
|
public int BssSize { get; private set; }
|
|
|
|
public int BssOffset => DataOffset + Data.Length;
|
|
|
|
[Flags]
|
|
private enum NsoFlags
|
|
{
|
|
IsTextCompressed = 1 << 0,
|
|
IsRoCompressed = 1 << 1,
|
|
IsDataCompressed = 1 << 2,
|
|
HasTextHash = 1 << 3,
|
|
HasRoHash = 1 << 4,
|
|
HasDataHash = 1 << 5
|
|
}
|
|
|
|
public NxStaticObject(Stream input)
|
|
{
|
|
BinaryReader reader = new BinaryReader(input);
|
|
|
|
input.Seek(0, SeekOrigin.Begin);
|
|
|
|
int nsoMagic = reader.ReadInt32();
|
|
int version = reader.ReadInt32();
|
|
int reserved = reader.ReadInt32();
|
|
int flagsMsk = reader.ReadInt32();
|
|
int textOffset = reader.ReadInt32();
|
|
int textMemOffset = reader.ReadInt32();
|
|
int textDecSize = reader.ReadInt32();
|
|
int modNameOffset = reader.ReadInt32();
|
|
int roOffset = reader.ReadInt32();
|
|
int roMemOffset = reader.ReadInt32();
|
|
int roDecSize = reader.ReadInt32();
|
|
int modNameSize = reader.ReadInt32();
|
|
int dataOffset = reader.ReadInt32();
|
|
int dataMemOffset = reader.ReadInt32();
|
|
int dataDecSize = reader.ReadInt32();
|
|
int bssSize = reader.ReadInt32();
|
|
|
|
byte[] buildId = reader.ReadBytes(0x20);
|
|
|
|
int textSize = reader.ReadInt32();
|
|
int roSize = reader.ReadInt32();
|
|
int dataSize = reader.ReadInt32();
|
|
|
|
input.Seek(0x24, SeekOrigin.Current);
|
|
|
|
int dynStrOffset = reader.ReadInt32();
|
|
int dynStrSize = reader.ReadInt32();
|
|
int dynSymOffset = reader.ReadInt32();
|
|
int dynSymSize = reader.ReadInt32();
|
|
|
|
byte[] textHash = reader.ReadBytes(0x20);
|
|
byte[] roHash = reader.ReadBytes(0x20);
|
|
byte[] dataHash = reader.ReadBytes(0x20);
|
|
|
|
NsoFlags flags = (NsoFlags)flagsMsk;
|
|
|
|
TextOffset = textMemOffset;
|
|
RoOffset = roMemOffset;
|
|
DataOffset = dataMemOffset;
|
|
BssSize = bssSize;
|
|
|
|
//Text segment
|
|
input.Seek(textOffset, SeekOrigin.Begin);
|
|
|
|
Text = reader.ReadBytes(textSize);
|
|
|
|
if (flags.HasFlag(NsoFlags.IsTextCompressed) && textSize != 0)
|
|
{
|
|
Text = Lz4.Decompress(Text, textDecSize);
|
|
}
|
|
|
|
//Read-only data segment
|
|
input.Seek(roOffset, SeekOrigin.Begin);
|
|
|
|
Ro = reader.ReadBytes(roSize);
|
|
|
|
if (flags.HasFlag(NsoFlags.IsRoCompressed) && roSize != 0)
|
|
{
|
|
Ro = Lz4.Decompress(Ro, roDecSize);
|
|
}
|
|
|
|
//Data segment
|
|
input.Seek(dataOffset, SeekOrigin.Begin);
|
|
|
|
Data = reader.ReadBytes(dataSize);
|
|
|
|
if (flags.HasFlag(NsoFlags.IsDataCompressed) && dataSize != 0)
|
|
{
|
|
Data = Lz4.Decompress(Data, dataDecSize);
|
|
}
|
|
}
|
|
}
|
|
} |