.net 如何通过反射发射计算.maxstack?
我正在生成一个包含大量switch语句的方法。我注意到在ildasm中.maxstack值非常高。我的理解是。maxstack是给定方法的最大堆栈深度?我在网上找不到太多的信息 在前两个示例中,最大堆栈大小超过4KB。在上一个示例中,最大堆栈大小为509,但实际最大深度似乎为10。为什么价值这么高?这只是jit的一个提示?拥有如此高的.maxstack是否意味着什么?我在互联网上读到的关于最大深度的所有信息都是错误的吗 使现代化 在我的第三个例子中似乎有一个错误。我没有验证或测试,在load参数之后有一个额外的推力。在我修正了maxstack是9之后。对于前两个示例,reflection.emit的最大堆栈大小保持在4K以上。在C编译器中,开关方法是9,而不是反射发射的4502 根据答案,它们似乎将每个基本块的最大深度相加,作为它们的总和,随着C编译器的计算更加正确。尽管如此高的价值,我仍然对其含义感到好奇.net 如何通过反射发射计算.maxstack?,.net,reflection,clr,reflection.emit,ilgenerator,.net,Reflection,Clr,Reflection.emit,Ilgenerator,我正在生成一个包含大量switch语句的方法。我注意到在ildasm中.maxstack值非常高。我的理解是。maxstack是给定方法的最大堆栈深度?我在网上找不到太多的信息 在前两个示例中,最大堆栈大小超过4KB。在上一个示例中,最大堆栈大小为509,但实际最大深度似乎为10。为什么价值这么高?这只是jit的一个提示?拥有如此高的.maxstack是否意味着什么?我在互联网上读到的关于最大深度的所有信息都是错误的吗 使现代化 在我的第三个例子中似乎有一个错误。我没有验证或测试,在load参数
class Program
{
static void Main(string[] args)
{
Foo();
Bar();
FooBar();
}
static void Foo()
{
// Increasing this value will increase the stack depth by the number of labels.
int caseStackDepth = 8;
string name = Path.ChangeExtension("foo", ".dll");
AssemblyName assemblyName = new AssemblyName("foo");
AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
ModuleBuilder module = assemblyBuilder.DefineDynamicModule(assemblyName.Name, name);
TypeBuilder type = module.DefineType("boo", System.Reflection.TypeAttributes.Class | System.Reflection.TypeAttributes.Public);
MethodBuilder method = type.DefineMethod(
"bar",
System.Reflection.MethodAttributes.Public | System.Reflection.MethodAttributes.Static,
typeof(int),
new [] { typeof(int) });
ILGenerator generator = method.GetILGenerator();
LocalBuilder result = generator.DeclareLocal(typeof(int));
Label[] labels = new Label[500];
for (int index = 0; index < labels.Length; index++)
{
labels[index] = generator.DefineLabel();
}
Label end = generator.DefineLabel();
generator.Emit(OpCodes.Ldarg_0);
generator.Emit(OpCodes.Switch, labels);
generator.Emit(OpCodes.Br, end);
for (int index = 0; index < labels.Length; index++)
{
generator.MarkLabel(labels[index]);
generator.Emit(OpCodes.Ldc_I4, index);
// Simulate stack depth.
for (int depth = 0; depth < caseStackDepth; depth++)
{
generator.Emit(OpCodes.Dup);
}
for (int depth = 0; depth < caseStackDepth; depth++)
{
generator.Emit(OpCodes.Add);
}
generator.Emit(OpCodes.Stloc, result);
generator.Emit(OpCodes.Br, end);
}
generator.MarkLabel(end);
generator.Emit(OpCodes.Ldloc, result);
generator.Emit(OpCodes.Ret);
type.CreateType();
assemblyBuilder.Save("foo.dll");
}
static void Bar()
{
// Increasing this value will increase the stack depth by the number of labels.
int caseStackDepth = 8;
string name = Path.ChangeExtension("bar", ".dll");
AssemblyName assemblyName = new AssemblyName("bar");
AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
ModuleBuilder module = assemblyBuilder.DefineDynamicModule(assemblyName.Name, name);
TypeBuilder type = module.DefineType("boo", System.Reflection.TypeAttributes.Class | System.Reflection.TypeAttributes.Public);
MethodBuilder method = type.DefineMethod(
"bar",
System.Reflection.MethodAttributes.Public | System.Reflection.MethodAttributes.Static,
typeof(int),
new[] { typeof(int) });
ILGenerator generator = method.GetILGenerator();
LocalBuilder result = generator.DeclareLocal(typeof(int));
Label end = generator.DefineLabel();
for (int index = 0; index < 500; index++)
{
Label equal = generator.DefineLabel();
Label notEqual = generator.DefineLabel();
generator.Emit(OpCodes.Ldarg_0);
generator.Emit(OpCodes.Ldc_I4, index);
generator.Emit(OpCodes.Beq, equal);
generator.Emit(OpCodes.Br, notEqual);
generator.MarkLabel(equal);
generator.Emit(OpCodes.Ldc_I4, index);
// Simulate stack depth.
for (int depth = 0; depth < caseStackDepth; depth++)
{
generator.Emit(OpCodes.Dup);
}
for (int depth = 0; depth < caseStackDepth; depth++)
{
generator.Emit(OpCodes.Add);
}
generator.Emit(OpCodes.Stloc, result);
generator.Emit(OpCodes.Br, end);
generator.MarkLabel(notEqual);
}
generator.MarkLabel(end);
generator.Emit(OpCodes.Ldloc, result);
generator.Emit(OpCodes.Ret);
type.CreateType();
assemblyBuilder.Save("bar.dll");
}
static void FooBar()
{
// Increasing this value will increase the stack depth by the number of labels.
int caseStackDepth = 8;
string name = Path.ChangeExtension("foobar", ".dll");
AssemblyName assemblyName = new AssemblyName("foobar");
AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
ModuleBuilder module = assemblyBuilder.DefineDynamicModule(assemblyName.Name, name);
TypeBuilder type = module.DefineType("boo", System.Reflection.TypeAttributes.Class | System.Reflection.TypeAttributes.Public);
MethodBuilder method = type.DefineMethod(
"bar",
System.Reflection.MethodAttributes.Public | System.Reflection.MethodAttributes.Static,
typeof(int),
new[] { typeof(int) });
ILGenerator generator = method.GetILGenerator();
LocalBuilder result = generator.DeclareLocal(typeof(int));
for (int index = 0; index < 500; index++)
{
generator.Emit(OpCodes.Ldarg_0);
//错误,这导致堆栈大小非常大。
//generator.EmitOpCodes.Ldc_I4,索引
// Simulate stack depth.
for (int depth = 0; depth < caseStackDepth; depth++)
{
generator.Emit(OpCodes.Dup);
}
for (int depth = 0; depth < caseStackDepth; depth++)
{
generator.Emit(OpCodes.Add);
}
generator.Emit(OpCodes.Stloc, result);
}
generator.Emit(OpCodes.Ldloc, result);
generator.Emit(OpCodes.Ret);
type.CreateType();
assemblyBuilder.Save("foobar.dll");
}
}
在中,我们可以找到跟踪所需堆栈大小的方法,称为。其中的评论似乎在说明:
//如果当前指令表示基本指令结束,则
//表示无条件分支,将m_maxMidStack添加到m_maxStackSize。
//m_maxStackSize最终将是
//每个基本块。
也就是说,没有人试图执行任何控制流分析。每个基本块所需的最大堆栈大小相加
我不知道这个大小异常高的任何实质性影响。查看,我们可以找到跟踪所需堆栈大小的方法,称为。其中的评论似乎在说明:
//如果当前指令表示基本指令结束,则
//表示无条件分支,将m_maxMidStack添加到m_maxStackSize。
//m_maxStackSize最终将是
//每个基本块。
也就是说,没有人试图执行任何控制流分析。每个基本块所需的最大堆栈大小相加
我不知道这个尺寸异常高会带来什么实质性影响。这个尺寸上存在连接错误。jit使用maxstack来确定方法是否可以内联
此服务器上存在连接错误。jit使用maxstack来确定方法是否可以内联