using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.IO; using System.Linq; using BinaryObjectScanner.Interfaces; using BinaryObjectScanner.Matching; using BinaryObjectScanner.Wrappers; namespace BinaryObjectScanner.Protection { public class LaserLok : IPathCheck, IPortableExecutableCheck { /// public string CheckPortableExecutable(string file, PortableExecutable pex, bool includeDebug) { // TODO: Add entry point check // https://github.com/horsicq/Detect-It-Easy/blob/master/db/PE/Laserlok.2.sg // TODO: Additional checks that may or may not be useful with the below // // These two can appear separately // \LASERLOK\LASERLOK.IN // C:\NOMOUSE.SP // // // :\LASERLOK\LASERLOK.IN + (char)0x00 + C:\NOMOUSE.SP // new ContentMatchSet(new byte?[] // { // 0x3A, 0x5C, 0x5C, 0x4C, 0x41, 0x53, 0x45, 0x52, // 0x4C, 0x4F, 0x4B, 0x5C, 0x5C, 0x4C, 0x41, 0x53, // 0x45, 0x52, 0x4C, 0x4F, 0x4B, 0x2E, 0x49, 0x4E, // 0x00, 0x43, 0x3A, 0x5C, 0x5C, 0x4E, 0x4F, 0x4D, // 0x4F, 0x55, 0x53, 0x45, 0x2E, 0x53, 0x50 // }, "LaserLok 3"), // // LASERLOK_INIT + (char)0xC + LASERLOK_RUN + (char)0xE + LASERLOK_CHECK + (char)0xF + LASERLOK_CHECK2 + (char)0xF + LASERLOK_CHECK3 // new ContentMatchSet(new byte?[] // { // 0x4C, 0x41, 0x53, 0x45, 0x52, 0x4C, 0x4F, 0x4B, // 0x5F, 0x49, 0x4E, 0x49, 0x54, 0x0C, 0x4C, 0x41, // 0x53, 0x45, 0x52, 0x4C, 0x4F, 0x4B, 0x5F, 0x52, // 0x55, 0x4E, 0x0E, 0x4C, 0x41, 0x53, 0x45, 0x52, // 0x4C, 0x4F, 0x4B, 0x5F, 0x43, 0x48, 0x45, 0x43, // 0x4B, 0x0F, 0x4C, 0x41, 0x53, 0x45, 0x52, 0x4C, // 0x4F, 0x4B, 0x5F, 0x43, 0x48, 0x45, 0x43, 0x4B, // 0x32, 0x0F, 0x4C, 0x41, 0x53, 0x45, 0x52, 0x4C, // 0x4F, 0x4B, 0x5F, 0x43, 0x48, 0x45, 0x43, 0x4B, // 0x33 // }, "LaserLok 5"), // Get the sections from the executable, if possible var sections = pex?.SectionTable; if (sections == null) return null; // Packed by SPEEnc V2 Asterios Parlamentas.PE byte?[] check = new byte?[] { 0x50, 0x61, 0x63, 0x6B, 0x65, 0x64, 0x20, 0x62, 0x79, 0x20, 0x53, 0x50, 0x45, 0x45, 0x6E, 0x63, 0x20, 0x56, 0x32, 0x20, 0x41, 0x73, 0x74, 0x65, 0x72, 0x69, 0x6F, 0x73, 0x20, 0x50, 0x61, 0x72, 0x6C, 0x61, 0x6D, 0x65, 0x6E, 0x74, 0x61, 0x73, 0x2E, 0x50, 0x45 }; int endDosStub = (int)pex.Stub_NewExeHeaderAddr; bool containsCheck = pex.StubExecutableData.FirstPosition(check, out int position); // Check the executable tables bool containsCheck2 = (pex.ImportTable?.HintNameTable.Any(hnte => hnte.Name == "GetModuleHandleA") ?? false) && (pex.ImportTable?.HintNameTable.Any(hnte => hnte.Name == "GetProcAddress") ?? false) && (pex.ImportTable?.HintNameTable.Any(hnte => hnte.Name == "LoadLibraryA") ?? false) && (pex.ImportTable?.ImportDirectoryTable.Any(idte => idte.Name == "KERNEL32.dll") ?? false); int position2 = -1; // Get the .text section, if it exists if (containsCheck2 && pex.ContainsSection(".text")) { // GetModuleHandleA + (char)0x00 + (char)0x00 + (char)0x00 + (char)0x00 + GetProcAddress + (char)0x00 + (char)0x00 + (char)0x00 + (char)0x00 + LoadLibraryA + (char)0x00 + (char)0x00 + KERNEL32.dll + (char)0x00 + ëy + (char)0x01 + SNIF/MPVI byte?[] check2 = new byte?[] { 0x47, 0x65, 0x74, 0x4D, 0x6F, 0x64, 0x75, 0x6C, 0x65, 0x48, 0x61, 0x6E, 0x64, 0x6C, 0x65, 0x41, 0x00, 0x00, 0x00, 0x00, 0x47, 0x65, 0x74, 0x50, 0x72, 0x6F, 0x63, 0x41, 0x64, 0x64, 0x72, 0x65, 0x73, 0x73, 0x00, 0x00, 0x00, 0x00, 0x4C, 0x6F, 0x61, 0x64, 0x4C, 0x69, 0x62, 0x72, 0x61, 0x72, 0x79, 0x41, 0x00, 0x00, 0x4B, 0x45, 0x52, 0x4E, 0x45, 0x4C, 0x33, 0x32, 0x2E, 0x64, 0x6C, 0x6C, 0x00, 0xEB, 0x79, 0x01, null, null, null, null, }; containsCheck2 = pex.GetFirstSectionData(".text").FirstPosition(check2, out position2); } else { // Otherwise, we reset the containsCheck2 value containsCheck2 = false; } if (containsCheck && containsCheck2) return $"LaserLok {GetVersion(pex.GetFirstSectionData(".text"), position2)} {GetBuild(pex.GetFirstSectionData(".text"), true)} [Check disc for physical ring]" + (includeDebug ? $" (Index {position}, {position2})" : string.Empty); else if (containsCheck && !containsCheck2) return $"LaserLok Marathon {GetBuild(pex.GetFirstSectionData(".text"), false)} [Check disc for physical ring]" + (includeDebug ? $" (Index {position})" : string.Empty); else if (!containsCheck && containsCheck2) return $"LaserLok {GetVersion(pex.GetFirstSectionData(".text"), --position2)} {GetBuild(pex.GetFirstSectionData(".text"), false)} [Check disc for physical ring]" + (includeDebug ? $" (Index {position2})" : string.Empty); return null; } /// public ConcurrentQueue CheckDirectoryPath(string path, IEnumerable files) { var matchers = new List { new PathMatchSet($"LASERLOK{Path.DirectorySeparatorChar}", "LaserLok [Check disc for physical ring]"), // TODO: Verify if these are OR or AND new PathMatchSet(new PathMatch($"{Path.DirectorySeparatorChar}NOMOUSE.SP", useEndsWith: true), GetVersion16Bit, "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch($"{Path.DirectorySeparatorChar}NOMOUSE.COM", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("l16dll.dll", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.in", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.o10", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.o11", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.o12", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.out", useEndsWith: true), "LaserLok [Check disc for physical ring]"), }; return MatchUtil.GetAllMatches(files, matchers, any: true); } /// public string CheckFilePath(string path) { var matchers = new List { new PathMatchSet(new PathMatch($"{Path.DirectorySeparatorChar}NOMOUSE.SP", useEndsWith: true), GetVersion16Bit, "LaserLok [Check disc for physical ring]"), // TODO: Verify if these are OR or AND new PathMatchSet(new PathMatch($"{Path.DirectorySeparatorChar}NOMOUSE.COM", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("l16dll.dll", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.in", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.o10", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.o11", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.o12", useEndsWith: true), "LaserLok [Check disc for physical ring]"), new PathMatchSet(new PathMatch("laserlok.out", useEndsWith: true), "LaserLok [Check disc for physical ring]"), }; return MatchUtil.GetFirstMatch(path, matchers, any: true); } private static string GetBuild(byte[] sectionContent, bool versionTwo) { if (sectionContent == null) return "(Build unknown)"; // Unkown + (char)0x00 + Unkown byte?[] check = new byte?[] { 0x55, 0x6E, 0x6B, 0x6F, 0x77, 0x6E, 0x00, 0x55, 0x6E, 0x6B, 0x6F, 0x77, 0x6E }; if (!sectionContent.FirstPosition(check, out int position)) return "(Build unknown)"; string year, month, day; if (versionTwo) { int index = position + 14; day = new string(new ArraySegment(sectionContent, index, 2).Select(b => (char)b).ToArray()); index += 3; month = new string(new ArraySegment(sectionContent, index, 2).Select(b => (char)b).ToArray()); index += 3; year = "20" + new string(new ArraySegment(sectionContent, index, 2).Select(b => (char)b).ToArray()); } else { int index = position + 13; day = new string(new ArraySegment(sectionContent, index, 2).Select(b => (char)b).ToArray()); index += 3; month = new string(new ArraySegment(sectionContent, index, 2).Select(b => (char)b).ToArray()); index += 3; year = "20" + new string(new ArraySegment(sectionContent, index, 2).Select(b => (char)b).ToArray()); } return $"(Build {year}-{month}-{day})"; } private static string GetVersion(byte[] sectionContent, int position) { // If we have invalid data if (sectionContent == null) return null; return new string(new ArraySegment(sectionContent, position + 76, 4).Select(b => (char)b).ToArray()); } public static string GetVersion16Bit(string firstMatchedString, IEnumerable files) { if (!File.Exists(firstMatchedString)) return string.Empty; using (var fs = File.Open(firstMatchedString, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) using (var br = new BinaryReader(fs)) { return GetVersion16Bit(br.ReadBytes((int)fs.Length)); } } private static string GetVersion16Bit(byte[] fileContent) { char[] version = new ArraySegment(fileContent, 71, 7).Select(b => (char)b).ToArray(); if (char.IsNumber(version[0]) && char.IsNumber(version[2]) && char.IsNumber(version[3])) { if (char.IsNumber(version[5]) && char.IsNumber(version[6])) return $"{version[0]}.{version[2]}{version[3]}.{version[5]}{version[6]}"; else if (char.IsNumber(version[5])) return $"{version[0]}.{version[2]}{version[3]}.{version[5]}"; return $"{version[0]}.{version[2]}{version[3]}"; } return string.Empty; } } }