使用 !dlk 命令调试死锁将返回未确认的线程名称



我试图使用 SOSEX 命令调试我进程中的死锁!dlk 我得到以下输出:

*DEADLOCK DETECTED*
CLR thread 0xac holds the lock on SyncBlock 00000012ac132068 OBJ:00000012830d66a0[System.Object]
and is waiting for the lock on SyncBlock 00000012ae4ba6b8 OBJ:00000012808391f8[System.Collections.Generic.Dictionary`2[[System.Guid, mscorlib],[CaptureServices.GenericInfrastructure.Controlling.CapManager.MO.CallRecorder.ICallRecorder, Capture Manager]]]
CLR thread 0x9a holds the lock on SyncBlock 00000012ae4ba6b8 OBJ:00000012808391f8[System.Collections.Generic.Dictionary`2[[System.Guid, mscorlib],[CaptureServices.GenericInfrastructure.Controlling.CapManager.MO.CallRecorder.ICallRecorder, Capture Manager]]]
and is waiting for the lock on SyncBlock 00000012ac132068 OBJ:00000012830d66a0[System.Object]

我希望能够获取这些线程的调用堆栈,但我在线程列表中找不到线程 ID(0xac、0x9a(。 我尝试了以下命令,并且上述线程未在任何结果中列出:

  1. ~- 应显示所有线程(托管和非托管(
  2. ~*e!clrstack- 应该显示所有线程的调用堆栈
  3. ~*e!dumpstack- dump all managed threads

有没有另一种方法可以查看在死锁中找到的线程的调用堆栈,我可能缺少?我什至尝试将线程 ID 转换为十进制,但未能找到匹配的线程 ID。

谢谢

以下文章 帮助我如何阅读 !dlk 命令的结果。 根据文章,例如,如果我们在线程0xac中发现死锁:

CLR **thread 0xac** holds the lock on SyncBlock 00000012ac132068 OBJ:00000012830d66a0[System.Object]

!thread 命令的结果有 3 个"ID"列:

  1. 逻辑线程 ID(第 1 列(
  2. CLR ID(第 2 列(
  3. OSID(第 3 列(

!dlk 中的线程 ID 实际上是 !thread 结果中的 CLR ID: (ac( = 172。CLR ID 172 是线程 177。

0:000> !threads
                                  Lock  
ID OSID ThreadOBJ           State GC Mode     GC Alloc Context                  Domain           Count Apt Exception
177  172 1188 00000012abfcaee0  3029220 Preemptive  0000000000000000:0000000000000000 00000012f0e94d70 3     MTA (Threadpool Worker) 

现在我可以通过命令切换到线程 177:~177s

并通过命令查看线程的调用堆栈:!clrstack

最新更新