2018-09-23 05:19:27 +02:00
|
|
|
|
using LibHac.Streams;
|
2018-09-23 06:46:15 +02:00
|
|
|
|
using Ryujinx.HLE.Loaders.Compression;
|
2018-09-23 05:19:27 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text;
|
|
|
|
|
|
|
|
|
|
namespace LibHac
|
|
|
|
|
{
|
|
|
|
|
public class Nso
|
|
|
|
|
{
|
|
|
|
|
public NsoSection[] Sections;
|
|
|
|
|
public RodataRelativeExtent[] RodataRelativeExtents;
|
|
|
|
|
public uint BssSize;
|
|
|
|
|
public byte[] BuildID = new byte[0x20];
|
|
|
|
|
|
2018-09-23 06:46:15 +02:00
|
|
|
|
private SharedStreamSource StreamSource;
|
2018-09-23 05:19:27 +02:00
|
|
|
|
|
|
|
|
|
public Nso(Stream stream)
|
|
|
|
|
{
|
2018-09-23 06:46:15 +02:00
|
|
|
|
StreamSource = new SharedStreamSource(stream);
|
|
|
|
|
BinaryReader reader = new BinaryReader(StreamSource.CreateStream());
|
2018-09-23 05:19:27 +02:00
|
|
|
|
if (reader.ReadAscii(4) != "NSO0")
|
|
|
|
|
throw new InvalidDataException("NSO magic is incorrect!");
|
|
|
|
|
reader.ReadUInt32(); // Version
|
|
|
|
|
reader.ReadUInt32(); // Reserved/Unused
|
|
|
|
|
BitArray flags = new BitArray(new int[] { (int) reader.ReadUInt32() });
|
2018-09-23 06:46:15 +02:00
|
|
|
|
NsoSection textSection = new NsoSection(StreamSource);
|
|
|
|
|
NsoSection rodataSection = new NsoSection(StreamSource);
|
|
|
|
|
NsoSection dataSection = new NsoSection(StreamSource);
|
2018-09-23 05:19:27 +02:00
|
|
|
|
textSection.IsCompressed = flags[0];
|
|
|
|
|
textSection.CheckHash = flags[3];
|
|
|
|
|
rodataSection.IsCompressed = flags[1];
|
|
|
|
|
rodataSection.CheckHash = flags[4];
|
|
|
|
|
dataSection.IsCompressed = flags[2];
|
|
|
|
|
dataSection.CheckHash = flags[5];
|
|
|
|
|
|
2018-09-23 09:24:14 +02:00
|
|
|
|
ReadSegmentHeader(textSection, reader);
|
2018-09-23 05:19:27 +02:00
|
|
|
|
reader.ReadUInt32(); // Module offset (TODO)
|
2018-09-23 09:24:14 +02:00
|
|
|
|
ReadSegmentHeader(rodataSection, reader);
|
2018-09-23 05:19:27 +02:00
|
|
|
|
reader.ReadUInt32(); // Module file size
|
2018-09-23 09:24:14 +02:00
|
|
|
|
ReadSegmentHeader(dataSection, reader);
|
2018-09-23 05:19:27 +02:00
|
|
|
|
BssSize = reader.ReadUInt32();
|
|
|
|
|
reader.Read(BuildID, 0, 0x20);
|
|
|
|
|
textSection.CompressedSize = reader.ReadUInt32();
|
|
|
|
|
rodataSection.CompressedSize = reader.ReadUInt32();
|
|
|
|
|
dataSection.CompressedSize = reader.ReadUInt32();
|
|
|
|
|
reader.ReadBytes(0x1C); // Padding
|
|
|
|
|
RodataRelativeExtents = new RodataRelativeExtent[]
|
|
|
|
|
{
|
2018-09-23 09:24:14 +02:00
|
|
|
|
ReadRodataRelativeExtent(reader), ReadRodataRelativeExtent(reader), ReadRodataRelativeExtent(reader)
|
2018-09-23 05:19:27 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
reader.Read(textSection.Hash, 0, 0x20);
|
|
|
|
|
reader.Read(rodataSection.Hash, 0, 0x20);
|
|
|
|
|
reader.Read(dataSection.Hash, 0, 0x20);
|
|
|
|
|
|
|
|
|
|
Sections = new NsoSection[] {textSection, rodataSection, dataSection };
|
2018-09-23 06:46:15 +02:00
|
|
|
|
reader.Close();
|
2018-09-23 05:19:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-23 09:24:14 +02:00
|
|
|
|
public void ReadSegmentHeader(NsoSection section, BinaryReader reader)
|
2018-09-23 05:19:27 +02:00
|
|
|
|
{
|
|
|
|
|
section.FileOffset = reader.ReadUInt32();
|
|
|
|
|
section.MemoryOffset = reader.ReadUInt32();
|
2018-09-23 06:46:15 +02:00
|
|
|
|
section.DecompressedSize = reader.ReadUInt32();
|
2018-09-23 05:19:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-23 09:24:14 +02:00
|
|
|
|
public RodataRelativeExtent ReadRodataRelativeExtent(BinaryReader reader)
|
2018-09-23 05:19:27 +02:00
|
|
|
|
{
|
|
|
|
|
RodataRelativeExtent extent = new RodataRelativeExtent();
|
|
|
|
|
extent.RegionRodataOffset = reader.ReadUInt32();
|
|
|
|
|
extent.RegionSize = reader.ReadUInt32();
|
|
|
|
|
return extent;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class NsoSection
|
|
|
|
|
{
|
2018-09-23 06:46:15 +02:00
|
|
|
|
private readonly SharedStreamSource StreamSource;
|
2018-09-23 05:19:27 +02:00
|
|
|
|
|
|
|
|
|
public bool IsCompressed,
|
|
|
|
|
CheckHash;
|
|
|
|
|
public uint FileOffset,
|
|
|
|
|
MemoryOffset,
|
|
|
|
|
DecompressedSize,
|
|
|
|
|
CompressedSize;
|
|
|
|
|
public byte[] Hash = new byte[0x20];
|
|
|
|
|
|
2018-09-23 06:46:15 +02:00
|
|
|
|
public NsoSection(SharedStreamSource streamSource)
|
2018-09-23 05:19:27 +02:00
|
|
|
|
{
|
2018-09-23 06:46:15 +02:00
|
|
|
|
StreamSource = streamSource;
|
2018-09-23 05:19:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Stream OpenCompressedStream()
|
|
|
|
|
{
|
2018-09-23 06:46:15 +02:00
|
|
|
|
return StreamSource.CreateStream(FileOffset, CompressedSize);
|
2018-09-23 05:19:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Stream OpenDecompressedStream()
|
|
|
|
|
{
|
2018-09-23 06:46:15 +02:00
|
|
|
|
return new MemoryStream(Decompress());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public byte[] Decompress()
|
|
|
|
|
{
|
|
|
|
|
byte[] compressed = new byte[CompressedSize];
|
|
|
|
|
OpenCompressedStream().Read(compressed, 0, (int) CompressedSize);
|
2018-09-23 06:52:30 +02:00
|
|
|
|
if (IsCompressed)
|
|
|
|
|
return Lz4.Decompress(compressed, (int)DecompressedSize);
|
|
|
|
|
else
|
|
|
|
|
return compressed;
|
2018-09-23 05:19:27 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class RodataRelativeExtent
|
|
|
|
|
{
|
|
|
|
|
public uint
|
|
|
|
|
RegionRodataOffset,
|
|
|
|
|
RegionSize;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|