|
@@ -8,7 +8,6 @@ import org.openzen.zenscript.codemodel.CompareType;
|
8
|
8
|
import org.openzen.zenscript.codemodel.FunctionParameter;
|
9
|
9
|
import org.openzen.zenscript.codemodel.expression.*;
|
10
|
10
|
import org.openzen.zenscript.codemodel.expression.switchvalue.VariantOptionSwitchValue;
|
11
|
|
-import org.openzen.zenscript.codemodel.generic.TypeParameter;
|
12
|
11
|
import org.openzen.zenscript.codemodel.member.ref.DefinitionMemberRef;
|
13
|
12
|
import org.openzen.zenscript.codemodel.member.ref.FieldMemberRef;
|
14
|
13
|
import org.openzen.zenscript.codemodel.statement.ReturnStatement;
|
|
@@ -1897,7 +1896,7 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void>, JavaNativ
|
1897
|
1896
|
final JavaWriter functionWriter;
|
1898
|
1897
|
|
1899
|
1898
|
//Bridge method!!!
|
1900
|
|
- if(!Objects.equals(methodInfo.descriptor, signature)) {
|
|
1899
|
+ if (!Objects.equals(methodInfo.descriptor, signature)) {
|
1901
|
1900
|
final JavaMethod bridgeMethodInfo = new JavaMethod(methodInfo.cls, methodInfo.kind, methodInfo.name, methodInfo.compile, methodInfo.descriptor, methodInfo.modifiers | JavaModifiers.BRIDGE | JavaModifiers.SYNTHETIC, methodInfo.genericResult, methodInfo.typeParameterArguments);
|
1902
|
1901
|
final JavaWriter bridgeWriter = new JavaWriter(expression.position, lambdaCW, bridgeMethodInfo, null, methodInfo.descriptor, null, "java/lang/Override");
|
1903
|
1902
|
bridgeWriter.start();
|
|
@@ -1909,10 +1908,12 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void>, JavaNativ
|
1909
|
1908
|
final FunctionParameter functionParameter = expression.header.parameters[i];
|
1910
|
1909
|
final Type type = context.getType(functionParameter.type);
|
1911
|
1910
|
bridgeWriter.load(type, i + 1);
|
1912
|
|
- bridgeWriter.checkCast(type);
|
|
1911
|
+ if (!CompilerUtils.isPrimitive(functionParameter.type.type)) {
|
|
1912
|
+ bridgeWriter.checkCast(type);
|
|
1913
|
+ }
|
1913
|
1914
|
}
|
1914
|
1915
|
|
1915
|
|
- bridgeWriter.invokeVirtual(methodInfo);
|
|
1916
|
+ bridgeWriter.invokeVirtual(new JavaMethod(JavaClass.fromInternalName(className, JavaClass.Kind.CLASS), methodInfo.kind, methodInfo.name, methodInfo.compile, signature, methodInfo.modifiers, methodInfo.genericResult));
|
1916
|
1917
|
if(expression.header.getReturnType().type != BasicTypeID.VOID) {
|
1917
|
1918
|
bridgeWriter.returnType(context.getType(expression.header.getReturnType()));
|
1918
|
1919
|
}
|
|
@@ -3324,7 +3325,7 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void>, JavaNativ
|
3324
|
3325
|
|
3325
|
3326
|
|
3326
|
3327
|
if (builtin == BuiltinID.ARRAY_CONSTRUCTOR_SIZED) {
|
3327
|
|
- type.elementType.type.accept(JavaDefaultExpressionTypeVisitor.INSTANCE).accept(this);
|
|
3328
|
+ type.elementType.type.getDefaultValue().accept(this);
|
3328
|
3329
|
} else {
|
3329
|
3330
|
expression.arguments.arguments[expression.arguments.arguments.length - 1].accept(this);
|
3330
|
3331
|
}
|
|
@@ -3338,117 +3339,117 @@ public class JavaExpressionVisitor implements ExpressionVisitor<Void>, JavaNativ
|
3338
|
3339
|
return;
|
3339
|
3340
|
}
|
3340
|
3341
|
case ARRAY_CONSTRUCTOR_LAMBDA: {
|
3341
|
|
- ArrayTypeID type = (ArrayTypeID) expression.type.type;
|
3342
|
|
-
|
3343
|
|
- if (type.dimension == 1) {
|
3344
|
|
- // array = new T[arguments[0]]
|
3345
|
|
- // lambda = arguments[1]
|
3346
|
|
- // for (int i = 0; i < array.length; i++)
|
3347
|
|
- // array[i] = lambda.invoke(i);
|
3348
|
|
- //
|
3349
|
|
- // NOTE: arguments[1] can be a FunctionExpression; this can be optimized by running it inline
|
3350
|
|
- throw new UnsupportedOperationException("Not yet supported!");
|
3351
|
|
- } else {
|
3352
|
|
- // TODO: implement
|
3353
|
|
- throw new UnsupportedOperationException("Not yet supported!");
|
3354
|
|
- }
|
|
3342
|
+
|
|
3343
|
+ //Labels
|
|
3344
|
+ final Label begin = new Label();
|
|
3345
|
+ final Label end = new Label();
|
|
3346
|
+ javaWriter.label(begin);
|
|
3347
|
+
|
|
3348
|
+ final Type ASMElementType = context.getType(expression.type);
|
|
3349
|
+ final int dimension = ((ArrayTypeID) expression.type.type).dimension;
|
|
3350
|
+ final int[] arraySizes = ArrayInitializerHelper.getArraySizeLocationsFromConstructor(dimension, expression.arguments.arguments, this);
|
|
3351
|
+ ArrayInitializerHelper.visitMultiDimArray(javaWriter, arraySizes, new int[dimension], dimension, ASMElementType, (elementType, counterLocations) -> {
|
|
3352
|
+ expression.arguments.arguments[dimension].accept(this);
|
|
3353
|
+ for (int counterLocation : counterLocations) {
|
|
3354
|
+ javaWriter.loadInt(counterLocation);
|
|
3355
|
+ }
|
|
3356
|
+ javaWriter.invokeInterface(context.getFunctionalInterface(expression.arguments.arguments[dimension].type));
|
|
3357
|
+ });
|
|
3358
|
+ javaWriter.label(end);
|
|
3359
|
+ return;
|
3355
|
3360
|
}
|
3356
|
|
- case ARRAY_CONSTRUCTOR_PROJECTED: {
|
3357
|
|
-
|
3358
|
|
-
|
|
3361
|
+ case ARRAY_CONSTRUCTOR_PROJECTED:
|
|
3362
|
+ case ARRAY_CONSTRUCTOR_PROJECTED_INDEXED: {
|
3359
|
3363
|
ArrayTypeID type = (ArrayTypeID) expression.type.type;
|
3360
|
|
-
|
|
3364
|
+
|
|
3365
|
+ //Labels
|
3361
|
3366
|
final Label begin = new Label();
|
3362
|
3367
|
final Label end = new Label();
|
3363
|
|
-
|
3364
|
3368
|
javaWriter.label(begin);
|
3365
|
|
- expression.arguments.arguments[0].accept(this); //Origin array
|
|
3369
|
+
|
|
3370
|
+ //Origin Array
|
|
3371
|
+ expression.arguments.arguments[0].accept(this);
|
3366
|
3372
|
final Type originArrayType = context.getType(expression.arguments.arguments[0].type);
|
3367
|
3373
|
final int originArrayLocation = javaWriter.local(originArrayType);
|
3368
|
3374
|
javaWriter.storeObject(originArrayLocation);
|
3369
|
3375
|
Type destinationArrayType = context.getType(expression.type);
|
3370
|
|
-
|
3371
|
|
-
|
3372
|
|
- final boolean canBeInlined = ArrayInitializerHelper.canBeInlined(expression.arguments.arguments[1]);
|
3373
|
|
- final Type functionType; //Only used if not inline able
|
3374
|
|
- final int functionLocation; //Only used if not inline able
|
3375
|
|
- if (!canBeInlined) {
|
3376
|
|
- expression.arguments.arguments[1].accept(this); //Projection Function
|
3377
|
|
- functionType = context.getType(expression.arguments.arguments[1].type);
|
3378
|
|
- functionLocation = javaWriter.local(functionType);
|
3379
|
|
- javaWriter.storeObject(functionLocation);
|
3380
|
|
- javaWriter.addVariableInfo(new JavaLocalVariableInfo(functionType, functionLocation, begin, "projectionFunction", end));
|
3381
|
|
- }
|
3382
|
|
-
|
3383
|
|
- final int[] arraySizes = ArrayInitializerHelper.getArraySizeLocationsProjected(type.dimension, originArrayType, originArrayLocation, javaWriter);
|
3384
|
|
- ArrayInitializerHelper.visitProjected(javaWriter, arraySizes, type.dimension, originArrayLocation, originArrayType, destinationArrayType,
|
3385
|
|
- (elementType, counterLocations) -> {
|
3386
|
|
- if (canBeInlined) {
|
|
3376
|
+
|
|
3377
|
+ final boolean indexed = builtin == BuiltinID.ARRAY_CONSTRUCTOR_PROJECTED_INDEXED;
|
|
3378
|
+ final boolean canBeInLined = ArrayInitializerHelper.canBeInLined(expression.arguments.arguments[1]);
|
|
3379
|
+ if (canBeInLined) {
|
|
3380
|
+ //We can inline, so do it
|
|
3381
|
+ final int[] arraySizes = ArrayInitializerHelper.getArraySizeLocationsProjected(type.dimension, originArrayType, originArrayLocation, javaWriter);
|
|
3382
|
+ final Type projectedElementType = Type.getType(originArrayType.getDescriptor().substring(type.dimension));
|
|
3383
|
+ ArrayInitializerHelper.visitProjected(javaWriter, arraySizes, type.dimension, originArrayLocation, originArrayType, destinationArrayType,
|
|
3384
|
+ (elementType, counterLocations) -> {
|
3387
|
3385
|
Label inlineBegin = new Label();
|
3388
|
3386
|
Label inlineEnd = new Label();
|
3389
|
3387
|
javaWriter.label(inlineBegin);
|
3390
|
|
- final Type projectedElementType = Type.getType(originArrayType.getDescriptor().substring(type.dimension));
|
|
3388
|
+
|
3391
|
3389
|
final int projectedElementLocal = javaWriter.local(projectedElementType);
|
3392
|
3390
|
javaWriter.store(projectedElementType, projectedElementLocal);
|
3393
|
|
-
|
3394
|
|
-
|
|
3391
|
+
|
|
3392
|
+
|
3395
|
3393
|
JavaExpressionVisitor visitor = new JavaExpressionVisitor(context, module, javaWriter) {
|
3396
|
3394
|
@Override
|
3397
|
3395
|
public Void visitGetFunctionParameter(GetFunctionParameterExpression expression) {
|
|
3396
|
+ if(indexed) {
|
|
3397
|
+ final JavaParameterInfo parameterInfo = module.getParameterInfo(expression.parameter);
|
|
3398
|
+ if (parameterInfo != null && parameterInfo.index <= type.dimension) {
|
|
3399
|
+ javaWriter.loadInt(counterLocations[parameterInfo.index - 1]);
|
|
3400
|
+ return null;
|
|
3401
|
+ }
|
|
3402
|
+ }
|
|
3403
|
+
|
3398
|
3404
|
javaWriter.load(projectedElementType, projectedElementLocal);
|
3399
|
3405
|
return null;
|
3400
|
3406
|
}
|
3401
|
3407
|
};
|
3402
|
|
-
|
|
3408
|
+
|
3403
|
3409
|
Expression funcExpression = expression.arguments.arguments[1];
|
3404
|
3410
|
while (funcExpression instanceof StorageCastExpression) {
|
3405
|
3411
|
funcExpression = ((StorageCastExpression) funcExpression).value;
|
3406
|
3412
|
}
|
3407
|
|
-
|
|
3413
|
+
|
3408
|
3414
|
if (funcExpression instanceof FunctionExpression && ((FunctionExpression) funcExpression).body instanceof ReturnStatement) {
|
|
3415
|
+ CompilerUtils.tagMethodParameters(context, module, ((FunctionExpression) funcExpression).header, false);
|
3409
|
3416
|
((ReturnStatement) ((FunctionExpression) funcExpression).body).value.accept(visitor);
|
3410
|
3417
|
javaWriter.addVariableInfo(new JavaLocalVariableInfo(projectedElementType, projectedElementLocal, inlineBegin, ((FunctionExpression) funcExpression).header.parameters[0].name, inlineEnd));
|
|
3418
|
+
|
3411
|
3419
|
} else throw new IllegalStateException("Trying to inline a non-inlineable expression");
|
3412
|
|
-
|
3413
|
|
-
|
|
3420
|
+
|
|
3421
|
+
|
3414
|
3422
|
javaWriter.label(inlineEnd);
|
3415
|
|
- } else {
|
|
3423
|
+ });
|
|
3424
|
+ } else {
|
|
3425
|
+ //We cannot inline, so get a hold of the function expression and apply it to every
|
|
3426
|
+ expression.arguments.arguments[1].accept(this); //Projection Function
|
|
3427
|
+ final Type functionType = context.getType(expression.arguments.arguments[1].type);
|
|
3428
|
+ final int functionLocation = javaWriter.local(functionType);
|
|
3429
|
+ javaWriter.storeObject(functionLocation);
|
|
3430
|
+ javaWriter.addVariableInfo(new JavaLocalVariableInfo(functionType, functionLocation, begin, "projectionFunction", end));
|
|
3431
|
+ final int[] arraySizes = ArrayInitializerHelper.getArraySizeLocationsProjected(type.dimension, originArrayType, originArrayLocation, javaWriter);
|
|
3432
|
+ ArrayInitializerHelper.visitProjected(javaWriter, arraySizes, type.dimension, originArrayLocation, originArrayType, destinationArrayType,
|
|
3433
|
+ (elementType, counterLocations) -> {
|
3416
|
3434
|
//Apply function here
|
3417
|
|
- //javaWriter.loadObject(functionLocation);
|
3418
|
|
- //javaWriter.swap();
|
3419
|
|
-
|
3420
|
|
- //TODO invoke?
|
3421
|
|
- //javaWriter.invokeVirtual(new JavaMethod(JavaClass.fromInternalName("lambda1", JavaClass.Kind.CLASS), JavaMethod.Kind.INSTANCE, "accept", true, "(Ljava/lang/String;)Ljava/lang/String;", 0, false));
|
3422
|
|
-
|
3423
|
|
- //FIXME Critical! Currently returning the same object!
|
3424
|
|
- //throw new UnsupportedOperationException("Cannot use projection functions yet!");
|
3425
|
|
- }
|
3426
|
|
- });
|
3427
|
|
-
|
3428
|
|
-
|
|
3435
|
+ javaWriter.loadObject(functionLocation);
|
|
3436
|
+ javaWriter.swap();
|
|
3437
|
+
|
|
3438
|
+ if(indexed) {
|
|
3439
|
+ for (int counterLocation : counterLocations) {
|
|
3440
|
+ javaWriter.loadInt(counterLocation);
|
|
3441
|
+ javaWriter.swap();
|
|
3442
|
+ }
|
|
3443
|
+ }
|
|
3444
|
+
|
|
3445
|
+ javaWriter.invokeInterface(context.getFunctionalInterface(expression.arguments.arguments[1].type));
|
|
3446
|
+ });
|
|
3447
|
+ }
|
|
3448
|
+
|
|
3449
|
+
|
3429
|
3450
|
javaWriter.label(end);
|
3430
|
3451
|
return;
|
3431
|
|
-
|
3432
|
|
- }
|
3433
|
|
- case ARRAY_CONSTRUCTOR_PROJECTED_INDEXED: {
|
3434
|
|
- ArrayTypeID type = (ArrayTypeID) expression.type.type;
|
3435
|
|
-
|
3436
|
|
- if (type.dimension == 1) {
|
3437
|
|
- // original = arguments[0] (this is an array)
|
3438
|
|
- // projector = arguments[1] (this is a lambda with 2 parameters)
|
3439
|
|
- // array = new T[original.length]
|
3440
|
|
- // for (int i = 0; i < array.length; i++)
|
3441
|
|
- // array[i] = projector(i, original[i]);
|
3442
|
|
- //
|
3443
|
|
- // NOTE: arguments[1] can be a FunctionExpression; this can be optimized by running it inline
|
3444
|
|
-
|
3445
|
|
- // TODO: implement in bytecode
|
3446
|
|
-
|
3447
|
|
- return;
|
3448
|
|
- } else {
|
3449
|
|
- // TODO: implement
|
3450
|
|
- throw new UnsupportedOperationException("Not yet supported!");
|
3451
|
|
- }
|
|
3452
|
+
|
3452
|
3453
|
}
|
3453
|
3454
|
case ARRAY_INDEXGET:
|
3454
|
3455
|
break;
|