using System; using System.IO; using System.Linq; using BinaryObjectScanner.Interfaces; using BinaryObjectScanner.Matching; using BinaryObjectScanner.Wrappers; namespace BinaryObjectScanner.Packer { /// /// Though not technically a packer, this detection is for any executables that include /// others in their resources in some uncompressed manner to be used at runtime. /// public class EmbeddedExecutable : IExtractable, IPortableExecutableCheck { /// public string CheckPortableExecutable(string file, PortableExecutable pex, bool includeDebug) { // Get the sections from the executable, if possible var sections = pex?.SectionTable; if (sections == null) return null; // Get the resources that have an executable signature if (pex.ResourceData?.Any(kvp => kvp.Value is byte[] ba && ba.StartsWith(BinaryObjectScanner.Models.MSDOS.Constants.SignatureBytes)) == true) return "Embedded Executable"; return null; } /// public string Extract(string file, bool includeDebug) { if (!File.Exists(file)) return null; using (var fs = File.Open(file, FileMode.Open, FileAccess.Read, FileShare.Read)) { return Extract(fs, file, includeDebug); } } /// public string Extract(Stream stream, string file, bool includeDebug) { try { // Parse into an executable again for easier extraction PortableExecutable pex = PortableExecutable.Create(stream); if (pex?.ResourceData == null) return null; // Get the resources that have an executable signature var resources = pex.ResourceData .Where(kvp => kvp.Value != null && kvp.Value is byte[]) .Where(kvp => (kvp.Value as byte[]).StartsWith(BinaryObjectScanner.Models.MSDOS.Constants.SignatureBytes)) .ToList(); string tempPath = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); Directory.CreateDirectory(tempPath); for (int i = 0; i < resources.Count; i++) { try { // Get the resource data var resource = resources[i]; byte[] data = resource.Value as byte[]; // Create the temp filename string tempFile = $"embedded_resource_{i}.bin"; tempFile = Path.Combine(tempPath, tempFile); // Write the resource data to a temp file using (Stream tempStream = File.Open(tempFile, FileMode.Create, FileAccess.Write, FileShare.ReadWrite)) { tempStream.Write(data, 0, data.Length); } } catch (Exception ex) { if (includeDebug) Console.WriteLine(ex); } } return tempPath; } catch (Exception ex) { if (includeDebug) Console.WriteLine(ex); return null; } } } }