2020-12-28 22:28:44 -06:00
|
|
|
|
// Decompiled with JetBrains decompiler
|
|
|
|
|
// Type: libWiiSharp.Ticket
|
|
|
|
|
// Assembly: libWiiSharp, Version=0.4.0.0, Culture=neutral, PublicKeyToken=null
|
|
|
|
|
// MVID: FBF36F3D-B5D6-481F-B5F5-1BD3C19E13B2
|
|
|
|
|
// Assembly location: C:\Users\theso\Downloads\NCPatcher\pack\libWiiSharp.dll
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Security.Cryptography;
|
|
|
|
|
|
|
|
|
|
namespace libWiiSharp
|
|
|
|
|
{
|
2021-02-06 21:19:03 -06:00
|
|
|
|
|
|
|
|
|
public enum CommonKeyType : byte
|
|
|
|
|
{
|
|
|
|
|
Standard = 0x00,
|
|
|
|
|
Korean = 0x01,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class Ticket : IDisposable
|
|
|
|
|
{
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private byte newKeyIndex;
|
|
|
|
|
private byte[] decryptedTitleKey = new byte[16];
|
|
|
|
|
private bool fakeSign;
|
|
|
|
|
private bool titleKeyChanged;
|
|
|
|
|
private byte[] newEncryptedTitleKey = new byte[0];
|
|
|
|
|
private bool reDecrypt;
|
|
|
|
|
private uint signatureExponent = 65537;
|
|
|
|
|
private byte[] signature = new byte[256];
|
|
|
|
|
private byte[] padding = new byte[60];
|
|
|
|
|
private byte[] issuer = new byte[64];
|
|
|
|
|
private byte[] unknown = new byte[63];
|
|
|
|
|
private byte[] encryptedTitleKey = new byte[16];
|
|
|
|
|
private byte unknown2;
|
|
|
|
|
private ulong ticketId;
|
|
|
|
|
private uint consoleId;
|
|
|
|
|
private ulong titleId;
|
|
|
|
|
private ushort unknown3 = ushort.MaxValue;
|
|
|
|
|
private ushort numOfDlc;
|
|
|
|
|
private ulong unknown4;
|
|
|
|
|
private byte padding2;
|
|
|
|
|
private byte commonKeyIndex;
|
|
|
|
|
private byte[] unknown5 = new byte[48];
|
|
|
|
|
private byte[] unknown6 = new byte[32];
|
|
|
|
|
private ushort padding3;
|
|
|
|
|
private uint enableTimeLimit;
|
|
|
|
|
private uint timeLimit;
|
|
|
|
|
private byte[] padding4 = new byte[88];
|
|
|
|
|
private bool isDisposed;
|
|
|
|
|
|
|
|
|
|
public byte[] TitleKey
|
|
|
|
|
{
|
|
|
|
|
get => decryptedTitleKey;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
decryptedTitleKey = value;
|
|
|
|
|
titleKeyChanged = true;
|
|
|
|
|
reDecrypt = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public CommonKeyType CommonKeyIndex
|
|
|
|
|
{
|
|
|
|
|
get => (CommonKeyType)newKeyIndex;
|
|
|
|
|
set => newKeyIndex = (byte)value;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public ulong TicketID
|
|
|
|
|
{
|
|
|
|
|
get => ticketId;
|
|
|
|
|
set => ticketId = value;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public uint ConsoleID
|
|
|
|
|
{
|
|
|
|
|
get => consoleId;
|
|
|
|
|
set => consoleId = value;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public ulong TitleID
|
|
|
|
|
{
|
|
|
|
|
get => titleId;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
titleId = value;
|
|
|
|
|
if (!reDecrypt)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ReDecryptTitleKey();
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public ushort NumOfDLC
|
|
|
|
|
{
|
|
|
|
|
get => numOfDlc;
|
|
|
|
|
set => numOfDlc = value;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public bool FakeSign
|
|
|
|
|
{
|
|
|
|
|
get => fakeSign;
|
|
|
|
|
set => fakeSign = value;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public bool TitleKeyChanged => titleKeyChanged;
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public event EventHandler<MessageEventArgs> Debug;
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
~Ticket() => Dispose(false);
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
Dispose(true);
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
protected virtual void Dispose(bool disposing)
|
|
|
|
|
{
|
|
|
|
|
if (disposing && !isDisposed)
|
|
|
|
|
{
|
|
|
|
|
decryptedTitleKey = null;
|
|
|
|
|
newEncryptedTitleKey = null;
|
|
|
|
|
signature = null;
|
|
|
|
|
padding = null;
|
|
|
|
|
issuer = null;
|
|
|
|
|
unknown = null;
|
|
|
|
|
encryptedTitleKey = null;
|
|
|
|
|
unknown5 = null;
|
|
|
|
|
unknown6 = null;
|
|
|
|
|
padding4 = null;
|
|
|
|
|
}
|
|
|
|
|
isDisposed = true;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public static Ticket Load(string pathToTicket)
|
|
|
|
|
{
|
|
|
|
|
return Ticket.Load(File.ReadAllBytes(pathToTicket));
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public static Ticket Load(byte[] ticket)
|
|
|
|
|
{
|
|
|
|
|
Ticket ticket1 = new Ticket();
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream(ticket);
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
ticket1.ParseTicket(memoryStream);
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
return ticket1;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public static Ticket Load(Stream ticket)
|
|
|
|
|
{
|
|
|
|
|
Ticket ticket1 = new Ticket();
|
|
|
|
|
ticket1.ParseTicket(ticket);
|
|
|
|
|
return ticket1;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void LoadFile(string pathToTicket)
|
|
|
|
|
{
|
|
|
|
|
LoadFile(File.ReadAllBytes(pathToTicket));
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void LoadFile(byte[] ticket)
|
|
|
|
|
{
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream(ticket);
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
ParseTicket(memoryStream);
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void LoadFile(Stream ticket)
|
|
|
|
|
{
|
|
|
|
|
ParseTicket(ticket);
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void Save(string savePath)
|
|
|
|
|
{
|
|
|
|
|
Save(savePath, false);
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void Save(string savePath, bool fakeSign)
|
|
|
|
|
{
|
|
|
|
|
if (fakeSign)
|
|
|
|
|
{
|
|
|
|
|
this.fakeSign = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (File.Exists(savePath))
|
|
|
|
|
{
|
|
|
|
|
File.Delete(savePath);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using FileStream fileStream = new FileStream(savePath, FileMode.Create);
|
|
|
|
|
WriteToStream(fileStream);
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public MemoryStream ToMemoryStream()
|
|
|
|
|
{
|
|
|
|
|
return ToMemoryStream(false);
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public MemoryStream ToMemoryStream(bool fakeSign)
|
|
|
|
|
{
|
|
|
|
|
if (fakeSign)
|
|
|
|
|
{
|
|
|
|
|
this.fakeSign = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream();
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
WriteToStream(memoryStream);
|
|
|
|
|
return memoryStream;
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public byte[] ToByteArray()
|
|
|
|
|
{
|
|
|
|
|
return ToByteArray(false);
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public byte[] ToByteArray(bool fakeSign)
|
|
|
|
|
{
|
|
|
|
|
if (fakeSign)
|
|
|
|
|
{
|
|
|
|
|
this.fakeSign = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream();
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
WriteToStream(memoryStream);
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
byte[] array = memoryStream.ToArray();
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
return array;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void SetTitleKey(string newTitleKey)
|
|
|
|
|
{
|
|
|
|
|
SetTitleKey(newTitleKey.ToCharArray());
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void SetTitleKey(char[] newTitleKey)
|
|
|
|
|
{
|
|
|
|
|
if (newTitleKey.Length != 16)
|
|
|
|
|
{
|
|
|
|
|
throw new Exception("The title key must be 16 characters long!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int index = 0; index < 16; ++index)
|
|
|
|
|
{
|
|
|
|
|
encryptedTitleKey[index] = (byte)newTitleKey[index];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DecryptTitleKey();
|
|
|
|
|
titleKeyChanged = true;
|
|
|
|
|
reDecrypt = true;
|
|
|
|
|
newEncryptedTitleKey = encryptedTitleKey;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public void SetTitleKey(byte[] newTitleKey)
|
|
|
|
|
{
|
|
|
|
|
encryptedTitleKey = newTitleKey.Length == 16 ? newTitleKey : throw new Exception("The title key must be 16 characters long!");
|
|
|
|
|
DecryptTitleKey();
|
|
|
|
|
titleKeyChanged = true;
|
|
|
|
|
reDecrypt = true;
|
|
|
|
|
newEncryptedTitleKey = newTitleKey;
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
public string GetUpperTitleID()
|
|
|
|
|
{
|
|
|
|
|
byte[] bytes = BitConverter.GetBytes(Shared.Swap((uint)titleId));
|
|
|
|
|
return new string(new char[4]
|
|
|
|
|
{
|
2020-12-28 22:28:44 -06:00
|
|
|
|
(char) bytes[0],
|
|
|
|
|
(char) bytes[1],
|
|
|
|
|
(char) bytes[2],
|
|
|
|
|
(char) bytes[3]
|
2021-02-06 22:53:40 -06:00
|
|
|
|
});
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private void WriteToStream(Stream writeStream)
|
2020-12-28 22:28:44 -06:00
|
|
|
|
{
|
2021-02-06 22:53:40 -06:00
|
|
|
|
FireDebug("Writing Ticket...");
|
|
|
|
|
FireDebug(" Encrypting Title Key...");
|
|
|
|
|
EncryptTitleKey();
|
|
|
|
|
FireDebug(" -> Decrypted Title Key: {0}", (object)Shared.ByteArrayToString(decryptedTitleKey));
|
|
|
|
|
FireDebug(" -> Encrypted Title Key: {0}", (object)Shared.ByteArrayToString(encryptedTitleKey));
|
|
|
|
|
if (fakeSign)
|
|
|
|
|
{
|
|
|
|
|
FireDebug(" Clearing Signature...");
|
|
|
|
|
signature = new byte[256];
|
|
|
|
|
}
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream();
|
|
|
|
|
memoryStream.Seek(0L, SeekOrigin.Begin);
|
|
|
|
|
FireDebug(" Writing Signature Exponent... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(signatureExponent)), 0, 4);
|
|
|
|
|
FireDebug(" Writing Signature... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(signature, 0, signature.Length);
|
|
|
|
|
FireDebug(" Writing Padding... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(padding, 0, padding.Length);
|
|
|
|
|
FireDebug(" Writing Issuer... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(issuer, 0, issuer.Length);
|
|
|
|
|
FireDebug(" Writing Unknown... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(unknown, 0, unknown.Length);
|
|
|
|
|
FireDebug(" Writing Title Key... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(encryptedTitleKey, 0, encryptedTitleKey.Length);
|
|
|
|
|
FireDebug(" Writing Unknown2... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.WriteByte(unknown2);
|
|
|
|
|
FireDebug(" Writing Ticket ID... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(ticketId)), 0, 8);
|
|
|
|
|
FireDebug(" Writing Console ID... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(consoleId)), 0, 4);
|
|
|
|
|
FireDebug(" Writing Title ID... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(titleId)), 0, 8);
|
|
|
|
|
FireDebug(" Writing Unknwon3... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(unknown3)), 0, 2);
|
|
|
|
|
FireDebug(" Writing NumOfDLC... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(numOfDlc)), 0, 2);
|
|
|
|
|
FireDebug(" Writing Unknwon4... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(unknown4)), 0, 8);
|
|
|
|
|
FireDebug(" Writing Padding2... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.WriteByte(padding2);
|
|
|
|
|
FireDebug(" Writing Common Key Index... (Offset: 0x{0})", (object)memoryStream.Position.ToString("x8").ToUpper());
|
|
|
|
|
memoryStream.WriteByte(commonKeyIndex);
|
|
|
|
|
object[] objArray1 = new object[1];
|
|
|
|
|
long position = memoryStream.Position;
|
|
|
|
|
objArray1[0] = position.ToString("x8").ToUpper();
|
|
|
|
|
FireDebug(" Writing Unknown5... (Offset: 0x{0})", objArray1);
|
|
|
|
|
memoryStream.Write(unknown5, 0, unknown5.Length);
|
|
|
|
|
object[] objArray2 = new object[1];
|
|
|
|
|
position = memoryStream.Position;
|
|
|
|
|
objArray2[0] = position.ToString("x8").ToUpper();
|
|
|
|
|
FireDebug(" Writing Unknown6... (Offset: 0x{0})", objArray2);
|
|
|
|
|
memoryStream.Write(unknown6, 0, unknown6.Length);
|
|
|
|
|
object[] objArray3 = new object[1];
|
|
|
|
|
position = memoryStream.Position;
|
|
|
|
|
objArray3[0] = position.ToString("x8").ToUpper();
|
|
|
|
|
FireDebug(" Writing Padding3... (Offset: 0x{0})", objArray3);
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(padding3)), 0, 2);
|
|
|
|
|
object[] objArray4 = new object[1];
|
|
|
|
|
position = memoryStream.Position;
|
|
|
|
|
objArray4[0] = position.ToString("x8").ToUpper();
|
|
|
|
|
FireDebug(" Writing Enable Time Limit... (Offset: 0x{0})", objArray4);
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(enableTimeLimit)), 0, 4);
|
|
|
|
|
object[] objArray5 = new object[1];
|
|
|
|
|
position = memoryStream.Position;
|
|
|
|
|
objArray5[0] = position.ToString("x8").ToUpper();
|
|
|
|
|
FireDebug(" Writing Time Limit... (Offset: 0x{0})", objArray5);
|
|
|
|
|
memoryStream.Write(BitConverter.GetBytes(Shared.Swap(timeLimit)), 0, 4);
|
|
|
|
|
object[] objArray6 = new object[1];
|
|
|
|
|
position = memoryStream.Position;
|
|
|
|
|
objArray6[0] = position.ToString("x8").ToUpper();
|
|
|
|
|
FireDebug(" Writing Padding4... (Offset: 0x{0})", objArray6);
|
|
|
|
|
memoryStream.Write(padding4, 0, padding4.Length);
|
|
|
|
|
byte[] array = memoryStream.ToArray();
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
if (fakeSign)
|
|
|
|
|
{
|
|
|
|
|
FireDebug(" Fakesigning Ticket...");
|
|
|
|
|
//byte[] numArray = new byte[20];
|
|
|
|
|
SHA1 shA1 = SHA1.Create();
|
|
|
|
|
for (ushort index = 0; index < ushort.MaxValue; ++index)
|
|
|
|
|
{
|
|
|
|
|
byte[] bytes = BitConverter.GetBytes(index);
|
|
|
|
|
array[498] = bytes[1];
|
|
|
|
|
array[499] = bytes[0];
|
|
|
|
|
if (shA1.ComputeHash(array)[0] == 0)
|
|
|
|
|
{
|
|
|
|
|
FireDebug(" -> Signed ({0})", (object)index);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (index == 65534)
|
|
|
|
|
{
|
|
|
|
|
FireDebug(" -> Signing Failed...");
|
|
|
|
|
throw new Exception("Fakesigning failed...");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
shA1.Clear();
|
|
|
|
|
}
|
|
|
|
|
writeStream.Seek(0L, SeekOrigin.Begin);
|
|
|
|
|
writeStream.Write(array, 0, array.Length);
|
|
|
|
|
FireDebug("Writing Ticket Finished...");
|
2020-12-28 22:28:44 -06:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private void ParseTicket(Stream ticketFile)
|
|
|
|
|
{
|
|
|
|
|
FireDebug("Parsing Ticket...");
|
|
|
|
|
ticketFile.Seek(0L, SeekOrigin.Begin);
|
|
|
|
|
byte[] buffer = new byte[8];
|
|
|
|
|
FireDebug(" Reading Signature Exponent... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 4);
|
|
|
|
|
signatureExponent = Shared.Swap(BitConverter.ToUInt32(buffer, 0));
|
|
|
|
|
FireDebug(" Reading Signature... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(signature, 0, signature.Length);
|
|
|
|
|
FireDebug(" Reading Padding... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(padding, 0, padding.Length);
|
|
|
|
|
FireDebug(" Reading Issuer... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(issuer, 0, issuer.Length);
|
|
|
|
|
FireDebug(" Reading Unknown... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(unknown, 0, unknown.Length);
|
|
|
|
|
FireDebug(" Reading Title Key... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(encryptedTitleKey, 0, encryptedTitleKey.Length);
|
|
|
|
|
FireDebug(" Reading Unknown2... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
unknown2 = (byte)ticketFile.ReadByte();
|
|
|
|
|
FireDebug(" Reading Ticket ID.. (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 8);
|
|
|
|
|
ticketId = Shared.Swap(BitConverter.ToUInt64(buffer, 0));
|
|
|
|
|
FireDebug(" Reading Console ID... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 4);
|
|
|
|
|
consoleId = Shared.Swap(BitConverter.ToUInt32(buffer, 0));
|
|
|
|
|
FireDebug(" Reading Title ID... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 8);
|
|
|
|
|
titleId = Shared.Swap(BitConverter.ToUInt64(buffer, 0));
|
|
|
|
|
FireDebug(" Reading Unknown3... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
FireDebug(" Reading NumOfDLC... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 4);
|
|
|
|
|
unknown3 = Shared.Swap(BitConverter.ToUInt16(buffer, 0));
|
|
|
|
|
numOfDlc = Shared.Swap(BitConverter.ToUInt16(buffer, 2));
|
|
|
|
|
FireDebug(" Reading Unknown4... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 8);
|
|
|
|
|
unknown4 = Shared.Swap(BitConverter.ToUInt64(buffer, 0));
|
|
|
|
|
FireDebug(" Reading Padding2... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
padding2 = (byte)ticketFile.ReadByte();
|
|
|
|
|
FireDebug(" Reading Common Key Index... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
commonKeyIndex = (byte)ticketFile.ReadByte();
|
|
|
|
|
newKeyIndex = commonKeyIndex;
|
|
|
|
|
FireDebug(" Reading Unknown5... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(unknown5, 0, unknown5.Length);
|
|
|
|
|
FireDebug(" Reading Unknown6... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(unknown6, 0, unknown6.Length);
|
|
|
|
|
FireDebug(" Reading Padding3... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 2);
|
|
|
|
|
padding3 = Shared.Swap(BitConverter.ToUInt16(buffer, 0));
|
|
|
|
|
FireDebug(" Reading Enable Time Limit... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
FireDebug(" Reading Time Limit... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(buffer, 0, 8);
|
|
|
|
|
enableTimeLimit = Shared.Swap(BitConverter.ToUInt32(buffer, 0));
|
|
|
|
|
timeLimit = Shared.Swap(BitConverter.ToUInt32(buffer, 4));
|
|
|
|
|
FireDebug(" Reading Padding4... (Offset: 0x{0})", (object)ticketFile.Position.ToString("x8").ToUpper());
|
|
|
|
|
ticketFile.Read(padding4, 0, padding4.Length);
|
|
|
|
|
FireDebug(" Decrypting Title Key...");
|
|
|
|
|
DecryptTitleKey();
|
|
|
|
|
FireDebug(" -> Encrypted Title Key: {0}", (object)Shared.ByteArrayToString(encryptedTitleKey));
|
|
|
|
|
FireDebug(" -> Decrypted Title Key: {0}", (object)Shared.ByteArrayToString(decryptedTitleKey));
|
|
|
|
|
FireDebug("Parsing Ticket Finished...");
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private void DecryptTitleKey()
|
|
|
|
|
{
|
|
|
|
|
byte[] numArray = commonKeyIndex == 1 ? CommonKey.GetKoreanKey() : CommonKey.GetStandardKey();
|
|
|
|
|
byte[] bytes = BitConverter.GetBytes(Shared.Swap(titleId));
|
|
|
|
|
Array.Resize<byte>(ref bytes, 16);
|
|
|
|
|
RijndaelManaged rijndaelManaged = new RijndaelManaged
|
|
|
|
|
{
|
|
|
|
|
Mode = CipherMode.CBC,
|
|
|
|
|
Padding = PaddingMode.None,
|
|
|
|
|
KeySize = 128,
|
|
|
|
|
BlockSize = 128,
|
|
|
|
|
Key = numArray,
|
|
|
|
|
IV = bytes
|
|
|
|
|
};
|
|
|
|
|
ICryptoTransform decryptor = rijndaelManaged.CreateDecryptor();
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream(encryptedTitleKey);
|
|
|
|
|
CryptoStream cryptoStream = new CryptoStream(memoryStream, decryptor, CryptoStreamMode.Read);
|
|
|
|
|
cryptoStream.Read(decryptedTitleKey, 0, decryptedTitleKey.Length);
|
|
|
|
|
cryptoStream.Dispose();
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
decryptor.Dispose();
|
|
|
|
|
rijndaelManaged.Clear();
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private void EncryptTitleKey()
|
|
|
|
|
{
|
|
|
|
|
commonKeyIndex = newKeyIndex;
|
|
|
|
|
byte[] numArray = commonKeyIndex == 1 ? CommonKey.GetKoreanKey() : CommonKey.GetStandardKey();
|
|
|
|
|
byte[] bytes = BitConverter.GetBytes(Shared.Swap(titleId));
|
|
|
|
|
Array.Resize<byte>(ref bytes, 16);
|
|
|
|
|
RijndaelManaged rijndaelManaged = new RijndaelManaged
|
|
|
|
|
{
|
|
|
|
|
Mode = CipherMode.CBC,
|
|
|
|
|
Padding = PaddingMode.None,
|
|
|
|
|
KeySize = 128,
|
|
|
|
|
BlockSize = 128,
|
|
|
|
|
Key = numArray,
|
|
|
|
|
IV = bytes
|
|
|
|
|
};
|
|
|
|
|
ICryptoTransform encryptor = rijndaelManaged.CreateEncryptor();
|
|
|
|
|
MemoryStream memoryStream = new MemoryStream(decryptedTitleKey);
|
|
|
|
|
CryptoStream cryptoStream = new CryptoStream(memoryStream, encryptor, CryptoStreamMode.Read);
|
|
|
|
|
cryptoStream.Read(encryptedTitleKey, 0, encryptedTitleKey.Length);
|
|
|
|
|
cryptoStream.Dispose();
|
|
|
|
|
memoryStream.Dispose();
|
|
|
|
|
encryptor.Dispose();
|
|
|
|
|
rijndaelManaged.Clear();
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private void ReDecryptTitleKey()
|
|
|
|
|
{
|
|
|
|
|
encryptedTitleKey = newEncryptedTitleKey;
|
|
|
|
|
DecryptTitleKey();
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
|
2021-02-06 22:53:40 -06:00
|
|
|
|
private void FireDebug(string debugMessage, params object[] args)
|
|
|
|
|
{
|
|
|
|
|
EventHandler<MessageEventArgs> debug = Debug;
|
|
|
|
|
if (debug == null)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
debug(new object(), new MessageEventArgs(string.Format(debugMessage, args)));
|
|
|
|
|
}
|
2020-12-28 22:28:44 -06:00
|
|
|
|
}
|
|
|
|
|
}
|