speed up compound assignments

This commit is contained in:
Robert Muir 2016-06-15 14:38:37 -04:00
parent a4036b8069
commit 786c3b0fa8
2 changed files with 28 additions and 9 deletions

View File

@ -241,19 +241,21 @@ public final class DefBootstrap {
switch(flavor) { switch(flavor) {
case UNARY_OPERATOR: case UNARY_OPERATOR:
case SHIFT_OPERATOR: case SHIFT_OPERATOR:
if ((flags & OPERATOR_COMPOUND_ASSIGNMENT) != 0) {
return lookupGeneric(); // XXX: optimize better.
}
// shifts are treated as unary, as java allows long arguments without a cast (but bits are ignored) // shifts are treated as unary, as java allows long arguments without a cast (but bits are ignored)
return DefMath.lookupUnary(args[0].getClass(), name); MethodHandle unary = DefMath.lookupUnary(args[0].getClass(), name);
case BINARY_OPERATOR:
if ((flags & OPERATOR_COMPOUND_ASSIGNMENT) != 0) { if ((flags & OPERATOR_COMPOUND_ASSIGNMENT) != 0) {
return lookupGeneric(); // XXX: optimize better. unary = DefMath.cast(args[0].getClass(), unary);
} }
return unary;
case BINARY_OPERATOR:
if (args[0] == null || args[1] == null) { if (args[0] == null || args[1] == null) {
return lookupGeneric(); // can handle nulls, if supported return lookupGeneric(); // can handle nulls, casts if supported
} else { } else {
return DefMath.lookupBinary(args[0].getClass(), args[1].getClass(), name); MethodHandle binary = DefMath.lookupBinary(args[0].getClass(), args[1].getClass(), name);
if ((flags & OPERATOR_COMPOUND_ASSIGNMENT) != 0) {
binary = DefMath.cast(args[0].getClass(), binary);
}
return binary;
} }
default: throw new AssertionError(); default: throw new AssertionError();
} }

View File

@ -1133,7 +1133,7 @@ public class DefMath {
return handle; return handle;
} }
/** Returns an appropriate method handle for a binary operator, based only promotion of the LHS and RHS arguments */ /** Returns an appropriate method handle for a binary operator, based on promotion of the LHS and RHS arguments */
public static MethodHandle lookupBinary(Class<?> classA, Class<?> classB, String name) { public static MethodHandle lookupBinary(Class<?> classA, Class<?> classB, String name) {
MethodHandle handle = TYPE_OP_MAPPING.get(promote(promote(unbox(classA)), promote(unbox(classB)))).get(name); MethodHandle handle = TYPE_OP_MAPPING.get(promote(promote(unbox(classA)), promote(unbox(classB)))).get(name);
if (handle == null) { if (handle == null) {
@ -1156,6 +1156,9 @@ public class DefMath {
static Object dynamicCast(Object returnValue, Object lhs) { static Object dynamicCast(Object returnValue, Object lhs) {
if (lhs != null) { if (lhs != null) {
Class<?> c = lhs.getClass(); Class<?> c = lhs.getClass();
if (c == returnValue.getClass()) {
return returnValue;
}
if (c == Integer.class) { if (c == Integer.class) {
return getNumber(returnValue).intValue(); return getNumber(returnValue).intValue();
} else if (c == Long.class) { } else if (c == Long.class) {
@ -1212,4 +1215,18 @@ public class DefMath {
// combine: f(x,y) -> g(f(x,y), x, y); // combine: f(x,y) -> g(f(x,y), x, y);
return MethodHandles.foldArguments(cast, generic); return MethodHandles.foldArguments(cast, generic);
} }
/** Forces a cast to class A for target (only if types differ) */
public static MethodHandle cast(Class<?> classA, MethodHandle target) {
MethodType newType = MethodType.methodType(classA).unwrap();
MethodType targetType = MethodType.methodType(target.type().returnType()).unwrap();
if (newType.returnType() == targetType.returnType()) {
return target; // no conversion
}
// this is safe for our uses of it here only, because we change just the return value,
// the original method itself does all the type checks correctly.
return MethodHandles.explicitCastArguments(target, target.type().changeReturnType(newType.returnType()));
}
} }