package org.graalvm.compiler.replacements.nodes;
import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_UNKNOWN;
import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_UNKNOWN;
import org.graalvm.compiler.core.common.GraalOptions;
import org.graalvm.compiler.core.common.type.StampFactory;
import org.graalvm.compiler.debug.GraalError;
import org.graalvm.compiler.graph.Node;
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.FixedWithNextNode;
import org.graalvm.compiler.nodes.ValueNode;
import org.graalvm.compiler.nodes.spi.LIRLowerable;
import org.graalvm.compiler.nodes.spi.Lowerable;
import org.graalvm.compiler.nodes.spi.LoweringTool;
import org.graalvm.compiler.nodes.spi.NodeLIRBuilderTool;
@NodeInfo(cycles = CYCLES_UNKNOWN, size = SIZE_UNKNOWN)
public final class AssertionNode extends FixedWithNextNode implements Lowerable, Canonicalizable, LIRLowerable {
public static final NodeClass<AssertionNode> TYPE = NodeClass.create(AssertionNode.class);
@Input ValueNode condition;
@Input ValueNode l1;
@Input ValueNode l2;
protected final boolean compileTimeAssertion;
protected final String message;
protected AssertionNode(@ConstantNodeParameter boolean compileTimeAssertion, ValueNode condition, @ConstantNodeParameter String message, @ConstantNodeParameter Object msgArg1,
@ConstantNodeParameter Object msgArg2,
ValueNode l1, ValueNode l2) {
super(TYPE, StampFactory.forVoid());
this.condition = condition;
this.compileTimeAssertion = compileTimeAssertion;
this.message = message + msgArg1 + msgArg2;
this.l1 = l1;
this.l2 = l2;
}
public ValueNode getL1() {
return l1;
}
public ValueNode getL2() {
return l2;
}
public ValueNode condition() {
return condition;
}
public String message() {
return message;
}
@Override
public Node canonical(CanonicalizerTool tool) {
if (condition.isConstant() && condition.asJavaConstant().asInt() != 0) {
return null;
}
return this;
}
@Override
public void lower(LoweringTool tool) {
if (!compileTimeAssertion) {
if (GraalOptions.ImmutableCode.getValue(getOptions())) {
graph().removeFixed(this);
} else {
tool.getLowerer().lower(this, tool);
}
}
}
@Override
public void generate(NodeLIRBuilderTool generator) {
assert compileTimeAssertion;
if (GraalOptions.ImmutableCode.getValue(getOptions())) {
return;
}
if (condition.isConstant()) {
if (condition.asJavaConstant().asInt() == 0) {
throw new GraalError("%s: failed compile-time assertion: %s", this, message);
}
} else {
throw new GraalError("%s: failed compile-time assertion (value %s): %s. Condition must be constant.", this, condition, message);
}
}
@NodeIntrinsic
public static native void assertion(@ConstantNodeParameter boolean compileTimeAssertion, boolean condition, @ConstantNodeParameter String message, @ConstantNodeParameter Object msgArg1,
@ConstantNodeParameter Object msgArg2, long arg1, long arg2);
public static void staticAssert(boolean condition, String message) {
assertion(true, condition, message, "", "", 0L, 0L);
}
public static void staticAssert(boolean condition, String message, Object msgArg1, Object msgArg2) {
assertion(true, condition, message, msgArg1, msgArg2, 0L, 0L);
}
public static void dynamicAssert(boolean condition, String message) {
assertion(false, condition, message, "", "", 0L, 0L);
}
public static void dynamicAssert(boolean condition, String message, long arg1, long arg2) {
assertion(false, condition, message, "", "", arg1, arg2);
}
}