/* This file is part of libWiiSharp
* Copyright (C) 2009 Leathl
* Copyright (C) 2020 Github Contributors
*
* libWiiSharp is free software: you can redistribute it and/or
* modify it under the terms of the GNU General Public License as published
* by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* libWiiSharp is distributed in the hope that it will be
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
using System;
using System.IO;
using System.Security.Cryptography;
namespace libWiiSharp
{
public class Headers
{
private static readonly uint imd5Magic = 1229800501;
private static readonly uint imetMagic = 1229800788;
///
/// Convert HeaderType to int to get it's Length.
///
public enum HeaderType
{
None = 0,
///
/// Used in banner.bin / icon.bin
///
IMD5 = 32,
///
/// Used in opening.bnr
///
ShortIMET = 1536,
///
/// Used in 00000000.app
///
IMET = 1600,
}
#region Public Functions
///
/// Checks a file for Headers.
///
///
///
public static Headers.HeaderType DetectHeader(string pathToFile)
{
return Headers.DetectHeader(File.ReadAllBytes(pathToFile));
}
///
/// Checks the byte array for Headers.
///
///
///
public static Headers.HeaderType DetectHeader(byte[] file)
{
if (file.Length > 68 && (int)Shared.Swap(BitConverter.ToUInt32(file, 64)) == (int)Headers.imetMagic)
{
return Headers.HeaderType.ShortIMET;
}
if (file.Length > 132 && (int)Shared.Swap(BitConverter.ToUInt32(file, 128)) == (int)Headers.imetMagic)
{
return Headers.HeaderType.IMET;
}
return file.Length > 4 && (int)Shared.Swap(BitConverter.ToUInt32(file, 0)) == (int)Headers.imd5Magic ? Headers.HeaderType.IMD5 : Headers.HeaderType.None;
}
///
/// Checks the stream for Headers.
///
///
///
public static Headers.HeaderType DetectHeader(Stream file)
{
byte[] buffer = new byte[4];
if (file.Length > 68L)
{
file.Seek(64L, SeekOrigin.Begin);
file.Read(buffer, 0, buffer.Length);
if ((int)Shared.Swap(BitConverter.ToUInt32(buffer, 0)) == (int)Headers.imetMagic)
{
return Headers.HeaderType.ShortIMET;
}
}
if (file.Length > 132L)
{
file.Seek(128L, SeekOrigin.Begin);
file.Read(buffer, 0, buffer.Length);
if ((int)Shared.Swap(BitConverter.ToUInt32(buffer, 0)) == (int)Headers.imetMagic)
{
return Headers.HeaderType.IMET;
}
}
if (file.Length > 4L)
{
file.Seek(0L, SeekOrigin.Begin);
file.Read(buffer, 0, buffer.Length);
if ((int)Shared.Swap(BitConverter.ToUInt32(buffer, 0)) == (int)Headers.imd5Magic)
{
return Headers.HeaderType.IMD5;
}
}
return Headers.HeaderType.None;
}
#endregion
public class IMET
{
private bool hashesMatch = true;
private bool isShortImet;
private readonly byte[] additionalPadding = new byte[64];
private readonly byte[] padding = new byte[64];
private readonly uint imetMagic = 1229800788;
private readonly uint sizeOfHeader = 1536;
private uint unknown = 3;
private uint iconSize;
private uint bannerSize;
private uint soundSize;
private uint flags;
private byte[] japaneseTitle = new byte[84];
private byte[] englishTitle = new byte[84];
private byte[] germanTitle = new byte[84];
private byte[] frenchTitle = new byte[84];
private byte[] spanishTitle = new byte[84];
private byte[] italianTitle = new byte[84];
private byte[] dutchTitle = new byte[84];
private readonly byte[] unknownTitle1 = new byte[84];
private readonly byte[] unknownTitle2 = new byte[84];
private byte[] koreanTitle = new byte[84];
private readonly byte[] padding2 = new byte[588];
private byte[] hash = new byte[16];
///
/// Short IMET has a padding of 64 bytes at the beginning while Long IMET has 128.
///
public bool IsShortIMET
{
get => isShortImet;
set => isShortImet = value;
}
///
/// The size of uncompressed icon.bin
///
public uint IconSize
{
get => iconSize;
set => iconSize = value;
}
///
/// The size of uncompressed banner.bin
///
public uint BannerSize
{
get => bannerSize;
set => bannerSize = value;
}
///
/// The size of uncompressed sound.bin
///
public uint SoundSize
{
get => soundSize;
set => soundSize = value;
}
///
/// The Japanese Title.
///
public string JapaneseTitle
{
get => ReturnTitleAsString(japaneseTitle);
set => SetTitleFromString(value, 0);
}
///
/// The English Title.
///
public string EnglishTitle
{
get => ReturnTitleAsString(englishTitle);
set => SetTitleFromString(value, 1);
}
///
/// The German Title.
///
public string GermanTitle
{
get => ReturnTitleAsString(germanTitle);
set => SetTitleFromString(value, 2);
}
///
/// The French Title.
///
public string FrenchTitle
{
get => ReturnTitleAsString(frenchTitle);
set => SetTitleFromString(value, 3);
}
///
/// The Spanish Title.
///
public string SpanishTitle
{
get => ReturnTitleAsString(spanishTitle);
set => SetTitleFromString(value, 4);
}
///
/// The Italian Title.
///
public string ItalianTitle
{
get => ReturnTitleAsString(italianTitle);
set => SetTitleFromString(value, 5);
}
///
/// The Dutch Title.
///
public string DutchTitle
{
get => ReturnTitleAsString(dutchTitle);
set => SetTitleFromString(value, 6);
}
///
/// The Korean Title.
///
public string KoreanTitle
{
get => ReturnTitleAsString(koreanTitle);
set => SetTitleFromString(value, 7);
}
///
/// All Titles as a string array.
///
public string[] AllTitles => new string[8]
{
JapaneseTitle,
EnglishTitle,
GermanTitle,
FrenchTitle,
SpanishTitle,
ItalianTitle,
DutchTitle,
KoreanTitle
};
///
/// When parsing an IMET header, this value will turn false if the hash stored in the header doesn't match the headers hash.
///
public bool HashesMatch => hashesMatch;
#region Public Functions
///
/// Loads the IMET Header of a file.
///
///
///
public static Headers.IMET Load(string pathToFile)
{
return Headers.IMET.Load(File.ReadAllBytes(pathToFile));
}
///
/// Loads the IMET Header of a byte array.
///
///
///
public static Headers.IMET Load(byte[] fileOrHeader)
{
Headers.HeaderType headerType = Headers.DetectHeader(fileOrHeader);
switch (headerType)
{
case Headers.HeaderType.ShortIMET:
case Headers.HeaderType.IMET:
Headers.IMET imet = new Headers.IMET();
if (headerType == Headers.HeaderType.ShortIMET)
{
imet.isShortImet = true;
}
MemoryStream memoryStream = new MemoryStream(fileOrHeader);
try
{
imet.ParseHeader(memoryStream);
}
catch
{
memoryStream.Dispose();
throw;
}
memoryStream.Dispose();
return imet;
default:
throw new Exception("No IMET Header found!");
}
}
///
/// Loads the IMET Header of a stream.
///
///
///
public static Headers.IMET Load(Stream fileOrHeader)
{
Headers.HeaderType headerType = Headers.DetectHeader(fileOrHeader);
switch (headerType)
{
case Headers.HeaderType.ShortIMET:
case Headers.HeaderType.IMET:
Headers.IMET imet = new Headers.IMET();
if (headerType == Headers.HeaderType.ShortIMET)
{
imet.isShortImet = true;
}
imet.ParseHeader(fileOrHeader);
return imet;
default:
throw new Exception("No IMET Header found!");
}
}
///
/// Creates a new IMET Header.
///
///
///
///
///
///
///
public static Headers.IMET Create(
bool isShortImet,
int iconSize,
int bannerSize,
int soundSize,
params string[] titles)
{
Headers.IMET imet = new Headers.IMET
{
isShortImet = isShortImet
};
for (int titleIndex = 0; titleIndex < titles.Length; ++titleIndex)
{
imet.SetTitleFromString(titles[titleIndex], titleIndex);
}
for (int length = titles.Length; length < 8; ++length)
{
imet.SetTitleFromString(titles.Length > 1 ? titles[1] : titles[0], length);
}
imet.iconSize = (uint)iconSize;
imet.bannerSize = (uint)bannerSize;
imet.soundSize = (uint)soundSize;
return imet;
}
///
/// Removes the IMET Header of a file.
///
///
public static void RemoveHeader(string pathToFile)
{
byte[] bytes = Headers.IMET.RemoveHeader(File.ReadAllBytes(pathToFile));
File.Delete(pathToFile);
File.WriteAllBytes(pathToFile, bytes);
}
///
/// Removes the IMET Header of a byte array.
///
///
///
public static byte[] RemoveHeader(byte[] file)
{
Headers.HeaderType headerType = Headers.DetectHeader(file);
switch (headerType)
{
case Headers.HeaderType.ShortIMET:
case Headers.HeaderType.IMET:
byte[] numArray = new byte[(int)(file.Length - headerType)];
Array.Copy(file, (int)headerType, numArray, 0, numArray.Length);
return numArray;
default:
throw new Exception("No IMET Header found!");
}
}
///
/// Sets all title to the given string.
///
///
public void SetAllTitles(string newTitle)
{
for (int titleIndex = 0; titleIndex < 10; ++titleIndex)
{
SetTitleFromString(newTitle, titleIndex);
}
}
///
/// Returns the Header as a memory stream.
///
///
public MemoryStream ToMemoryStream()
{
MemoryStream memoryStream = new MemoryStream();
try
{
WriteToStream(memoryStream);
return memoryStream;
}
catch
{
memoryStream.Dispose();
throw;
}
}
///
/// Returns the Header as a byte array.
///
///
public byte[] ToByteArray()
{
return ToMemoryStream().ToArray();
}
///
/// Writes the Header to the given stream.
///
///
public void Write(Stream writeStream)
{
WriteToStream(writeStream);
}
///
/// Changes the Titles.
///
///
public void ChangeTitles(params string[] newTitles)
{
for (int titleIndex = 0; titleIndex < newTitles.Length; ++titleIndex)
{
SetTitleFromString(newTitles[titleIndex], titleIndex);
}
for (int length = newTitles.Length; length < 8; ++length)
{
SetTitleFromString(newTitles.Length > 1 ? newTitles[1] : newTitles[0], length);
}
}
///
/// Returns a string array with the Titles.
///
///
public string[] GetTitles()
{
return new string[8]
{
JapaneseTitle,
EnglishTitle,
GermanTitle,
FrenchTitle,
SpanishTitle,
ItalianTitle,
DutchTitle,
KoreanTitle
};
}
#endregion
#region Private Functions
private void WriteToStream(Stream writeStream)
{
writeStream.Seek(0L, SeekOrigin.Begin);
if (!isShortImet)
{
writeStream.Write(additionalPadding, 0, additionalPadding.Length);
}
writeStream.Write(padding, 0, padding.Length);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(imetMagic)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(sizeOfHeader)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(unknown)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(iconSize)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(bannerSize)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(soundSize)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(flags)), 0, 4);
writeStream.Write(japaneseTitle, 0, japaneseTitle.Length);
writeStream.Write(englishTitle, 0, englishTitle.Length);
writeStream.Write(germanTitle, 0, germanTitle.Length);
writeStream.Write(frenchTitle, 0, frenchTitle.Length);
writeStream.Write(spanishTitle, 0, spanishTitle.Length);
writeStream.Write(italianTitle, 0, italianTitle.Length);
writeStream.Write(dutchTitle, 0, dutchTitle.Length);
writeStream.Write(unknownTitle1, 0, unknownTitle1.Length);
writeStream.Write(unknownTitle2, 0, unknownTitle2.Length);
writeStream.Write(koreanTitle, 0, koreanTitle.Length);
writeStream.Write(padding2, 0, padding2.Length);
int position = (int)writeStream.Position;
hash = new byte[16];
writeStream.Write(hash, 0, hash.Length);
byte[] numArray = new byte[writeStream.Position];
writeStream.Seek(0L, SeekOrigin.Begin);
writeStream.Read(numArray, 0, numArray.Length);
ComputeHash(numArray, !isShortImet ? 64 : 0);
writeStream.Seek(position, SeekOrigin.Begin);
writeStream.Write(hash, 0, hash.Length);
}
private void ComputeHash(byte[] headerBytes, int hashPos)
{
MD5 md5 = MD5.Create();
hash = md5.ComputeHash(headerBytes, hashPos, 1536);
md5.Clear();
}
private void ParseHeader(Stream headerStream)
{
headerStream.Seek(0L, SeekOrigin.Begin);
byte[] buffer1 = new byte[4];
if (!isShortImet)
{
headerStream.Read(additionalPadding, 0, additionalPadding.Length);
}
headerStream.Read(padding, 0, padding.Length);
headerStream.Read(buffer1, 0, 4);
if ((int)Shared.Swap(BitConverter.ToUInt32(buffer1, 0)) != (int)imetMagic)
{
throw new Exception("Invalid Magic!");
}
headerStream.Read(buffer1, 0, 4);
if ((int)Shared.Swap(BitConverter.ToUInt32(buffer1, 0)) != (int)sizeOfHeader)
{
throw new Exception("Invalid Header Size!");
}
headerStream.Read(buffer1, 0, 4);
unknown = Shared.Swap(BitConverter.ToUInt32(buffer1, 0));
headerStream.Read(buffer1, 0, 4);
iconSize = Shared.Swap(BitConverter.ToUInt32(buffer1, 0));
headerStream.Read(buffer1, 0, 4);
bannerSize = Shared.Swap(BitConverter.ToUInt32(buffer1, 0));
headerStream.Read(buffer1, 0, 4);
soundSize = Shared.Swap(BitConverter.ToUInt32(buffer1, 0));
headerStream.Read(buffer1, 0, 4);
flags = Shared.Swap(BitConverter.ToUInt32(buffer1, 0));
headerStream.Read(japaneseTitle, 0, japaneseTitle.Length);
headerStream.Read(englishTitle, 0, englishTitle.Length);
headerStream.Read(germanTitle, 0, germanTitle.Length);
headerStream.Read(frenchTitle, 0, frenchTitle.Length);
headerStream.Read(spanishTitle, 0, spanishTitle.Length);
headerStream.Read(italianTitle, 0, italianTitle.Length);
headerStream.Read(dutchTitle, 0, dutchTitle.Length);
headerStream.Read(unknownTitle1, 0, unknownTitle1.Length);
headerStream.Read(unknownTitle2, 0, unknownTitle2.Length);
headerStream.Read(koreanTitle, 0, koreanTitle.Length);
headerStream.Read(padding2, 0, padding2.Length);
headerStream.Read(this.hash, 0, this.hash.Length);
headerStream.Seek(-16L, SeekOrigin.Current);
headerStream.Write(new byte[16], 0, 16);
byte[] buffer2 = new byte[headerStream.Length];
headerStream.Seek(0L, SeekOrigin.Begin);
headerStream.Read(buffer2, 0, buffer2.Length);
MD5 md5 = MD5.Create();
byte[] hash = md5.ComputeHash(buffer2, !isShortImet ? 64 : 0, 1536);
md5.Clear();
hashesMatch = Shared.CompareByteArrays(hash, this.hash);
}
private string ReturnTitleAsString(byte[] title)
{
string empty = string.Empty;
for (int index = 0; index < 84; index += 2)
{
char ch = BitConverter.ToChar(new byte[2]
{
title[index + 1],
title[index]
}, 0);
if (ch != char.MinValue)
{
empty += ch.ToString();
}
}
return empty;
}
private void SetTitleFromString(string title, int titleIndex)
{
byte[] numArray = new byte[84];
for (int index = 0; index < title.Length; ++index)
{
byte[] bytes = BitConverter.GetBytes(title[index]);
numArray[index * 2 + 1] = bytes[0];
numArray[index * 2] = bytes[1];
}
switch (titleIndex)
{
case 0:
japaneseTitle = numArray;
break;
case 1:
englishTitle = numArray;
break;
case 2:
germanTitle = numArray;
break;
case 3:
frenchTitle = numArray;
break;
case 4:
spanishTitle = numArray;
break;
case 5:
italianTitle = numArray;
break;
case 6:
dutchTitle = numArray;
break;
case 7:
koreanTitle = numArray;
break;
}
}
#endregion
}
public class IMD5
{
private readonly uint imd5Magic = 1229800501;
private uint fileSize;
private readonly byte[] padding = new byte[8];
private byte[] hash = new byte[16];
///
/// The size of the file without the IMD5 Header.
///
public uint FileSize => fileSize;
///
/// The hash of the file without the IMD5 Header.
///
public byte[] Hash => hash;
private IMD5()
{
}
#region Public Functions
///
/// Loads the IMD5 Header of a file.
///
///
///
public static Headers.IMD5 Load(string pathToFile)
{
return Headers.IMD5.Load(File.ReadAllBytes(pathToFile));
}
///
/// Loads the IMD5 Header of a byte array.
///
///
///
public static Headers.IMD5 Load(byte[] fileOrHeader)
{
if (Headers.DetectHeader(fileOrHeader) != Headers.HeaderType.IMD5)
{
throw new Exception("No IMD5 Header found!");
}
Headers.IMD5 imD5 = new Headers.IMD5();
MemoryStream memoryStream = new MemoryStream(fileOrHeader);
try
{
imD5.ParseHeader(memoryStream);
}
catch
{
memoryStream.Dispose();
throw;
}
memoryStream.Dispose();
return imD5;
}
///
/// Loads the IMD5 Header of a stream.
///
///
///
public static Headers.IMD5 Load(Stream fileOrHeader)
{
if (Headers.DetectHeader(fileOrHeader) != Headers.HeaderType.IMD5)
{
throw new Exception("No IMD5 Header found!");
}
Headers.IMD5 imD5 = new Headers.IMD5();
imD5.ParseHeader(fileOrHeader);
return imD5;
}
///
/// Creates a new IMD5 Header.
///
///
///
public static Headers.IMD5 Create(byte[] file)
{
IMD5 imD5 = new IMD5
{
fileSize = (uint)file.Length
};
imD5.ComputeHash(file);
return imD5;
}
///
/// Adds an IMD5 Header to a file.
///
///
public static void AddHeader(string pathToFile)
{
byte[] buffer = Headers.IMD5.AddHeader(File.ReadAllBytes(pathToFile));
File.Delete(pathToFile);
using FileStream fileStream = new FileStream(pathToFile, FileMode.Create);
fileStream.Write(buffer, 0, buffer.Length);
}
///
/// Adds an IMD5 Header to a byte array.
///
///
///
public static byte[] AddHeader(byte[] file)
{
Headers.IMD5 imD5 = Headers.IMD5.Create(file);
MemoryStream memoryStream1 = new MemoryStream();
MemoryStream memoryStream2 = memoryStream1;
imD5.WriteToStream(memoryStream2);
memoryStream1.Write(file, 0, file.Length);
byte[] array = memoryStream1.ToArray();
memoryStream1.Dispose();
return array;
}
///
/// Removes the IMD5 Header of a file.
///
///
public static void RemoveHeader(string pathToFile)
{
byte[] buffer = Headers.IMD5.RemoveHeader(File.ReadAllBytes(pathToFile));
File.Delete(pathToFile);
using FileStream fileStream = new FileStream(pathToFile, FileMode.Create);
fileStream.Write(buffer, 0, buffer.Length);
}
///
/// Removes the IMD5 Header of a byte array.
///
///
///
public static byte[] RemoveHeader(byte[] file)
{
MemoryStream memoryStream = new MemoryStream();
memoryStream.Write(file, 32, file.Length - 32);
byte[] array = memoryStream.ToArray();
memoryStream.Dispose();
return array;
}
///
/// Returns the IMD5 Header as a memory stream.
///
///
public MemoryStream ToMemoryStream()
{
MemoryStream memoryStream = new MemoryStream();
try
{
WriteToStream(memoryStream);
return memoryStream;
}
catch
{
memoryStream.Dispose();
throw;
}
}
///
/// Returns the IMD5 Header as a byte array.
///
///
public byte[] ToByteArray()
{
return ToMemoryStream().ToArray();
}
///
/// Writes the IMD5 Header to the given stream.
///
///
public void Write(Stream writeStream)
{
WriteToStream(writeStream);
}
#endregion
#region Private Functions
private void WriteToStream(Stream writeStream)
{
writeStream.Seek(0L, SeekOrigin.Begin);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(imd5Magic)), 0, 4);
writeStream.Write(BitConverter.GetBytes(Shared.Swap(fileSize)), 0, 4);
writeStream.Write(padding, 0, padding.Length);
writeStream.Write(hash, 0, hash.Length);
}
private void ComputeHash(byte[] bytesToHash)
{
MD5 md5 = MD5.Create();
hash = md5.ComputeHash(bytesToHash);
md5.Clear();
}
private void ParseHeader(Stream headerStream)
{
headerStream.Seek(0L, SeekOrigin.Begin);
byte[] buffer = new byte[4];
headerStream.Read(buffer, 0, 4);
if ((int)Shared.Swap(BitConverter.ToUInt32(buffer, 0)) != (int)imd5Magic)
{
throw new Exception("Invalid Magic!");
}
headerStream.Read(buffer, 0, 4);
fileSize = Shared.Swap(BitConverter.ToUInt32(buffer, 0));
headerStream.Read(padding, 0, padding.Length);
headerStream.Read(hash, 0, hash.Length);
}
#endregion
}
}
}