gpt4 book ai didi

.net - 如何使用 SOS 和 Windbg 列出内存转储中的枚举值?

转载 作者:行者123 更新时间:2023-12-01 23:10:54 27 4
gpt4 key购买 nike

我有一个小型转储,我已将 Windbg 附加到其中。小型转储来自在 IIS 上运行的 .NET 4.6.1 ASP.NET 站点。我想获取枚举的定义,但每当我获取类的 MethodTable 时,我只会得到以下内容。

0:000> !DumpMT /d 256db60c
EEClass: 256c773c
Module: 201fcfb0
Name: MyDll.eDefaultRelatedObjects
mdToken: 02000029
File: C:\Windows\Microsoft.NET\Framework\v4.0.30319\Temporary ASP.NET Files\root\303e164d\216bec4f\assembly\dl3\bdb2a421\004bd941_fee1d501\MyDll.dll
BaseSize: 0xc
ComponentSize: 0x0
Slots in VTable: 23
Number of IFaces in IFaceMap: 3


0:000> !DumpClass /d 256c773c
Class Name: MyDll.eDefaultRelatedObjects
mdToken: 02000029
File: C:\Windows\Microsoft.NET\Framework\v4.0.30319\Temporary ASP.NET Files\root\303e164d\216bec4f\assembly\dl3\bdb2a421\004bd941_fee1d501\MyDll.dll
Parent Class: 717f17cc
Module: 201fcfb0
Method Table: 256db60c
Vtable Slots: 17
Total Method Slots: 17
Class Attributes: 101
Transparency: Critical
NumInstanceFields: 1
NumStaticFields: 0
MT Field Offset Type VT Attr Value Name
71c9f54c 400055f 124 System.Char[] 0 shared static enumSeperatorCharArray
>> Domain:Value 09671520:NotInit 306d85b0:0d8e822c 306da248:NotInit 306d98c0:NotInit 306dc868:138690b0 <<
71ca0994 400007d 4 System.Int32 1 instance value__

我查看了应用程序域中的值,但它们只是逗号。

0:000> !DumpObj /d 0d8e822c
Name: System.Char[]
MethodTable: 71c9f54c
EEClass: 71874c84
Size: 14(0xe) bytes
Array: Rank 1, Number of elements 1, Type Char (Print Array)
Content: ,
Fields:
None

我需要做什么才能从堆上的对象中获取枚举的定义方式?

编辑1:如果这有影响的话,我可以访问 PDB。

最佳答案

先简单的事情:

0:000> !DumpObj /d 0d8e822c
[...]
Content: ,

您在这里所做的是:列出 enumSeperatorCharArray 的值。它与您的枚举定义无关。所有枚举都有它。

SOS IMHO 没有办法列出枚举定义。您需要为了那个原因。

这是调试 session :

0:006> .loadby sos clr
0:006> .load D:\mylongpath\sosex.dll

0:006> !dumpheap -type YourEnum
Address MT Size
02cf2480 01154dc4 12

Statistics:
MT Count TotalSize Class Name
01154dc4 1 12 DebuggingEnumDefinition.YourEnum
Total 1 objects

所以只有一个对象,并且可以像您一样查看它。在输出的最后,您可以看到它的十进制值,即 65,并没有多大帮助。

0:006> !DumpObj /d 02cf2480
Name: DebuggingEnumDefinition.YourEnum
[...]
61bf42a8 4000001 4 System.Int32 1 instance 65 value__

使用 SOSEX 的 !mdt,您可以列出枚举常量:

0:006> !mdt DebuggingEnumDefinition.YourEnum
DebuggingEnumDefinition.YourEnum
[s]enumSeperatorCharArray: char[]
AppDomain 'DebuggingEnumDefinition.exe' (00c8dc18): <uninitialized>
[s]enumSeperator: string
AppDomain 'DebuggingEnumDefinition.exe' (00c8dc18): <Field def not loaded>
value__: int
[s]EnumVal1: DebuggingEnumDefinition.YourEnum
AppDomain 'DebuggingEnumDefinition.exe' (00c8dc18): <Field def not loaded>
[s]EnumVal2: DebuggingEnumDefinition.YourEnum
AppDomain 'DebuggingEnumDefinition.exe' (00c8dc18): <Field def not loaded>
[s]EnumVal3: DebuggingEnumDefinition.YourEnum
AppDomain 'DebuggingEnumDefinition.exe' (00c8dc18): <Field def not loaded>

其实我也预料到了数值。

您还可以将 !mdt 与对象的地址一起使用,以便获得其常量和十六进制值 (0x41 == 65):

0:006> !mdt 02cf2480
0x41 (EnumVal3) (DebuggingEnumDefinition.YourEnum)
<小时/>

通过更改进程内存,您可能可以生成枚举值的映射。

以下生成从 0 到 127 的循环:

.for (r $t0=0; @$t0<0n128; r $t0 = @$t0+1) { }

在循环内,您可以更改枚举的值(我忘记了 +4 是否与位数相关):

ed 03402470+4 @$t0

接下来可以使用!mdt来分析修改后的值

!mdt 03402470

完整的命令是

.for (r $t0=0; @$t0<0n128; r $t0 = @$t0+1) { ed 03402470+4 @$t0; !mdt 03402470}

输出将如下所示

0x0 (EnumVal1) (DebuggingEnumDefinition.YourEnum)
0x1 (EnumVal2) (DebuggingEnumDefinition.YourEnum)
0x2 (DebuggingEnumDefinition.YourEnum)
0x3 (DebuggingEnumDefinition.YourEnum)
[...]
0x7f (DebuggingEnumDefinition.YourEnum)

接下来,您可能只过滤有效的案例。请注意,我们可以通过包含 ) (.

来区分有效条目和无效条目。

这就是 WinDbg 中的脚本变得有点难看的地方......

.echo Just a test

显示一些东西来说明原理。

.shell -ci ".echo Just a test" findstr "Just"

使用DOS命令findstr来过滤包含特定单词的行。

接下来,将 .echo 替换为之前的完整命令,并将“Just”替换为 ) (。因为 findstr 也是一个奇怪的程序,您实际上需要 ).( 因为否则它会将其视为两个单独的搜索词。

.shell -ci ".for (r $t0=0; @$t0<0n128; r $t0 = @$t0+1) { ed 03402470+4 @$t0; !mdt 03402470}" findstr ").("

yippieh,输出是:

0x0 (EnumVal1) (DebuggingEnumDefinition.YourEnum)
0x1 (EnumVal2) (DebuggingEnumDefinition.YourEnum)
0x41 (EnumVal3) (DebuggingEnumDefinition.YourEnum)
.shell: Process exited

真是一次冒险!

<小时/>

我使用的源代码,以防万一......

using System;
using System.Collections;

namespace DebuggingEnumDefinition
{
class Program
{
static void Main()
{
var somwehere = new ArrayList() { YourEnum.EnumVal3 };
Console.WriteLine("There should be an enum on the heap now.");
Console.ReadLine();
Console.WriteLine(somwehere[0]); // Just fix the unused variable issue
}
}

enum YourEnum
{
EnumVal1,
EnumVal2,
EnumVal3=65
}
}

关于.net - 如何使用 SOS 和 Windbg 列出内存转储中的枚举值?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60212867/

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