mirror of
https://github.com/SabreTools/BinaryObjectScanner.git
synced 2026-02-07 21:30:13 +00:00
175 lines
6.0 KiB
C#
175 lines
6.0 KiB
C#
using System.IO;
|
|
using System.Text;
|
|
using BinaryObjectScanner.Models.MSDOS;
|
|
using BinaryObjectScanner.Utilities;
|
|
using static BinaryObjectScanner.Models.MSDOS.Constants;
|
|
|
|
namespace BinaryObjectScanner.Builders
|
|
{
|
|
public static class MSDOS
|
|
{
|
|
#region Byte Data
|
|
|
|
/// <summary>
|
|
/// Parse a byte array into an MS-DOS executable
|
|
/// </summary>
|
|
/// <param name="data">Byte array to parse</param>
|
|
/// <param name="offset">Offset into the byte array</param>
|
|
/// <returns>Filled executable on success, null on error</returns>
|
|
public static Executable ParseExecutable(byte[] data, int offset)
|
|
{
|
|
// 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 parse that
|
|
MemoryStream dataStream = new MemoryStream(data, offset, data.Length - offset);
|
|
return ParseExecutable(dataStream);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Stream Data
|
|
|
|
/// <summary>
|
|
/// Parse a Stream into an MS-DOS executable
|
|
/// </summary>
|
|
/// <param name="data">Stream to parse</param>
|
|
/// <returns>Filled executable on success, null on error</returns>
|
|
public static Executable ParseExecutable(Stream data)
|
|
{
|
|
// If the data is invalid
|
|
if (data == null || data.Length == 0 || !data.CanSeek || !data.CanRead)
|
|
return null;
|
|
|
|
// If the offset is out of bounds
|
|
if (data.Position < 0 || data.Position >= data.Length)
|
|
return null;
|
|
|
|
// Cache the current offset
|
|
int initialOffset = (int)data.Position;
|
|
|
|
// Create a new executable to fill
|
|
var executable = new Executable();
|
|
|
|
#region Executable Header
|
|
|
|
// Try to parse the executable header
|
|
var executableHeader = ParseExecutableHeader(data);
|
|
if (executableHeader == null)
|
|
return null;
|
|
|
|
// Set the executable header
|
|
executable.Header = executableHeader;
|
|
|
|
#endregion
|
|
|
|
#region Relocation Table
|
|
|
|
// If the offset for the relocation table doesn't exist
|
|
int tableAddress = initialOffset + executableHeader.RelocationTableAddr;
|
|
if (tableAddress >= data.Length)
|
|
return executable;
|
|
|
|
// Try to parse the relocation table
|
|
data.Seek(tableAddress, SeekOrigin.Begin);
|
|
var relocationTable = ParseRelocationTable(data, executableHeader.RelocationItems);
|
|
if (relocationTable == null)
|
|
return null;
|
|
|
|
// Set the relocation table
|
|
executable.RelocationTable = relocationTable;
|
|
|
|
#endregion
|
|
|
|
// Return the executable
|
|
return executable;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse a Stream into an MS-DOS executable header
|
|
/// </summary>
|
|
/// <param name="data">Stream to parse</param>
|
|
/// <returns>Filled executable header on success, null on error</returns>
|
|
private static ExecutableHeader ParseExecutableHeader(Stream data)
|
|
{
|
|
// TODO: Use marshalling here instead of building
|
|
var header = new ExecutableHeader();
|
|
|
|
#region Standard Fields
|
|
|
|
byte[] magic = data.ReadBytes(2);
|
|
header.Magic = Encoding.ASCII.GetString(magic);
|
|
if (header.Magic != SignatureString)
|
|
return null;
|
|
|
|
header.LastPageBytes = data.ReadUInt16();
|
|
header.Pages = data.ReadUInt16();
|
|
header.RelocationItems = data.ReadUInt16();
|
|
header.HeaderParagraphSize = data.ReadUInt16();
|
|
header.MinimumExtraParagraphs = data.ReadUInt16();
|
|
header.MaximumExtraParagraphs = data.ReadUInt16();
|
|
header.InitialSSValue = data.ReadUInt16();
|
|
header.InitialSPValue = data.ReadUInt16();
|
|
header.Checksum = data.ReadUInt16();
|
|
header.InitialIPValue = data.ReadUInt16();
|
|
header.InitialCSValue = data.ReadUInt16();
|
|
header.RelocationTableAddr = data.ReadUInt16();
|
|
header.OverlayNumber = data.ReadUInt16();
|
|
|
|
#endregion
|
|
|
|
// If we don't have enough data for PE extensions
|
|
if (data.Position >= data.Length || data.Length - data.Position < 36)
|
|
return header;
|
|
|
|
#region PE Extensions
|
|
|
|
header.Reserved1 = new ushort[4];
|
|
for (int i = 0; i < header.Reserved1.Length; i++)
|
|
{
|
|
header.Reserved1[i] = data.ReadUInt16();
|
|
}
|
|
header.OEMIdentifier = data.ReadUInt16();
|
|
header.OEMInformation = data.ReadUInt16();
|
|
header.Reserved2 = new ushort[10];
|
|
for (int i = 0; i < header.Reserved2.Length; i++)
|
|
{
|
|
header.Reserved2[i] = data.ReadUInt16();
|
|
}
|
|
header.NewExeHeaderAddr = data.ReadUInt32();
|
|
|
|
#endregion
|
|
|
|
return header;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse a Stream into a relocation table
|
|
/// </summary>
|
|
/// <param name="data">Stream to parse</param>
|
|
/// <param name="count">Number of relocation table entries to read</param>
|
|
/// <returns>Filled relocation table on success, null on error</returns>
|
|
private static RelocationEntry[] ParseRelocationTable(Stream data, int count)
|
|
{
|
|
// TODO: Use marshalling here instead of building
|
|
var relocationTable = new RelocationEntry[count];
|
|
|
|
for (int i = 0; i < count; i++)
|
|
{
|
|
var entry = new RelocationEntry();
|
|
entry.Offset = data.ReadUInt16();
|
|
entry.Segment = data.ReadUInt16();
|
|
relocationTable[i] = entry;
|
|
}
|
|
|
|
return relocationTable;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
} |