-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathInstructions.cs
267 lines (219 loc) · 7.39 KB
/
Instructions.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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
// Copyright Microsoft, 2017
// Copyright Carl Walsh, 2021
using System;
using System.Collections.Generic;
using System.Linq;
// CodeNode is written though reflection
#pragma warning disable 0649 // CS0649: Field '...' is never assigned to
//TODO(link) Link branch targets
//TODO(method) §III.1.7.2 validate branch targets are valid offsets
//TODO(method) §III.1.3 validate stack depth doesn't go negative or violate maxstack §III.1.7.4
//TODO(method) §III.1.5 validate operand type with needed manual conversions
//TODO(method) §III.1.7.4 validate branching stack depth is consistant
//TODO(method) §III.1.7.5 unseen op after unconditional branch the stack is assumed to have depth zero
//TODO(method) §III.1.8.1.3 validate all branches have mergable types in the stack
// TODO?????checking that methods return with 0 or 1 elements on the stack
// III CIL Instruction Set
[Ecma("III.1.7.1")]
sealed class InstructionStream : CodeNode
{
Dictionary<int, Op> ops = new Dictionary<int, Op>();
Method method;
public InstructionStream(Method method) {
this.method = method;
}
protected override void InnerRead() {
while (Bytes.Stream.Position - Start < method.CodeSize) {
var op = Bytes.ReadClass<Op>();
op.NodeName = $"Op[{Children.Count}]";
Children.Add(op);
ops.Add(op.Start, op);
}
//TODO(fixme) call ValidateStack() here but fix all the NotImplementedException
Description = string.Join("\n", Children.Take(10).Select(n => n.Description));
}
void ValidateStack() {
int? stack = null;
foreach (Op op in Children) {
if (op.Def.controlFlow == "RETURN") {
// stack should be 1 if return type != void
if (stack > 1) {
op.Errors.Add($"Stack depth is {stack} on RETURN");
}
stack = null;
continue;
}
if (!stack.HasValue) stack = 0;
ValidateOp(op, stack.Value);
stack = Op.DataPopPush(op.Def, stack.Value);
switch (op.Def.controlFlow) {
case "NEXT":
case "CALL":
case "META":
case "BREAK": // Debugger breakpoint
break;
case "BRANCH":
ValidateBranch(op, stack.Value);
stack = null;
break;
case "COND_BRANCH":
ValidateBranch(op, stack.Value);
break;
case "THROW":
throw new NotImplementedException();
case "RETURN":
stack = null;
break;
default:
throw new InvalidOperationException(op.Def.controlFlow);
}
if (op.Def.controlFlow != "NEXT") throw new NotImplementedException(op.Def.controlFlow);
}
}
static void ValidateOp(Op op, int stack) {
if (op.StackCalculated) {
if (op.StartStack != stack) {
op.Errors.Add($"Stack depth {stack} != {op.StartStack}");
}
} else {
op.StartStack = stack;
}
}
void ValidateBranch(Op op, int stack) {
foreach (var targetArg in op.Children.Skip(1)) {
var target = targetArg.GetInt32() + op.End;
if (!ops.TryGetValue(target, out var targetOp)) {
op.Errors.Add($"Branch target {target:X2} not found");
return;
}
targetArg.Link = targetOp;
ValidateOp(targetOp, stack);
}
}
}
[Ecma("III.1.9")]
sealed class MetadataToken : CodeNode
{
public UInt24 Offset;
public byte Table;
public UserStringHeapIndex Index;
protected override void InnerRead() {
var origPos = Bytes.Stream.Position;
AddChild(nameof(Offset));
AddChild(nameof(Table));
if (Table == 0x70) {
Children.Clear();
// Reposition stream, and read "XX XX 00 70"
Bytes.Stream.Position = origPos;
AddChild(nameof(Index));
if (Bytes.Read<byte>() != 0)
throw new NotImplementedException("Too big UserStringHeapIndex");
Index.End++;
AddChild(nameof(Table));
Children.Last().Description = "UserStringHeapIndex";
NodeValue = Index.NodeValue;
} else {
var flag = (MetadataTableFlags)(1L << Table);
Children.Last().Description = flag.ToString();
var link = Bytes.TildeStream.GetCodeNode(flag, Offset.IntValue - 1); // indexed by 1
Children.First().Link = link;
NodeValue = link.NodeValue;
}
}
}
sealed class Op : CodeNode
{
public byte Opcode;
public OpCode Def { get; set; }
public int StartStack { get; set; } = -1; // would be better to use types
public bool StackCalculated => StartStack != -1;
public override string EcmaSection => Def.ecma;
protected override void InnerRead() {
AddChild(nameof(Opcode));
if (Opcode == 0xFE) {
var secondByte = Bytes.Read<byte>();
var opcodeNode = Children.Single();
opcodeNode.End = (int)Bytes.Stream.Position;
opcodeNode.NodeValue = opcodeNode.NodeValue + " " + secondByte.GetString();
Def = OpCode.SecondByte(secondByte);
} else {
Def = OpCode.FirstByte(Opcode);
}
Children.Single().Description = Def.name;
Description = ReadInLineArguments();
if (Children.Count == 1) {
NodeValue = Children.Single().NodeValue;
Children.Clear();
}
}
// §VI.C.2
public static int DataPopPush(OpCode def, int stack) {
//TODO(method) ensure that invoking instance function pops one extra.
//TODO(method) Also, not sure where the code should go, but ldarg.0 in instance method gives this
if (def.stackPop == "VarPop") throw new NotImplementedException();
foreach (var s in def.stackPop.Split("Pop").Where(s => s != "")) {
stack -= GetStackElem(s);
}
if (def.stackPush == "VarPush") throw new NotImplementedException();
foreach (var s in def.stackPush.Split("Push").Where(s => s != "")) {
stack += GetStackElem(s);
}
static int GetStackElem(string s) {
return s switch {
"0" => 0,
"1" => 1,
"I" => 1,
"I8" => 1,
"R4" => 1,
"R8" => 1,
"Ref" => 1,
_ => throw new InvalidOperationException(s),
};
}
return stack;
}
string ReadInLineArguments() => Def.opParams switch {
// MAYBE use OpCode const fields
"InlineBrTarget" => With<uint>(),
"InlineField" => WithToken(),
"InlineI" => With<int>(),
"InlineI8" => With<long>(),
"InlineMethod" => WithToken(),
"InlineNone" => Def.name,
"InlineR" => With<double>(),
"InlineSig" => WithToken(),
"InlineString" => WithToken(),
"InlineSwitch" => SwitchOp(),
"InlineTok" => WithToken(),
"InlineType" => WithToken(),
"InlineVar" => With<ushort>(),
"ShortInlineBrTarget" => With<sbyte>(),
"ShortInlineI" => With<sbyte>(),
"ShortInlineR" => With<float>(),
"ShortInlineVar" => With<byte>(),
_ => throw new InvalidOperationException(Def.opParams),
};
string With<T>() where T : struct {
var value = Bytes.ReadClass<StructNode<T>>();
value.NodeName = "Value";
Children.Add(value);
return $"{Def.name} {value.NodeValue}";
}
public MetadataToken Token;
string WithToken() {
AddChild(nameof(Token));
return $"{Def.name} {Token.NodeValue}";
}
public uint Count;
//TODO(link) link each switch target to op. Using StructNode[] keeps each row its own size
public StructNode<int>[] Targets;
string SwitchOp() {
AddChild(nameof(Count));
AddChild(nameof(Targets));
return $"switch ({string.Join(", ", Targets.Select(n => n.t))})";
}
protected override int GetCount(string field) => field switch {
nameof(Targets) => (int)Count,
_ => base.GetCount(field),
};
}