// ReSharper disable UnusedVariable UnusedMember.Local using System; using System.Collections.Generic; using System.IO; using LibHac; using LibHac.Nand; using LibHac.Savefile; namespace NandReader { public static class Program { public static void Main(string[] args) { if (args.Length != 1) { Console.WriteLine("Usage: NandReader raw_nand_dump_file"); return; } GetTitleKeys(args[0]); } private static void GetTitleKeys(string nandFile) { using (var logger = new ProgressBar()) using (var stream = new FileStream(nandFile, FileMode.Open, FileAccess.Read)) { var keyset = OpenKeyset(); var nand = new Nand(stream, keyset); var prodinfo = nand.OpenProdInfo(); var calibration = new Calibration(prodinfo); keyset.EticketExtKeyRsa = Crypto.DecryptRsaKey(calibration.EticketExtKeyRsa, keyset.EticketRsaKek); var tickets = GetTickets(keyset, nand, logger); foreach (var ticket in tickets) { var key = ticket.GetTitleKey(keyset); logger.LogMessage($"{ticket.RightsId.ToHexString()},{key.ToHexString()}"); } } } private static void ReadSwitchFs(string nandFile) { using (var logger = new ProgressBar()) using (var stream = new FileStream(nandFile, FileMode.Open, FileAccess.Read)) { var keyset = OpenKeyset(); var nand = new Nand(stream, keyset); var user = nand.OpenSystemPartition(); var sdfs = new SwitchFs(keyset, user); } } private static void ReadCalibration(string nandFile) { using (var logger = new ProgressBar()) using (var stream = new FileStream(nandFile, FileMode.Open, FileAccess.Read)) { var keyset = OpenKeyset(); var nand = new Nand(stream, keyset); var prodinfo = nand.OpenProdInfo(); var calibration = new Calibration(prodinfo); } } private static void DumpTickets(string nandFile) { using (var logger = new ProgressBar()) using (var stream = new FileStream(nandFile, FileMode.Open, FileAccess.Read)) { var keyset = OpenKeyset(); var nand = new Nand(stream, keyset); var tickets = GetTickets(keyset, nand, logger); Directory.CreateDirectory("tickets"); foreach (var ticket in tickets) { var filename = Path.Combine("tickets", $"{ticket.RightsId.ToHexString()}.tik"); File.WriteAllBytes(filename, ticket.File); } } } private static Ticket[] GetTickets(Keyset keyset, Nand nand, IProgressReport logger = null) { var tickets = new List(); var system = nand.OpenSystemPartition(); var saveE1File = system.OpenFile("save\\80000000000000E1", FileMode.Open, FileAccess.Read); tickets.AddRange(ReadTickets(keyset, saveE1File)); var saveE2 = system.OpenFile("save\\80000000000000E2", FileMode.Open, FileAccess.Read); tickets.AddRange(ReadTickets(keyset, saveE2)); logger?.LogMessage($"Found {tickets.Count} tickets"); return tickets.ToArray(); } private static List ReadTickets(Keyset keyset, Stream savefile) { var tickets = new List(); var save = new Savefile(keyset, savefile); var ticketList = new BinaryReader(save.OpenFile("/ticket_list.bin")); var ticketFile = new BinaryReader(save.OpenFile("/ticket.bin")); var titleId = ticketList.ReadUInt64(); while (titleId != ulong.MaxValue) { ticketList.BaseStream.Position += 0x18; var start = ticketFile.BaseStream.Position; tickets.Add(new Ticket(ticketFile)); ticketFile.BaseStream.Position = start + 0x400; titleId = ticketList.ReadUInt64(); } return tickets; } private static Keyset OpenKeyset() { var home = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile); var homeKeyFile = Path.Combine(home, ".switch", "prod.keys"); var homeTitleKeyFile = Path.Combine(home, ".switch", "title.keys"); var homeConsoleKeyFile = Path.Combine(home, ".switch", "console.keys"); string keyFile = null; string titleKeyFile = null; string consoleKeyFile = null; if (File.Exists(homeKeyFile)) { keyFile = homeKeyFile; } if (File.Exists(homeTitleKeyFile)) { titleKeyFile = homeTitleKeyFile; } if (File.Exists(homeConsoleKeyFile)) { consoleKeyFile = homeConsoleKeyFile; } return ExternalKeys.ReadKeyFile(keyFile, titleKeyFile, consoleKeyFile); } } }