mirror of
https://github.com/Thealexbarney/LibHac.git
synced 2024-11-14 10:49:41 +01:00
3d50085e22
*Create an IStorage interface and Storage abstract class to use instead of Stream * Improve AES-XTS performance by ~16x * Double AES-CTR performance: 800 MB/s -> 1600 MB/s on a 6700K * Add AES-XTS tests * Add AES benchmark and AES-CTR writing * Add support for a hashed FAT in save files * Add option to export decrypted NCA * Allow opening decrypted package1 and package2 * Make sure romfs disposal can cascade all the way down * Validate NCA, NPDM and package2 signatures
146 lines
5.5 KiB
C#
146 lines
5.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Management;
|
|
using System.Windows.Input;
|
|
using GalaSoft.MvvmLight;
|
|
using GalaSoft.MvvmLight.Command;
|
|
using LibHac;
|
|
using LibHac.IO;
|
|
using LibHac.IO.Save;
|
|
using LibHac.Nand;
|
|
|
|
namespace NandReaderGui.ViewModel
|
|
{
|
|
public class NandViewModel : ViewModelBase
|
|
{
|
|
public List<DiskInfo> Disks { get; } = new List<DiskInfo>();
|
|
public ICommand OpenCommand { get; set; }
|
|
public DiskInfo SelectedDisk { get; set; }
|
|
|
|
public NandViewModel()
|
|
{
|
|
OpenCommand = new RelayCommand(Open);
|
|
|
|
var query = new WqlObjectQuery("SELECT * FROM Win32_DiskDrive");
|
|
using (var searcher = new ManagementObjectSearcher(query))
|
|
{
|
|
foreach (ManagementBaseObject drive in searcher.Get())
|
|
{
|
|
if (drive.GetPropertyValue("Size") == null) continue;
|
|
var info = new DiskInfo();
|
|
info.PhysicalName = (string)drive.GetPropertyValue("Name");
|
|
info.Name = (string)drive.GetPropertyValue("Caption");
|
|
info.Model = (string)drive.GetPropertyValue("Model");
|
|
//todo Why is Windows returning small sizes? https://stackoverflow.com/questions/15051660
|
|
info.Length = (long)((ulong)drive.GetPropertyValue("Size"));
|
|
info.SectorSize = (int)((uint)drive.GetPropertyValue("BytesPerSector"));
|
|
info.DisplaySize = Util.GetBytesReadable((long)((ulong)drive.GetPropertyValue("Size")));
|
|
|
|
Disks.Add(info);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void Open()
|
|
{
|
|
DiskInfo disk = SelectedDisk;
|
|
var storage = new CachedStorage(new DeviceStream(disk.PhysicalName, disk.Length).AsStorage(), disk.SectorSize * 100, 4, true);
|
|
storage.SetReadOnly();
|
|
Stream stream = storage.AsStream();
|
|
|
|
Keyset keyset = OpenKeyset();
|
|
var nand = new Nand(stream, keyset);
|
|
|
|
Stream prodinfo = nand.OpenProdInfo();
|
|
var calibration = new Calibration(prodinfo);
|
|
|
|
keyset.EticketExtKeyRsa = Crypto.DecryptRsaKey(calibration.EticketExtKeyRsa, keyset.EticketRsaKek);
|
|
Ticket[] tickets = GetTickets(keyset, nand);
|
|
|
|
using (var outStream = new StreamWriter("titlekeys.txt"))
|
|
{
|
|
foreach (Ticket ticket in tickets)
|
|
{
|
|
byte[] key = ticket.GetTitleKey(keyset);
|
|
outStream.WriteLine($"{ticket.RightsId.ToHexString()},{key.ToHexString()}");
|
|
}
|
|
}
|
|
}
|
|
|
|
private static Ticket[] GetTickets(Keyset keyset, Nand nand, IProgressReport logger = null)
|
|
{
|
|
var tickets = new List<Ticket>();
|
|
NandPartition system = nand.OpenSystemPartition();
|
|
|
|
Stream saveE1File = system.OpenFile("save\\80000000000000E1", FileMode.Open, FileAccess.Read);
|
|
tickets.AddRange(ReadTickets(keyset, saveE1File));
|
|
|
|
Stream 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<Ticket> ReadTickets(Keyset keyset, Stream savefile)
|
|
{
|
|
var tickets = new List<Ticket>();
|
|
var save = new Savefile(keyset, savefile.AsStorage(), IntegrityCheckLevel.None, true);
|
|
var ticketList = new BinaryReader(save.OpenFile("/ticket_list.bin").AsStream());
|
|
var ticketFile = new BinaryReader(save.OpenFile("/ticket.bin").AsStream());
|
|
|
|
ulong titleId = ticketList.ReadUInt64();
|
|
while (titleId != ulong.MaxValue)
|
|
{
|
|
ticketList.BaseStream.Position += 0x18;
|
|
long start = ticketFile.BaseStream.Position;
|
|
tickets.Add(new Ticket(ticketFile));
|
|
ticketFile.BaseStream.Position = start + 0x400;
|
|
titleId = ticketList.ReadUInt64();
|
|
}
|
|
|
|
return tickets;
|
|
}
|
|
|
|
private static Keyset OpenKeyset()
|
|
{
|
|
string home = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
|
|
string homeKeyFile = Path.Combine(home, ".switch", "prod.keys");
|
|
string homeTitleKeyFile = Path.Combine(home, ".switch", "title.keys");
|
|
string 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);
|
|
}
|
|
}
|
|
|
|
public class DiskInfo
|
|
{
|
|
public string PhysicalName { get; set; }
|
|
public string Name { get; set; }
|
|
public string Model { get; set; }
|
|
public long Length { get; set; }
|
|
public int SectorSize { get; set; }
|
|
public string DisplaySize { get; set; }
|
|
public string Display => $"{Name} ({DisplaySize})";
|
|
}
|
|
}
|