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 int[] 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 int[32]; for (int index = 0; index < Capabilities.Length; index++) { Capabilities[index] = (int)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; } } }