2018-09-13 03:28:50 +02:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
2018-09-14 00:58:40 +02:00
|
|
|
|
using System.Text;
|
2018-09-13 03:28:50 +02:00
|
|
|
|
using LibHac;
|
2018-11-19 05:20:34 +01:00
|
|
|
|
using LibHac.IO;
|
2018-09-14 00:58:40 +02:00
|
|
|
|
using static hactoolnet.Print;
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
namespace hactoolnet
|
|
|
|
|
{
|
|
|
|
|
internal static class ProcessXci
|
|
|
|
|
{
|
|
|
|
|
public static void Process(Context ctx)
|
|
|
|
|
{
|
|
|
|
|
using (var file = new FileStream(ctx.Options.InFile, FileMode.Open, FileAccess.Read))
|
|
|
|
|
{
|
2018-11-19 05:20:34 +01:00
|
|
|
|
var xci = new Xci(ctx.Keyset, file.AsStorage());
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
2018-09-14 00:58:40 +02:00
|
|
|
|
ctx.Logger.LogMessage(xci.Print());
|
|
|
|
|
|
2018-09-13 03:28:50 +02:00
|
|
|
|
if (ctx.Options.RootDir != null)
|
|
|
|
|
{
|
|
|
|
|
xci.RootPartition?.Extract(ctx.Options.RootDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.UpdateDir != null)
|
|
|
|
|
{
|
|
|
|
|
xci.UpdatePartition?.Extract(ctx.Options.UpdateDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.NormalDir != null)
|
|
|
|
|
{
|
|
|
|
|
xci.NormalPartition?.Extract(ctx.Options.NormalDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.SecureDir != null)
|
|
|
|
|
{
|
|
|
|
|
xci.SecurePartition?.Extract(ctx.Options.SecureDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.LogoDir != null)
|
|
|
|
|
{
|
|
|
|
|
xci.LogoPartition?.Extract(ctx.Options.LogoDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.OutDir != null && xci.RootPartition != null)
|
|
|
|
|
{
|
2018-10-03 00:25:58 +02:00
|
|
|
|
XciPartition root = xci.RootPartition;
|
2018-09-13 03:28:50 +02:00
|
|
|
|
if (root == null)
|
|
|
|
|
{
|
|
|
|
|
ctx.Logger.LogMessage("Could not find root partition");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-03 00:25:58 +02:00
|
|
|
|
foreach (PfsFileEntry sub in root.Files)
|
2018-09-13 03:28:50 +02:00
|
|
|
|
{
|
|
|
|
|
var subPfs = new Pfs(root.OpenFile(sub));
|
2018-10-03 00:25:58 +02:00
|
|
|
|
string subDir = Path.Combine(ctx.Options.OutDir, sub.Name);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
subPfs.Extract(subDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.ExefsOutDir != null || ctx.Options.ExefsOut != null)
|
|
|
|
|
{
|
2018-10-03 00:25:58 +02:00
|
|
|
|
Nca mainNca = GetXciMainNca(xci, ctx);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
if (mainNca == null)
|
|
|
|
|
{
|
|
|
|
|
ctx.Logger.LogMessage("Could not find Program NCA");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-13 04:28:05 +02:00
|
|
|
|
NcaSection exefsSection = mainNca.Sections[(int)ProgramPartitionType.Code];
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
if (exefsSection == null)
|
|
|
|
|
{
|
|
|
|
|
ctx.Logger.LogMessage("NCA has no ExeFS section");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.ExefsOutDir != null)
|
|
|
|
|
{
|
2018-10-09 22:33:56 +02:00
|
|
|
|
mainNca.ExtractSection(exefsSection.SectionNum, ctx.Options.ExefsOutDir, ctx.Options.IntegrityLevel, ctx.Logger);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.ExefsOut != null)
|
|
|
|
|
{
|
2018-10-09 22:33:56 +02:00
|
|
|
|
mainNca.ExportSection(exefsSection.SectionNum, ctx.Options.ExefsOut, ctx.Options.Raw, ctx.Options.IntegrityLevel, ctx.Logger);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.RomfsOutDir != null || ctx.Options.RomfsOut != null)
|
|
|
|
|
{
|
2018-10-03 00:25:58 +02:00
|
|
|
|
Nca mainNca = GetXciMainNca(xci, ctx);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
if (mainNca == null)
|
|
|
|
|
{
|
|
|
|
|
ctx.Logger.LogMessage("Could not find Program NCA");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-03 00:25:58 +02:00
|
|
|
|
NcaSection romfsSection = mainNca.Sections.FirstOrDefault(x => x.Type == SectionType.Romfs);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
if (romfsSection == null)
|
|
|
|
|
{
|
|
|
|
|
ctx.Logger.LogMessage("NCA has no RomFS section");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.RomfsOutDir != null)
|
|
|
|
|
{
|
2018-11-19 05:20:34 +01:00
|
|
|
|
var romfs = new Romfs(mainNca.OpenSection(romfsSection.SectionNum, false, ctx.Options.IntegrityLevel, true));
|
2018-09-13 03:28:50 +02:00
|
|
|
|
romfs.Extract(ctx.Options.RomfsOutDir, ctx.Logger);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ctx.Options.RomfsOut != null)
|
|
|
|
|
{
|
2018-10-09 22:33:56 +02:00
|
|
|
|
mainNca.ExportSection(romfsSection.SectionNum, ctx.Options.RomfsOut, ctx.Options.Raw, ctx.Options.IntegrityLevel, ctx.Logger);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Nca GetXciMainNca(Xci xci, Context ctx)
|
|
|
|
|
{
|
|
|
|
|
if (xci.SecurePartition == null)
|
|
|
|
|
{
|
|
|
|
|
ctx.Logger.LogMessage("Could not find secure partition");
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Nca mainNca = null;
|
|
|
|
|
|
2018-10-03 00:25:58 +02:00
|
|
|
|
foreach (PfsFileEntry fileEntry in xci.SecurePartition.Files.Where(x => x.Name.EndsWith(".nca")))
|
2018-09-13 03:28:50 +02:00
|
|
|
|
{
|
2018-11-19 05:20:34 +01:00
|
|
|
|
IStorage ncaStorage = xci.SecurePartition.OpenFile(fileEntry);
|
|
|
|
|
var nca = new Nca(ctx.Keyset, ncaStorage, true);
|
2018-09-13 03:28:50 +02:00
|
|
|
|
|
|
|
|
|
if (nca.Header.ContentType == ContentType.Program)
|
|
|
|
|
{
|
|
|
|
|
mainNca = nca;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return mainNca;
|
|
|
|
|
}
|
2018-09-14 00:58:40 +02:00
|
|
|
|
|
|
|
|
|
private static string Print(this Xci xci)
|
|
|
|
|
{
|
|
|
|
|
const int colLen = 36;
|
|
|
|
|
|
|
|
|
|
var sb = new StringBuilder();
|
|
|
|
|
sb.AppendLine();
|
|
|
|
|
|
|
|
|
|
sb.AppendLine("XCI:");
|
|
|
|
|
|
|
|
|
|
PrintItem(sb, colLen, "Magic:", xci.Header.Magic);
|
2018-11-19 05:20:34 +01:00
|
|
|
|
PrintItem(sb, colLen, $"Header Signature{xci.Header.SignatureValidity.GetValidityString()}:", xci.Header.Signature);
|
|
|
|
|
PrintItem(sb, colLen, $"Header Hash{xci.Header.PartitionFsHeaderValidity.GetValidityString()}:", xci.Header.PartitionFsHeaderHash);
|
2018-09-14 00:58:40 +02:00
|
|
|
|
PrintItem(sb, colLen, "Cartridge Type:", GetCartridgeType(xci.Header.RomSize));
|
|
|
|
|
PrintItem(sb, colLen, "Cartridge Size:", $"0x{Util.MediaToReal(xci.Header.ValidDataEndPage + 1):x12}");
|
|
|
|
|
PrintItem(sb, colLen, "Header IV:", xci.Header.AesCbcIv);
|
|
|
|
|
|
|
|
|
|
foreach (XciPartition partition in xci.Partitions.OrderBy(x => x.Offset))
|
|
|
|
|
{
|
|
|
|
|
PrintPartition(sb, colLen, partition);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sb.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void PrintPartition(StringBuilder sb, int colLen, XciPartition partition)
|
|
|
|
|
{
|
|
|
|
|
const int fileNameLen = 57;
|
|
|
|
|
|
2018-09-21 14:37:30 +02:00
|
|
|
|
sb.AppendLine($"{GetDisplayName(partition.Name)} Partition:{partition.HashValidity.GetValidityString()}");
|
2018-09-14 00:58:40 +02:00
|
|
|
|
PrintItem(sb, colLen, " Magic:", partition.Header.Magic);
|
|
|
|
|
PrintItem(sb, colLen, " Offset:", $"{partition.Offset:x12}");
|
|
|
|
|
PrintItem(sb, colLen, " Number of files:", partition.Files.Length);
|
|
|
|
|
|
|
|
|
|
if (partition.Files.Length > 0 && partition.Files.Length < 100)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < partition.Files.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
PfsFileEntry file = partition.Files[i];
|
|
|
|
|
|
|
|
|
|
string label = i == 0 ? " Files:" : "";
|
2018-09-21 14:37:30 +02:00
|
|
|
|
string offsets = $"{file.Offset:x12}-{file.Offset + file.Size:x12}{file.HashValidity.GetValidityString()}";
|
2018-09-14 00:58:40 +02:00
|
|
|
|
string data = $"{partition.Name}:/{file.Name}".PadRight(fileNameLen) + offsets;
|
|
|
|
|
|
|
|
|
|
PrintItem(sb, colLen, label, data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static string GetDisplayName(string name)
|
|
|
|
|
{
|
|
|
|
|
switch (name)
|
|
|
|
|
{
|
|
|
|
|
case "rootpt": return "Root";
|
|
|
|
|
case "update": return "Update";
|
|
|
|
|
case "normal": return "Normal";
|
|
|
|
|
case "secure": return "Secure";
|
|
|
|
|
case "logo": return "Logo";
|
|
|
|
|
default: return name;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static string GetCartridgeType(RomSize size)
|
|
|
|
|
{
|
|
|
|
|
switch (size)
|
|
|
|
|
{
|
|
|
|
|
case RomSize.Size1Gb: return "1GB";
|
|
|
|
|
case RomSize.Size2Gb: return "2GB";
|
|
|
|
|
case RomSize.Size4Gb: return "4GB";
|
|
|
|
|
case RomSize.Size8Gb: return "8GB";
|
|
|
|
|
case RomSize.Size16Gb: return "16GB";
|
|
|
|
|
case RomSize.Size32Gb: return "32GB";
|
|
|
|
|
default: return string.Empty;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-09-13 03:28:50 +02:00
|
|
|
|
}
|
|
|
|
|
}
|