Files
BinaryObjectScanner/BinaryObjectScanner/Protection/NEACProtect.cs
2023-11-22 13:28:13 -05:00

97 lines
4.1 KiB
C#

#if NET40_OR_GREATER || NETCOREAPP
using System.Collections.Concurrent;
#endif
using System.Collections.Generic;
using BinaryObjectScanner.Interfaces;
using SabreTools.Matching;
using SabreTools.Serialization.Wrappers;
namespace BinaryObjectScanner.Protection
{
/// <summary>
/// NEAC Protect is an anti-cheat currently only known to be used in the game "NARAKA: BLADEPOINT".
/// It requires HVCI to be disabled before play.
/// As the game's publisher is NetEase Games, the name of the protection may be short for "NetEase AntiCheat".
/// This is further supported by the fact that the "NeacSafe.sys" driver is signed by NetEase.
/// Another program by NetEase, MuMu Player, seems to have also included a "NetEase AC" at one point(https://www.pcgamingwiki.com/wiki/MuMu_Player).
/// There is also a separate DRM service provided by NetEase called Yidun (http://dun.163.com + http://dun.163.com/locale/en), though this seems to be unrelated.
///
/// Additional resources:
/// https://www.pcgamingwiki.com/wiki/Naraka:_Bladepoint
/// https://github.com/SteamDatabase/FileDetectionRuleSets/pull/235
/// https://www.protondb.com/app/1203220
/// </summary>
public class NEACProtect : IPathCheck, IPortableExecutableCheck
{
/// <inheritdoc/>
public string? CheckPortableExecutable(string file, PortableExecutable pex, bool includeDebug)
{
// Most of the relevant executables are highly obfuscated, making executable detection mostly impractical.
// Get the sections from the executable, if possible
var sections = pex.Model.SectionTable;
if (sections == null)
return null;
// Get the .neac0 and .neac1 sections, if they exist.
// Found in "NeacSafe64.sys" and "NeacSafe.sys".
bool neac0Section = pex.ContainsSection(".neac0", exact: true);
bool neac1Section = pex.ContainsSection(".neac1", exact: true);
// Check if either .neac section is present.
if (neac0Section || neac1Section)
return "NEAC Protect";
var name = pex.ProductName;
// Found in "NeacSafe64.sys" and "NeacSafe.sys".
// TODO: Fix Product Name not being properly grabbed from the file.
if (!string.IsNullOrEmpty(name) && name!.Contains("neacsafe"))
return "NEAC Protect";
return null;
}
/// <inheritdoc/>
#if NET20 || NET35
public Queue<string> CheckDirectoryPath(string path, IEnumerable<string>? files)
#else
public ConcurrentQueue<string> CheckDirectoryPath(string path, IEnumerable<string>? files)
#endif
{
var matchers = new List<PathMatchSet>
{
// Found installed in the main game folder.
new(new FilePathMatch("NeacClient.exe"), "NEAC Protect"),
new(new FilePathMatch("NeacInterface.dll"), "NEAC Protect"),
new(new FilePathMatch("NeacSafe64.sys"), "NEAC Protect"),
// Found installed in "System32\drivers".
new(new FilePathMatch("NeacSafe.sys"), "NEAC Protect"),
// Known associated log files: "NeacSafe.log", "Neac.log", "NeacDll.log", "NeacLoader.log", and "NeacBak.log".
};
return MatchUtil.GetAllMatches(files, matchers, any: true);
}
/// <inheritdoc/>
public string? CheckFilePath(string path)
{
var matchers = new List<PathMatchSet>
{
// Found installed in the main game folder.
new(new FilePathMatch("NeacClient.exe"), "NEAC Protect"),
new(new FilePathMatch("NeacInterface.dll"), "NEAC Protect"),
new(new FilePathMatch("NeacSafe64.sys"), "NEAC Protect"),
// Found installed in "System32\drivers".
new(new FilePathMatch("NeacSafe.sys"), "NEAC Protect"),
// Known associated log files: "NeacSafe.log", "Neac.log", "NeacDll.log", "NeacLoader.log", and "NeacBak.log".
};
return MatchUtil.GetFirstMatch(path, matchers, any: true);
}
}
}