gpt4 book ai didi

c# - 从调试进程 MainModule 获取符号

转载 作者:行者123 更新时间:2023-12-02 03:29:45 26 4
gpt4 key购买 nike

我开始用 C# 编写一个调试器,以调试操作系统上的任何进程。目前,它只能处理断点(HW、SW 和内存),但现在我想显示该进程的操作码。

我的第一次尝试是使用nidsasm(NASM),但这并不合适,因为启动后a.Net应用程序汇编程序指令与ndisasm不同(使用CheatEngine测试)。

所以我搜索了一段时间,从 dbghelp.dll 中找到了一些方法,可以调用这些方法来列出所有加载的模块和符号(加上基地址)。好吧,我的尝试是,用 SharpDisasm 单独反汇编所有模块。

我使用ProcessModuleCollection module = ProcessData.Instance.MPMR.ReadProcess.Modules;来获取进程的所有加载模块。这非常有效。

现在我尝试加载 MainModule 的符号,但此时我坚持执行。我使用 p/Invoke 和其他必要的函数(如 SymInitialize)实现了 SymEnumSymbols 函数。

当我使用 BaseAddress(例如“User32.dll”)调用它时,所有符号都会完美打印,但对于 MainModule,我没有得到任何符号。

这是 CheatEngine 的屏幕截图: Symbols gained from Cheat Engine

如您所见,有一些像“Form1_Load”这样的符号,我的实现中没有得到这些符号。

这是必要的代码示例:

if (!DebugApi.SymInitialize(ProcessData.Instance.MPMR.M_hProcess, null, false))
{
var err = Marshal.GetLastWin32Error();

//throw new Exception("GetMemoryInfo failed : GetLastError() : " + new Win32Exception(err).Message);
Console.WriteLine("GetMemoryInfo failed : GetLastError() : " + new Win32Exception(err).Message);
return;
}

if (!DebugApi.SymEnumSymbols(ProcessData.Instance.MPMR.M_hProcess, (ulong)ProcessData.Instance.MPMR.ReadProcess.MainModule.BaseAddress, "!", DebugApi.EnumSyms, IntPtr.Zero))
{
var err = Marshal.GetLastWin32Error();

//throw new Exception("GetMemoryInfo failed : GetLastError() : " + new Win32Exception(err).Message);
Console.WriteLine("GetMemoryInfo failed : GetLastError() : " + new Win32Exception(err).Message);
return;
}

DebugApi.SymCleanup(ProcessData.Instance.MPMR.M_hProcess);

还有我的 DebugApi,带有所有必要的 p/Invoke 函数。

public class DebugApi
{

[DllImport("dbghelp.dll", SetLastError = true, CharSet = CharSet.Unicode)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool SymInitialize(IntPtr hProcess, string UserSearchPath, [MarshalAs(UnmanagedType.Bool)]bool fInvadeProcess);

[DllImport("dbghelp.dll", SetLastError = true, CharSet = CharSet.Unicode)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool SymCleanup(IntPtr hProcess);

[DllImport("dbghelp.dll", SetLastError = true, CharSet = CharSet.Unicode)]
public static extern ulong SymLoadModuleEx(IntPtr hProcess, IntPtr hFile, string ImageName, string ModuleName, long BaseOfDll, int DllSize, IntPtr Data, int Flags);

[DllImport("dbghelp.dll", SetLastError = true, CharSet = CharSet.Unicode)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool SymEnumSymbols(IntPtr hProcess, ulong BaseOfDll, string Mask, PSYM_ENUMERATESYMBOLS_CALLBACK EnumSymbolsCallback, IntPtr UserContext);

public delegate bool PSYM_ENUMERATESYMBOLS_CALLBACK(ref SYMBOL_INFO pSymInfo, uint SymbolSize, IntPtr UserContext);

public static bool EnumSyms(ref SYMBOL_INFO pSymInfo, uint SymbolSize, IntPtr UserContext)
{
Console.Out.WriteLine("Name: " + pSymInfo.Name);
return true;
}

[Flags]
public enum SymFlag : uint
{
VALUEPRESENT = 0x00000001,
REGISTER = 0x00000008,
REGREL = 0x00000010,
FRAMEREL = 0x00000020,
PARAMETER = 0x00000040,
LOCAL = 0x00000080,
CONSTANT = 0x00000100,
EXPORT = 0x00000200,
FORWARDER = 0x00000400,
FUNCTION = 0x00000800,
VIRTUAL = 0x00001000,
THUNK = 0x00002000,
TLSREL = 0x00004000,
}

[Flags]
public enum SymTagEnum : uint
{
Null,
Exe,
Compiland,
CompilandDetails,
CompilandEnv,
Function,
Block,
Data,
Annotation,
Label,
PublicSymbol,
UDT,
Enum,
FunctionType,
PointerType,
ArrayType,
BaseType,
Typedef,
BaseClass,
Friend,
FunctionArgType,
FuncDebugStart,
FuncDebugEnd,
UsingNamespace,
VTableShape,
VTable,
Custom,
Thunk,
CustomType,
ManagedType,
Dimension
};

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
public struct SYMBOL_INFO
{
public uint SizeOfStruct;
public uint TypeIndex;
public ulong Reserved1;
public ulong Reserved2;
public uint Reserved3;
public uint Size;
public ulong ModBase;
public SymFlag Flags;
public ulong Value;
public ulong Address;
public uint Register;
public uint Scope;
public SymTagEnum Tag;
public int NameLen;
public int MaxNameLen;

[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 1024)]
public string Name;
}

}

我的函数应该没问题,因为它可以与其他模块一起使用(例如加载的 dll)。也许我不理解 .Net 可执行文件的符号概念或丢失了某些内容。

最佳答案

您是否正在寻找 System.Diagnostics.SymbolStore.ISymbolScope。看看类(class)SymbolAccess ,您可以使用它来访问返回 ISymbolVariable[] 的 ISymbolScope.GetLocals() 和 GetChildren(),这次再次返回一个名为 ISymbolVariable[] 的数组

现在,另一组有趣的引用代码示例是调试器扩展,可让您“嗅探”如图所示的值 here

关于c# - 从调试进程 MainModule 获取符号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33481591/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com