Mercurial > hg > truffle
view graal/com.oracle.graal.nodes/src/com/oracle/graal/nodes/java/TypeCheckNode.java @ 21495:0b76c55e1cef
Fix folding of type check node.
author | Thomas Wuerthinger <thomas.wuerthinger@oracle.com> |
---|---|
date | Tue, 26 May 2015 18:34:22 +0200 |
parents | d094ea7e0433 |
children | 31fc2fce38f3 |
line wrap: on
line source
/* * Copyright (c) 2009, 2015, 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.graal.nodes.java; import com.oracle.graal.api.meta.*; import com.oracle.graal.api.meta.Assumptions.AssumptionResult; import com.oracle.graal.compiler.common.type.*; import com.oracle.graal.graph.*; import com.oracle.graal.graph.spi.*; import com.oracle.graal.nodeinfo.*; import com.oracle.graal.nodes.*; import com.oracle.graal.nodes.calc.*; import com.oracle.graal.nodes.spi.*; /** * The {@code TypeCheckNode} represents a test equivalent to (o != null && o.getClass() == type). */ @NodeInfo public final class TypeCheckNode extends UnaryOpLogicNode implements Lowerable, Virtualizable { public static final NodeClass<TypeCheckNode> TYPE = NodeClass.create(TypeCheckNode.class); protected final ResolvedJavaType type; public TypeCheckNode(ResolvedJavaType type, ValueNode object) { super(TYPE, object); this.type = type; assert type != null; assert type.isConcrete() || type.isArray(); } public static LogicNode create(ResolvedJavaType type, ValueNode object) { ObjectStamp objectStamp = (ObjectStamp) object.stamp(); LogicNode constantValue = findSynonym(type, objectStamp.type(), objectStamp.nonNull(), objectStamp.isExactType()); if (constantValue != null) { return constantValue; } else { return new TypeCheckNode(type, object); } } @Override public void lower(LoweringTool tool) { tool.getLowerer().lower(this, tool); } @Override public ValueNode canonical(CanonicalizerTool tool, ValueNode forValue) { if (!(forValue.stamp() instanceof ObjectStamp)) { return this; } ObjectStamp objectStamp = (ObjectStamp) forValue.stamp(); if (objectStamp.alwaysNull()) { return LogicConstantNode.contradiction(); } ResolvedJavaType stampType = objectStamp.type(); if (stampType != null) { ValueNode result = check(forValue, stampType, objectStamp.nonNull(), objectStamp.isExactType()); if (result != null) { return result; } Assumptions assumptions = graph() == null ? null : graph().getAssumptions(); if (assumptions != null) { AssumptionResult<ResolvedJavaType> leafConcreteSubtype = stampType.findLeafConcreteSubtype(); if (leafConcreteSubtype != null) { result = check(forValue, leafConcreteSubtype.getResult(), objectStamp.nonNull(), true); if (result != null) { assumptions.record(leafConcreteSubtype); return result; } } } } return this; } private ValueNode check(ValueNode forValue, ResolvedJavaType inputType, boolean nonNull, boolean exactType) { ValueNode result = findSynonym(type(), inputType, nonNull, exactType); if (result != null) { return result; } if (type().equals(inputType)) { boolean mightBeNull = !nonNull; if (exactType && mightBeNull) { // the instanceof matches if the object is non-null, so return true // depending on the null-ness. return LogicNegationNode.create(new IsNullNode(forValue)); } } return null; } public static LogicNode findSynonym(ResolvedJavaType type, ResolvedJavaType inputType, boolean nonNull, boolean exactType) { if (inputType == null) { return null; } if (type.equals(inputType)) { if (nonNull && exactType) { // the type matches, so return true return LogicConstantNode.tautology(); } } else { if (exactType) { // since this type check failed for an exact type we know that it can never // succeed at run time. we also don't care about null values, since they will // also make the check fail. return LogicConstantNode.contradiction(); } } return null; } /** * Gets the type being tested. */ public ResolvedJavaType type() { return type; } @Override public void virtualize(VirtualizerTool tool) { State state = tool.getObjectState(getValue()); if (state != null) { tool.replaceWithValue(LogicConstantNode.forBoolean(type().equals(state.getVirtualObject().type()), graph())); } } @Override public Stamp getSucceedingStampForValue(boolean negated) { if (negated) { return null; } else { return StampFactory.exactNonNull(type); } } @Override public TriState tryFold(Stamp valueStamp) { if (valueStamp instanceof ObjectStamp) { ObjectStamp objectStamp = (ObjectStamp) valueStamp; if (objectStamp.alwaysNull()) { return TriState.FALSE; } ResolvedJavaType objectType = objectStamp.type(); if (objectType != null) { ResolvedJavaType instanceofType = type; if (instanceofType.equals(objectType)) { if (objectStamp.nonNull() && (objectStamp.isExactType() || objectType.isFinal())) { return TriState.TRUE; } } else { if (objectStamp.isExactType()) { return TriState.FALSE; } } } } return TriState.UNKNOWN; } }