5

我从 ASP.NET 应用程序返回的堆栈跟踪(从 ArgumentNullException 生成)给人的印象是下面代码的最后一行发生了错误。据我所知,这是不可能的,但如果 JIT 优化了对 的调用Bar,这将导致不同的堆栈跟踪,它会解释一切。我确定它不是 c# 编译器,因为 CIL 看起来像我所期望的那样。JIT 编译器是否有可能删除对 Bar 的调用?

c# 4,.NET 4.0.30319.1,ASP.NET 4.0.30319.1

编辑: 我应该提到这是一个发布配置,Optimize code = on,Debug Info = pdb-only。

Stack Trace:

[ArgumentNullException: Value cannot be null. Parameter name: value]
CreateHiddenField(HtmlTextWriter tr, String name, String value) in Foo.cs:129
Foo(IHttpContext context, HtmlTextWriter writer) in Foo.cs:106

private static void Foo(IHttpContext context, HtmlTextWriter writer)
{ // line 103
  Bar(writer, AuthorizationServerResponseDetailsHttpRequestParser.RequestSAMLFieldName, context);
  Bar(writer, AuthorizationServerResponseDetailsHttpRequestParser.RequestTargetFieldName, context);
  // line 106 - blank line in source code.
  CreateHiddenField(tr, name, string.Empty); // looks like its here
}

private static void Bar(HtmlTextWriter tr,string name, IHttpContext context)
{ // line 116
   #region Sanitation
   if (tr == null) { throw new System.ArgumentNullException("tr"); }
   if (name == null) { throw new System.ArgumentNullException("name"); }
   if (context == null) { throw new System.ArgumentNullException("context"); }
   #endregion

   CreateHiddenField(tr, name, context.RequestQueryString(name));
}

private static void CreateHiddenField(HtmlTextWriter tr, string name, string value)
{ // line 127
   #region Sanitation
   if (tr == null) { throw new System.ArgumentNullException("tr"); }
   if (name == null) { throw new System.ArgumentNullException("name"); }
   if (value == null) { throw new System.ArgumentNullException("value"); }
   #endregion

   // payload...
}
4

2 回答 2

4

根据http://www.hanselman.com/blog/ReleaseISNOTDebug64bitOptimizationsAndCMethodInliningInReleaseBuildCallStacks.aspx,如果 JITter 内联方法调用,它确实会在堆栈跟踪中折叠。如果您不希望它内联(这可能不是一个好主意),您可以使用

[MethodImpl(MethodImplOptions.NoInlining)] 

在方法上。对于 exe,您还可以使用 INI 文件告诉 JITter 生成跟踪信息(在链接页面的底部),但我不确定这对 ASP.NET 应用程序如何工作。

于 2013-04-11T18:09:34.010 回答
2

我从未见过 JIT 删除方法调用,只是内联它。通常,它会删除未使用的局部变量,但我相当肯定 JIT 不能静态分析调用方法的所有副作用,所以我认为它永远不会删除方法调用。

我的猜测是 null 值被传递给内部调用的方法,该方法CreateHiddenField恰好也有一个参数value。因为默认情况下在发布模式下启用内联,所以您不能真正信任堆栈跟踪(或发生异常的假定行号)。您的问题很可能CreateHiddenField在隐形堆栈内部或更深处。:)

您可以禁用优化以更好地了解堆栈跟踪。 MSDN 这里有一些说明

于 2013-04-11T18:18:04.443 回答