/**
* Steamless - Copyright (c) 2015 - 2022 atom0s [atom0s@live.com]
*
* This work is licensed under the Creative Commons Attribution-NonCommercial-NoDerivatives 4.0 International License.
* To view a copy of this license, visit http://creativecommons.org/licenses/by-nc-nd/4.0/ or send a letter to
* Creative Commons, PO Box 1866, Mountain View, CA 94042, USA.
*
* By using Steamless, you agree to the above license and its terms.
*
* Attribution - You must give appropriate credit, provide a link to the license and indicate if changes were
* made. You must do so in any reasonable manner, but not in any way that suggests the licensor
* endorses you or your use.
*
* Non-Commercial - You may not use the material (Steamless) for commercial purposes.
*
* No-Derivatives - If you remix, transform, or build upon the material (Steamless), you may not distribute the
* modified material. You are, however, allowed to submit the modified works back to the original
* Steamless project in attempt to have it added to the original project.
*
* You may not apply legal terms or technological measures that legally restrict others
* from doing anything the license permits.
*
* No warranties are given.
*/
namespace Steamless.API.PE32
{
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
///
/// Portable Executable (32bit) Class
///
public class Pe32File
{
///
/// Default Constructor
///
public Pe32File()
{
}
///
/// Overloaded Constructor
///
///
public Pe32File(string file)
{
this.FilePath = file;
}
///
/// Parses a Win32 PE file.
///
///
///
public bool Parse(string file = null)
{
// Prepare the class variables..
if (file != null)
this.FilePath = file;
this.FileData = null;
this.DosHeader = new NativeApi32.ImageDosHeader32();
this.NtHeaders = new NativeApi32.ImageNtHeaders32();
this.DosStubSize = 0;
this.DosStubOffset = 0;
this.DosStubData = null;
this.Sections = new List();
this.SectionData = new List();
this.TlsDirectory = new NativeApi32.ImageTlsDirectory32();
this.TlsCallbacks = new List();
// Ensure a file path has been set..
if (string.IsNullOrEmpty(this.FilePath) || !File.Exists(this.FilePath))
return false;
// Read the file data..
this.FileData = File.ReadAllBytes(this.FilePath);
// Ensure we have valid data by the overall length..
if (this.FileData.Length < (Marshal.SizeOf(typeof(NativeApi32.ImageDosHeader32)) + Marshal.SizeOf(typeof(NativeApi32.ImageNtHeaders32))))
return false;
// Read the file headers..
this.DosHeader = Pe32Helpers.GetStructure(this.FileData);
this.NtHeaders = Pe32Helpers.GetStructure(this.FileData, this.DosHeader.e_lfanew);
// Validate the headers..
if (!this.DosHeader.IsValid || !this.NtHeaders.IsValid)
return false;
// Read and store the dos header if it exists..
this.DosStubSize = (uint)(this.DosHeader.e_lfanew - Marshal.SizeOf(typeof(NativeApi32.ImageDosHeader32)));
if (this.DosStubSize > 0)
{
this.DosStubOffset = (uint)Marshal.SizeOf(typeof(NativeApi32.ImageDosHeader32));
this.DosStubData = new byte[this.DosStubSize];
Array.Copy(this.FileData, this.DosStubOffset, this.DosStubData, 0, this.DosStubSize);
}
// Read the file sections..
for (var x = 0; x < this.NtHeaders.FileHeader.NumberOfSections; x++)
{
var section = Pe32Helpers.GetSection(this.FileData, x, this.DosHeader, this.NtHeaders);
this.Sections.Add(section);
// Get the sections data..
var sectionData = new byte[this.GetAlignment(section.SizeOfRawData, this.NtHeaders.OptionalHeader.FileAlignment)];
Array.Copy(this.FileData, section.PointerToRawData, sectionData, 0, section.SizeOfRawData);
this.SectionData.Add(sectionData);
}
try
{
// Obtain the file overlay if one exists..
var lastSection = this.Sections.Last();
var fileSize = lastSection.SizeOfRawData + lastSection.PointerToRawData;
if (fileSize < this.FileData.Length)
{
this.OverlayData = new byte[this.FileData.Length - fileSize];
Array.Copy(this.FileData, fileSize, this.OverlayData, 0, this.FileData.Length - fileSize);
}
}
catch
{
return false;
}
// Read the files Tls information if available..
if (this.NtHeaders.OptionalHeader.TLSTable.VirtualAddress != 0)
{
// Get the file offset to the Tls data..
var tls = this.NtHeaders.OptionalHeader.TLSTable;
var addr = this.GetFileOffsetFromRva(tls.VirtualAddress);
// Read the Tls directory..
this.TlsDirectory = Pe32Helpers.GetStructure(this.FileData, (int)addr);
// Read the Tls callbacks..
addr = this.GetRvaFromVa(this.TlsDirectory.AddressOfCallBacks);
addr = this.GetFileOffsetFromRva(addr);
// Loop until we hit a null pointer..
var count = 0;
while (true)
{
var callback = BitConverter.ToUInt32(this.FileData, (int)addr + (count * 4));
if (callback == 0)
break;
this.TlsCallbacks.Add(callback);
count++;
}
}
return true;
}
///
/// Determines if the current file is 64bit.
///
///
public bool IsFile64Bit()
{
return (this.NtHeaders.FileHeader.Machine & (uint)NativeApi32.MachineType.X64) == (uint)NativeApi32.MachineType.X64;
}
///
/// Determines if the file has a section containing the given name.
///
///
///
public bool HasSection(string name)
{
return this.Sections.Any(s => string.Compare(s.SectionName, name, StringComparison.InvariantCultureIgnoreCase) == 0);
}
///
/// Obtains a section by its name.
///
///
///
public NativeApi32.ImageSectionHeader32 GetSection(string name)
{
return this.Sections.FirstOrDefault(s => string.Compare(s.SectionName, name, StringComparison.InvariantCultureIgnoreCase) == 0);
}
///
/// Obtains the owner section of the given rva.
///
///
///
public NativeApi32.ImageSectionHeader32 GetOwnerSection(uint rva)
{
foreach (var s in this.Sections)
{
var size = s.VirtualSize;
if (size == 0)
size = s.SizeOfRawData;
if ((rva >= s.VirtualAddress) && (rva < s.VirtualAddress + size))
return s;
}
return default(NativeApi32.ImageSectionHeader32);
}
///
/// Obtains the owner section of the given rva.
///
///
///
public NativeApi32.ImageSectionHeader32 GetOwnerSection(ulong rva)
{
foreach (var s in this.Sections)
{
var size = s.VirtualSize;
if (size == 0)
size = s.SizeOfRawData;
if ((rva >= s.VirtualAddress) && (rva < s.VirtualAddress + size))
return s;
}
return default(NativeApi32.ImageSectionHeader32);
}
///
/// Obtains a sections data by its index.
///
///
///
public byte[] GetSectionData(int index)
{
if (index < 0 || index > this.Sections.Count)
return null;
return this.SectionData[index];
}
///
/// Obtains a sections data by its name.
///
///
///
public byte[] GetSectionData(string name)
{
for (var x = 0; x < this.Sections.Count; x++)
{
if (string.Compare(this.Sections[x].SectionName, name, StringComparison.InvariantCultureIgnoreCase) == 0)
return this.SectionData[x];
}
return null;
}
///
/// Gets a sections index by its name.
///
///
///
public int GetSectionIndex(string name)
{
for (var x = 0; x < this.Sections.Count; x++)
{
if (string.Compare(this.Sections[x].SectionName, name, StringComparison.InvariantCultureIgnoreCase) == 0)
return x;
}
return -1;
}
///
/// Gets a sections index by its name.
///
///
///
public int GetSectionIndex(NativeApi32.ImageSectionHeader32 section)
{
return this.Sections.IndexOf(section);
}
///
/// Removes a section from the files section list.
///
///
///
public bool RemoveSection(NativeApi32.ImageSectionHeader32 section)
{
var index = this.Sections.IndexOf(section);
if (index == -1)
return false;
this.Sections.RemoveAt(index);
this.SectionData.RemoveAt(index);
return true;
}
///
/// Rebuilds the sections by aligning them as needed. Updates the Nt headers to
/// correct the new SizeOfImage after alignment is completed.
///
public void RebuildSections()
{
for (var x = 0; x < this.Sections.Count; x++)
{
// Obtain the current section and realign the data..
var section = this.Sections[x];
section.VirtualAddress = this.GetAlignment(section.VirtualAddress, this.NtHeaders.OptionalHeader.SectionAlignment);
section.VirtualSize = this.GetAlignment(section.VirtualSize, this.NtHeaders.OptionalHeader.SectionAlignment);
section.PointerToRawData = this.GetAlignment(section.PointerToRawData, this.NtHeaders.OptionalHeader.FileAlignment);
section.SizeOfRawData = this.GetAlignment(section.SizeOfRawData, this.NtHeaders.OptionalHeader.FileAlignment);
// Store the sections updates..
this.Sections[x] = section;
}
// Update the size of the image..
var ntHeaders = this.NtHeaders;
ntHeaders.OptionalHeader.SizeOfImage = this.Sections.Last().VirtualAddress + this.Sections.Last().VirtualSize;
this.NtHeaders = ntHeaders;
}
///
/// Obtains the relative virtual address from the given virtual address.
///
///
///
public uint GetRvaFromVa(uint va)
{
return va - this.NtHeaders.OptionalHeader.ImageBase;
}
///
/// Obtains the file offset from the given relative virtual address.
///
///
///
public uint GetFileOffsetFromRva(uint rva)
{
var section = this.GetOwnerSection(rva);
return (rva - (section.VirtualAddress - section.PointerToRawData));
}
///
/// Aligns the value based on the given alignment.
///
///
///
///
public uint GetAlignment(uint val, uint align)
{
return (((val + align - 1) / align) * align);
}
///
/// Gets or sets the path to the file being processed.
///
public string FilePath { get; set; }
///
/// Gets or sets the raw file data read from disk.
///
public byte[] FileData { get; set; }
///
/// Gets or sets the dos header of the file.
///
public NativeApi32.ImageDosHeader32 DosHeader { get; set; }
///
/// Gets or sets the NT headers of the file.
///
public NativeApi32.ImageNtHeaders32 NtHeaders { get; set; }
///
/// Gets or sets the optional dos stub size.
///
public uint DosStubSize { get; set; }
///
/// Gets or sets the optional dos stub offset.
///
public uint DosStubOffset { get; set; }
///
/// Gets or sets the optional dos stub data.
///
public byte[] DosStubData { get; set; }
///
/// Gets or sets the sections of the file.
///
public List Sections;
///
/// Gets or sets the section data of the file.
///
public List SectionData;
///
/// Gets or sets the overlay data of the file.
///
public byte[] OverlayData;
///
/// Gets or sets the Tls directory of the file.
///
public NativeApi32.ImageTlsDirectory32 TlsDirectory { get; set; }
///
/// Gets or sets a list of Tls callbacks of the file.
///
public List TlsCallbacks { get; set; }
}
}