From e2a240b284d399c4f7d15db4730f349df8688260 Mon Sep 17 00:00:00 2001 From: Mayuki Sawatari Date: Sun, 10 Nov 2019 00:36:38 +0900 Subject: [PATCH 1/2] feat: Hide a invoke helper method in a stack trace. --- .../Server/Hubs/StreamingHubHandler.cs | 3 +- src/MagicOnion/Server/InvokeHelper.cs | 72 +++++++++++++++++++ src/MagicOnion/Server/MethodHandler.cs | 3 +- 3 files changed, 74 insertions(+), 4 deletions(-) create mode 100644 src/MagicOnion/Server/InvokeHelper.cs diff --git a/src/MagicOnion/Server/Hubs/StreamingHubHandler.cs b/src/MagicOnion/Server/Hubs/StreamingHubHandler.cs index a9d9e46c0..98d3dbd74 100644 --- a/src/MagicOnion/Server/Hubs/StreamingHubHandler.cs +++ b/src/MagicOnion/Server/Hubs/StreamingHubHandler.cs @@ -167,8 +167,7 @@ Func BuildMethodBodyWithFilter(Func newFilter.Invoke(ctx, next_); + next = new InvokeHelper>(newFilter.Invoke, next).GetDelegate(); } return next; diff --git a/src/MagicOnion/Server/InvokeHelper.cs b/src/MagicOnion/Server/InvokeHelper.cs new file mode 100644 index 000000000..0c2cb1dff --- /dev/null +++ b/src/MagicOnion/Server/InvokeHelper.cs @@ -0,0 +1,72 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Reflection.Emit; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; +using System.Text; +using System.Threading.Tasks; + +namespace MagicOnion.Server +{ + internal class InvokeHelper + where TDelegate : Delegate + { + public Func Invoke; + public TDelegate Next; + + private static readonly Func, TDelegate> InvokeNextFactory; + + static InvokeHelper() + { + var fieldInvoke = typeof(InvokeHelper).GetField("Invoke"); + var fieldNext = typeof(InvokeHelper).GetField("Next"); + var methodInvoke = typeof(Func).GetMethod("Invoke"); + + if (RuntimeInformation.FrameworkDescription.StartsWith(".NET Core") && Environment.Version > new Version(3, 0, 0)) + { + // HACK: If the app is running on .NET Core 2.2 or earlier, the runtime hides dynamic method in the stack trace. + var method = new DynamicMethod("InvokeNext", typeof(ValueTask), new[] { typeof(InvokeHelper), typeof(TArg1) }, restrictedSkipVisibility: true); + { + var il = method.GetILGenerator(); + + // arg0.Invoke; + il.Emit(OpCodes.Ldarg_0); + il.Emit(OpCodes.Ldfld, fieldInvoke); + + // return arg0.Invoke(arg1) + il.Emit(OpCodes.Ldarg_1); + il.Emit(OpCodes.Ldarg_0); + il.Emit(OpCodes.Ldfld, fieldNext); + il.Emit(OpCodes.Callvirt, methodInvoke); + il.Emit(OpCodes.Ret); + } + + InvokeNextFactory = (helper) => (TDelegate)method.CreateDelegate(typeof(TDelegate), helper); + + } + else + { + // HACK: If the app is running on .NET Core 3.0 or later, the runtime hides `AggressiveInlining` method in the stack trace. (If the app is running on .NET Framework 4.x, This hack does not affect.) + // https://github.com/dotnet/coreclr/blob/release/3.0/src/System.Private.CoreLib/shared/System/Diagnostics/StackTrace.cs#L343-L350 + InvokeNextFactory = (helper) => + { + var invokeNext = new Func(helper.InvokeNext); + return (TDelegate)Delegate.CreateDelegate(typeof(TDelegate), invokeNext.Target, invokeNext.Method); + }; + } + } + + public InvokeHelper(Func invoke, TDelegate next) + { + Invoke = invoke; + Next = next; + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + [DebuggerHidden] + private ValueTask InvokeNext(TArg1 arg1) => Invoke(arg1, Next); + + public TDelegate GetDelegate() => InvokeNextFactory(this); + } +} diff --git a/src/MagicOnion/Server/MethodHandler.cs b/src/MagicOnion/Server/MethodHandler.cs index 23fb29608..507fc0157 100644 --- a/src/MagicOnion/Server/MethodHandler.cs +++ b/src/MagicOnion/Server/MethodHandler.cs @@ -292,8 +292,7 @@ Func BuildMethodBodyWithFilter(Func newFilter.Invoke(ctx, next_); + next = new InvokeHelper>(newFilter.Invoke, next).GetDelegate(); } return next; From 3a39f7f19813b3216b9bed6ae7d7a3c3af045c3f Mon Sep 17 00:00:00 2001 From: Mayuki Sawatari Date: Sun, 10 Nov 2019 01:45:28 +0900 Subject: [PATCH 2/2] fix: .NET Core 2.x reports ".NET Core 4.x" --- src/MagicOnion/Server/InvokeHelper.cs | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/MagicOnion/Server/InvokeHelper.cs b/src/MagicOnion/Server/InvokeHelper.cs index 0c2cb1dff..18dc9f673 100644 --- a/src/MagicOnion/Server/InvokeHelper.cs +++ b/src/MagicOnion/Server/InvokeHelper.cs @@ -23,18 +23,19 @@ static InvokeHelper() var fieldNext = typeof(InvokeHelper).GetField("Next"); var methodInvoke = typeof(Func).GetMethod("Invoke"); - if (RuntimeInformation.FrameworkDescription.StartsWith(".NET Core") && Environment.Version > new Version(3, 0, 0)) + if (RuntimeInformation.FrameworkDescription.StartsWith(".NET Core 4")) /* .NET Core 2.x returns ".NET Core 4.x.y.z" */ { // HACK: If the app is running on .NET Core 2.2 or earlier, the runtime hides dynamic method in the stack trace. var method = new DynamicMethod("InvokeNext", typeof(ValueTask), new[] { typeof(InvokeHelper), typeof(TArg1) }, restrictedSkipVisibility: true); { var il = method.GetILGenerator(); - // arg0.Invoke; + // invoke = arg0.Invoke; il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, fieldInvoke); - // return arg0.Invoke(arg1) + // next = arg0.Next; + // return invoke(arg1, next); il.Emit(OpCodes.Ldarg_1); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldfld, fieldNext);