Coo*_*lue 8 c# .net-4.0 compiler-optimization windows-10 visual-studio-2015
我正在逐步完成WPF源代码,用我自己的代码调试一些问题.我有源和pdb(我使用dotpeek作为符号服务器,所以我有完整的pdbs源代码),我可以进入WPF代码没问题.我感兴趣的模块是PresentationFramework和System.Xaml.调试体验非常糟糕,因为框架模块已经过优化(通常是一件好事!).
我的(非常vauge)理解是他们在安装时用VS或其他任何东西预先用ngen.exe进行JIT编辑......这导致混淆.
据我了解,我可以使用ngen.exe(从Visual Studio文件夹启动的Developer Command Prompt)卸载本机映像文件.例如...
ngen uninstall PresentationFramework
Uninstalling assembly PresentationFramework, Version=3.0.0.0, Culture=Neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=msil
Uninstalling assembly PresentationFramework, Version=4.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35
Uninstalling assembly PresentationFramework, Version=4.0.0.0, Culture=Neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=msil
Run Code Online (Sandbox Code Playgroud)
我认为这将迫使C#编译器恢复到MSIL版本并且JIT在运行时编译它们.我还假设我可以禁用运行时JIT优化,这就是我如何获得不错的调试体验.
但是,卸载目标本机映像证明很麻烦.Ngen做到了,但就调试体验而言,它没有任何影响.
当我尝试再次卸载相同的模块时,我被告知它们没有安装 - 这是令人鼓舞的 - 但我还得到一个依赖于它们的文件列表,很多dll和exe文件以及一些本机图像文件(大约十个)和以下信息......
You may need to uninstall these assembly roots in order to delete the native image for PresentationFramework
Error: The specified assembly is not installed.
Run Code Online (Sandbox Code Playgroud)
所以,我假设我需要找到这十个文件的根,并从那里开始卸载所有内容.如果存在一系列依赖关系,这可能会很快失控.
假设我可以获得未经优化的模块,为了抑制JIT优化,我将ini文件添加到与我想要逐步执行的模块相同的文件夹中,例如,C:\Windows\Microsoft.NET\assembly\GAC_MSIL\PresentationFramework\v4.0_4.0.0.0__31bf3856ad364e35我已经
PresentationFramework.ini
[.NET Framework Debugging Control]
GenerateTrackingInfo=1
AllowOptimize=0
Run Code Online (Sandbox Code Playgroud)
在工具/选项/调试/常规/我已禁用启用源服务器支持,并在启用模块加载时抑制JIT优化.
在Project Options中,我在Build部分中禁用了Optimize代码,我还在Build/Advanced中将Debugging信息设置为full.
我是在正确的轨道上吗?在VS的某个地方是否有配置选项我可以告诉它使用dll文件并忽略积极优化的原生图像?
感谢@HansPassant 的帮助,我终于弄清楚了。
本机图像文件存储在 C:\Windows\Assembly\ 中,但有一个名为 shfusion.exe 的 Windows 扩展,它会阻止正常访问该文件夹。通过从提升的 cmd 窗口运行以下命令可以修复此问题...
regsvr32 /u C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\shfusion.dll
Run Code Online (Sandbox Code Playgroud)
接下来,我通过使用 Process Hacker/Properties 上下文菜单检查 devenv 进程使用的 .NET 程序集,准确确认 Visual Studio 正在使用哪些版本的本机映像。它有一个“路径”和一个“本机图像路径”列,因此我可以准确地看到 VS 正在使用哪些 NI。
然后我重命名了我不想优化的模块的文件夹......
确保我在 VS 选项/调试中具有正确的设置,包括抑制 JIT 优化...

请注意,如果您按照我的问题“禁用特定 MSIL 模块的JIT 优化”中所述添加 ini 文件,则无需选中“抑制JIT 优化”框。
并且修复了对象被优化和跳过行的调试问题。
正如这个答案中所解释的,我可以将这样的表达式放在断点操作中,它每次都会起作用......
{((EventSetterNull_SO_41604891_2670182.MainWindow)System.Windows.Application.Current.MainWindow).Logger.LogMemberTry(data, new [] \{"Name", "_value"\}, "XamlNodeList.Add ")}
Run Code Online (Sandbox Code Playgroud)