当遇到第一次机会OutOfMemoryException时尝试使用DebugDiag进行转储时遇到问题。所以我写了一个应用程序,我可以用来创建内存不足的情况,并遵循以下说明:
但我没有得到第一次机会转储,我只是得到第二次机会转储。当我从DebugDiag查看日志文件时,我得到了这个:
[6/16/2013 9:54:04 PM] First chance exception - 0xe06d7363 caused by thread with System ID: 4628
[6/16/2013 9:54:04 PM] First chance exception - 0xe0434352 caused by thread with System ID: 4628
[6/16/2013 9:54:05 PM] Unable to determine CLR exception type
ExceptionObjHexAddr = 0x00000000`00000000
bInnerException = False
DumpObject Output = Invalid parameter 0x00000000`00000000
ChildEBP RetAddr Args to Child
002bdee4 6a44c93f e0434352 00000001 00000005 KERNELBASE!RaiseException+0x58
002bdf88 6a573b17 00000000 20b578f4 002be04c clr!RaiseTheExceptionInternalOnly+0x276
002bdfb8 6a5e5589 20b54734 002be090 00000000 clr!UnwindAndContinueRethrowHelperAfterCatch+0x83
002be058 003c0a3a 00000000 00000000 0233d174 clr!JIT_NewArr1+0x1af
... removed some rows, lots of data ...
OS Thread Id: 0x1214 (0)
Child SP IP Call Site
002bdfd4 7554c41f [Frame: 002bdfd4]
002be060 003c0a3a
... removed some rows, lots of data ...
Error requesting GC Heap data
Unable to determine bounds of gc heap
然后我得到了这个:
[6/16/2013 9:54:05 PM] CLR Exception Type - ''
[6/16/2013 9:54:05 PM] First chance exception - 0xe0434352 caused by thread with System ID: 4628
[6/16/2013 9:54:05 PM] Unable to determine CLR exception type
然后我终于得到了
[6/16/2013 9:54:05 PM] CLR Exception Type - ''
[6/16/2013 9:54:05 PM] C:\Windows\Microsoft.NET\Framework\v4.0.30319\diasymreader.dll loaded at 0x615c0000
[6/16/2013 9:54:13 PM] Second chance exception - 0xe0434352 caused by thread with System ID: 4628
看起来它可能能够获取异常对象的地址,它是0,所以当脚本调用DumpObject时它无法找到异常信息。
我读取这些日志条目的方式是从malloc或其他东西获得原生的第一次机会异常,然后跟进OutOfMemoryException的CLR异常。我试图弄清楚第一次机会异常的第二个是什么,我的代码看起来像这样:
private void OnGrowMemoryCommand(int growMemorySize)
{
try
{
_heldMemoryChunks.Add(new byte[growMemorySize * 1024 * 1024]);
}
catch (Exception)
{
throw;
}
TotalMemorySize += growMemorySize;
}
此代码是从WPF按钮上的命令触发的。因此,源自此代码的任何异常都会导致TargetInvocationException,我认为这是第一次机会异常中的第二个。然后最后从throw块中得到第二次机会异常,它的类型为TargetInvocationException。
所以我开始看第二次机会转储文件。我将它加载到windbg中,然后发出以下命令:
.symfix C:\symcache
.loadby sos clr
.reload
!pe
Exception object: 023caf9c
Exception type: System.Reflection.TargetInvocationException
Message: Exception has been thrown by the target of an invocation.
InnerException: System.OutOfMemoryException, Use !PrintException 023c9928 to see more.
我可以看到上面的假设是由第二次机会异常是TargetInvocationException支持的,但为什么DebugDiag不能获得CLR异常类型?为了进行健全检查,我尝试了一个实时调试会话。所以我启动应用程序并附加,然后发出这些命令。
.symfix C:\symcache
.loadby sos clr
.reload
!threads
Failed to request ThreadStore
!dumpheap
The garbage collector data structures are not in a valid state for traversal.
It is either in the "plan phase," where objects are being moved around, or
we are at the initialization or shutdown of the gc heap. Commands related to
displaying, finding or traversing objects as well as gc heap segments may not
work properly. !dumpheap and !verifyheap may incorrectly complain of heap
consistency errors.
Error requesting GC Heap data
Unable to build snapshot of the garbage collector state
完全被冲洗了。所以我开始研究这个问题。
此网址表明它可能是多个CLR实例:
所以我发出这些命令:
.cordll
CLR DLL status: Loaded DLL C:\Windows\Microsoft.NET\Framework\v4.0.30319\mscordacwks.dll
这对我来说很奇怪,我认为从4.0开始,mscorwks被抛弃了支持clr。是mscordacwks 4.5吗?
我发出了这个命令:
lmvm mscordacwks
但是加载了clr:
lmvm clr
start end module name
6a350000 6a9e2000 clr (pdb symbols) C:\symcache\clr.pdb\97FD69E1786F42F9A541C81D81AC96852\clr.pdb
Loaded symbol image file: C:\Windows\Microsoft.NET\Framework\v4.0.30319\clr.dll
Image path: C:\Windows\Microsoft.NET\Framework\v4.0.30319\clr.dll
Image name: clr.dll
Timestamp: Fri Mar 29 00:13:44 2013 (51553118)
CheckSum: 0069496E
ImageSize: 00692000
File version: 4.0.30319.18047
Product version: 4.0.30319.18047
File flags: 8 (Mask 3F) Private
File OS: 4 Unknown Win32
File type: 2.0 Dll
File date: 00000000.00000000
Translations: 0409.04b0
CompanyName: Microsoft Corporation
ProductName: Microsoft® .NET Framework
InternalName: clr.dll
OriginalFilename: clr.dll
ProductVersion: 4.0.30319.18047
FileVersion: 4.0.30319.18047 built by: FX45RTMGDR
PrivateBuild: DDBLD316
FileDescription: Microsoft .NET Runtime Common Language Runtime - WorkStation
LegalCopyright: © Microsoft Corporation. All rights reserved.
Comments: Flavor=Retail
所以我认为我没有加载多个CLR。
所以我认为导致我的实时调试问题的同样事情导致我无法解决第一次机会问题。有什么想法吗?
答案 0 :(得分:2)
我在两台机器上遇到了同样的错误。两者都配备了.NET 4.5,所以我认为这是它不起作用的原因。
原来这是DebugDiag Scripts中的一个错误。它无法在其中一个脚本中检索CLR异常的名称。但是,此问题是可修复的,因为每个DebugDiag规则都会创建一个可以修改的vbs脚本。它使用相同的命令,也可以在WinDbg中使用。以下是如何解决它:
C:\Program Files\DebugDiag\Scripts\CrashRule_<rulename>.vbs
GetCLRExceptionType
按如下方式修改:
Function GetCLRExceptionType(ByVal ExceptionObjHexAddr, ByVal bInnerException)
Dim Output, Lines, i
If Debugger.IsClrExtensionMissing Then
WriteToLog "Unable to determine CLR exception type - extension dll could not be loaded."
Else
' Output = Debugger.Execute("!DumpObj " & ExceptionObjHexAddr) ' Does not work in .NET 4.5
Output = Debugger.Execute("!pe") ' FIX .NET45
Lines = Split(Output, Chr(10))
For i = 0 To UBound(Lines)
If bInnerException Then
If InStr(Lines(i), "_innerException") <> 0 Then
Tokens = Split(Lines(i), " ")
For j = 0 To UBound(Tokens)
If Len(Tokens(j)) = 8 Then
GetCLRExceptionType = GetCLRExceptionType(Tokens(j), False)
Exit For
End If
Next
End If
ElseIf Len(Lines(i)) >= 7 Then
If InStr(Lines(i), "Exception type:") = 1 Then ' FIX .NET45
GetCLRExceptionType = Trim(Mid(Lines(i), 16)) ' FIX .NET45
Exit For
End If
End If
Next
If GetCLRExceptionType = "" Then
If g_ClrExceptionTypeFailureLogCount < MAX_CLR_EXCEPTION_TYPE_FAILURE_LOG_ENTRIES Then
g_ClrExceptionTypeFailureLogCount = g_ClrExceptionTypeFailureLogCount + 1
WriteToLog "Unable to determine CLR exception type" & vbcrlf & _
"ExceptionObjHexAddr = " & ExceptionObjHexAddr & vbcrlf & _
"bInnerException = " & bInnerException & vbcrlf & _
"DumpObject Output = " & Output & vbcrlf & _
Debugger.Execute("kb100") & vbcrlf & _
Debugger.Execute("!clrstack") & vbcrlf & _
Debugger.Execute("!dso")
End If
End If
End If
End Function
默认情况下,!DumpObj
用于转储异常对象的内容。但是,在.NET 4.5中发生了一些变化,显然,无法再提取异常的类型。相反,在此处放置!pe
命令并从其结果中解析异常类型。
要使!pe
有效,您需要执行.loadby sos clr
。我将其添加到Sub Debugger_OnLoadModule
:
Sub Debugger_OnLoadModule(ByVal NewModule)
WriteToLog NewModule.ImageName & " loaded at " & Debugger.GetAs32BitHexString(NewModule.Base)
Select Case UCase(NewModule.ModuleName)
Case "MSCORWKS", "MSCORSVR", "CLR", "CORECLR"
UpdateDeferredManagedBreakpoints
End Select
Debugger.Execute(".loadby sos clr")
End Sub
进行更改后,您需要重新启动流程或停用/重新激活规则以应用更改。
另外:请注意,如果您在DebugDiag GUI中修改规则,您的更改将被完全覆盖。