package org.graalvm.compiler.nodes.extended;
import static org.graalvm.compiler.core.common.GraalOptions.GeneratePIC;
import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_2;
import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_1;
import org.graalvm.compiler.core.common.type.AbstractPointerStamp;
import org.graalvm.compiler.core.common.type.ObjectStamp;
import org.graalvm.compiler.core.common.type.Stamp;
import org.graalvm.compiler.core.common.type.TypeReference;
import org.graalvm.compiler.graph.NodeClass;
import org.graalvm.compiler.graph.spi.Canonicalizable;
import org.graalvm.compiler.graph.spi.CanonicalizerTool;
import org.graalvm.compiler.nodeinfo.NodeInfo;
import org.graalvm.compiler.nodes.ConstantNode;
import org.graalvm.compiler.nodes.NodeView;
import org.graalvm.compiler.nodes.ValueNode;
import org.graalvm.compiler.nodes.calc.FloatingNode;
import org.graalvm.compiler.nodes.spi.Lowerable;
import org.graalvm.compiler.nodes.spi.StampProvider;
import org.graalvm.compiler.nodes.spi.Virtualizable;
import org.graalvm.compiler.nodes.spi.VirtualizerTool;
import org.graalvm.compiler.nodes.type.StampTool;
import jdk.vm.ci.meta.ConstantReflectionProvider;
import jdk.vm.ci.meta.MetaAccessProvider;
@NodeInfo(cycles = CYCLES_2, size = SIZE_1)
public final class LoadHubNode extends FloatingNode implements Lowerable, Canonicalizable, Virtualizable {
public static final NodeClass<LoadHubNode> TYPE = NodeClass.create(LoadHubNode.class);
@Input ValueNode value;
public ValueNode getValue() {
return value;
}
private static AbstractPointerStamp hubStamp(StampProvider stampProvider, ValueNode value) {
assert value.stamp(NodeView.DEFAULT) instanceof ObjectStamp;
return stampProvider.createHubStamp(((ObjectStamp) value.stamp(NodeView.DEFAULT)));
}
public static ValueNode create(ValueNode value, StampProvider stampProvider, MetaAccessProvider metaAccess, ConstantReflectionProvider constantReflection) {
final AbstractPointerStamp stamp = hubStamp(stampProvider, value);
return create(value, stamp, metaAccess, constantReflection);
}
public static ValueNode create(ValueNode value, AbstractPointerStamp stamp, MetaAccessProvider metaAccess, ConstantReflectionProvider constantReflection) {
ValueNode synonym = findSynonym(value, stamp, metaAccess, constantReflection);
if (synonym != null) {
return synonym;
}
return new LoadHubNode(stamp, value);
}
public LoadHubNode(@InjectedNodeParameter StampProvider stampProvider, ValueNode value) {
this(hubStamp(stampProvider, value), value);
}
public LoadHubNode(Stamp stamp, ValueNode value) {
super(TYPE, stamp);
this.value = value;
}
@Override
public ValueNode canonical(CanonicalizerTool tool) {
if (!GeneratePIC.getValue(tool.getOptions())) {
NodeView view = NodeView.from(tool);
MetaAccessProvider metaAccess = tool.getMetaAccess();
ValueNode curValue = getValue();
ValueNode newNode = findSynonym(curValue, stamp(view), metaAccess, tool.getConstantReflection());
if (newNode != null) {
return newNode;
}
}
return this;
}
public static ValueNode findSynonym(ValueNode curValue, Stamp stamp, MetaAccessProvider metaAccess, ConstantReflectionProvider constantReflection) {
if (!GeneratePIC.getValue(curValue.getOptions())) {
TypeReference type = StampTool.typeReferenceOrNull(curValue);
if (type != null && type.isExact()) {
return ConstantNode.forConstant(stamp, constantReflection.asObjectHub(type.getType()), metaAccess);
}
}
return null;
}
@Override
public void virtualize(VirtualizerTool tool) {
if (!GeneratePIC.getValue(tool.getOptions())) {
ValueNode alias = tool.getAlias(getValue());
TypeReference type = StampTool.typeReferenceOrNull(alias);
if (type != null && type.isExact()) {
tool.replaceWithValue(ConstantNode.forConstant(stamp(NodeView.DEFAULT), tool.getConstantReflection().asObjectHub(type.getType()), tool.getMetaAccess(), graph()));
}
}
}
}