Mercurial > hg > graal-compiler
view graal/com.oracle.max.graal.compiler/src/com/oracle/max/graal/compiler/ir/Conditional.java @ 3490:ce7cf6d16b38
Reintroduce Conditional (IfOp) using the new infrastructure, fix the Cmov support in the backend, use x86 cmov when possible, make it work with floating point comparison conditions, make MaterializeNode a subclass of Conditional, fix nasty bug related to introducing LIR jumps in an existing LIRBlock in MaterializeNode/Conditional
Canonicalize FDConversionNode if input is constant
author | Gilles Duboscq <gilles.duboscq@oracle.com> |
---|---|
date | Fri, 05 Aug 2011 17:40:35 +0200 |
parents | a64b615ba630 |
children | e04e88087e98 |
line wrap: on
line source
/* * 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.oracle.max.graal.compiler.ir; import com.oracle.max.graal.compiler.*; import com.oracle.max.graal.compiler.debug.*; import com.oracle.max.graal.compiler.gen.*; import com.oracle.max.graal.compiler.gen.LIRGenerator.LIRGeneratorOp; import com.oracle.max.graal.compiler.phases.CanonicalizerPhase.CanonicalizerOp; import com.oracle.max.graal.compiler.phases.CanonicalizerPhase.NotifyReProcess; import com.oracle.max.graal.compiler.util.*; import com.oracle.max.graal.compiler.value.*; import com.oracle.max.graal.graph.*; import com.sun.cri.bytecode.*; import com.sun.cri.ci.*; /** * The {@code Conditional} class represents a comparison that yields one of two values. * Note that these nodes are not built directly from the bytecode but are introduced * by conditional expression elimination. */ public class Conditional extends Binary { private static final int INPUT_COUNT = 2; private static final int INPUT_CONDITION = 0; private static final int INPUT_STATE = 1; private static final int SUCCESSOR_COUNT = 0; @Override protected int inputCount() { return super.inputCount() + INPUT_COUNT; } @Override protected int successorCount() { return super.successorCount() + SUCCESSOR_COUNT; } public BooleanNode condition() { return (BooleanNode) inputs().get(super.inputCount() + INPUT_CONDITION); } public void setCondition(BooleanNode n) { inputs().set(super.inputCount() + INPUT_CONDITION, n); } public FrameState stateDuring() { return (FrameState) inputs().get(super.inputCount() + INPUT_STATE); } public void setStateDuring(FrameState n) { inputs().set(super.inputCount() + INPUT_STATE, n); } /** * Constructs a new IfOp. * @param x the instruction producing the first value to be compared * @param condition the condition of the comparison * @param y the instruction producing the second value to be compared * @param trueValue the value produced if the condition is true * @param falseValue the value produced if the condition is false */ public Conditional(BooleanNode condition, Value trueValue, Value falseValue, Graph graph) { // TODO: return the appropriate bytecode IF_ICMPEQ, etc super(trueValue.kind.meet(falseValue.kind), Bytecodes.ILLEGAL, trueValue, falseValue, INPUT_COUNT, SUCCESSOR_COUNT, graph); setCondition(condition); } // for copying private Conditional(CiKind kind, Graph graph) { super(kind, Bytecodes.ILLEGAL, null, null, INPUT_COUNT, SUCCESSOR_COUNT, graph); } public Value trueValue() { return x(); } public Value falseValue() { return y(); } public Value setTrueValue(Value value) { return setX(value); } public Value setFalseValue(Value value) { return setY(value); } @Override public boolean valueEqual(Node i) { if (i instanceof Conditional) { return super.valueEqual(i); } return false; } @Override public void print(LogStream out) { out.print(x()). print(' '). print(condition()). print(' '). print(y()). print(" ? "). print(trueValue()). print(" : "). print(falseValue()); } @Override public Node copy(Graph into) { Conditional x = new Conditional(kind, into); return x; } @SuppressWarnings("unchecked") @Override public <T extends Op> T lookup(Class<T> clazz) { if (clazz == CanonicalizerOp.class) { return (T) CANONICALIZER; } if (clazz == LIRGeneratorOp.class) { return (T) LIRGEN; } return super.lookup(clazz); } private static final CanonicalizerOp CANONICALIZER = new CanonicalizerOp() { @Override public Node canonical(Node node, NotifyReProcess reProcess) { Conditional conditional = (Conditional) node; BooleanNode condition = conditional.condition(); Value trueValue = conditional.trueValue(); Value falseValue = conditional.falseValue(); if (condition instanceof Constant) { Constant c = (Constant) condition; if (c.asConstant().asBoolean()) { return trueValue; } else { return falseValue; } } if (trueValue == falseValue) { return trueValue; } if (!(conditional instanceof MaterializeNode) && trueValue instanceof Constant && falseValue instanceof Constant && trueValue.kind == CiKind.Int && falseValue.kind == CiKind.Int) { int trueInt = trueValue.asConstant().asInt(); int falseInt = falseValue.asConstant().asInt(); if (trueInt == 0 && falseInt == 1) { if (GraalOptions.TraceCanonicalizer) { TTY.println("> Conditional canon'ed to ~Materialize"); } reProcess.reProccess(condition); // because we negate it MaterializeNode materializeNode = new MaterializeNode(new NegateBooleanNode(condition, node.graph()), node.graph()); materializeNode.setStateDuring(conditional.stateDuring()); return materializeNode; } else if (trueInt == 1 && falseInt == 0) { if (GraalOptions.TraceCanonicalizer) { TTY.println("> Conditional canon'ed to Materialize"); } MaterializeNode materializeNode = new MaterializeNode(condition, node.graph()); materializeNode.setStateDuring(conditional.stateDuring()); return materializeNode; } } else if (falseValue instanceof Constant && !(trueValue instanceof Constant)) { conditional.setTrueValue(falseValue); conditional.setFalseValue(trueValue); condition = new NegateBooleanNode(condition, node.graph()); conditional.setCondition(condition); } return conditional; } }; private static final LIRGeneratorOp LIRGEN = new LIRGeneratorOp() { @Override public void generate(Node n, LIRGenerator generator) { Conditional conditional = (Conditional) n; BooleanNode condition = conditional.condition(); CiVariable result = generator.createResultVariable(conditional); // try to use cmp + cmov first Condition cond = null; CiValue left = null; CiValue right = null; boolean floating = false; boolean unOrderedIsSecond = false; boolean negate = false; while (condition instanceof NegateBooleanNode) { negate = !negate; condition = ((NegateBooleanNode) condition).value(); } if (condition instanceof Compare) { Compare compare = (Compare) condition; Value x = compare.x(); Value y = compare.y(); cond = compare.condition(); if (x.kind.isFloatOrDouble()) { floating = true; unOrderedIsSecond = !compare.unorderedIsTrue(); cond = generator.floatingPointCondition(cond); } left = generator.load(x); if (!generator.canInlineAsConstant(y)) { right = generator.load(y); } else { right = generator.makeOperand(y); } } else if (condition instanceof IsNonNull) { IsNonNull isNonNull = (IsNonNull) condition; left = generator.load(isNonNull.object()); right = CiConstant.NULL_OBJECT; cond = Condition.NE; } else if (condition instanceof Constant) { generator.lir().move(result, condition.asConstant()); } else { throw Util.shouldNotReachHere("Currently not implemented because we can not create blocks during LIRGen : " + condition); } CiValue tVal = generator.makeOperand(conditional.trueValue()); CiValue fVal = generator.makeOperand(conditional.falseValue()); if (cond != null) { if (negate) { cond = cond.negate(); } assert left != null && right != null; generator.lir().cmp(cond, left, right); if (floating) { generator.lir().fcmove(cond, tVal, fVal, result, unOrderedIsSecond); } else { generator.lir().cmove(cond, tVal, fVal, result); } } } }; }