|
|
@@ -2,15 +2,16 @@ package org.openzen.zenscript.javabytecode.compiler;
|
|
2
|
2
|
|
|
3
|
3
|
import org.objectweb.asm.Label;
|
|
4
|
4
|
import org.objectweb.asm.Type;
|
|
|
5
|
+import org.openzen.zenscript.codemodel.CompareType;
|
|
5
|
6
|
import org.openzen.zenscript.codemodel.expression.*;
|
|
6
|
7
|
import org.openzen.zenscript.codemodel.member.DefinitionMember;
|
|
7
|
8
|
import org.openzen.zenscript.codemodel.type.DefinitionTypeID;
|
|
|
9
|
+import org.openzen.zenscript.codemodel.type.ITypeID;
|
|
8
|
10
|
import org.openzen.zenscript.implementations.IntRange;
|
|
9
|
11
|
import org.openzen.zenscript.javabytecode.*;
|
|
10
|
12
|
import org.openzen.zenscript.shared.CompileException;
|
|
11
|
13
|
import org.openzen.zenscript.shared.CompileExceptionCode;
|
|
12
|
14
|
|
|
13
|
|
-
|
|
14
|
15
|
import java.util.Map;
|
|
15
|
16
|
|
|
16
|
17
|
public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
|
@@ -27,8 +28,34 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
27
|
28
|
this.isInit = isInit;
|
|
28
|
29
|
}
|
|
29
|
30
|
|
|
|
31
|
+ private static Class<?> getForEquals(ITypeID id) {
|
|
|
32
|
+ if (CompilerUtils.isPrimitive(id))
|
|
|
33
|
+ return id.accept(JavaTypeClassVisitor.INSTANCE);
|
|
|
34
|
+ return Object.class;
|
|
|
35
|
+ }
|
|
|
36
|
+
|
|
30
|
37
|
@Override
|
|
31
|
38
|
public Void visitAndAnd(AndAndExpression expression) {
|
|
|
39
|
+ Label end = new Label();
|
|
|
40
|
+ Label onFalse = new Label();
|
|
|
41
|
+
|
|
|
42
|
+ expression.left.accept(this);
|
|
|
43
|
+
|
|
|
44
|
+ javaWriter.ifEQ(onFalse);
|
|
|
45
|
+ expression.right.accept(this);
|
|
|
46
|
+
|
|
|
47
|
+ // //these two calls are redundant but make decompiled code look better. Keep?
|
|
|
48
|
+ // javaWriter.ifEQ(onFalse);
|
|
|
49
|
+ // javaWriter.iConst1();
|
|
|
50
|
+
|
|
|
51
|
+ javaWriter.goTo(end);
|
|
|
52
|
+
|
|
|
53
|
+ javaWriter.label(onFalse);
|
|
|
54
|
+ javaWriter.iConst0();
|
|
|
55
|
+
|
|
|
56
|
+
|
|
|
57
|
+ javaWriter.label(end);
|
|
|
58
|
+
|
|
32
|
59
|
return null;
|
|
33
|
60
|
}
|
|
34
|
61
|
|
|
|
@@ -48,6 +75,13 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
48
|
75
|
|
|
49
|
76
|
@Override
|
|
50
|
77
|
public Void visitCompare(BasicCompareExpression expression) {
|
|
|
78
|
+ expression.left.accept(this);
|
|
|
79
|
+ expression.right.accept(this);
|
|
|
80
|
+ final Type operatorType = Type.getType(CompareType.class);
|
|
|
81
|
+ javaWriter.getStaticField(operatorType.getInternalName(), expression.operator.name(), operatorType.getDescriptor());
|
|
|
82
|
+
|
|
|
83
|
+ javaWriter.invokeStatic(ZenUtils.class, "compare", boolean.class, getForEquals(expression.left.type), getForEquals(expression.right.type), CompareType.class);
|
|
|
84
|
+
|
|
51
|
85
|
return null;
|
|
52
|
86
|
}
|
|
53
|
87
|
|
|
|
@@ -65,6 +99,12 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
65
|
99
|
|
|
66
|
100
|
@Override
|
|
67
|
101
|
public Void visitCallStatic(CallStaticExpression expression) {
|
|
|
102
|
+ for (Expression argument : expression.arguments.arguments) {
|
|
|
103
|
+ argument.accept(this);
|
|
|
104
|
+ }
|
|
|
105
|
+ //TODO: Test with actual static method
|
|
|
106
|
+ final JavaMethodInfo info = expression.member.getTag(JavaMethodInfo.class);
|
|
|
107
|
+ javaWriter.invokeStatic(info.javaClass.internalClassName, info.name, info.signature);
|
|
68
|
108
|
return null;
|
|
69
|
109
|
}
|
|
70
|
110
|
|
|
|
@@ -93,34 +133,65 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
93
|
133
|
public Void visitCapturedThis(CapturedThisExpression expression) {
|
|
94
|
134
|
return null;
|
|
95
|
135
|
}
|
|
96
|
|
-
|
|
97
|
|
- @Override
|
|
|
136
|
+
|
|
|
137
|
+ @Override
|
|
98
|
138
|
public Void visitCast(CastExpression expression) {
|
|
99
|
|
- expression.target.accept(this);
|
|
|
139
|
+ expression.target.accept(this);
|
|
100
|
140
|
if (!checkAndExecuteByteCodeImplementation(expression.member) && !checkAndExecuteMethodInfo(expression.member))
|
|
101
|
141
|
throw new IllegalStateException("Call target has no method info!");
|
|
102
|
|
-
|
|
|
142
|
+
|
|
103
|
143
|
return null;
|
|
104
|
|
- }
|
|
|
144
|
+ }
|
|
105
|
145
|
|
|
106
|
146
|
@Override
|
|
107
|
147
|
public Void visitCheckNull(CheckNullExpression expression) {
|
|
|
148
|
+ final Label end = new Label();
|
|
|
149
|
+ expression.value.accept(this);
|
|
|
150
|
+ javaWriter.dup();
|
|
|
151
|
+ javaWriter.ifNonNull(end);
|
|
|
152
|
+ javaWriter.pop();
|
|
|
153
|
+ javaWriter.newObject(NullPointerException.class);
|
|
|
154
|
+ javaWriter.dup();
|
|
|
155
|
+ javaWriter.constant("Tried to convert a null value to nonnull type " + expression.type.accept(JavaTypeClassVisitor.INSTANCE).getSimpleName());
|
|
|
156
|
+ javaWriter.invokeSpecial(NullPointerException.class, "<init>", "(Ljava/lang/String;)V");
|
|
|
157
|
+ javaWriter.aThrow();
|
|
|
158
|
+ javaWriter.label(end);
|
|
|
159
|
+
|
|
108
|
160
|
return null;
|
|
109
|
161
|
}
|
|
110
|
162
|
|
|
111
|
163
|
@Override
|
|
112
|
164
|
public Void visitCoalesce(CoalesceExpression expression) {
|
|
|
165
|
+ final Label end = new Label();
|
|
|
166
|
+ expression.left.accept(this);
|
|
|
167
|
+ javaWriter.dup();
|
|
|
168
|
+ javaWriter.ifNonNull(end);
|
|
|
169
|
+ javaWriter.pop();
|
|
|
170
|
+ expression.right.accept(this);
|
|
|
171
|
+ javaWriter.label(end);
|
|
113
|
172
|
return null;
|
|
114
|
173
|
}
|
|
115
|
174
|
|
|
116
|
175
|
@Override
|
|
117
|
176
|
public Void visitConditional(ConditionalExpression expression) {
|
|
|
177
|
+ final Label end = new Label();
|
|
|
178
|
+ final Label onElse = new Label();
|
|
|
179
|
+ expression.condition.accept(this);
|
|
|
180
|
+ javaWriter.ifEQ(onElse);
|
|
|
181
|
+ expression.ifThen.accept(this);
|
|
|
182
|
+ javaWriter.goTo(end);
|
|
|
183
|
+ javaWriter.label(onElse);
|
|
|
184
|
+ expression.ifElse.accept(this);
|
|
|
185
|
+ javaWriter.label(end);
|
|
118
|
186
|
return null;
|
|
119
|
187
|
}
|
|
120
|
188
|
|
|
121
|
189
|
@Override
|
|
122
|
190
|
public Void visitConstantBool(ConstantBoolExpression expression) {
|
|
123
|
|
- getJavaWriter().constant(expression.value);
|
|
|
191
|
+ if (expression.value)
|
|
|
192
|
+ javaWriter.iConst1();
|
|
|
193
|
+ else
|
|
|
194
|
+ javaWriter.iConst0();
|
|
124
|
195
|
return null;
|
|
125
|
196
|
}
|
|
126
|
197
|
|
|
|
@@ -209,11 +280,19 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
209
|
280
|
|
|
210
|
281
|
@Override
|
|
211
|
282
|
public Void visitConstructorSuperCall(ConstructorSuperCallExpression expression) {
|
|
|
283
|
+ javaWriter.loadObject(0);
|
|
|
284
|
+ for (Expression argument : expression.arguments.arguments) {
|
|
|
285
|
+ argument.accept(this);
|
|
|
286
|
+ }
|
|
|
287
|
+ //No super calls in enums possible, and that's already handled in the enum constructor itself.
|
|
|
288
|
+ javaWriter.invokeSpecial(expression.objectType.accept(JavaTypeClassVisitor.INSTANCE), "<init>", CompilerUtils.calcDesc(expression.constructor.header, false));
|
|
212
|
289
|
return null;
|
|
213
|
290
|
}
|
|
214
|
291
|
|
|
215
|
292
|
@Override
|
|
216
|
293
|
public Void visitEnumConstant(EnumConstantExpression expression) {
|
|
|
294
|
+ final Type type = expression.type.accept(JavaTypeVisitor.INSTANCE);
|
|
|
295
|
+ javaWriter.getStaticField(type.getInternalName(), expression.value.name, type.getDescriptor());
|
|
217
|
296
|
return null;
|
|
218
|
297
|
}
|
|
219
|
298
|
|
|
|
@@ -224,6 +303,7 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
224
|
303
|
|
|
225
|
304
|
@Override
|
|
226
|
305
|
public Void visitGenericCompare(GenericCompareExpression expression) {
|
|
|
306
|
+ //TODO: What am I supposed to do here?
|
|
227
|
307
|
return null;
|
|
228
|
308
|
}
|
|
229
|
309
|
|
|
|
@@ -237,7 +317,9 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
237
|
317
|
|
|
238
|
318
|
@Override
|
|
239
|
319
|
public Void visitGetFunctionParameter(GetFunctionParameterExpression expression) {
|
|
240
|
|
- javaWriter.load(Type.getType(expression.parameter.type.accept(JavaTypeClassVisitor.INSTANCE)), expression.parameter.index + 1);
|
|
|
320
|
+ //TODO is Static?
|
|
|
321
|
+ final boolean isStatic = false;
|
|
|
322
|
+ javaWriter.load(Type.getType(expression.parameter.type.accept(JavaTypeClassVisitor.INSTANCE)), isStatic ? expression.parameter.index : expression.parameter.index + 1);
|
|
241
|
323
|
return null;
|
|
242
|
324
|
}
|
|
243
|
325
|
|
|
|
@@ -262,16 +344,16 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
262
|
344
|
public Void visitGetter(GetterExpression expression) {
|
|
263
|
345
|
return null;
|
|
264
|
346
|
}
|
|
265
|
|
-
|
|
266
|
|
- @Override
|
|
267
|
|
- public Void visitGlobal(GlobalExpression expression) {
|
|
268
|
|
- return expression.resolution.accept(this);
|
|
269
|
|
- }
|
|
270
|
|
-
|
|
271
|
|
- @Override
|
|
272
|
|
- public Void visitGlobalCall(GlobalCallExpression expression) {
|
|
273
|
|
- return expression.resolution.accept(this);
|
|
274
|
|
- }
|
|
|
347
|
+
|
|
|
348
|
+ @Override
|
|
|
349
|
+ public Void visitGlobal(GlobalExpression expression) {
|
|
|
350
|
+ return expression.resolution.accept(this);
|
|
|
351
|
+ }
|
|
|
352
|
+
|
|
|
353
|
+ @Override
|
|
|
354
|
+ public Void visitGlobalCall(GlobalCallExpression expression) {
|
|
|
355
|
+ return expression.resolution.accept(this);
|
|
|
356
|
+ }
|
|
275
|
357
|
|
|
276
|
358
|
@Override
|
|
277
|
359
|
public Void visitInterfaceCast(InterfaceCastExpression expression) {
|
|
|
@@ -289,7 +371,7 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
289
|
371
|
|
|
290
|
372
|
@Override
|
|
291
|
373
|
public Void visitMakeConst(MakeConstExpression expression) {
|
|
292
|
|
-
|
|
|
374
|
+ //TODO: What am I supposed to do here?
|
|
293
|
375
|
return null;
|
|
294
|
376
|
}
|
|
295
|
377
|
|
|
|
@@ -311,7 +393,7 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
311
|
393
|
@Override
|
|
312
|
394
|
public Void visitNew(NewExpression expression) {
|
|
313
|
395
|
final String type;
|
|
314
|
|
- if(expression.type instanceof DefinitionTypeID)
|
|
|
396
|
+ if (expression.type instanceof DefinitionTypeID)
|
|
315
|
397
|
type = ((DefinitionTypeID) expression.type).definition.name;
|
|
316
|
398
|
else
|
|
317
|
399
|
type = Type.getDescriptor(expression.type.accept(JavaTypeClassVisitor.INSTANCE));
|
|
|
@@ -337,6 +419,26 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
337
|
419
|
|
|
338
|
420
|
@Override
|
|
339
|
421
|
public Void visitOrOr(OrOrExpression expression) {
|
|
|
422
|
+ Label end = new Label();
|
|
|
423
|
+ Label onTrue = new Label();
|
|
|
424
|
+
|
|
|
425
|
+ expression.left.accept(this);
|
|
|
426
|
+
|
|
|
427
|
+ javaWriter.ifNE(onTrue);
|
|
|
428
|
+ expression.right.accept(this);
|
|
|
429
|
+
|
|
|
430
|
+ // //these two calls are redundant but make decompiled code look better. Keep?
|
|
|
431
|
+ // javaWriter.ifNE(onTrue);
|
|
|
432
|
+ // javaWriter.iConst0();
|
|
|
433
|
+
|
|
|
434
|
+ javaWriter.goTo(end);
|
|
|
435
|
+
|
|
|
436
|
+ javaWriter.label(onTrue);
|
|
|
437
|
+ javaWriter.iConst1();
|
|
|
438
|
+
|
|
|
439
|
+
|
|
|
440
|
+ javaWriter.label(end);
|
|
|
441
|
+
|
|
340
|
442
|
return null;
|
|
341
|
443
|
}
|
|
342
|
444
|
|
|
|
@@ -367,6 +469,10 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
367
|
469
|
|
|
368
|
470
|
@Override
|
|
369
|
471
|
public Void visitSetFunctionParameter(SetFunctionParameterExpression expression) {
|
|
|
472
|
+ //TODO is static?
|
|
|
473
|
+ final boolean isStatic = false;
|
|
|
474
|
+ expression.value.accept(this);
|
|
|
475
|
+ javaWriter.store(expression.type.accept(JavaTypeVisitor.INSTANCE), isStatic ? expression.parameter.index : expression.parameter.index + 1);
|
|
370
|
476
|
return null;
|
|
371
|
477
|
}
|
|
372
|
478
|
|
|
|
@@ -418,6 +524,8 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void> {
|
|
418
|
524
|
|
|
419
|
525
|
@Override
|
|
420
|
526
|
public Void visitWrapOptional(WrapOptionalExpression expression) {
|
|
|
527
|
+ //TODO What am I supposed to do here?
|
|
|
528
|
+ expression.value.accept(this);
|
|
421
|
529
|
return null;
|
|
422
|
530
|
}
|
|
423
|
531
|
|