comparison 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
comparison
equal deleted inserted replaced
2508:fea94949e0a2 2509:16b9a8b5ad39
1 /*
2 * Copyright (c) 2009, 2011, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 */
23 package com.sun.c1x.opt;
24
25 import static com.sun.cri.bytecode.Bytecodes.*;
26
27 import java.lang.reflect.*;
28 import java.util.*;
29
30 import com.sun.c1x.*;
31 import com.sun.c1x.ir.*;
32 import com.sun.c1x.util.*;
33 import com.sun.cri.ci.*;
34 import com.sun.cri.ri.*;
35
36 /**
37 * The {@code Canonicalizer} reduces instructions to a canonical form by folding constants,
38 * putting constants on the right side of commutative operators, simplifying conditionals,
39 * and several other transformations.
40 *
41 * @author Ben L. Titzer
42 */
43 public class Canonicalizer extends DefaultValueVisitor {
44
45 final RiRuntime runtime;
46 final RiMethod method;
47 final CiTarget target;
48 Value canonical;
49 List<Instruction> extra;
50
51 public Canonicalizer(RiRuntime runtime, RiMethod method, CiTarget target) {
52 this.runtime = runtime;
53 this.method = method;
54 this.target = target;
55 }
56
57 public Value canonicalize(Instruction original) {
58 this.canonical = original;
59 this.extra = null;
60 original.accept(this);
61 return this.canonical;
62 }
63
64 public List<Instruction> extra() {
65 return extra;
66 }
67
68 private <T extends Instruction> T addInstr(T x) {
69 if (extra == null) {
70 extra = new LinkedList<Instruction>();
71 }
72 extra.add(x);
73 return x;
74 }
75
76 private Constant intInstr(int v) {
77 return addInstr(Constant.forInt(v));
78 }
79
80 private Constant longInstr(long v) {
81 return addInstr(Constant.forLong(v));
82 }
83
84 private Constant wordInstr(long v) {
85 return addInstr(Constant.forWord(v));
86 }
87
88 private Value setCanonical(Value x) {
89 return canonical = x;
90 }
91
92 private Value setIntConstant(int val) {
93 return canonical = Constant.forInt(val);
94 }
95
96 private Value setConstant(CiConstant val) {
97 return canonical = new Constant(val);
98 }
99
100 private Value setBooleanConstant(boolean val) {
101 return canonical = Constant.forBoolean(val);
102 }
103
104 private Value setObjectConstant(Object val) {
105 if (C1XOptions.SupportObjectConstants) {
106 return canonical = Constant.forObject(val);
107 }
108 return canonical;
109 }
110
111 private Value setLongConstant(long val) {
112 return canonical = Constant.forLong(val);
113 }
114
115 private Value setFloatConstant(float val) {
116 return canonical = Constant.forFloat(val);
117 }
118
119 private Value setDoubleConstant(double val) {
120 return canonical = Constant.forDouble(val);
121 }
122
123 private Value setWordConstant(long val) {
124 return canonical = Constant.forDouble(val);
125 }
126
127 private void moveConstantToRight(Op2 x) {
128 if (x.x().isConstant() && isCommutative(x.opcode)) {
129 x.swapOperands();
130 }
131 }
132
133 private void visitOp2(Op2 i) {
134 final Value x = i.x();
135 final Value y = i.y();
136
137 if (x == y) {
138 // the left and right operands are the same value, try reducing some operations
139 switch (i.opcode) {
140 case ISUB: setIntConstant(0); return;
141 case LSUB: setLongConstant(0); return;
142 case IAND: // fall through
143 case LAND: // fall through
144 case IOR: // fall through
145 case LOR: setCanonical(x); return;
146 case IXOR: setIntConstant(0); return;
147 case LXOR: setLongConstant(0); return;
148 }
149 }
150
151 CiKind kind = x.kind;
152 if (x.isConstant() && y.isConstant()) {
153 // both operands are constants, try constant folding
154 switch (kind) {
155 case Int: {
156 Integer val = foldIntOp2(i.opcode, x.asConstant().asInt(), y.asConstant().asInt());
157 if (val != null) {
158 setIntConstant(val); // the operation was successfully folded to an int
159 return;
160 }
161 break;
162 }
163 case Long: {
164 Long val = foldLongOp2(i.opcode, x.asConstant().asLong(), y.asConstant().asLong());
165 if (val != null) {
166 setLongConstant(val); // the operation was successfully folded to a long
167 return;
168 }
169 break;
170 }
171 case Float: {
172 if (C1XOptions.CanonicalizeFloatingPoint) {
173 // try to fold a floating point operation
174 Float val = foldFloatOp2(i.opcode, x.asConstant().asFloat(), y.asConstant().asFloat());
175 if (val != null) {
176 setFloatConstant(val); // the operation was successfully folded to a float
177 return;
178 }
179 }
180 break;
181 }
182 case Double: {
183 if (C1XOptions.CanonicalizeFloatingPoint) {
184 // try to fold a floating point operation
185 Double val = foldDoubleOp2(i.opcode, x.asConstant().asDouble(), y.asConstant().asDouble());
186 if (val != null) {
187 setDoubleConstant(val); // the operation was successfully folded to a double
188 return;
189 }
190 }
191 break;
192 }
193 case Word: {
194 CiConstant val = runtime.foldWordOperation(i.opcode, new CiMethodInvokeArguments() {
195 int argIndex;
196 @Override
197 public CiConstant nextArg() {
198 if (argIndex == 0) {
199 return x.asConstant();
200 }
201 if (argIndex == 1) {
202 return y.asConstant();
203 }
204 argIndex++;
205 return null;
206 }
207 });
208
209 if (val != null) {
210 setConstant(val); // the operation was successfully folded to a word
211 return;
212 }
213 break;
214 }
215 }
216 }
217
218 // if there is a constant on the left and the operation is commutative, move it to the right
219 moveConstantToRight(i);
220
221 if (i.y().isConstant()) {
222 // the right side is a constant, try strength reduction
223 switch (kind) {
224 case Int: {
225 if (reduceIntOp2(i, i.x(), i.y().asConstant().asInt()) != null) {
226 return;
227 }
228 break;
229 }
230 case Long: {
231 if (reduceLongOp2(i, i.x(), i.y().asConstant().asLong()) != null) {
232 return;
233 }
234 break;
235 }
236 case Word: {
237 if (reduceWordOp2(i, i.x(), i.y().asConstant().asLong()) != null) {
238 return;
239 }
240 break;
241 }
242 // XXX: note that other cases are possible, but harder
243 // floating point operations need to be extra careful
244 }
245 }
246 assert Util.archKindsEqual(i, canonical);
247 }
248
249 private Value reduceIntOp2(Op2 original, Value x, int y) {
250 // attempt to reduce a binary operation with a constant on the right
251 int opcode = original.opcode;
252 switch (opcode) {
253 case IADD: return y == 0 ? setCanonical(x) : null;
254 case ISUB: return y == 0 ? setCanonical(x) : null;
255 case IMUL: {
256 if (y == 1) {
257 return setCanonical(x);
258 }
259 if (y > 0 && (y & y - 1) == 0 && C1XOptions.CanonicalizeMultipliesToShifts) {
260 // strength reduce multiply by power of 2 to shift operation
261 return setCanonical(new ShiftOp(ISHL, x, intInstr(CiUtil.log2(y))));
262 }
263 return y == 0 ? setIntConstant(0) : null;
264 }
265 case IDIV: return y == 1 ? setCanonical(x) : null;
266 case IREM: return y == 1 ? setCanonical(x) : null;
267 case IAND: {
268 if (y == -1) {
269 return setCanonical(x);
270 }
271 return y == 0 ? setIntConstant(0) : null;
272 }
273 case IOR: {
274 if (y == -1) {
275 return setIntConstant(-1);
276 }
277 return y == 0 ? setCanonical(x) : null;
278 }
279 case IXOR: return y == 0 ? setCanonical(x) : null;
280 case ISHL: return reduceShift(false, opcode, IUSHR, x, y);
281 case ISHR: return reduceShift(false, opcode, 0, x, y);
282 case IUSHR: return reduceShift(false, opcode, ISHL, x, y);
283 }
284 return null;
285 }
286
287 private Value reduceShift(boolean islong, int opcode, int reverse, Value x, long y) {
288 int mod = islong ? 0x3f : 0x1f;
289 long shift = y & mod;
290 if (shift == 0) {
291 return setCanonical(x);
292 }
293 if (x instanceof ShiftOp) {
294 // this is a chained shift operation ((e shift e) shift K)
295 ShiftOp s = (ShiftOp) x;
296 if (s.y().isConstant()) {
297 long z = s.y().asConstant().asLong();
298 if (s.opcode == opcode) {
299 // this is a chained shift operation (e >> C >> K)
300 y = y + z;
301 shift = y & mod;
302 if (shift == 0) {
303 return setCanonical(s.x());
304 }
305 // reduce to (e >> (C + K))
306 return setCanonical(new ShiftOp(opcode, s.x(), intInstr((int) shift)));
307 }
308 if (s.opcode == reverse && y == z) {
309 // this is a chained shift of the form (e >> K << K)
310 if (islong) {
311 long mask = -1;
312 if (opcode == LUSHR) {
313 mask = mask >>> y;
314 } else {
315 mask = mask << y;
316 }
317 // reduce to (e & mask)
318 return setCanonical(new LogicOp(LAND, s.x(), longInstr(mask)));
319 } else {
320 int mask = -1;
321 if (opcode == IUSHR) {
322 mask = mask >>> y;
323 } else {
324 mask = mask << y;
325 }
326 return setCanonical(new LogicOp(IAND, s.x(), intInstr(mask)));
327 }
328 }
329 }
330 }
331 if (y != shift) {
332 // (y & mod) != y
333 return setCanonical(new ShiftOp(opcode, x, intInstr((int) shift)));
334 }
335 return null;
336 }
337
338 private Value reduceLongOp2(Op2 original, Value x, long y) {
339 // attempt to reduce a binary operation with a constant on the right
340 int opcode = original.opcode;
341 switch (opcode) {
342 case LADD: return y == 0 ? setCanonical(x) : null;
343 case LSUB: return y == 0 ? setCanonical(x) : null;
344 case LMUL: {
345 if (y == 1) {
346 return setCanonical(x);
347 }
348 if (y > 0 && (y & y - 1) == 0 && C1XOptions.CanonicalizeMultipliesToShifts) {
349 // strength reduce multiply by power of 2 to shift operation
350 return setCanonical(new ShiftOp(LSHL, x, intInstr(CiUtil.log2(y))));
351 }
352 return y == 0 ? setLongConstant(0) : null;
353 }
354 case LDIV: return y == 1 ? setCanonical(x) : null;
355 case LREM: return y == 1 ? setCanonical(x) : null;
356 case LAND: {
357 if (y == -1) {
358 return setCanonical(x);
359 }
360 return y == 0 ? setLongConstant(0) : null;
361 }
362 case LOR: {
363 if (y == -1) {
364 return setLongConstant(-1);
365 }
366 return y == 0 ? setCanonical(x) : null;
367 }
368 case LXOR: return y == 0 ? setCanonical(x) : null;
369 case LSHL: return reduceShift(true, opcode, LUSHR, x, y);
370 case LSHR: return reduceShift(true, opcode, 0, x, y);
371 case LUSHR: return reduceShift(true, opcode, LSHL, x, y);
372 }
373 return null;
374 }
375
376 private Value reduceWordOp2(Op2 original, Value x, long y) {
377 if (y == 0) {
378 // Defer to arithmetic exception at runtime
379 return null;
380 }
381 // attempt to reduce a binary operation with a constant on the right
382 int opcode = original.opcode;
383 switch (opcode) {
384 case WDIVI:
385 case WDIV: {
386 if (y == 1) {
387 return setCanonical(x);
388 }
389 if (CiUtil.isPowerOf2(y)) {
390 return setCanonical(new ShiftOp(target.arch.is64bit() ? LUSHR : IUSHR, x, intInstr(CiUtil.log2(y))));
391 }
392 break;
393 }
394 case WREMI: {
395 if (y == 1) {
396 return setCanonical(intInstr(0));
397 }
398 if (CiUtil.isPowerOf2(y)) {
399 int mask = (int) y - 1;
400 if (target.arch.is64bit()) {
401 Convert l2i = new Convert(L2I, x, CiKind.Int);
402 addInstr(l2i);
403 return setCanonical(new LogicOp(IAND, l2i, intInstr(mask)));
404 }
405 return setCanonical(new LogicOp(CiKind.Int, IAND, x, intInstr(mask)));
406 }
407 break;
408 }
409 case WREM: {
410 if (y == 1) {
411 return setCanonical(wordInstr(0));
412 }
413 if (CiUtil.isPowerOf2(y)) {
414 if (target.arch.is64bit()) {
415 long mask = y - 1L;
416 return setCanonical(new LogicOp(LAND, x, longInstr(mask)));
417 }
418 int mask = (int) y - 1;
419 return setCanonical(new LogicOp(IAND, x, intInstr(mask)));
420 }
421 break;
422 }
423 }
424 return null;
425 }
426
427 private boolean inCurrentBlock(Value x) {
428 if (x instanceof Instruction) {
429 Instruction i = (Instruction) x;
430 int max = 4; // XXX: anything special about 4? seems like a tunable heuristic
431 while (max > 0 && i != null && !(i instanceof BlockEnd)) {
432 i = i.next();
433 max--;
434 }
435 return i == null;
436 }
437 return true;
438 }
439
440 private Value eliminateNarrowing(CiKind kind, Convert c) {
441 Value nv = null;
442 switch (c.opcode) {
443 case I2B:
444 if (kind == CiKind.Byte) {
445 nv = c.value();
446 }
447 break;
448 case I2S:
449 if (kind == CiKind.Short || kind == CiKind.Byte) {
450 nv = c.value();
451 }
452 break;
453 case I2C:
454 if (kind == CiKind.Char || kind == CiKind.Byte) {
455 nv = c.value();
456 }
457 break;
458 }
459 return nv;
460 }
461
462 @Override
463 public void visitLoadField(LoadField i) {
464 if (!i.isLoaded() || !C1XOptions.CanonicalizeConstantFields) {
465 return;
466 }
467 if (i.isStatic()) {
468 RiField field = i.field();
469 CiConstant value = field.constantValue(null);
470 if (value != null) {
471 if (method.isClassInitializer()) {
472 // don't do canonicalization in the <clinit> method
473 return;
474 }
475 setConstant(value);
476 }
477 } else {
478 RiField field = i.field();
479 if (i.object().isConstant()) {
480 CiConstant value = field.constantValue(i.object().asConstant());
481 if (value != null) {
482 setConstant(value);
483 }
484 }
485 }
486 }
487
488 @Override
489 public void visitStoreField(StoreField i) {
490 if (C1XOptions.CanonicalizeNarrowingInStores) {
491 // Eliminate narrowing conversions emitted by javac which are unnecessary when
492 // writing the value to a field that is packed
493 Value v = i.value();
494 if (v instanceof Convert) {
495 Value nv = eliminateNarrowing(i.field().kind(), (Convert) v);
496 // limit this optimization to the current basic block
497 // XXX: why is this limited to the current block?
498 if (nv != null && inCurrentBlock(v)) {
499 setCanonical(new StoreField(i.object(), i.field(), nv, i.isStatic(),
500 i.stateBefore(), i.isLoaded()));
501 }
502 }
503 }
504 }
505
506 @Override
507 public void visitArrayLength(ArrayLength i) {
508 // we can compute the length of the array statically if the object
509 // is a NewArray of a constant, or if the object is a constant reference
510 // (either by itself or loaded from a constant value field)
511 Value array = i.array();
512 if (array instanceof NewArray) {
513 // the array is a NewArray; check if it has a constant length
514 NewArray newArray = (NewArray) array;
515 Value length = newArray.length();
516 if (length instanceof Constant) {
517 // note that we don't use the Constant instruction itself
518 // as that would cause problems with liveness later
519 int actualLength = length.asConstant().asInt();
520 setIntConstant(actualLength);
521 }
522 } else if (array instanceof LoadField) {
523 // the array is a load of a field; check if it is a constant
524 LoadField load = (LoadField) array;
525 CiConstant cons = load.constantValue();
526 if (cons != null && cons.isNonNull()) {
527 setIntConstant(runtime.getArrayLength(cons));
528 }
529 } else if (array.isConstant()) {
530 // the array itself is a constant object reference
531 CiConstant obj = array.asConstant();
532 if (obj.isNonNull()) {
533 setIntConstant(runtime.getArrayLength(obj));
534 }
535 }
536 }
537
538 @Override
539 public void visitStoreIndexed(StoreIndexed i) {
540 Value array = i.array();
541 Value value = i.value();
542 if (C1XOptions.CanonicalizeNarrowingInStores) {
543 // Eliminate narrowing conversions emitted by javac which are unnecessary when
544 // writing the value to an array (which is packed)
545 Value v = value;
546 if (v instanceof Convert) {
547 Value nv = eliminateNarrowing(i.elementKind(), (Convert) v);
548 if (nv != null && inCurrentBlock(v)) {
549 setCanonical(new StoreIndexed(array, i.index(), i.length(), i.elementKind(), nv, i.stateBefore()));
550 }
551 }
552 }
553 if (C1XOptions.CanonicalizeArrayStoreChecks && i.elementKind() == CiKind.Object) {
554 if (value.isNullConstant()) {
555 i.eliminateStoreCheck();
556 } else {
557 RiType exactType = array.exactType();
558 if (exactType != null && exactType.isResolved()) {
559 if (exactType.componentType().superType() == null) {
560 // the exact type of the array is Object[] => no check is necessary
561 i.eliminateStoreCheck();
562 } else {
563 RiType declaredType = value.declaredType();
564 if (declaredType != null && declaredType.isResolved() && declaredType.isSubtypeOf(exactType.componentType())) {
565 // the value being stored has a known type
566 i.eliminateStoreCheck();
567 }
568 }
569 }
570 }
571 }
572 if (i.index().isConstant() && i.length() != null && i.length().isConstant()) {
573 int index = i.index().asConstant().asInt();
574 if (index >= 0 && index < i.length().asConstant().asInt()) {
575 i.eliminateBoundsCheck();
576 }
577 }
578 }
579
580 @Override
581 public void visitLoadIndexed(LoadIndexed i) {
582 if (i.index().isConstant() && i.length() != null && i.length().isConstant()) {
583 int index = i.index().asConstant().asInt();
584 if (index >= 0 && index < i.length().asConstant().asInt()) {
585 i.eliminateBoundsCheck();
586 }
587 }
588 }
589
590 @Override
591 public void visitNegateOp(NegateOp i) {
592 CiKind vt = i.x().kind;
593 Value v = i.x();
594 if (i.x().isConstant()) {
595 switch (vt) {
596 case Int: setIntConstant(-v.asConstant().asInt()); break;
597 case Long: setLongConstant(-v.asConstant().asLong()); break;
598 case Float: setFloatConstant(-v.asConstant().asFloat()); break;
599 case Double: setDoubleConstant(-v.asConstant().asDouble()); break;
600 }
601 }
602 assert vt == canonical.kind;
603 }
604
605 @Override
606 public void visitArithmeticOp(ArithmeticOp i) {
607 visitOp2(i);
608 }
609
610 @Override
611 public void visitShiftOp(ShiftOp i) {
612 visitOp2(i);
613 }
614
615 @Override
616 public void visitLogicOp(LogicOp i) {
617 visitOp2(i);
618 }
619
620 @Override
621 public void visitCompareOp(CompareOp i) {
622 if (i.kind.isVoid()) {
623 return;
624 }
625 // we can reduce a compare op if the two inputs are the same,
626 // or if both are constants
627 Value x = i.x();
628 Value y = i.y();
629 CiKind xt = x.kind;
630 if (x == y) {
631 // x and y are generated by the same instruction
632 switch (xt) {
633 case Long: setIntConstant(0); return;
634 case Float:
635 if (x.isConstant()) {
636 float xval = x.asConstant().asFloat(); // get the actual value of x (and y since x == y)
637 Integer val = foldFloatCompare(i.opcode, xval, xval);
638 assert val != null : "invalid opcode in float compare op";
639 setIntConstant(val);
640 return;
641 }
642 break;
643 case Double:
644 if (x.isConstant()) {
645 double xval = x.asConstant().asDouble(); // get the actual value of x (and y since x == y)
646 Integer val = foldDoubleCompare(i.opcode, xval, xval);
647 assert val != null : "invalid opcode in double compare op";
648 setIntConstant(val);
649 return;
650 }
651 break;
652 // note that there are no integer CompareOps
653 }
654 }
655 if (x.isConstant() && y.isConstant()) {
656 // both x and y are constants
657 switch (xt) {
658 case Long:
659 setIntConstant(foldLongCompare(x.asConstant().asLong(), y.asConstant().asLong()));
660 break;
661 case Float: {
662 Integer val = foldFloatCompare(i.opcode, x.asConstant().asFloat(), y.asConstant().asFloat());
663 assert val != null : "invalid opcode in float compare op";
664 setIntConstant(val);
665 break;
666 }
667 case Double: {
668 Integer val = foldDoubleCompare(i.opcode, x.asConstant().asDouble(), y.asConstant().asDouble());
669 assert val != null : "invalid opcode in float compare op";
670 setIntConstant(val);
671 break;
672 }
673 }
674 }
675 assert Util.archKindsEqual(i, canonical);
676 }
677
678 @Override
679 public void visitIfOp(IfOp i) {
680 moveConstantToRight(i);
681 }
682
683 @Override
684 public void visitConvert(Convert i) {
685 Value v = i.value();
686 if (v.isConstant()) {
687 // fold conversions between primitive types
688 // Checkstyle: stop
689 switch (i.opcode) {
690 case I2B: setIntConstant ((byte) v.asConstant().asInt()); return;
691 case I2S: setIntConstant ((short) v.asConstant().asInt()); return;
692 case I2C: setIntConstant ((char) v.asConstant().asInt()); return;
693 case I2L: setLongConstant ( v.asConstant().asInt()); return;
694 case I2F: setFloatConstant ( v.asConstant().asInt()); return;
695 case L2I: setIntConstant ((int) v.asConstant().asLong()); return;
696 case L2F: setFloatConstant ( v.asConstant().asLong()); return;
697 case L2D: setDoubleConstant( v.asConstant().asLong()); return;
698 case F2D: setDoubleConstant( v.asConstant().asFloat()); return;
699 case F2I: setIntConstant ((int) v.asConstant().asFloat()); return;
700 case F2L: setLongConstant ((long) v.asConstant().asFloat()); return;
701 case D2F: setFloatConstant ((float) v.asConstant().asDouble()); return;
702 case D2I: setIntConstant ((int) v.asConstant().asDouble()); return;
703 case D2L: setLongConstant ((long) v.asConstant().asDouble()); return;
704 }
705 // Checkstyle: resume
706 }
707
708 CiKind kind = CiKind.Illegal;
709 if (v instanceof LoadField) {
710 // remove redundant conversions from field loads of the correct type
711 kind = ((LoadField) v).field().kind();
712 } else if (v instanceof LoadIndexed) {
713 // remove redundant conversions from array loads of the correct type
714 kind = ((LoadIndexed) v).elementKind();
715 } else if (v instanceof Convert) {
716 // remove chained redundant conversions
717 Convert c = (Convert) v;
718 switch (c.opcode) {
719 case I2B: kind = CiKind.Byte; break;
720 case I2S: kind = CiKind.Short; break;
721 case I2C: kind = CiKind.Char; break;
722 }
723 }
724
725 if (kind != CiKind.Illegal) {
726 // if any of the above matched
727 switch (i.opcode) {
728 case I2B:
729 if (kind == CiKind.Byte) {
730 setCanonical(v);
731 }
732 break;
733 case I2S:
734 if (kind == CiKind.Byte || kind == CiKind.Short) {
735 setCanonical(v);
736 }
737 break;
738 case I2C:
739 if (kind == CiKind.Char) {
740 setCanonical(v);
741 }
742 break;
743 }
744 }
745
746 if (v instanceof Op2) {
747 // check if the operation was IAND with a constant; it may have narrowed the value already
748 Op2 op = (Op2) v;
749 // constant should be on right hand side if there is one
750 if (op.opcode == IAND && op.y().isConstant()) {
751 int safebits = 0;
752 int mask = op.y().asConstant().asInt();
753 switch (i.opcode) {
754 case I2B: safebits = 0x7f; break;
755 case I2S: safebits = 0x7fff; break;
756 case I2C: safebits = 0xffff; break;
757 }
758 if (safebits != 0 && (mask & ~safebits) == 0) {
759 // the mask already cleared all the upper bits necessary.
760 setCanonical(v);
761 }
762 }
763 }
764 }
765
766 @Override
767 public void visitNullCheck(NullCheck i) {
768 Value o = i.object();
769 if (o.isNonNull()) {
770 // if the instruction producing the object was a new, no check is necessary
771 setCanonical(o);
772 } else if (o.isConstant()) {
773 // if the object is a constant, check if it is nonnull
774 CiConstant c = o.asConstant();
775 if (c.kind.isObject() && !c.isNull()) {
776 setCanonical(o);
777 }
778 }
779 }
780
781 @Override
782 public void visitInvoke(Invoke i) {
783 if (C1XOptions.CanonicalizeFoldableMethods) {
784 RiMethod method = i.target();
785 if (method.isResolved()) {
786 // only try to fold resolved method invocations
787 CiConstant result = foldInvocation(runtime, i.target(), i.arguments());
788 if (result != null) {
789 // folding was successful
790 setCanonical(new Constant(result));
791 }
792 }
793 }
794 }
795
796 @Override
797 public void visitCheckCast(CheckCast i) {
798 // we can remove a redundant check cast if it is an object constant or the exact type is known
799 if (i.targetClass().isResolved()) {
800 Value o = i.object();
801 RiType type = o.exactType();
802 if (type == null) {
803 type = o.declaredType();
804 }
805 if (type != null && type.isResolved() && type.isSubtypeOf(i.targetClass())) {
806 // cast is redundant if exact type or declared type is already a subtype of the target type
807 setCanonical(o);
808 }
809 if (o.isConstant()) {
810 final CiConstant obj = o.asConstant();
811 if (obj.isNull()) {
812 // checkcast of null is null
813 setCanonical(o);
814 } else if (C1XOptions.CanonicalizeObjectCheckCast) {
815 if (i.targetClass().isInstance(obj)) {
816 // fold the cast if it will succeed
817 setCanonical(o);
818 }
819 }
820 }
821 }
822 }
823
824 @Override
825 public void visitInstanceOf(InstanceOf i) {
826 // we can fold an instanceof if it is an object constant or the exact type is known
827 if (i.targetClass().isResolved()) {
828 Value o = i.object();
829 RiType exact = o.exactType();
830 if (exact != null && exact.isResolved() && o.isNonNull()) {
831 setIntConstant(exact.isSubtypeOf(i.targetClass()) ? 1 : 0);
832 } else if (o.isConstant()) {
833 final CiConstant obj = o.asConstant();
834 if (obj.isNull()) {
835 // instanceof of null is false
836 setIntConstant(0);
837 } else if (C1XOptions.CanonicalizeObjectInstanceOf) {
838 // fold the instanceof test
839 setIntConstant(i.targetClass().isInstance(obj) ? 1 : 0);
840 }
841 }
842 }
843 }
844
845 @Override
846 public void visitIntrinsic(Intrinsic i) {
847 if (!C1XOptions.CanonicalizeIntrinsics) {
848 return;
849 }
850 if (!foldIntrinsic(i)) {
851 // folding did not work, try recognizing special intrinsics
852 reduceIntrinsic(i);
853 }
854 assert Util.archKindsEqual(i, canonical);
855 }
856
857 private void reduceIntrinsic(Intrinsic i) {
858 Value[] args = i.arguments();
859 C1XIntrinsic intrinsic = i.intrinsic();
860 if (intrinsic == C1XIntrinsic.java_lang_Class$isInstance) {
861 // try to convert a call to Class.isInstance() into an InstanceOf
862 RiType type = getTypeOf(args[0]);
863 if (type != null) {
864 setCanonical(new InstanceOf(type, Constant.forObject(type.getEncoding(RiType.Representation.TypeInfo)), args[1], i.stateBefore()));
865 return;
866 }
867 }
868 if (intrinsic == C1XIntrinsic.java_lang_reflect_Array$newArray) {
869 // try to convert a call to Array.newInstance() into a NewObjectArray or NewTypeArray
870 RiType type = getTypeOf(args[0]);
871 if (type != null) {
872 if (type.kind() == CiKind.Object) {
873 setCanonical(new NewObjectArray(type, args[1], i.stateBefore()));
874 } else {
875 RiType elementType = runtime.asRiType(type.kind());
876 setCanonical(new NewTypeArray(args[1], elementType, i.stateBefore()));
877 }
878 return;
879 }
880 }
881 assert Util.archKindsEqual(i, canonical);
882 }
883
884 private boolean foldIntrinsic(Intrinsic i) {
885 Value[] args = i.arguments();
886 for (Value arg : args) {
887 if (arg != null && !arg.isConstant()) {
888 // one input is not constant, give up
889 return true;
890 }
891 }
892 switch (i.intrinsic()) {
893 // do not use reflection here due to efficiency and potential bootstrap problems
894 case java_lang_Object$hashCode: {
895 Object object = argAsObject(args, 0);
896 if (object != null) {
897 setIntConstant(System.identityHashCode(object));
898 }
899 return true;
900 }
901 case java_lang_Object$getClass: {
902 Object object = argAsObject(args, 0);
903 if (object != null) {
904 setObjectConstant(object.getClass());
905 }
906 return true;
907 }
908
909 // java.lang.Class
910 case java_lang_Class$isAssignableFrom: {
911 Class<?> javaClass = argAsClass(args, 0);
912 Class<?> otherClass = argAsClass(args, 1);
913 if (javaClass != null && otherClass != null) {
914 setBooleanConstant(javaClass.isAssignableFrom(otherClass));
915 }
916 return true;
917 }
918 case java_lang_Class$isInstance: {
919 Class<?> javaClass = argAsClass(args, 0);
920 Object object = argAsObject(args, 1);
921 if (javaClass != null && object != null) {
922 setBooleanConstant(javaClass.isInstance(object));
923 }
924 return true;
925 }
926 case java_lang_Class$getModifiers: {
927 Class<?> javaClass = argAsClass(args, 0);
928 if (javaClass != null) {
929 setIntConstant(javaClass.getModifiers());
930 }
931 return true;
932 }
933 case java_lang_Class$isInterface: {
934 Class<?> javaClass = argAsClass(args, 0);
935 if (javaClass != null) {
936 setBooleanConstant(javaClass.isInterface());
937 }
938 return true;
939 }
940 case java_lang_Class$isArray: {
941 Class<?> javaClass = argAsClass(args, 0);
942 if (javaClass != null) {
943 setBooleanConstant(javaClass.isArray());
944 }
945 return true;
946 }
947 case java_lang_Class$isPrimitive: {
948 Class<?> javaClass = argAsClass(args, 0);
949 if (javaClass != null) {
950 setBooleanConstant(javaClass.isPrimitive());
951 }
952 return true;
953 }
954 case java_lang_Class$getSuperclass: {
955 Class<?> javaClass = argAsClass(args, 0);
956 if (javaClass != null) {
957 setObjectConstant(javaClass.getSuperclass());
958 }
959 return true;
960 }
961 case java_lang_Class$getComponentType: {
962 Class<?> javaClass = argAsClass(args, 0);
963 if (javaClass != null) {
964 setObjectConstant(javaClass.getComponentType());
965 }
966 return true;
967 }
968
969 // java.lang.Math
970 case java_lang_Math$abs: setDoubleConstant(Math.abs(argAsDouble(args, 0))); return true;
971 case java_lang_Math$sin: setDoubleConstant(Math.sin(argAsDouble(args, 0))); return true;
972 case java_lang_Math$cos: setDoubleConstant(Math.cos(argAsDouble(args, 0))); return true;
973 case java_lang_Math$tan: setDoubleConstant(Math.tan(argAsDouble(args, 0))); return true;
974 case java_lang_Math$atan2: setDoubleConstant(Math.atan2(argAsDouble(args, 0), argAsDouble(args, 2))); return true;
975 case java_lang_Math$sqrt: setDoubleConstant(Math.sqrt(argAsDouble(args, 0))); return true;
976 case java_lang_Math$log: setDoubleConstant(Math.log(argAsDouble(args, 0))); return true;
977 case java_lang_Math$log10: setDoubleConstant(Math.log10(argAsDouble(args, 0))); return true;
978 case java_lang_Math$pow: setDoubleConstant(Math.pow(argAsDouble(args, 0), argAsDouble(args, 2))); return true;
979 case java_lang_Math$exp: setDoubleConstant(Math.exp(argAsDouble(args, 0))); return true;
980 case java_lang_Math$min: setIntConstant(Math.min(argAsInt(args, 0), argAsInt(args, 1))); return true;
981 case java_lang_Math$max: setIntConstant(Math.max(argAsInt(args, 0), argAsInt(args, 1))); return true;
982
983 // java.lang.Float
984 case java_lang_Float$floatToRawIntBits: setIntConstant(Float.floatToRawIntBits(argAsFloat(args, 0))); return true;
985 case java_lang_Float$floatToIntBits: setIntConstant(Float.floatToIntBits(argAsFloat(args, 0))); return true;
986 case java_lang_Float$intBitsToFloat: setFloatConstant(Float.intBitsToFloat(argAsInt(args, 0))); return true;
987
988 // java.lang.Double
989 case java_lang_Double$doubleToRawLongBits: setLongConstant(Double.doubleToRawLongBits(argAsDouble(args, 0))); return true;
990 case java_lang_Double$doubleToLongBits: setLongConstant(Double.doubleToLongBits(argAsDouble(args, 0))); return true;
991 case java_lang_Double$longBitsToDouble: setDoubleConstant(Double.longBitsToDouble(argAsLong(args, 0))); return true;
992
993 // java.lang.Integer
994 case java_lang_Integer$bitCount: setIntConstant(Integer.bitCount(argAsInt(args, 0))); return true;
995 case java_lang_Integer$reverseBytes: setIntConstant(Integer.reverseBytes(argAsInt(args, 0))); return true;
996
997 // java.lang.Long
998 case java_lang_Long$bitCount: setIntConstant(Long.bitCount(argAsLong(args, 0))); return true;
999 case java_lang_Long$reverseBytes: setLongConstant(Long.reverseBytes(argAsLong(args, 0))); return true;
1000
1001 // java.lang.System
1002 case java_lang_System$identityHashCode: {
1003 Object object = argAsObject(args, 0);
1004 if (object != null) {
1005 setIntConstant(System.identityHashCode(object));
1006 }
1007 return true;
1008 }
1009
1010 // java.lang.reflect.Array
1011 case java_lang_reflect_Array$getLength: {
1012 Object object = argAsObject(args, 0);
1013 if (object != null && object.getClass().isArray()) {
1014 setIntConstant(Array.getLength(object));
1015 }
1016 return true;
1017 }
1018 }
1019 return false;
1020 }
1021
1022 @Override
1023 public void visitIf(If i) {
1024 if (i.x().isConstant()) {
1025 // move constant to the right
1026 i.swapOperands();
1027 }
1028 Value l = i.x();
1029 Value r = i.y();
1030
1031 if (l == r && !l.kind.isFloatOrDouble()) {
1032 // this is a comparison of x op x
1033 // No opt for float/double due to NaN case
1034 reduceReflexiveIf(i);
1035 return;
1036 }
1037
1038 CiKind rt = r.kind;
1039
1040 Condition ifcond = i.condition();
1041 if (l.isConstant() && r.isConstant()) {
1042 // fold comparisons between constants and convert to Goto
1043 Boolean result = ifcond.foldCondition(l.asConstant(), r.asConstant(), runtime);
1044 if (result != null) {
1045 setCanonical(new Goto(i.successor(result), i.stateAfter(), i.isSafepoint()));
1046 return;
1047 }
1048 }
1049
1050 if (r.isConstant() && rt.isInt()) {
1051 // attempt to reduce comparisons with constant on right side
1052 if (l instanceof CompareOp) {
1053 // attempt to reduce If ((a cmp b) op const)
1054 reduceIfCompareOpConstant(i, r.asConstant());
1055 }
1056 }
1057
1058 if (isNullConstant(r) && l.isNonNull()) {
1059 // this is a comparison of null against something that is not null
1060 if (ifcond == Condition.EQ) {
1061 // new() == null is always false
1062 setCanonical(new Goto(i.falseSuccessor(), i.stateAfter(), i.isSafepoint()));
1063 } else if (ifcond == Condition.NE) {
1064 // new() != null is always true
1065 setCanonical(new Goto(i.trueSuccessor(), i.stateAfter(), i.isSafepoint()));
1066 }
1067 }
1068 }
1069
1070 private boolean isNullConstant(Value r) {
1071 return r.isConstant() && r.asConstant().isNull();
1072 }
1073
1074 private void reduceIfCompareOpConstant(If i, CiConstant rtc) {
1075 Condition ifcond = i.condition();
1076 Value l = i.x();
1077 CompareOp cmp = (CompareOp) l;
1078 boolean unorderedIsLess = cmp.opcode == FCMPL || cmp.opcode == DCMPL;
1079 BlockBegin lssSucc = i.successor(ifcond.foldCondition(CiConstant.forInt(-1), rtc, runtime));
1080 BlockBegin eqlSucc = i.successor(ifcond.foldCondition(CiConstant.forInt(0), rtc, runtime));
1081 BlockBegin gtrSucc = i.successor(ifcond.foldCondition(CiConstant.forInt(1), rtc, runtime));
1082 BlockBegin nanSucc = unorderedIsLess ? lssSucc : gtrSucc;
1083 // Note: At this point all successors (lssSucc, eqlSucc, gtrSucc, nanSucc) are
1084 // equal to x->tsux() or x->fsux(). Furthermore, nanSucc equals either
1085 // lssSucc or gtrSucc.
1086 if (lssSucc == eqlSucc && eqlSucc == gtrSucc) {
1087 // all successors identical => simplify to: Goto
1088 setCanonical(new Goto(lssSucc, i.stateAfter(), i.isSafepoint()));
1089 } else {
1090 // two successors differ and two successors are the same => simplify to: If (x cmp y)
1091 // determine new condition & successors
1092 Condition cond;
1093 BlockBegin tsux;
1094 BlockBegin fsux;
1095 if (lssSucc == eqlSucc) {
1096 cond = Condition.LE;
1097 tsux = lssSucc;
1098 fsux = gtrSucc;
1099 } else if (lssSucc == gtrSucc) {
1100 cond = Condition.NE;
1101 tsux = lssSucc;
1102 fsux = eqlSucc;
1103 } else if (eqlSucc == gtrSucc) {
1104 cond = Condition.GE;
1105 tsux = eqlSucc;
1106 fsux = lssSucc;
1107 } else {
1108 throw Util.shouldNotReachHere();
1109 }
1110 // TODO: the state after is incorrect here: should it be preserved from the original if?
1111 If canon = new If(cmp.x(), cond, nanSucc == tsux, cmp.y(), tsux, fsux, cmp.stateBefore(), i.isSafepoint());
1112 if (cmp.x() == cmp.y()) {
1113 // re-canonicalize the new if
1114 visitIf(canon);
1115 } else {
1116 setCanonical(canon);
1117 }
1118 }
1119 }
1120
1121 private void reduceReflexiveIf(If i) {
1122 // simplify reflexive comparisons If (x op x) to Goto
1123 BlockBegin succ;
1124 switch (i.condition()) {
1125 case EQ: succ = i.successor(true); break;
1126 case NE: succ = i.successor(false); break;
1127 case LT: succ = i.successor(false); break;
1128 case LE: succ = i.successor(true); break;
1129 case GT: succ = i.successor(false); break;
1130 case GE: succ = i.successor(true); break;
1131 default:
1132 throw Util.shouldNotReachHere();
1133 }
1134 setCanonical(new Goto(succ, i.stateAfter(), i.isSafepoint()));
1135 }
1136
1137 @Override
1138 public void visitTableSwitch(TableSwitch i) {
1139 Value v = i.value();
1140 if (v.isConstant()) {
1141 // fold a table switch over a constant by replacing it with a goto
1142 int val = v.asConstant().asInt();
1143 BlockBegin succ = i.defaultSuccessor();
1144 if (val >= i.lowKey() && val <= i.highKey()) {
1145 succ = i.successors().get(val - i.lowKey());
1146 }
1147 setCanonical(new Goto(succ, i.stateAfter(), i.isSafepoint()));
1148 return;
1149 }
1150 int max = i.numberOfCases();
1151 if (max == 0) {
1152 // replace switch with Goto
1153 if (v instanceof Instruction) {
1154 // TODO: is it necessary to add the instruction explicitly?
1155 addInstr((Instruction) v);
1156 }
1157 setCanonical(new Goto(i.defaultSuccessor(), i.stateAfter(), i.isSafepoint()));
1158 return;
1159 }
1160 if (max == 1) {
1161 // replace switch with If
1162 Constant key = intInstr(i.lowKey());
1163 If newIf = new If(v, Condition.EQ, false, key, i.successors().get(0), i.defaultSuccessor(), null, i.isSafepoint());
1164 newIf.setStateAfter(i.stateAfter());
1165 setCanonical(newIf);
1166 }
1167 }
1168
1169 @Override
1170 public void visitLookupSwitch(LookupSwitch i) {
1171 Value v = i.value();
1172 if (v.isConstant()) {
1173 // fold a lookup switch over a constant by replacing it with a goto
1174 int val = v.asConstant().asInt();
1175 BlockBegin succ = i.defaultSuccessor();
1176 for (int j = 0; j < i.numberOfCases(); j++) {
1177 if (val == i.keyAt(j)) {
1178 succ = i.successors().get(j);
1179 break;
1180 }
1181 }
1182 setCanonical(new Goto(succ, i.stateAfter(), i.isSafepoint()));
1183 return;
1184 }
1185 int max = i.numberOfCases();
1186 if (max == 0) {
1187 // replace switch with Goto
1188 if (v instanceof Instruction) {
1189 addInstr((Instruction) v); // the value expression may produce side effects
1190 }
1191 setCanonical(new Goto(i.defaultSuccessor(), i.stateAfter(), i.isSafepoint()));
1192 return;
1193 }
1194 if (max == 1) {
1195 // replace switch with If
1196 Constant key = intInstr(i.keyAt(0));
1197 If newIf = new If(v, Condition.EQ, false, key, i.successors().get(0), i.defaultSuccessor(), null, i.isSafepoint());
1198 newIf.setStateAfter(i.stateAfter());
1199 setCanonical(newIf);
1200 }
1201 }
1202
1203 private void visitUnsafeRawOp(UnsafeRawOp i) {
1204 if (i.base() instanceof ArithmeticOp) {
1205 // if the base is an arithmetic op, try reducing
1206 ArithmeticOp root = (ArithmeticOp) i.base();
1207 if (!root.isLive() && root.opcode == LADD) {
1208 // match unsafe(x + y) if the x + y is not pinned
1209 // try reducing (x + y) and (y + x)
1210 Value y = root.y();
1211 Value x = root.x();
1212 if (reduceRawOp(i, x, y) || reduceRawOp(i, y, x)) {
1213 // the operation was reduced
1214 return;
1215 }
1216 if (y instanceof Convert) {
1217 // match unsafe(x + (long) y)
1218 Convert convert = (Convert) y;
1219 if (convert.opcode == I2L && convert.value().kind.isInt()) {
1220 // the conversion is redundant
1221 setUnsafeRawOp(i, x, convert.value(), 0);
1222 }
1223 }
1224 }
1225 }
1226 }
1227
1228 private boolean reduceRawOp(UnsafeRawOp i, Value base, Value index) {
1229 if (index instanceof Convert) {
1230 // skip any conversion operations
1231 index = ((Convert) index).value();
1232 }
1233 if (index instanceof ShiftOp) {
1234 // try to match the index as a shift by a constant
1235 ShiftOp shift = (ShiftOp) index;
1236 CiKind st = shift.y().kind;
1237 if (shift.y().isConstant() && st.isInt()) {
1238 int val = shift.y().asConstant().asInt();
1239 switch (val) {
1240 case 0: // fall through
1241 case 1: // fall through
1242 case 2: // fall through
1243 case 3: return setUnsafeRawOp(i, base, shift.x(), val);
1244 }
1245 }
1246 }
1247 if (index instanceof ArithmeticOp) {
1248 // try to match the index as a multiply by a constant
1249 // note that this case will not happen if C1XOptions.CanonicalizeMultipliesToShifts is true
1250 ArithmeticOp arith = (ArithmeticOp) index;
1251 CiKind st = arith.y().kind;
1252 if (arith.opcode == IMUL && arith.y().isConstant() && st.isInt()) {
1253 int val = arith.y().asConstant().asInt();
1254 switch (val) {
1255 case 1: return setUnsafeRawOp(i, base, arith.x(), 0);
1256 case 2: return setUnsafeRawOp(i, base, arith.x(), 1);
1257 case 4: return setUnsafeRawOp(i, base, arith.x(), 2);
1258 case 8: return setUnsafeRawOp(i, base, arith.x(), 3);
1259 }
1260 }
1261 }
1262
1263 return false;
1264 }
1265
1266 private boolean setUnsafeRawOp(UnsafeRawOp i, Value base, Value index, int log2scale) {
1267 i.setBase(base);
1268 i.setIndex(index);
1269 i.setLog2Scale(log2scale);
1270 return true;
1271 }
1272
1273 @Override
1274 public void visitUnsafeGetRaw(UnsafeGetRaw i) {
1275 if (C1XOptions.CanonicalizeUnsafes) {
1276 visitUnsafeRawOp(i);
1277 }
1278 }
1279
1280 @Override
1281 public void visitUnsafePutRaw(UnsafePutRaw i) {
1282 if (C1XOptions.CanonicalizeUnsafes) {
1283 visitUnsafeRawOp(i);
1284 }
1285 }
1286
1287 private Object argAsObject(Value[] args, int index) {
1288 CiConstant c = args[index].asConstant();
1289 if (c != null) {
1290 return runtime.asJavaObject(c);
1291 }
1292 return null;
1293 }
1294
1295 private Class<?> argAsClass(Value[] args, int index) {
1296 CiConstant c = args[index].asConstant();
1297 if (c != null) {
1298 return runtime.asJavaClass(c);
1299 }
1300 return null;
1301 }
1302
1303 private double argAsDouble(Value[] args, int index) {
1304 return args[index].asConstant().asDouble();
1305 }
1306
1307 private float argAsFloat(Value[] args, int index) {
1308 return args[index].asConstant().asFloat();
1309 }
1310
1311 private int argAsInt(Value[] args, int index) {
1312 return args[index].asConstant().asInt();
1313 }
1314
1315 private long argAsLong(Value[] args, int index) {
1316 return args[index].asConstant().asLong();
1317 }
1318
1319 public static CiConstant foldInvocation(RiRuntime runtime, RiMethod method, final Value[] args) {
1320 CiConstant result = runtime.invoke(method, new CiMethodInvokeArguments() {
1321 int i;
1322 @Override
1323 public CiConstant nextArg() {
1324 if (i >= args.length) {
1325 return null;
1326 }
1327 Value arg = args[i++];
1328 if (arg == null) {
1329 if (i >= args.length) {
1330 return null;
1331 }
1332 arg = args[i++];
1333 assert arg != null;
1334 }
1335 return arg.isConstant() ? arg.asConstant() : null;
1336 }
1337 });
1338 if (result != null) {
1339 C1XMetrics.MethodsFolded++;
1340 }
1341 return result;
1342 }
1343
1344 @Override
1345 public void visitTypeEqualityCheck(TypeEqualityCheck i) {
1346 if (i.condition == Condition.EQ && i.left() == i.right()) {
1347 setCanonical(null);
1348 }
1349 }
1350
1351 @Override
1352 public void visitBoundsCheck(BoundsCheck b) {
1353 Value index = b.index();
1354 Value length = b.length();
1355
1356 if (index.isConstant() && length.isConstant()) {
1357 int i = index.asConstant().asInt();
1358 int l = index.asConstant().asInt();
1359 Condition c = b.condition;
1360 if (c.check(i, l)) {
1361 setCanonical(null);
1362 }
1363 }
1364 }
1365
1366 private RiType getTypeOf(Value x) {
1367 if (x.isConstant()) {
1368 return runtime.getTypeOf(x.asConstant());
1369 }
1370 return null;
1371 }
1372 }