using System.IO; using System.Text; namespace BinaryObjectScanner.Wrappers { public class Nitro : WrapperBase { #region Descriptive Properties /// public override string DescriptionString => "Nintendo DS/DSi Cart Image"; #endregion #region Pass-Through Properties #region Common Header /// #if NET48 public string GameTitle => _model.CommonHeader.GameTitle; #else public string? GameTitle => _model.CommonHeader?.GameTitle; #endif /// #if NET48 public uint GameCode => _model.CommonHeader.GameCode; #else public uint? GameCode => _model.CommonHeader?.GameCode; #endif /// #if NET48 public string MakerCode => _model.CommonHeader.MakerCode; #else public string? MakerCode => _model.CommonHeader?.MakerCode; #endif /// #if NET48 public SabreTools.Models.Nitro.Unitcode UnitCode => _model.CommonHeader.UnitCode; #else public SabreTools.Models.Nitro.Unitcode? UnitCode => _model.CommonHeader?.UnitCode; #endif /// #if NET48 public byte EncryptionSeedSelect => _model.CommonHeader.EncryptionSeedSelect; #else public byte? EncryptionSeedSelect => _model.CommonHeader?.EncryptionSeedSelect; #endif /// #if NET48 public byte DeviceCapacity => _model.CommonHeader.DeviceCapacity; #else public byte? DeviceCapacity => _model.CommonHeader?.DeviceCapacity; #endif /// #if NET48 public byte[] Reserved1 => _model.CommonHeader.Reserved1; #else public byte[]? Reserved1 => _model.CommonHeader?.Reserved1; #endif /// #if NET48 public ushort GameRevision => _model.CommonHeader.GameRevision; #else public ushort? GameRevision => _model.CommonHeader?.GameRevision; #endif /// #if NET48 public byte RomVersion => _model.CommonHeader.RomVersion; #else public byte? RomVersion => _model.CommonHeader?.RomVersion; #endif /// #if NET48 public byte InternalFlags => _model.CommonHeader.InternalFlags; #else public byte? InternalFlags => _model.CommonHeader?.InternalFlags; #endif /// #if NET48 public uint ARM9RomOffset => _model.CommonHeader.ARM9RomOffset; #else public uint? ARM9RomOffset => _model.CommonHeader?.ARM9RomOffset; #endif /// #if NET48 public uint ARM9EntryAddress => _model.CommonHeader.ARM9EntryAddress; #else public uint? ARM9EntryAddress => _model.CommonHeader?.ARM9EntryAddress; #endif /// #if NET48 public uint ARM9LoadAddress => _model.CommonHeader.ARM9LoadAddress; #else public uint? ARM9LoadAddress => _model.CommonHeader?.ARM9LoadAddress; #endif /// #if NET48 public uint ARM9Size => _model.CommonHeader.ARM9Size; #else public uint? ARM9Size => _model.CommonHeader?.ARM9Size; #endif /// #if NET48 public uint ARM7RomOffset => _model.CommonHeader.ARM7RomOffset; #else public uint? ARM7RomOffset => _model.CommonHeader?.ARM7RomOffset; #endif /// #if NET48 public uint ARM7EntryAddress => _model.CommonHeader.ARM7EntryAddress; #else public uint? ARM7EntryAddress => _model.CommonHeader?.ARM7EntryAddress; #endif /// #if NET48 public uint ARM7LoadAddress => _model.CommonHeader.ARM7LoadAddress; #else public uint? ARM7LoadAddress => _model.CommonHeader?.ARM7LoadAddress; #endif /// #if NET48 public uint ARM7Size => _model.CommonHeader.ARM7Size; #else public uint? ARM7Size => _model.CommonHeader?.ARM7Size; #endif /// #if NET48 public uint FileNameTableOffset => _model.CommonHeader.FileNameTableOffset; #else public uint? FileNameTableOffset => _model.CommonHeader?.FileNameTableOffset; #endif /// #if NET48 public uint FileNameTableLength => _model.CommonHeader.FileNameTableLength; #else public uint? FileNameTableLength => _model.CommonHeader?.FileNameTableLength; #endif /// #if NET48 public uint FileAllocationTableOffset => _model.CommonHeader.FileAllocationTableOffset; #else public uint? FileAllocationTableOffset => _model.CommonHeader?.FileAllocationTableOffset; #endif /// #if NET48 public uint FileAllocationTableLength => _model.CommonHeader.FileAllocationTableLength; #else public uint? FileAllocationTableLength => _model.CommonHeader?.FileAllocationTableLength; #endif /// #if NET48 public uint ARM9OverlayOffset => _model.CommonHeader.ARM9OverlayOffset; #else public uint? ARM9OverlayOffset => _model.CommonHeader?.ARM9OverlayOffset; #endif /// #if NET48 public uint ARM9OverlayLength => _model.CommonHeader.ARM9OverlayLength; #else public uint? ARM9OverlayLength => _model.CommonHeader?.ARM9OverlayLength; #endif /// #if NET48 public uint ARM7OverlayOffset => _model.CommonHeader.ARM7OverlayOffset; #else public uint? ARM7OverlayOffset => _model.CommonHeader?.ARM7OverlayOffset; #endif /// #if NET48 public uint ARM7OverlayLength => _model.CommonHeader.ARM7OverlayLength; #else public uint? ARM7OverlayLength => _model.CommonHeader?.ARM7OverlayLength; #endif /// #if NET48 public uint NormalCardControlRegisterSettings => _model.CommonHeader.NormalCardControlRegisterSettings; #else public uint? NormalCardControlRegisterSettings => _model.CommonHeader?.NormalCardControlRegisterSettings; #endif /// #if NET48 public uint SecureCardControlRegisterSettings => _model.CommonHeader.SecureCardControlRegisterSettings; #else public uint? SecureCardControlRegisterSettings => _model.CommonHeader?.SecureCardControlRegisterSettings; #endif /// #if NET48 public uint IconBannerOffset => _model.CommonHeader.IconBannerOffset; #else public uint? IconBannerOffset => _model.CommonHeader?.IconBannerOffset; #endif /// #if NET48 public ushort SecureAreaCRC => _model.CommonHeader.SecureAreaCRC; #else public ushort? SecureAreaCRC => _model.CommonHeader?.SecureAreaCRC; #endif /// #if NET48 public ushort SecureTransferTimeout => _model.CommonHeader.SecureTransferTimeout; #else public ushort? SecureTransferTimeout => _model.CommonHeader?.SecureTransferTimeout; #endif /// #if NET48 public uint ARM9Autoload => _model.CommonHeader.ARM9Autoload; #else public uint? ARM9Autoload => _model.CommonHeader?.ARM9Autoload; #endif /// #if NET48 public uint ARM7Autoload => _model.CommonHeader.ARM7Autoload; #else public uint? ARM7Autoload => _model.CommonHeader?.ARM7Autoload; #endif /// #if NET48 public byte[] SecureDisable => _model.CommonHeader.SecureDisable; #else public byte[]? SecureDisable => _model.CommonHeader?.SecureDisable; #endif /// #if NET48 public uint NTRRegionRomSize => _model.CommonHeader.NTRRegionRomSize; #else public uint? NTRRegionRomSize => _model.CommonHeader?.NTRRegionRomSize; #endif /// #if NET48 public uint HeaderSize => _model.CommonHeader.HeaderSize; #else public uint? HeaderSize => _model.CommonHeader?.HeaderSize; #endif /// #if NET48 public byte[] Reserved2 => _model.CommonHeader.Reserved2; #else public byte[]? Reserved2 => _model.CommonHeader?.Reserved2; #endif /// #if NET48 public byte[] NintendoLogo => _model.CommonHeader.NintendoLogo; #else public byte[]? NintendoLogo => _model.CommonHeader?.NintendoLogo; #endif /// #if NET48 public ushort NintendoLogoCRC => _model.CommonHeader.NintendoLogoCRC; #else public ushort? NintendoLogoCRC => _model.CommonHeader?.NintendoLogoCRC; #endif /// #if NET48 public ushort HeaderCRC => _model.CommonHeader.HeaderCRC; #else public ushort? HeaderCRC => _model.CommonHeader?.HeaderCRC; #endif /// #if NET48 public byte[] DebuggerReserved => _model.CommonHeader.DebuggerReserved; #else public byte[]? DebuggerReserved => _model.CommonHeader?.DebuggerReserved; #endif #endregion #region Extended DSi Header /// #if NET48 public uint[] GlobalMBK15Settings => _model.ExtendedDSiHeader?.GlobalMBK15Settings; #else public uint[]? GlobalMBK15Settings => _model.ExtendedDSiHeader?.GlobalMBK15Settings; #endif /// #if NET48 public uint[] LocalMBK68SettingsARM9 => _model.ExtendedDSiHeader?.LocalMBK68SettingsARM9; #else public uint[]? LocalMBK68SettingsARM9 => _model.ExtendedDSiHeader?.LocalMBK68SettingsARM9; #endif /// #if NET48 public uint[] LocalMBK68SettingsARM7 => _model.ExtendedDSiHeader?.LocalMBK68SettingsARM7; #else public uint[]? LocalMBK68SettingsARM7 => _model.ExtendedDSiHeader?.LocalMBK68SettingsARM7; #endif /// public uint? GlobalMBK9Setting => _model.ExtendedDSiHeader?.GlobalMBK9Setting; /// public uint? RegionFlags => _model.ExtendedDSiHeader?.RegionFlags; /// public uint? AccessControl => _model.ExtendedDSiHeader?.AccessControl; /// public uint? ARM7SCFGEXTMask => _model.ExtendedDSiHeader?.ARM7SCFGEXTMask; /// public uint? ReservedFlags => _model.ExtendedDSiHeader?.ReservedFlags; /// public uint? ARM9iRomOffset => _model.ExtendedDSiHeader?.ARM9iRomOffset; /// public uint? Reserved3 => _model.ExtendedDSiHeader?.Reserved3; /// public uint? ARM9iLoadAddress => _model.ExtendedDSiHeader?.ARM9iLoadAddress; /// public uint? ARM9iSize => _model.ExtendedDSiHeader?.ARM9iSize; /// public uint? ARM7iRomOffset => _model.ExtendedDSiHeader?.ARM7iRomOffset; /// public uint? Reserved4 => _model.ExtendedDSiHeader?.Reserved4; /// public uint? ARM7iLoadAddress => _model.ExtendedDSiHeader?.ARM7iLoadAddress; /// public uint? ARM7iSize => _model.ExtendedDSiHeader?.ARM7iSize; /// public uint? DigestNTRRegionOffset => _model.ExtendedDSiHeader?.DigestNTRRegionOffset; /// public uint? DigestNTRRegionLength => _model.ExtendedDSiHeader?.DigestNTRRegionLength; /// public uint? DigestTWLRegionOffset => _model.ExtendedDSiHeader?.DigestTWLRegionOffset; /// public uint? DigestTWLRegionLength => _model.ExtendedDSiHeader?.DigestTWLRegionLength; /// public uint? DigestSectorHashtableRegionOffset => _model.ExtendedDSiHeader?.DigestSectorHashtableRegionOffset; /// public uint? DigestSectorHashtableRegionLength => _model.ExtendedDSiHeader?.DigestSectorHashtableRegionLength; /// public uint? DigestBlockHashtableRegionOffset => _model.ExtendedDSiHeader?.DigestBlockHashtableRegionOffset; /// public uint? DigestBlockHashtableRegionLength => _model.ExtendedDSiHeader?.DigestBlockHashtableRegionLength; /// public uint? DigestSectorSize => _model.ExtendedDSiHeader?.DigestSectorSize; /// public uint? DigestBlockSectorCount => _model.ExtendedDSiHeader?.DigestBlockSectorCount; /// public uint? IconBannerSize => _model.ExtendedDSiHeader?.IconBannerSize; /// public uint? Unknown1 => _model.ExtendedDSiHeader?.Unknown1; /// public uint? ModcryptArea1Offset => _model.ExtendedDSiHeader?.ModcryptArea1Offset; /// public uint? ModcryptArea1Size => _model.ExtendedDSiHeader?.ModcryptArea1Size; /// public uint? ModcryptArea2Offset => _model.ExtendedDSiHeader?.ModcryptArea2Offset; /// public uint? ModcryptArea2Size => _model.ExtendedDSiHeader?.ModcryptArea2Size; /// #if NET48 public byte[] TitleID => _model.ExtendedDSiHeader?.TitleID; #else public byte[]? TitleID => _model.ExtendedDSiHeader?.TitleID; #endif /// public uint? DSiWarePublicSavSize => _model.ExtendedDSiHeader?.DSiWarePublicSavSize; /// public uint? DSiWarePrivateSavSize => _model.ExtendedDSiHeader?.DSiWarePrivateSavSize; /// #if NET48 public byte[] ReservedZero => _model.ExtendedDSiHeader?.ReservedZero; #else public byte[]? ReservedZero => _model.ExtendedDSiHeader?.ReservedZero; #endif /// #if NET48 public byte[] Unknown2 => _model.ExtendedDSiHeader?.Unknown2; #else public byte[]? Unknown2 => _model.ExtendedDSiHeader?.Unknown2; #endif /// #if NET48 public byte[] ARM9WithSecureAreaSHA1HMACHash => _model.ExtendedDSiHeader?.ARM9WithSecureAreaSHA1HMACHash; #else public byte[]? ARM9WithSecureAreaSHA1HMACHash => _model.ExtendedDSiHeader?.ARM9WithSecureAreaSHA1HMACHash; #endif /// #if NET48 public byte[] ARM7SHA1HMACHash => _model.ExtendedDSiHeader?.ARM7SHA1HMACHash; #else public byte[]? ARM7SHA1HMACHash => _model.ExtendedDSiHeader?.ARM7SHA1HMACHash; #endif /// #if NET48 public byte[] DigestMasterSHA1HMACHash => _model.ExtendedDSiHeader?.DigestMasterSHA1HMACHash; #else public byte[]? DigestMasterSHA1HMACHash => _model.ExtendedDSiHeader?.DigestMasterSHA1HMACHash; #endif /// #if NET48 public byte[] BannerSHA1HMACHash => _model.ExtendedDSiHeader?.BannerSHA1HMACHash; #else public byte[]? BannerSHA1HMACHash => _model.ExtendedDSiHeader?.BannerSHA1HMACHash; #endif /// #if NET48 public byte[] ARM9iDecryptedSHA1HMACHash => _model.ExtendedDSiHeader?.ARM9iDecryptedSHA1HMACHash; #else public byte[]? ARM9iDecryptedSHA1HMACHash => _model.ExtendedDSiHeader?.ARM9iDecryptedSHA1HMACHash; #endif /// #if NET48 public byte[] ARM7iDecryptedSHA1HMACHash => _model.ExtendedDSiHeader?.ARM7iDecryptedSHA1HMACHash; #else public byte[]? ARM7iDecryptedSHA1HMACHash => _model.ExtendedDSiHeader?.ARM7iDecryptedSHA1HMACHash; #endif /// #if NET48 public byte[] Reserved5 => _model.ExtendedDSiHeader?.Reserved5; #else public byte[]? Reserved5 => _model.ExtendedDSiHeader?.Reserved5; #endif /// #if NET48 public byte[] ARM9NoSecureAreaSHA1HMACHash => _model.ExtendedDSiHeader?.ARM9NoSecureAreaSHA1HMACHash; #else public byte[]? ARM9NoSecureAreaSHA1HMACHash => _model.ExtendedDSiHeader?.ARM9NoSecureAreaSHA1HMACHash; #endif /// #if NET48 public byte[] Reserved6 => _model.ExtendedDSiHeader?.Reserved6; #else public byte[]? Reserved6 => _model.ExtendedDSiHeader?.Reserved6; #endif /// #if NET48 public byte[] ReservedAndUnchecked => _model.ExtendedDSiHeader?.ReservedAndUnchecked; #else public byte[]? ReservedAndUnchecked => _model.ExtendedDSiHeader?.ReservedAndUnchecked; #endif /// #if NET48 public byte[] RSASignature => _model.ExtendedDSiHeader?.RSASignature; #else public byte[]? RSASignature => _model.ExtendedDSiHeader?.RSASignature; #endif #endregion #region Secure Area /// #if NET48 public byte[] SecureArea => _model.SecureArea; #else public byte[]? SecureArea => _model.SecureArea; #endif #endregion #region Name Table /// #if NET48 public SabreTools.Models.Nitro.FolderAllocationTableEntry[] FolderAllocationTable => _model.NameTable.FolderAllocationTable; #else public SabreTools.Models.Nitro.FolderAllocationTableEntry?[]? FolderAllocationTable => _model.NameTable?.FolderAllocationTable; #endif /// #if NET48 public SabreTools.Models.Nitro.NameListEntry[] NameList => _model.NameTable.NameList; #else public SabreTools.Models.Nitro.NameListEntry?[]? NameList => _model.NameTable?.NameList; #endif #endregion #region File Allocation Table /// #if NET48 public SabreTools.Models.Nitro.FileAllocationTableEntry[] FileAllocationTable => _model.FileAllocationTable; #else public SabreTools.Models.Nitro.FileAllocationTableEntry?[]? FileAllocationTable => _model.FileAllocationTable; #endif #endregion #endregion #region Constructors /// #if NET48 public Nitro(SabreTools.Models.Nitro.Cart model, byte[] data, int offset) #else public Nitro(SabreTools.Models.Nitro.Cart? model, byte[]? data, int offset) #endif : base(model, data, offset) { // All logic is handled by the base class } /// #if NET48 public Nitro(SabreTools.Models.Nitro.Cart model, Stream data) #else public Nitro(SabreTools.Models.Nitro.Cart? model, Stream? data) #endif : base(model, data) { // All logic is handled by the base class } /// /// Create a NDS cart image from a byte array and offset /// /// Byte array representing the archive /// Offset within the array to parse /// A NDS cart image wrapper on success, null on failure #if NET48 public static Nitro Create(byte[] data, int offset) #else public static Nitro? Create(byte[]? data, int offset) #endif { // If the data is invalid if (data == null) return null; // If the offset is out of bounds if (offset < 0 || offset >= data.Length) return null; // Create a memory stream and use that MemoryStream dataStream = new MemoryStream(data, offset, data.Length - offset); return Create(dataStream); } /// /// Create a NDS cart image from a Stream /// /// Stream representing the archive /// A NDS cart image wrapper on success, null on failure #if NET48 public static Nitro Create(Stream data) #else public static Nitro? Create(Stream? data) #endif { // If the data is invalid if (data == null || data.Length == 0 || !data.CanSeek || !data.CanRead) return null; var archive = new SabreTools.Serialization.Streams.Nitro().Deserialize(data); if (archive == null) return null; try { return new Nitro(archive, data); } catch { return null; } } #endregion #region Printing /// public override StringBuilder PrettyPrint() { StringBuilder builder = new StringBuilder(); Printing.Nitro.Print(builder, _model); return builder; } #if NET6_0_OR_GREATER /// public override string ExportJSON() => System.Text.Json.JsonSerializer.Serialize(_model, _jsonSerializerOptions); #endif #endregion } }