using System.IO; using System.Text; namespace BinaryObjectScanner.Wrappers { public class MSDOS : WrapperBase { #region Descriptive Properties /// public override string DescriptionString => "MS-DOS Executable"; #endregion #region Pass-Through Properties #region Header /// #if NET48 public string Magic => this.Model.Header.Magic; #else public string? Magic => this.Model.Header?.Magic; #endif /// #if NET48 public ushort LastPageBytes => this.Model.Header.LastPageBytes; #else public ushort? LastPageBytes => this.Model.Header?.LastPageBytes; #endif /// #if NET48 public ushort Pages => this.Model.Header.Pages; #else public ushort? Pages => this.Model.Header?.Pages; #endif /// #if NET48 public ushort RelocationItems => this.Model.Header.RelocationItems; #else public ushort? RelocationItems => this.Model.Header?.RelocationItems; #endif /// #if NET48 public ushort HeaderParagraphSize => this.Model.Header.HeaderParagraphSize; #else public ushort? HeaderParagraphSize => this.Model.Header?.HeaderParagraphSize; #endif /// #if NET48 public ushort MinimumExtraParagraphs => this.Model.Header.MinimumExtraParagraphs; #else public ushort? MinimumExtraParagraphs => this.Model.Header?.MinimumExtraParagraphs; #endif /// #if NET48 public ushort MaximumExtraParagraphs => this.Model.Header.MaximumExtraParagraphs; #else public ushort? MaximumExtraParagraphs => this.Model.Header?.MaximumExtraParagraphs; #endif /// #if NET48 public ushort InitialSSValue => this.Model.Header.InitialSSValue; #else public ushort? InitialSSValue => this.Model.Header?.InitialSSValue; #endif /// #if NET48 public ushort InitialSPValue => this.Model.Header.InitialSPValue; #else public ushort? InitialSPValue => this.Model.Header?.InitialSPValue; #endif /// #if NET48 public ushort Checksum => this.Model.Header.Checksum; #else public ushort? Checksum => this.Model.Header?.Checksum; #endif /// #if NET48 public ushort InitialIPValue => this.Model.Header.InitialIPValue; #else public ushort? InitialIPValue => this.Model.Header?.InitialIPValue; #endif /// #if NET48 public ushort InitialCSValue => this.Model.Header.InitialCSValue; #else public ushort? InitialCSValue => this.Model.Header?.InitialCSValue; #endif /// #if NET48 public ushort RelocationTableAddr => this.Model.Header.RelocationTableAddr; #else public ushort? RelocationTableAddr => this.Model.Header?.RelocationTableAddr; #endif /// #if NET48 public ushort OverlayNumber => this.Model.Header.OverlayNumber; #else public ushort? OverlayNumber => this.Model.Header?.OverlayNumber; #endif #endregion #region PE Extensions /// #if NET48 public ushort[] Reserved1 => this.Model.Header.Reserved1; #else public ushort[]? Reserved1 => this.Model.Header?.Reserved1; #endif /// #if NET48 public ushort OEMIdentifier => this.Model.Header.OEMIdentifier; #else public ushort? OEMIdentifier => this.Model.Header?.OEMIdentifier; #endif /// #if NET48 public ushort OEMInformation => this.Model.Header.OEMInformation; #else public ushort? OEMInformation => this.Model.Header?.OEMInformation; #endif /// #if NET48 public ushort[] Reserved2 => this.Model.Header.Reserved2; #else public ushort[]? Reserved2 => this.Model.Header?.Reserved2; #endif /// #if NET48 public uint NewExeHeaderAddr => this.Model.Header.NewExeHeaderAddr; #else public uint? NewExeHeaderAddr => this.Model.Header?.NewExeHeaderAddr; #endif #endregion #region Relocation Table /// #if NET48 public SabreTools.Models.MSDOS.RelocationEntry[] RelocationTable => this.Model.RelocationTable; #else public SabreTools.Models.MSDOS.RelocationEntry?[]? RelocationTable => this.Model.RelocationTable; #endif #endregion #endregion #region Constructors /// #if NET48 public MSDOS(SabreTools.Models.MSDOS.Executable model, byte[] data, int offset) #else public MSDOS(SabreTools.Models.MSDOS.Executable? model, byte[]? data, int offset) #endif : base(model, data, offset) { // All logic is handled by the base class } /// #if NET48 public MSDOS(SabreTools.Models.MSDOS.Executable model, Stream data) #else public MSDOS(SabreTools.Models.MSDOS.Executable? model, Stream? data) #endif : base(model, data) { // All logic is handled by the base class }/// /// Create an MS-DOS executable from a byte array and offset /// /// Byte array representing the executable /// Offset within the array to parse /// An MS-DOS executable wrapper on success, null on failure #if NET48 public static MSDOS Create(byte[] data, int offset) #else public static MSDOS? 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 an MS-DOS executable from a Stream /// /// Stream representing the executable /// An MS-DOS executable wrapper on success, null on failure #if NET48 public static MSDOS Create(Stream data) #else public static MSDOS? Create(Stream? data) #endif { // If the data is invalid if (data == null || data.Length == 0 || !data.CanSeek || !data.CanRead) return null; var executable = new SabreTools.Serialization.Streams.MSDOS().Deserialize(data); if (executable == null) return null; try { return new MSDOS(executable, data); } catch { return null; } } #endregion #region Printing /// public override StringBuilder PrettyPrint() { StringBuilder builder = new StringBuilder(); Printing.MSDOS.Print(builder, this.Model); return builder; } #endregion } }