diff graal/GraalCompiler/src/com/sun/c1x/opt/Canonicalizer.java @ 2509:16b9a8b5ad39

Renamings Runtime=>GraalRuntime and Compiler=>GraalCompiler
author Thomas Wuerthinger <thomas@wuerthinger.net>
date Wed, 27 Apr 2011 11:50:44 +0200
parents graal/Compiler/src/com/sun/c1x/opt/Canonicalizer.java@9ec15d6914ca
children f6125fb5bfbc
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/graal/GraalCompiler/src/com/sun/c1x/opt/Canonicalizer.java	Wed Apr 27 11:50:44 2011 +0200
@@ -0,0 +1,1372 @@
+/*
+ * Copyright (c) 2009, 2011, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+package com.sun.c1x.opt;
+
+import static com.sun.cri.bytecode.Bytecodes.*;
+
+import java.lang.reflect.*;
+import java.util.*;
+
+import com.sun.c1x.*;
+import com.sun.c1x.ir.*;
+import com.sun.c1x.util.*;
+import com.sun.cri.ci.*;
+import com.sun.cri.ri.*;
+
+/**
+ * The {@code Canonicalizer} reduces instructions to a canonical form by folding constants,
+ * putting constants on the right side of commutative operators, simplifying conditionals,
+ * and several other transformations.
+ *
+ * @author Ben L. Titzer
+ */
+public class Canonicalizer extends DefaultValueVisitor {
+
+    final RiRuntime runtime;
+    final RiMethod method;
+    final CiTarget target;
+    Value canonical;
+    List<Instruction> extra;
+
+    public Canonicalizer(RiRuntime runtime, RiMethod method, CiTarget target) {
+        this.runtime = runtime;
+        this.method = method;
+        this.target = target;
+    }
+
+    public Value canonicalize(Instruction original) {
+        this.canonical = original;
+        this.extra = null;
+        original.accept(this);
+        return this.canonical;
+    }
+
+    public List<Instruction> extra() {
+        return extra;
+    }
+
+    private <T extends Instruction> T addInstr(T x) {
+        if (extra == null) {
+            extra = new LinkedList<Instruction>();
+        }
+        extra.add(x);
+        return x;
+    }
+
+    private Constant intInstr(int v) {
+        return addInstr(Constant.forInt(v));
+    }
+
+    private Constant longInstr(long v) {
+        return addInstr(Constant.forLong(v));
+    }
+
+    private Constant wordInstr(long v) {
+        return addInstr(Constant.forWord(v));
+    }
+
+    private Value setCanonical(Value x) {
+        return canonical = x;
+    }
+
+    private Value setIntConstant(int val) {
+        return canonical = Constant.forInt(val);
+    }
+
+    private Value setConstant(CiConstant val) {
+        return canonical = new Constant(val);
+    }
+
+    private Value setBooleanConstant(boolean val) {
+        return canonical = Constant.forBoolean(val);
+    }
+
+    private Value setObjectConstant(Object val) {
+        if (C1XOptions.SupportObjectConstants) {
+            return canonical = Constant.forObject(val);
+        }
+        return canonical;
+    }
+
+    private Value setLongConstant(long val) {
+        return canonical = Constant.forLong(val);
+    }
+
+    private Value setFloatConstant(float val) {
+        return canonical = Constant.forFloat(val);
+    }
+
+    private Value setDoubleConstant(double val) {
+        return canonical = Constant.forDouble(val);
+    }
+
+    private Value setWordConstant(long val) {
+        return canonical = Constant.forDouble(val);
+    }
+
+    private void moveConstantToRight(Op2 x) {
+        if (x.x().isConstant() && isCommutative(x.opcode)) {
+            x.swapOperands();
+        }
+    }
+
+    private void visitOp2(Op2 i) {
+        final Value x = i.x();
+        final Value y = i.y();
+
+        if (x == y) {
+            // the left and right operands are the same value, try reducing some operations
+            switch (i.opcode) {
+                case ISUB: setIntConstant(0); return;
+                case LSUB: setLongConstant(0); return;
+                case IAND: // fall through
+                case LAND: // fall through
+                case IOR:  // fall through
+                case LOR: setCanonical(x); return;
+                case IXOR: setIntConstant(0); return;
+                case LXOR: setLongConstant(0); return;
+            }
+        }
+
+        CiKind kind = x.kind;
+        if (x.isConstant() && y.isConstant()) {
+            // both operands are constants, try constant folding
+            switch (kind) {
+                case Int: {
+                    Integer val = foldIntOp2(i.opcode, x.asConstant().asInt(), y.asConstant().asInt());
+                    if (val != null) {
+                        setIntConstant(val); // the operation was successfully folded to an int
+                        return;
+                    }
+                    break;
+                }
+                case Long: {
+                    Long val = foldLongOp2(i.opcode, x.asConstant().asLong(), y.asConstant().asLong());
+                    if (val != null) {
+                        setLongConstant(val); // the operation was successfully folded to a long
+                        return;
+                    }
+                    break;
+                }
+                case Float: {
+                    if (C1XOptions.CanonicalizeFloatingPoint) {
+                        // try to fold a floating point operation
+                        Float val = foldFloatOp2(i.opcode, x.asConstant().asFloat(), y.asConstant().asFloat());
+                        if (val != null) {
+                            setFloatConstant(val); // the operation was successfully folded to a float
+                            return;
+                        }
+                    }
+                    break;
+                }
+                case Double: {
+                    if (C1XOptions.CanonicalizeFloatingPoint) {
+                        // try to fold a floating point operation
+                        Double val = foldDoubleOp2(i.opcode, x.asConstant().asDouble(), y.asConstant().asDouble());
+                        if (val != null) {
+                            setDoubleConstant(val); // the operation was successfully folded to a double
+                            return;
+                        }
+                    }
+                    break;
+                }
+                case Word: {
+                    CiConstant val = runtime.foldWordOperation(i.opcode, new CiMethodInvokeArguments() {
+                        int argIndex;
+                        @Override
+                        public CiConstant nextArg() {
+                            if (argIndex == 0) {
+                                return x.asConstant();
+                            }
+                            if (argIndex == 1) {
+                                return y.asConstant();
+                            }
+                            argIndex++;
+                            return null;
+                        }
+                    });
+
+                    if (val != null) {
+                        setConstant(val); // the operation was successfully folded to a word
+                        return;
+                    }
+                    break;
+                }
+            }
+        }
+
+        // if there is a constant on the left and the operation is commutative, move it to the right
+        moveConstantToRight(i);
+
+        if (i.y().isConstant()) {
+            // the right side is a constant, try strength reduction
+            switch (kind) {
+                case Int: {
+                    if (reduceIntOp2(i, i.x(), i.y().asConstant().asInt()) != null) {
+                        return;
+                    }
+                    break;
+                }
+                case Long: {
+                    if (reduceLongOp2(i, i.x(), i.y().asConstant().asLong()) != null) {
+                        return;
+                    }
+                    break;
+                }
+                case Word: {
+                    if (reduceWordOp2(i, i.x(), i.y().asConstant().asLong()) != null) {
+                        return;
+                    }
+                    break;
+                }
+                // XXX: note that other cases are possible, but harder
+                // floating point operations need to be extra careful
+            }
+        }
+        assert Util.archKindsEqual(i, canonical);
+    }
+
+    private Value reduceIntOp2(Op2 original, Value x, int y) {
+        // attempt to reduce a binary operation with a constant on the right
+        int opcode = original.opcode;
+        switch (opcode) {
+            case IADD: return y == 0 ? setCanonical(x) : null;
+            case ISUB: return y == 0 ? setCanonical(x) : null;
+            case IMUL: {
+                if (y == 1) {
+                    return setCanonical(x);
+                }
+                if (y > 0 && (y & y - 1) == 0 && C1XOptions.CanonicalizeMultipliesToShifts) {
+                    // strength reduce multiply by power of 2 to shift operation
+                    return setCanonical(new ShiftOp(ISHL, x, intInstr(CiUtil.log2(y))));
+                }
+                return y == 0 ? setIntConstant(0) : null;
+            }
+            case IDIV: return y == 1 ? setCanonical(x) : null;
+            case IREM: return y == 1 ? setCanonical(x) : null;
+            case IAND: {
+                if (y == -1) {
+                    return setCanonical(x);
+                }
+                return y == 0 ? setIntConstant(0) : null;
+            }
+            case IOR: {
+                if (y == -1) {
+                    return setIntConstant(-1);
+                }
+                return y == 0 ? setCanonical(x) : null;
+            }
+            case IXOR: return y == 0 ? setCanonical(x) : null;
+            case ISHL: return reduceShift(false, opcode, IUSHR, x, y);
+            case ISHR: return reduceShift(false, opcode, 0, x, y);
+            case IUSHR: return reduceShift(false, opcode, ISHL, x, y);
+        }
+        return null;
+    }
+
+    private Value reduceShift(boolean islong, int opcode, int reverse, Value x, long y) {
+        int mod = islong ? 0x3f : 0x1f;
+        long shift = y & mod;
+        if (shift == 0) {
+            return setCanonical(x);
+        }
+        if (x instanceof ShiftOp) {
+            // this is a chained shift operation ((e shift e) shift K)
+            ShiftOp s = (ShiftOp) x;
+            if (s.y().isConstant()) {
+                long z = s.y().asConstant().asLong();
+                if (s.opcode == opcode) {
+                    // this is a chained shift operation (e >> C >> K)
+                    y = y + z;
+                    shift = y & mod;
+                    if (shift == 0) {
+                        return setCanonical(s.x());
+                    }
+                    // reduce to (e >> (C + K))
+                    return setCanonical(new ShiftOp(opcode, s.x(), intInstr((int) shift)));
+                }
+                if (s.opcode == reverse && y == z) {
+                    // this is a chained shift of the form (e >> K << K)
+                    if (islong) {
+                        long mask = -1;
+                        if (opcode == LUSHR) {
+                            mask = mask >>> y;
+                        } else {
+                            mask = mask << y;
+                        }
+                        // reduce to (e & mask)
+                        return setCanonical(new LogicOp(LAND, s.x(), longInstr(mask)));
+                    } else {
+                        int mask = -1;
+                        if (opcode == IUSHR) {
+                            mask = mask >>> y;
+                        } else {
+                            mask = mask << y;
+                        }
+                        return setCanonical(new LogicOp(IAND, s.x(), intInstr(mask)));
+                    }
+                }
+            }
+        }
+        if (y != shift) {
+            // (y & mod) != y
+            return setCanonical(new ShiftOp(opcode, x, intInstr((int) shift)));
+        }
+        return null;
+    }
+
+    private Value reduceLongOp2(Op2 original, Value x, long y) {
+        // attempt to reduce a binary operation with a constant on the right
+        int opcode = original.opcode;
+        switch (opcode) {
+            case LADD: return y == 0 ? setCanonical(x) : null;
+            case LSUB: return y == 0 ? setCanonical(x) : null;
+            case LMUL: {
+                if (y == 1) {
+                    return setCanonical(x);
+                }
+                if (y > 0 && (y & y - 1) == 0 && C1XOptions.CanonicalizeMultipliesToShifts) {
+                    // strength reduce multiply by power of 2 to shift operation
+                    return setCanonical(new ShiftOp(LSHL, x, intInstr(CiUtil.log2(y))));
+                }
+                return y == 0 ? setLongConstant(0) : null;
+            }
+            case LDIV: return y == 1 ? setCanonical(x) : null;
+            case LREM: return y == 1 ? setCanonical(x) : null;
+            case LAND: {
+                if (y == -1) {
+                    return setCanonical(x);
+                }
+                return y == 0 ? setLongConstant(0) : null;
+            }
+            case LOR: {
+                if (y == -1) {
+                    return setLongConstant(-1);
+                }
+                return y == 0 ? setCanonical(x) : null;
+            }
+            case LXOR: return y == 0 ? setCanonical(x) : null;
+            case LSHL: return reduceShift(true, opcode, LUSHR, x, y);
+            case LSHR: return reduceShift(true, opcode, 0, x, y);
+            case LUSHR: return reduceShift(true, opcode, LSHL, x, y);
+        }
+        return null;
+    }
+
+    private Value reduceWordOp2(Op2 original, Value x, long y) {
+        if (y == 0) {
+            // Defer to arithmetic exception at runtime
+            return null;
+        }
+        // attempt to reduce a binary operation with a constant on the right
+        int opcode = original.opcode;
+        switch (opcode) {
+            case WDIVI:
+            case WDIV: {
+                if (y == 1) {
+                    return setCanonical(x);
+                }
+                if (CiUtil.isPowerOf2(y)) {
+                    return setCanonical(new ShiftOp(target.arch.is64bit() ? LUSHR : IUSHR, x, intInstr(CiUtil.log2(y))));
+                }
+                break;
+            }
+            case WREMI: {
+                if (y == 1) {
+                    return setCanonical(intInstr(0));
+                }
+                if (CiUtil.isPowerOf2(y)) {
+                    int mask = (int) y - 1;
+                    if (target.arch.is64bit()) {
+                        Convert l2i = new Convert(L2I, x, CiKind.Int);
+                        addInstr(l2i);
+                        return setCanonical(new LogicOp(IAND, l2i, intInstr(mask)));
+                    }
+                    return setCanonical(new LogicOp(CiKind.Int, IAND, x, intInstr(mask)));
+                }
+                break;
+            }
+            case WREM: {
+                if (y == 1) {
+                    return setCanonical(wordInstr(0));
+                }
+                if (CiUtil.isPowerOf2(y)) {
+                    if (target.arch.is64bit()) {
+                        long mask = y - 1L;
+                        return setCanonical(new LogicOp(LAND, x, longInstr(mask)));
+                    }
+                    int mask = (int) y - 1;
+                    return setCanonical(new LogicOp(IAND, x, intInstr(mask)));
+                }
+                break;
+            }
+        }
+        return null;
+    }
+
+    private boolean inCurrentBlock(Value x) {
+        if (x instanceof Instruction) {
+            Instruction i = (Instruction) x;
+            int max = 4; // XXX: anything special about 4? seems like a tunable heuristic
+            while (max > 0 && i != null && !(i instanceof BlockEnd)) {
+                i = i.next();
+                max--;
+            }
+            return i == null;
+        }
+        return true;
+    }
+
+    private Value eliminateNarrowing(CiKind kind, Convert c) {
+        Value nv = null;
+        switch (c.opcode) {
+            case I2B:
+                if (kind == CiKind.Byte) {
+                    nv = c.value();
+                }
+                break;
+            case I2S:
+                if (kind == CiKind.Short || kind == CiKind.Byte) {
+                    nv = c.value();
+                }
+                break;
+            case I2C:
+                if (kind == CiKind.Char || kind == CiKind.Byte) {
+                    nv = c.value();
+                }
+                break;
+        }
+        return nv;
+    }
+
+    @Override
+    public void visitLoadField(LoadField i) {
+        if (!i.isLoaded() || !C1XOptions.CanonicalizeConstantFields) {
+            return;
+        }
+        if (i.isStatic()) {
+            RiField field = i.field();
+            CiConstant value = field.constantValue(null);
+            if (value != null) {
+                if (method.isClassInitializer()) {
+                    // don't do canonicalization in the <clinit> method
+                    return;
+                }
+                setConstant(value);
+            }
+        } else {
+            RiField field = i.field();
+            if (i.object().isConstant()) {
+                CiConstant value = field.constantValue(i.object().asConstant());
+                if (value != null) {
+                    setConstant(value);
+                }
+            }
+        }
+    }
+
+    @Override
+    public void visitStoreField(StoreField i) {
+        if (C1XOptions.CanonicalizeNarrowingInStores) {
+            // Eliminate narrowing conversions emitted by javac which are unnecessary when
+            // writing the value to a field that is packed
+            Value v = i.value();
+            if (v instanceof Convert) {
+                Value nv = eliminateNarrowing(i.field().kind(), (Convert) v);
+                // limit this optimization to the current basic block
+                // XXX: why is this limited to the current block?
+                if (nv != null && inCurrentBlock(v)) {
+                    setCanonical(new StoreField(i.object(), i.field(), nv, i.isStatic(),
+                                                i.stateBefore(), i.isLoaded()));
+                }
+            }
+        }
+    }
+
+    @Override
+    public void visitArrayLength(ArrayLength i) {
+        // we can compute the length of the array statically if the object
+        // is a NewArray of a constant, or if the object is a constant reference
+        // (either by itself or loaded from a constant value field)
+        Value array = i.array();
+        if (array instanceof NewArray) {
+            // the array is a NewArray; check if it has a constant length
+            NewArray newArray = (NewArray) array;
+            Value length = newArray.length();
+            if (length instanceof Constant) {
+                // note that we don't use the Constant instruction itself
+                // as that would cause problems with liveness later
+                int actualLength = length.asConstant().asInt();
+                setIntConstant(actualLength);
+            }
+        } else if (array instanceof LoadField) {
+            // the array is a load of a field; check if it is a constant
+            LoadField load = (LoadField) array;
+            CiConstant cons = load.constantValue();
+            if (cons != null && cons.isNonNull()) {
+                setIntConstant(runtime.getArrayLength(cons));
+            }
+        } else if (array.isConstant()) {
+            // the array itself is a constant object reference
+            CiConstant obj = array.asConstant();
+            if (obj.isNonNull()) {
+                setIntConstant(runtime.getArrayLength(obj));
+            }
+        }
+    }
+
+    @Override
+    public void visitStoreIndexed(StoreIndexed i) {
+        Value array = i.array();
+        Value value = i.value();
+        if (C1XOptions.CanonicalizeNarrowingInStores) {
+            // Eliminate narrowing conversions emitted by javac which are unnecessary when
+            // writing the value to an array (which is packed)
+            Value v = value;
+            if (v instanceof Convert) {
+                Value nv = eliminateNarrowing(i.elementKind(), (Convert) v);
+                if (nv != null && inCurrentBlock(v)) {
+                    setCanonical(new StoreIndexed(array, i.index(), i.length(), i.elementKind(), nv, i.stateBefore()));
+                }
+            }
+        }
+        if (C1XOptions.CanonicalizeArrayStoreChecks && i.elementKind() == CiKind.Object) {
+            if (value.isNullConstant()) {
+                i.eliminateStoreCheck();
+            } else {
+                RiType exactType = array.exactType();
+                if (exactType != null && exactType.isResolved()) {
+                    if (exactType.componentType().superType() == null) {
+                        // the exact type of the array is Object[] => no check is necessary
+                        i.eliminateStoreCheck();
+                    } else {
+                        RiType declaredType = value.declaredType();
+                        if (declaredType != null && declaredType.isResolved() && declaredType.isSubtypeOf(exactType.componentType())) {
+                            // the value being stored has a known type
+                            i.eliminateStoreCheck();
+                        }
+                    }
+                }
+            }
+        }
+        if (i.index().isConstant() && i.length() != null && i.length().isConstant()) {
+            int index = i.index().asConstant().asInt();
+            if (index >= 0 && index < i.length().asConstant().asInt()) {
+                i.eliminateBoundsCheck();
+            }
+        }
+    }
+
+    @Override
+    public void visitLoadIndexed(LoadIndexed i) {
+        if (i.index().isConstant() && i.length() != null && i.length().isConstant()) {
+            int index = i.index().asConstant().asInt();
+            if (index >= 0 && index < i.length().asConstant().asInt()) {
+                i.eliminateBoundsCheck();
+            }
+        }
+    }
+
+    @Override
+    public void visitNegateOp(NegateOp i) {
+        CiKind vt = i.x().kind;
+        Value v = i.x();
+        if (i.x().isConstant()) {
+            switch (vt) {
+                case Int: setIntConstant(-v.asConstant().asInt()); break;
+                case Long: setLongConstant(-v.asConstant().asLong()); break;
+                case Float: setFloatConstant(-v.asConstant().asFloat()); break;
+                case Double: setDoubleConstant(-v.asConstant().asDouble()); break;
+            }
+        }
+        assert vt == canonical.kind;
+    }
+
+    @Override
+    public void visitArithmeticOp(ArithmeticOp i) {
+        visitOp2(i);
+    }
+
+    @Override
+    public void visitShiftOp(ShiftOp i) {
+        visitOp2(i);
+    }
+
+    @Override
+    public void visitLogicOp(LogicOp i) {
+        visitOp2(i);
+    }
+
+    @Override
+    public void visitCompareOp(CompareOp i) {
+        if (i.kind.isVoid()) {
+            return;
+        }
+        // we can reduce a compare op if the two inputs are the same,
+        // or if both are constants
+        Value x = i.x();
+        Value y = i.y();
+        CiKind xt = x.kind;
+        if (x == y) {
+            // x and y are generated by the same instruction
+            switch (xt) {
+                case Long: setIntConstant(0); return;
+                case Float:
+                    if (x.isConstant()) {
+                        float xval = x.asConstant().asFloat(); // get the actual value of x (and y since x == y)
+                        Integer val = foldFloatCompare(i.opcode, xval, xval);
+                        assert val != null : "invalid opcode in float compare op";
+                        setIntConstant(val);
+                        return;
+                    }
+                    break;
+                case Double:
+                    if (x.isConstant()) {
+                        double xval = x.asConstant().asDouble(); // get the actual value of x (and y since x == y)
+                        Integer val = foldDoubleCompare(i.opcode, xval, xval);
+                        assert val != null : "invalid opcode in double compare op";
+                        setIntConstant(val);
+                        return;
+                    }
+                    break;
+                // note that there are no integer CompareOps
+            }
+        }
+        if (x.isConstant() && y.isConstant()) {
+            // both x and y are constants
+            switch (xt) {
+                case Long:
+                    setIntConstant(foldLongCompare(x.asConstant().asLong(), y.asConstant().asLong()));
+                    break;
+                case Float: {
+                    Integer val = foldFloatCompare(i.opcode, x.asConstant().asFloat(), y.asConstant().asFloat());
+                    assert val != null : "invalid opcode in float compare op";
+                    setIntConstant(val);
+                    break;
+                }
+                case Double: {
+                    Integer val = foldDoubleCompare(i.opcode, x.asConstant().asDouble(), y.asConstant().asDouble());
+                    assert val != null : "invalid opcode in float compare op";
+                    setIntConstant(val);
+                    break;
+                }
+            }
+        }
+        assert Util.archKindsEqual(i, canonical);
+    }
+
+    @Override
+    public void visitIfOp(IfOp i) {
+        moveConstantToRight(i);
+    }
+
+    @Override
+    public void visitConvert(Convert i) {
+        Value v = i.value();
+        if (v.isConstant()) {
+            // fold conversions between primitive types
+            // Checkstyle: stop
+            switch (i.opcode) {
+                case I2B: setIntConstant   ((byte)   v.asConstant().asInt()); return;
+                case I2S: setIntConstant   ((short)  v.asConstant().asInt()); return;
+                case I2C: setIntConstant   ((char)   v.asConstant().asInt()); return;
+                case I2L: setLongConstant  (         v.asConstant().asInt()); return;
+                case I2F: setFloatConstant (         v.asConstant().asInt()); return;
+                case L2I: setIntConstant   ((int)    v.asConstant().asLong()); return;
+                case L2F: setFloatConstant (         v.asConstant().asLong()); return;
+                case L2D: setDoubleConstant(         v.asConstant().asLong()); return;
+                case F2D: setDoubleConstant(         v.asConstant().asFloat()); return;
+                case F2I: setIntConstant   ((int)    v.asConstant().asFloat()); return;
+                case F2L: setLongConstant  ((long)   v.asConstant().asFloat()); return;
+                case D2F: setFloatConstant ((float)  v.asConstant().asDouble()); return;
+                case D2I: setIntConstant   ((int)    v.asConstant().asDouble()); return;
+                case D2L: setLongConstant  ((long)   v.asConstant().asDouble()); return;
+            }
+            // Checkstyle: resume
+        }
+
+        CiKind kind = CiKind.Illegal;
+        if (v instanceof LoadField) {
+            // remove redundant conversions from field loads of the correct type
+            kind = ((LoadField) v).field().kind();
+        } else if (v instanceof LoadIndexed) {
+            // remove redundant conversions from array loads of the correct type
+            kind = ((LoadIndexed) v).elementKind();
+        } else if (v instanceof Convert) {
+            // remove chained redundant conversions
+            Convert c = (Convert) v;
+            switch (c.opcode) {
+                case I2B: kind = CiKind.Byte; break;
+                case I2S: kind = CiKind.Short; break;
+                case I2C: kind = CiKind.Char; break;
+            }
+        }
+
+        if (kind != CiKind.Illegal) {
+            // if any of the above matched
+            switch (i.opcode) {
+                case I2B:
+                    if (kind == CiKind.Byte) {
+                        setCanonical(v);
+                    }
+                    break;
+                case I2S:
+                    if (kind == CiKind.Byte || kind == CiKind.Short) {
+                        setCanonical(v);
+                    }
+                    break;
+                case I2C:
+                    if (kind == CiKind.Char) {
+                        setCanonical(v);
+                    }
+                    break;
+            }
+        }
+
+        if (v instanceof Op2) {
+            // check if the operation was IAND with a constant; it may have narrowed the value already
+            Op2 op = (Op2) v;
+            // constant should be on right hand side if there is one
+            if (op.opcode == IAND && op.y().isConstant()) {
+                int safebits = 0;
+                int mask = op.y().asConstant().asInt();
+                switch (i.opcode) {
+                    case I2B: safebits = 0x7f; break;
+                    case I2S: safebits = 0x7fff; break;
+                    case I2C: safebits = 0xffff; break;
+                }
+                if (safebits != 0 && (mask & ~safebits) == 0) {
+                    // the mask already cleared all the upper bits necessary.
+                    setCanonical(v);
+                }
+            }
+        }
+    }
+
+    @Override
+    public void visitNullCheck(NullCheck i) {
+        Value o = i.object();
+        if (o.isNonNull()) {
+            // if the instruction producing the object was a new, no check is necessary
+            setCanonical(o);
+        } else if (o.isConstant()) {
+            // if the object is a constant, check if it is nonnull
+            CiConstant c = o.asConstant();
+            if (c.kind.isObject() && !c.isNull()) {
+                setCanonical(o);
+            }
+        }
+    }
+
+    @Override
+    public void visitInvoke(Invoke i) {
+        if (C1XOptions.CanonicalizeFoldableMethods) {
+            RiMethod method = i.target();
+            if (method.isResolved()) {
+                // only try to fold resolved method invocations
+                CiConstant result = foldInvocation(runtime, i.target(), i.arguments());
+                if (result != null) {
+                    // folding was successful
+                    setCanonical(new Constant(result));
+                }
+            }
+        }
+    }
+
+    @Override
+    public void visitCheckCast(CheckCast i) {
+        // we can remove a redundant check cast if it is an object constant or the exact type is known
+        if (i.targetClass().isResolved()) {
+            Value o = i.object();
+            RiType type = o.exactType();
+            if (type == null) {
+                type = o.declaredType();
+            }
+            if (type != null && type.isResolved() && type.isSubtypeOf(i.targetClass())) {
+                // cast is redundant if exact type or declared type is already a subtype of the target type
+                setCanonical(o);
+            }
+            if (o.isConstant()) {
+                final CiConstant obj = o.asConstant();
+                if (obj.isNull()) {
+                    // checkcast of null is null
+                    setCanonical(o);
+                } else if (C1XOptions.CanonicalizeObjectCheckCast) {
+                    if (i.targetClass().isInstance(obj)) {
+                        // fold the cast if it will succeed
+                        setCanonical(o);
+                    }
+                }
+            }
+        }
+    }
+
+    @Override
+    public void visitInstanceOf(InstanceOf i) {
+        // we can fold an instanceof if it is an object constant or the exact type is known
+        if (i.targetClass().isResolved()) {
+            Value o = i.object();
+            RiType exact = o.exactType();
+            if (exact != null && exact.isResolved() && o.isNonNull()) {
+                setIntConstant(exact.isSubtypeOf(i.targetClass()) ? 1 : 0);
+            } else if (o.isConstant()) {
+                final CiConstant obj = o.asConstant();
+                if (obj.isNull()) {
+                    // instanceof of null is false
+                    setIntConstant(0);
+                } else if (C1XOptions.CanonicalizeObjectInstanceOf) {
+                    // fold the instanceof test
+                    setIntConstant(i.targetClass().isInstance(obj) ? 1 : 0);
+                }
+            }
+        }
+    }
+
+    @Override
+    public void visitIntrinsic(Intrinsic i) {
+        if (!C1XOptions.CanonicalizeIntrinsics) {
+            return;
+        }
+        if (!foldIntrinsic(i)) {
+            // folding did not work, try recognizing special intrinsics
+            reduceIntrinsic(i);
+        }
+        assert Util.archKindsEqual(i, canonical);
+    }
+
+    private void reduceIntrinsic(Intrinsic i) {
+        Value[] args = i.arguments();
+        C1XIntrinsic intrinsic = i.intrinsic();
+        if (intrinsic == C1XIntrinsic.java_lang_Class$isInstance) {
+            // try to convert a call to Class.isInstance() into an InstanceOf
+            RiType type = getTypeOf(args[0]);
+            if (type != null) {
+                setCanonical(new InstanceOf(type, Constant.forObject(type.getEncoding(RiType.Representation.TypeInfo)), args[1], i.stateBefore()));
+                return;
+            }
+        }
+        if (intrinsic == C1XIntrinsic.java_lang_reflect_Array$newArray) {
+            // try to convert a call to Array.newInstance() into a NewObjectArray or NewTypeArray
+            RiType type = getTypeOf(args[0]);
+            if (type != null) {
+                if (type.kind() == CiKind.Object) {
+                    setCanonical(new NewObjectArray(type, args[1], i.stateBefore()));
+                } else {
+                    RiType elementType = runtime.asRiType(type.kind());
+                    setCanonical(new NewTypeArray(args[1], elementType, i.stateBefore()));
+                }
+                return;
+            }
+        }
+        assert Util.archKindsEqual(i, canonical);
+    }
+
+    private boolean foldIntrinsic(Intrinsic i) {
+        Value[] args = i.arguments();
+        for (Value arg : args) {
+            if (arg != null && !arg.isConstant()) {
+                // one input is not constant, give up
+                return true;
+            }
+        }
+        switch (i.intrinsic()) {
+            // do not use reflection here due to efficiency and potential bootstrap problems
+            case java_lang_Object$hashCode: {
+                Object object = argAsObject(args, 0);
+                if (object != null) {
+                    setIntConstant(System.identityHashCode(object));
+                }
+                return true;
+            }
+            case java_lang_Object$getClass: {
+                Object object = argAsObject(args, 0);
+                if (object != null) {
+                    setObjectConstant(object.getClass());
+                }
+                return true;
+            }
+
+            // java.lang.Class
+            case java_lang_Class$isAssignableFrom: {
+                Class<?> javaClass = argAsClass(args, 0);
+                Class<?> otherClass = argAsClass(args, 1);
+                if (javaClass != null && otherClass != null) {
+                    setBooleanConstant(javaClass.isAssignableFrom(otherClass));
+                }
+                return true;
+            }
+            case java_lang_Class$isInstance: {
+                Class<?> javaClass = argAsClass(args, 0);
+                Object object = argAsObject(args, 1);
+                if (javaClass != null && object != null) {
+                    setBooleanConstant(javaClass.isInstance(object));
+                }
+                return true;
+            }
+            case java_lang_Class$getModifiers: {
+                Class<?> javaClass = argAsClass(args, 0);
+                if (javaClass != null) {
+                    setIntConstant(javaClass.getModifiers());
+                }
+                return true;
+            }
+            case java_lang_Class$isInterface: {
+                Class<?> javaClass = argAsClass(args, 0);
+                if (javaClass != null) {
+                    setBooleanConstant(javaClass.isInterface());
+                }
+                return true;
+            }
+            case java_lang_Class$isArray: {
+                Class<?> javaClass = argAsClass(args, 0);
+                if (javaClass != null) {
+                    setBooleanConstant(javaClass.isArray());
+                }
+                return true;
+            }
+            case java_lang_Class$isPrimitive: {
+                Class<?> javaClass = argAsClass(args, 0);
+                if (javaClass != null) {
+                    setBooleanConstant(javaClass.isPrimitive());
+                }
+                return true;
+            }
+            case java_lang_Class$getSuperclass: {
+                Class<?> javaClass = argAsClass(args, 0);
+                if (javaClass != null) {
+                    setObjectConstant(javaClass.getSuperclass());
+                }
+                return true;
+            }
+            case java_lang_Class$getComponentType: {
+                Class<?> javaClass = argAsClass(args, 0);
+                if (javaClass != null) {
+                    setObjectConstant(javaClass.getComponentType());
+                }
+                return true;
+            }
+
+            // java.lang.Math
+            case java_lang_Math$abs:   setDoubleConstant(Math.abs(argAsDouble(args, 0))); return true;
+            case java_lang_Math$sin:   setDoubleConstant(Math.sin(argAsDouble(args, 0))); return true;
+            case java_lang_Math$cos:   setDoubleConstant(Math.cos(argAsDouble(args, 0))); return true;
+            case java_lang_Math$tan:   setDoubleConstant(Math.tan(argAsDouble(args, 0))); return true;
+            case java_lang_Math$atan2: setDoubleConstant(Math.atan2(argAsDouble(args, 0), argAsDouble(args, 2))); return true;
+            case java_lang_Math$sqrt:  setDoubleConstant(Math.sqrt(argAsDouble(args, 0))); return true;
+            case java_lang_Math$log:   setDoubleConstant(Math.log(argAsDouble(args, 0))); return true;
+            case java_lang_Math$log10: setDoubleConstant(Math.log10(argAsDouble(args, 0))); return true;
+            case java_lang_Math$pow:   setDoubleConstant(Math.pow(argAsDouble(args, 0), argAsDouble(args, 2))); return true;
+            case java_lang_Math$exp:   setDoubleConstant(Math.exp(argAsDouble(args, 0))); return true;
+            case java_lang_Math$min:   setIntConstant(Math.min(argAsInt(args, 0), argAsInt(args, 1))); return true;
+            case java_lang_Math$max:   setIntConstant(Math.max(argAsInt(args, 0), argAsInt(args, 1))); return true;
+
+            // java.lang.Float
+            case java_lang_Float$floatToRawIntBits: setIntConstant(Float.floatToRawIntBits(argAsFloat(args, 0))); return true;
+            case java_lang_Float$floatToIntBits: setIntConstant(Float.floatToIntBits(argAsFloat(args, 0))); return true;
+            case java_lang_Float$intBitsToFloat: setFloatConstant(Float.intBitsToFloat(argAsInt(args, 0))); return true;
+
+            // java.lang.Double
+            case java_lang_Double$doubleToRawLongBits: setLongConstant(Double.doubleToRawLongBits(argAsDouble(args, 0))); return true;
+            case java_lang_Double$doubleToLongBits: setLongConstant(Double.doubleToLongBits(argAsDouble(args, 0))); return true;
+            case java_lang_Double$longBitsToDouble: setDoubleConstant(Double.longBitsToDouble(argAsLong(args, 0))); return true;
+
+            // java.lang.Integer
+            case java_lang_Integer$bitCount: setIntConstant(Integer.bitCount(argAsInt(args, 0))); return true;
+            case java_lang_Integer$reverseBytes: setIntConstant(Integer.reverseBytes(argAsInt(args, 0))); return true;
+
+            // java.lang.Long
+            case java_lang_Long$bitCount: setIntConstant(Long.bitCount(argAsLong(args, 0))); return true;
+            case java_lang_Long$reverseBytes: setLongConstant(Long.reverseBytes(argAsLong(args, 0))); return true;
+
+            // java.lang.System
+            case java_lang_System$identityHashCode: {
+                Object object = argAsObject(args, 0);
+                if (object != null) {
+                    setIntConstant(System.identityHashCode(object));
+                }
+                return true;
+            }
+
+            // java.lang.reflect.Array
+            case java_lang_reflect_Array$getLength: {
+                Object object = argAsObject(args, 0);
+                if (object != null && object.getClass().isArray()) {
+                    setIntConstant(Array.getLength(object));
+                }
+                return true;
+            }
+        }
+        return false;
+    }
+
+    @Override
+    public void visitIf(If i) {
+        if (i.x().isConstant()) {
+            // move constant to the right
+            i.swapOperands();
+        }
+        Value l = i.x();
+        Value r = i.y();
+
+        if (l == r && !l.kind.isFloatOrDouble()) {
+            // this is a comparison of x op x
+            // No opt for float/double due to NaN case
+            reduceReflexiveIf(i);
+            return;
+        }
+
+        CiKind rt = r.kind;
+
+        Condition ifcond = i.condition();
+        if (l.isConstant() && r.isConstant()) {
+            // fold comparisons between constants and convert to Goto
+            Boolean result = ifcond.foldCondition(l.asConstant(), r.asConstant(), runtime);
+            if (result != null) {
+                setCanonical(new Goto(i.successor(result), i.stateAfter(), i.isSafepoint()));
+                return;
+            }
+        }
+
+        if (r.isConstant() && rt.isInt()) {
+            // attempt to reduce comparisons with constant on right side
+            if (l instanceof CompareOp) {
+                // attempt to reduce If ((a cmp b) op const)
+                reduceIfCompareOpConstant(i, r.asConstant());
+            }
+        }
+
+        if (isNullConstant(r) && l.isNonNull()) {
+            // this is a comparison of null against something that is not null
+            if (ifcond == Condition.EQ) {
+                // new() == null is always false
+                setCanonical(new Goto(i.falseSuccessor(), i.stateAfter(), i.isSafepoint()));
+            } else if (ifcond == Condition.NE) {
+                // new() != null is always true
+                setCanonical(new Goto(i.trueSuccessor(), i.stateAfter(), i.isSafepoint()));
+            }
+        }
+    }
+
+    private boolean isNullConstant(Value r) {
+        return r.isConstant() && r.asConstant().isNull();
+    }
+
+    private void reduceIfCompareOpConstant(If i, CiConstant rtc) {
+        Condition ifcond = i.condition();
+        Value l = i.x();
+        CompareOp cmp = (CompareOp) l;
+        boolean unorderedIsLess = cmp.opcode == FCMPL || cmp.opcode == DCMPL;
+        BlockBegin lssSucc = i.successor(ifcond.foldCondition(CiConstant.forInt(-1), rtc, runtime));
+        BlockBegin eqlSucc = i.successor(ifcond.foldCondition(CiConstant.forInt(0), rtc, runtime));
+        BlockBegin gtrSucc = i.successor(ifcond.foldCondition(CiConstant.forInt(1), rtc, runtime));
+        BlockBegin nanSucc = unorderedIsLess ? lssSucc : gtrSucc;
+        // Note: At this point all successors (lssSucc, eqlSucc, gtrSucc, nanSucc) are
+        //       equal to x->tsux() or x->fsux(). Furthermore, nanSucc equals either
+        //       lssSucc or gtrSucc.
+        if (lssSucc == eqlSucc && eqlSucc == gtrSucc) {
+            // all successors identical => simplify to: Goto
+            setCanonical(new Goto(lssSucc, i.stateAfter(), i.isSafepoint()));
+        } else {
+            // two successors differ and two successors are the same => simplify to: If (x cmp y)
+            // determine new condition & successors
+            Condition cond;
+            BlockBegin tsux;
+            BlockBegin fsux;
+            if (lssSucc == eqlSucc) {
+                cond = Condition.LE;
+                tsux = lssSucc;
+                fsux = gtrSucc;
+            } else if (lssSucc == gtrSucc) {
+                cond = Condition.NE;
+                tsux = lssSucc;
+                fsux = eqlSucc;
+            } else if (eqlSucc == gtrSucc) {
+                cond = Condition.GE;
+                tsux = eqlSucc;
+                fsux = lssSucc;
+            } else {
+                throw Util.shouldNotReachHere();
+            }
+            // TODO: the state after is incorrect here: should it be preserved from the original if?
+            If canon = new If(cmp.x(), cond, nanSucc == tsux, cmp.y(), tsux, fsux, cmp.stateBefore(), i.isSafepoint());
+            if (cmp.x() == cmp.y()) {
+                // re-canonicalize the new if
+                visitIf(canon);
+            } else {
+                setCanonical(canon);
+            }
+        }
+    }
+
+    private void reduceReflexiveIf(If i) {
+        // simplify reflexive comparisons If (x op x) to Goto
+        BlockBegin succ;
+        switch (i.condition()) {
+            case EQ: succ = i.successor(true); break;
+            case NE: succ = i.successor(false); break;
+            case LT: succ = i.successor(false); break;
+            case LE: succ = i.successor(true); break;
+            case GT: succ = i.successor(false); break;
+            case GE: succ = i.successor(true); break;
+            default:
+                throw Util.shouldNotReachHere();
+        }
+        setCanonical(new Goto(succ, i.stateAfter(), i.isSafepoint()));
+    }
+
+    @Override
+    public void visitTableSwitch(TableSwitch i) {
+        Value v = i.value();
+        if (v.isConstant()) {
+            // fold a table switch over a constant by replacing it with a goto
+            int val = v.asConstant().asInt();
+            BlockBegin succ = i.defaultSuccessor();
+            if (val >= i.lowKey() && val <= i.highKey()) {
+                succ = i.successors().get(val - i.lowKey());
+            }
+            setCanonical(new Goto(succ, i.stateAfter(), i.isSafepoint()));
+            return;
+        }
+        int max = i.numberOfCases();
+        if (max == 0) {
+            // replace switch with Goto
+            if (v instanceof Instruction) {
+                // TODO: is it necessary to add the instruction explicitly?
+                addInstr((Instruction) v);
+            }
+            setCanonical(new Goto(i.defaultSuccessor(), i.stateAfter(), i.isSafepoint()));
+            return;
+        }
+        if (max == 1) {
+            // replace switch with If
+            Constant key = intInstr(i.lowKey());
+            If newIf = new If(v, Condition.EQ, false, key, i.successors().get(0), i.defaultSuccessor(), null, i.isSafepoint());
+            newIf.setStateAfter(i.stateAfter());
+            setCanonical(newIf);
+        }
+    }
+
+    @Override
+    public void visitLookupSwitch(LookupSwitch i) {
+        Value v = i.value();
+        if (v.isConstant()) {
+            // fold a lookup switch over a constant by replacing it with a goto
+            int val = v.asConstant().asInt();
+            BlockBegin succ = i.defaultSuccessor();
+            for (int j = 0; j < i.numberOfCases(); j++) {
+                if (val == i.keyAt(j)) {
+                    succ = i.successors().get(j);
+                    break;
+                }
+            }
+            setCanonical(new Goto(succ, i.stateAfter(), i.isSafepoint()));
+            return;
+        }
+        int max = i.numberOfCases();
+        if (max == 0) {
+            // replace switch with Goto
+            if (v instanceof Instruction) {
+                addInstr((Instruction) v); // the value expression may produce side effects
+            }
+            setCanonical(new Goto(i.defaultSuccessor(), i.stateAfter(), i.isSafepoint()));
+            return;
+        }
+        if (max == 1) {
+            // replace switch with If
+            Constant key = intInstr(i.keyAt(0));
+            If newIf = new If(v, Condition.EQ, false, key, i.successors().get(0), i.defaultSuccessor(), null, i.isSafepoint());
+            newIf.setStateAfter(i.stateAfter());
+            setCanonical(newIf);
+        }
+    }
+
+    private void visitUnsafeRawOp(UnsafeRawOp i) {
+        if (i.base() instanceof ArithmeticOp) {
+            // if the base is an arithmetic op, try reducing
+            ArithmeticOp root = (ArithmeticOp) i.base();
+            if (!root.isLive() && root.opcode == LADD) {
+                // match unsafe(x + y) if the x + y is not pinned
+                // try reducing (x + y) and (y + x)
+                Value y = root.y();
+                Value x = root.x();
+                if (reduceRawOp(i, x, y) || reduceRawOp(i, y, x)) {
+                    // the operation was reduced
+                    return;
+                }
+                if (y instanceof Convert) {
+                    // match unsafe(x + (long) y)
+                    Convert convert = (Convert) y;
+                    if (convert.opcode == I2L && convert.value().kind.isInt()) {
+                        // the conversion is redundant
+                        setUnsafeRawOp(i, x, convert.value(), 0);
+                    }
+                }
+            }
+        }
+    }
+
+    private boolean reduceRawOp(UnsafeRawOp i, Value base, Value index) {
+        if (index instanceof Convert) {
+            // skip any conversion operations
+            index = ((Convert) index).value();
+        }
+        if (index instanceof ShiftOp) {
+            // try to match the index as a shift by a constant
+            ShiftOp shift = (ShiftOp) index;
+            CiKind st = shift.y().kind;
+            if (shift.y().isConstant() && st.isInt()) {
+                int val = shift.y().asConstant().asInt();
+                switch (val) {
+                    case 0: // fall through
+                    case 1: // fall through
+                    case 2: // fall through
+                    case 3: return setUnsafeRawOp(i, base, shift.x(), val);
+                }
+            }
+        }
+        if (index instanceof ArithmeticOp) {
+            // try to match the index as a multiply by a constant
+            // note that this case will not happen if C1XOptions.CanonicalizeMultipliesToShifts is true
+            ArithmeticOp arith = (ArithmeticOp) index;
+            CiKind st = arith.y().kind;
+            if (arith.opcode == IMUL && arith.y().isConstant() && st.isInt()) {
+                int val = arith.y().asConstant().asInt();
+                switch (val) {
+                    case 1: return setUnsafeRawOp(i, base, arith.x(), 0);
+                    case 2: return setUnsafeRawOp(i, base, arith.x(), 1);
+                    case 4: return setUnsafeRawOp(i, base, arith.x(), 2);
+                    case 8: return setUnsafeRawOp(i, base, arith.x(), 3);
+                }
+            }
+        }
+
+        return false;
+    }
+
+    private boolean setUnsafeRawOp(UnsafeRawOp i, Value base, Value index, int log2scale) {
+        i.setBase(base);
+        i.setIndex(index);
+        i.setLog2Scale(log2scale);
+        return true;
+    }
+
+    @Override
+    public void visitUnsafeGetRaw(UnsafeGetRaw i) {
+        if (C1XOptions.CanonicalizeUnsafes) {
+            visitUnsafeRawOp(i);
+        }
+    }
+
+    @Override
+    public void visitUnsafePutRaw(UnsafePutRaw i) {
+        if (C1XOptions.CanonicalizeUnsafes) {
+            visitUnsafeRawOp(i);
+        }
+    }
+
+    private Object argAsObject(Value[] args, int index) {
+        CiConstant c = args[index].asConstant();
+        if (c != null) {
+            return runtime.asJavaObject(c);
+        }
+        return null;
+    }
+
+    private Class<?> argAsClass(Value[] args, int index) {
+        CiConstant c = args[index].asConstant();
+        if (c != null) {
+            return runtime.asJavaClass(c);
+        }
+        return null;
+    }
+
+    private double argAsDouble(Value[] args, int index) {
+        return args[index].asConstant().asDouble();
+    }
+
+    private float argAsFloat(Value[] args, int index) {
+        return args[index].asConstant().asFloat();
+    }
+
+    private int argAsInt(Value[] args, int index) {
+        return args[index].asConstant().asInt();
+    }
+
+    private long argAsLong(Value[] args, int index) {
+        return args[index].asConstant().asLong();
+    }
+
+    public static CiConstant foldInvocation(RiRuntime runtime, RiMethod method, final Value[] args) {
+        CiConstant result = runtime.invoke(method, new CiMethodInvokeArguments() {
+            int i;
+            @Override
+            public CiConstant nextArg() {
+                if (i >= args.length) {
+                    return null;
+                }
+                Value arg = args[i++];
+                if (arg == null) {
+                    if (i >= args.length) {
+                        return null;
+                    }
+                    arg = args[i++];
+                    assert arg != null;
+                }
+                return arg.isConstant() ? arg.asConstant() : null;
+            }
+        });
+        if (result != null) {
+            C1XMetrics.MethodsFolded++;
+        }
+        return result;
+    }
+
+    @Override
+    public void visitTypeEqualityCheck(TypeEqualityCheck i) {
+        if (i.condition == Condition.EQ && i.left() == i.right()) {
+            setCanonical(null);
+        }
+    }
+
+    @Override
+    public void visitBoundsCheck(BoundsCheck b) {
+        Value index = b.index();
+        Value length = b.length();
+
+        if (index.isConstant() && length.isConstant()) {
+            int i = index.asConstant().asInt();
+            int l = index.asConstant().asInt();
+            Condition c = b.condition;
+            if (c.check(i, l)) {
+                setCanonical(null);
+            }
+        }
+    }
+
+    private RiType getTypeOf(Value x) {
+        if (x.isConstant()) {
+            return runtime.getTypeOf(x.asConstant());
+        }
+        return null;
+    }
+}