using System.IO; using SabreTools.Data.Models.N3DS; namespace SabreTools.Wrappers { public partial class CIA : WrapperBase { #region Descriptive Properties /// public override string DescriptionString => "CTR Importable Archive (CIA)"; #endregion #region Constructors /// public CIA(Data.Models.N3DS.CIA model, byte[] data) : base(model, data) { } /// public CIA(Data.Models.N3DS.CIA model, byte[] data, int offset) : base(model, data, offset) { } /// public CIA(Data.Models.N3DS.CIA model, byte[] data, int offset, int length) : base(model, data, offset, length) { } /// public CIA(Data.Models.N3DS.CIA model, Stream data) : base(model, data) { } /// public CIA(Data.Models.N3DS.CIA model, Stream data, long offset) : base(model, data, offset) { } /// public CIA(Data.Models.N3DS.CIA model, Stream data, long offset, long length) : base(model, data, offset, length) { } #endregion #region Static Constructors /// /// Create a CIA archive from a byte array and offset /// /// Byte array representing the archive /// Offset within the array to parse /// A CIA archive wrapper on success, null on failure public static CIA? Create(byte[]? data, int offset) { // If the data is invalid if (data is null || data.Length == 0) return null; // If the offset is out of bounds if (offset < 0 || offset >= data.Length) return null; // Create a memory stream and use that var dataStream = new MemoryStream(data, offset, data.Length - offset); return Create(dataStream); } /// /// Create a CIA archive from a Stream /// /// Stream representing the archive /// A CIA archive wrapper on success, null on failure public static CIA? Create(Stream? data) { // If the data is invalid if (data is null || !data.CanRead) return null; try { // Cache the current offset long currentOffset = data.Position; var model = new Serialization.Readers.CIA().Deserialize(data); if (model is null) return null; return new CIA(model, data, currentOffset); } catch { return null; } } #endregion // TODO: Hook these up for external use #region Currently Unused Extensions #region Ticket /// /// Denotes if the ticket denotes a demo or not /// public static bool IsDemo(Ticket? ticket) { if (ticket?.Limits is null || ticket.Limits.Length == 0) return false; return ticket.Limits[0] == 0x0004; } /// /// Denotes if the max playcount for a demo /// public static uint PlayCount(Ticket ticket) { if (ticket?.Limits is null || ticket.Limits.Length == 0) return 0; return ticket.Limits[1]; } #endregion #endregion } }