package org.graalvm.compiler.nodes;
import java.util.stream.Collectors;
import org.graalvm.compiler.core.common.type.Stamp;
import org.graalvm.compiler.core.common.type.StampFactory;
import org.graalvm.compiler.graph.NodeClass;
import org.graalvm.compiler.graph.NodeInputList;
import org.graalvm.compiler.nodeinfo.NodeInfo;
import org.graalvm.compiler.nodes.spi.ArrayLengthProvider;
import org.graalvm.compiler.nodes.type.StampTool;
import org.graalvm.compiler.nodes.util.GraphUtil;
@NodeInfo(nameTemplate = "Phi({i#values})")
public class ValuePhiNode extends PhiNode implements ArrayLengthProvider {
public static final NodeClass<ValuePhiNode> TYPE = NodeClass.create(ValuePhiNode.class);
@Input protected NodeInputList<ValueNode> values;
public ValuePhiNode(Stamp stamp, AbstractMergeNode merge) {
this(TYPE, stamp, merge);
}
protected ValuePhiNode(NodeClass<? extends ValuePhiNode> c, Stamp stamp, AbstractMergeNode merge) {
super(c, stamp, merge);
assert stamp != StampFactory.forVoid();
values = new NodeInputList<>(this);
}
public ValuePhiNode(Stamp stamp, AbstractMergeNode merge, ValueNode[] values) {
super(TYPE, stamp, merge);
assert stamp != StampFactory.forVoid();
this.values = new NodeInputList<>(this, values);
}
@Override
public NodeInputList<ValueNode> values() {
return values;
}
@Override
public boolean inferStamp() {
Stamp valuesStamp = StampTool.meetOrNull(values(), this);
if (valuesStamp == null) {
valuesStamp = stamp;
} else if (stamp.isCompatible(valuesStamp)) {
valuesStamp = stamp.join(valuesStamp);
}
return updateStamp(valuesStamp);
}
@Override
public ValueNode length() {
if (merge() instanceof LoopBeginNode) {
return null;
}
ValueNode length = null;
for (ValueNode input : values()) {
ValueNode l = GraphUtil.arrayLength(input);
if (l == null) {
return null;
}
if (length == null) {
length = l;
} else if (length != l) {
return null;
}
}
return length;
}
@Override
public boolean verify() {
Stamp s = null;
for (ValueNode input : values()) {
if (s == null) {
s = input.stamp();
} else {
if (!s.isCompatible(input.stamp())) {
fail("Phi Input Stamps are not compatible. Phi:%s inputs:%s", this, values().stream().map(x -> x.toString() + ":" + x.stamp()).collect(Collectors.joining(", ")));
}
}
}
return super.verify();
}
}