aboutsummaryrefslogtreecommitdiff
path: root/src/Ryujinx.HLE/Loaders/Executables/KipExecutable.cs
diff options
context:
space:
mode:
Diffstat (limited to 'src/Ryujinx.HLE/Loaders/Executables/KipExecutable.cs')
-rw-r--r--src/Ryujinx.HLE/Loaders/Executables/KipExecutable.cs86
1 files changed, 86 insertions, 0 deletions
diff --git a/src/Ryujinx.HLE/Loaders/Executables/KipExecutable.cs b/src/Ryujinx.HLE/Loaders/Executables/KipExecutable.cs
new file mode 100644
index 00000000..ad2b681c
--- /dev/null
+++ b/src/Ryujinx.HLE/Loaders/Executables/KipExecutable.cs
@@ -0,0 +1,86 @@
+using LibHac.Common;
+using LibHac.Fs;
+using LibHac.Kernel;
+using System;
+
+namespace Ryujinx.HLE.Loaders.Executables
+{
+ class KipExecutable : IExecutable
+ {
+ public byte[] Program { get; }
+ public Span<byte> Text => Program.AsSpan((int)TextOffset, (int)TextSize);
+ public Span<byte> Ro => Program.AsSpan((int)RoOffset, (int)RoSize);
+ public Span<byte> Data => Program.AsSpan((int)DataOffset, (int)DataSize);
+
+ public uint TextOffset { get; }
+ public uint RoOffset { get; }
+ public uint DataOffset { get; }
+ public uint BssOffset { get; }
+
+ public uint TextSize { get; }
+ public uint RoSize { get; }
+ public uint DataSize { get; }
+ public uint BssSize { get; }
+
+ public uint[] Capabilities { get; }
+ public bool UsesSecureMemory { get; }
+ public bool Is64BitAddressSpace { get; }
+ public bool Is64Bit { get; }
+ public ulong ProgramId { get; }
+ public byte Priority { get; }
+ public int StackSize { get; }
+ public byte IdealCoreId { get; }
+ public int Version { get; }
+ public string Name { get; }
+
+ public KipExecutable(in SharedRef<IStorage> inStorage)
+ {
+ KipReader reader = new KipReader();
+
+ reader.Initialize(in inStorage).ThrowIfFailure();
+
+ TextOffset = (uint)reader.Segments[0].MemoryOffset;
+ RoOffset = (uint)reader.Segments[1].MemoryOffset;
+ DataOffset = (uint)reader.Segments[2].MemoryOffset;
+ BssOffset = (uint)reader.Segments[3].MemoryOffset;
+ BssSize = (uint)reader.Segments[3].Size;
+
+ StackSize = reader.StackSize;
+
+ UsesSecureMemory = reader.UsesSecureMemory;
+ Is64BitAddressSpace = reader.Is64BitAddressSpace;
+ Is64Bit = reader.Is64Bit;
+
+ ProgramId = reader.ProgramId;
+ Priority = reader.Priority;
+ IdealCoreId = reader.IdealCoreId;
+ Version = reader.Version;
+ Name = reader.Name.ToString();
+
+ Capabilities = new uint[32];
+
+ for (int index = 0; index < Capabilities.Length; index++)
+ {
+ Capabilities[index] = reader.Capabilities[index];
+ }
+
+ reader.GetSegmentSize(KipReader.SegmentType.Data, out int uncompressedSize).ThrowIfFailure();
+ Program = new byte[DataOffset + uncompressedSize];
+
+ TextSize = DecompressSection(reader, KipReader.SegmentType.Text, TextOffset, Program);
+ RoSize = DecompressSection(reader, KipReader.SegmentType.Ro, RoOffset, Program);
+ DataSize = DecompressSection(reader, KipReader.SegmentType.Data, DataOffset, Program);
+ }
+
+ private static uint DecompressSection(KipReader reader, KipReader.SegmentType segmentType, uint offset, byte[] program)
+ {
+ reader.GetSegmentSize(segmentType, out int uncompressedSize).ThrowIfFailure();
+
+ var span = program.AsSpan((int)offset, uncompressedSize);
+
+ reader.ReadSegment(segmentType, span).ThrowIfFailure();
+
+ return (uint)uncompressedSize;
+ }
+ }
+} \ No newline at end of file