-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathExample4.cs
107 lines (95 loc) · 4.07 KB
/
Example4.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
/* CilTools.BytecodeAnalysis library example: generating dynamic methods using existing method as template
* Copyright (c) 2021, MSDN.WhiteKnight (https://github.com/MSDN-WhiteKnight)
* License: BSD 2.0 */
using System;
using System.Collections.Generic;
using System.Reflection;
using System.Reflection.Emit;
using System.Linq;
using CilTools.BytecodeAnalysis;
using CilTools.BytecodeAnalysis.Extensions;
namespace CilToolsTest
{
class Program
{
public static void DoSomething(string s)
{
//placeholder method
}
public static void TemplateMethod() //template for a dynamic method
{
Random rnd = new Random();
for (int i = 0; i < 3; i++)
{
try
{
int a = rnd.Next(0, 9);
int b = rnd.Next(10, 15);
DoSomething((a).ToString());
DoSomething("+");
DoSomething((b).ToString());
DoSomething("=");
DoSomething((a + b).ToString());
}
catch (Exception ex)
{
DoSomething(ex.Message);
}
finally
{
DoSomething(";\r\n");
}
}
}
static void Main(string[] args)
{
//create two dynamic methods and get their IL Generators
DynamicMethod dm1 = new DynamicMethod("Method1", typeof(void), new Type[] { }, typeof(Program).Module);
ILGenerator ilg = dm1.GetILGenerator(512);
DynamicMethod dm2 = new DynamicMethod("Method2", typeof(void), new Type[] { }, typeof(Program).Module);
ILGenerator ilg2 = dm2.GetILGenerator(512);
//build CIL graph for template method
CilGraph graph = typeof(Program).GetMethod("TemplateMethod").GetCilGraph();
Console.WriteLine(graph);
//emit two dynamic methods based on the template
graph.EmitTo(ilg, (instr) => {
if ((instr.OpCode.Equals(OpCodes.Call) || instr.OpCode.Equals(OpCodes.Callvirt)) &&
instr.ReferencedMember.Name == "DoSomething")
{
//replace every DoSomething call by Console.Write call
ilg.EmitCall(OpCodes.Call,
typeof(Console).GetMethods().Where((x) =>
{
return x.Name == "Write" && x.GetParameters().Length == 1 &&
x.GetParameters()[0].ParameterType == typeof(string);
}).First(),
null);
return true; //handled
}
else return false;
});
graph.EmitTo(ilg2, (instr) => {
if ((instr.OpCode.Equals(OpCodes.Call) || instr.OpCode.Equals(OpCodes.Callvirt)) &&
instr.ReferencedMember.Name == "DoSomething")
{
//replace every DoSomething call by System.Diagnostics.Debug.Write call
ilg2.EmitCall(OpCodes.Call,
typeof(System.Diagnostics.Debug).GetMethods().Where((x) =>
{
return x.Name == "Write" && x.GetParameters().Length == 1 &&
x.GetParameters()[0].ParameterType == typeof(string);
}).First(),
null);
return true; //handled
}
else return false;
});
//create and execute delegates for both dynamic methods
Action deleg = (Action)dm1.CreateDelegate(typeof(Action));
deleg();
deleg = (Action)dm2.CreateDelegate(typeof(Action));
deleg();
Console.ReadKey();
}
}
}