io.micronaut.sourcegen.bytecode.AbstractConditionalWriter Maven / Gradle / Ivy
/*
* Copyright 2017-2024 original authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.micronaut.sourcegen.bytecode;
import io.micronaut.core.annotation.Internal;
import io.micronaut.sourcegen.bytecode.expression.ExpressionWriter;
import io.micronaut.sourcegen.model.ExpressionDef;
import io.micronaut.sourcegen.model.JavaIdioms;
import io.micronaut.sourcegen.model.TypeDef;
import org.objectweb.asm.Label;
import org.objectweb.asm.Type;
import org.objectweb.asm.commons.GeneratorAdapter;
/**
* The common condition writer methods.
*
* @author Denis Stepanov
* @since 1.5
*/
@Internal
public abstract class AbstractConditionalWriter {
protected static void pushElseConditionalExpression(GeneratorAdapter generatorAdapter,
MethodContext context,
ExpressionDef expressionDef,
Label elseLabel) {
if (expressionDef instanceof ExpressionDef.ConditionExpressionDef conditionExpressionDef) {
if (expressionDef instanceof ExpressionDef.InstanceOf instanceOf) {
ExpressionWriter.writeExpression(generatorAdapter, context, instanceOf.expression());
generatorAdapter.instanceOf(TypeUtils.getType(instanceOf.instanceType(), context.objectDef()));
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.NE, elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.And andExpressionDef) {
pushElseConditionalExpression(generatorAdapter, context, andExpressionDef.left(), elseLabel);
pushElseConditionalExpression(generatorAdapter, context, andExpressionDef.right(), elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.Or orExpressionDef) {
Label ifLabel = new Label();
pushIfConditionalExpression(generatorAdapter, context, orExpressionDef.left(), ifLabel);
pushIfConditionalExpression(generatorAdapter, context, orExpressionDef.right(), ifLabel);
generatorAdapter.goTo(elseLabel);
generatorAdapter.visitLabel(ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.Condition condition) {
ExpressionWriter.writeExpression(generatorAdapter, context, condition.left());
ExpressionWriter.writeExpression(generatorAdapter, context, condition.right());
Type conditionType = TypeUtils.getType(condition.left().type(), context.objectDef());
generatorAdapter.ifCmp(conditionType, getInvertConditionOp(condition.operator()), elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsNull isNull) {
ExpressionWriter.writeExpression(generatorAdapter, context, isNull.expression());
generatorAdapter.ifNonNull(elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsNotNull isNotNull) {
ExpressionWriter.writeExpression(generatorAdapter, context, isNotNull.expression());
generatorAdapter.ifNull(elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsTrue isTrue) {
ExpressionWriter.writeExpression(generatorAdapter, context, isTrue.expression());
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.NE, elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsFalse isFalse) {
ExpressionWriter.writeExpression(generatorAdapter, context, isFalse.expression());
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.EQ, elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.EqualsReferentially equalsReferentially) {
pushEqualsReferentially(generatorAdapter, context, equalsReferentially, elseLabel, GeneratorAdapter.NE);
return;
}
if (expressionDef instanceof ExpressionDef.EqualsStructurally equalsStructurally) {
pushEqualsStructurally(generatorAdapter, context, equalsStructurally, elseLabel, GeneratorAdapter.NE);
return;
}
throw new UnsupportedOperationException("Unrecognized conditional expression: " + conditionExpressionDef);
}
if (!expressionDef.type().equals(TypeDef.Primitive.BOOLEAN) && !expressionDef.type().equals(TypeDef.Primitive.BOOLEAN.wrapperType())) {
throw new IllegalStateException("Conditional expression should produce a boolean: " + expressionDef);
}
ExpressionWriter.writeExpressionCheckCast(generatorAdapter, context, expressionDef, TypeDef.Primitive.BOOLEAN);
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.NE, elseLabel);
}
private static void pushIfConditionalExpression(GeneratorAdapter generatorAdapter,
MethodContext context,
ExpressionDef expressionDef,
Label ifLabel) {
if (expressionDef instanceof ExpressionDef.ConditionExpressionDef conditionExpressionDef) {
if (expressionDef instanceof ExpressionDef.InstanceOf instanceOf) {
ExpressionWriter.writeExpression(generatorAdapter, context, instanceOf.expression());
generatorAdapter.instanceOf(TypeUtils.getType(instanceOf.instanceType(), context.objectDef()));
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.EQ, ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.And andExpressionDef) {
Label elseLabel = new Label();
pushElseConditionalExpression(generatorAdapter, context, andExpressionDef.left(), elseLabel);
pushElseConditionalExpression(generatorAdapter, context, andExpressionDef.right(), elseLabel);
generatorAdapter.goTo(ifLabel);
generatorAdapter.visitLabel(elseLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.Or orExpressionDef) {
pushIfConditionalExpression(generatorAdapter, context, orExpressionDef.left(), ifLabel);
pushIfConditionalExpression(generatorAdapter, context, orExpressionDef.right(), ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.Condition condition) {
ExpressionWriter.writeExpression(generatorAdapter, context, condition.left());
ExpressionWriter.writeExpression(generatorAdapter, context, condition.right());
Type conditionType = TypeUtils.getType(condition.left().type(), context.objectDef());
generatorAdapter.ifCmp(conditionType, getConditionOp(condition.operator()), ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsNull isNull) {
ExpressionWriter.writeExpression(generatorAdapter, context, isNull.expression());
generatorAdapter.ifNull(ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsNotNull isNotNull) {
ExpressionWriter.writeExpression(generatorAdapter, context, isNotNull.expression());
generatorAdapter.ifNonNull(ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsTrue isTrue) {
ExpressionWriter.writeExpression(generatorAdapter, context, isTrue.expression());
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.EQ, ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.IsFalse isFalse) {
ExpressionWriter.writeExpression(generatorAdapter, context, isFalse.expression());
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.NE, ifLabel);
return;
}
if (conditionExpressionDef instanceof ExpressionDef.EqualsReferentially equalsReferentially) {
pushEqualsReferentially(generatorAdapter, context, equalsReferentially, ifLabel, GeneratorAdapter.EQ);
return;
}
if (expressionDef instanceof ExpressionDef.EqualsStructurally equalsStructurally) {
pushEqualsStructurally(generatorAdapter, context, equalsStructurally, ifLabel, GeneratorAdapter.EQ);
return;
}
throw new UnsupportedOperationException("Unrecognized conditional expression: " + conditionExpressionDef);
}
if (!expressionDef.type().equals(TypeDef.Primitive.BOOLEAN) && !expressionDef.type().equals(TypeDef.Primitive.BOOLEAN.wrapperType())) {
throw new IllegalStateException("Conditional expression should produce a boolean: " + expressionDef);
}
ExpressionWriter.writeExpressionCheckCast(generatorAdapter, context, expressionDef, TypeDef.Primitive.BOOLEAN);
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, GeneratorAdapter.EQ, ifLabel);
}
private static void pushEqualsStructurally(GeneratorAdapter generatorAdapter, MethodContext context, ExpressionDef.EqualsStructurally equalsStructurally, Label ifLabel, int op) {
TypeDef leftType = equalsStructurally.instance().type();
TypeDef rightType = equalsStructurally.other().type();
if (leftType.isPrimitive()) {
pushEqualsReferentially(generatorAdapter, context, equalsStructurally.instance(), equalsStructurally.other().cast(leftType), ifLabel, op);
return;
}
if (rightType.isPrimitive()) {
pushEqualsReferentially(generatorAdapter, context, equalsStructurally.instance().cast(rightType), equalsStructurally.other(), ifLabel, op);
return;
}
pushEqualsStructurally(generatorAdapter, context, equalsStructurally);
generatorAdapter.push(true);
generatorAdapter.ifCmp(Type.BOOLEAN_TYPE, op, ifLabel);
}
private static void pushEqualsReferentially(GeneratorAdapter generatorAdapter, MethodContext context,
ExpressionDef.EqualsReferentially equalsReferentially,
Label label, int op) {
ExpressionDef left = equalsReferentially.instance();
ExpressionDef right = equalsReferentially.other();
pushEqualsReferentially(generatorAdapter, context, left, right, label, op);
}
private static void pushEqualsReferentially(GeneratorAdapter generatorAdapter,
MethodContext context,
ExpressionDef left,
ExpressionDef right,
Label label,
int op) {
TypeDef leftType = left.type();
ExpressionWriter.writeExpression(generatorAdapter, context, left);
TypeDef rightType = right.type();
ExpressionWriter.writeExpression(generatorAdapter, context, right);
if (leftType instanceof TypeDef.Primitive p1 && rightType instanceof TypeDef.Primitive) {
generatorAdapter.ifCmp(TypeUtils.getType(p1), op, label);
} else {
generatorAdapter.ifCmp(TypeUtils.OBJECT_TYPE, op, label);
}
}
private static void pushEqualsStructurally(GeneratorAdapter generatorAdapter, MethodContext context, ExpressionDef.EqualsStructurally equalsStructurally) {
ExpressionWriter.writeExpressionCheckCast(generatorAdapter, context, JavaIdioms.equalsStructurally(equalsStructurally), TypeDef.Primitive.BOOLEAN);
}
private static int getInvertConditionOp(String op) {
int conditionOp = getConditionOp(op);
return switch (conditionOp) {
case GeneratorAdapter.EQ -> GeneratorAdapter.NE;
case GeneratorAdapter.NE -> GeneratorAdapter.EQ;
default ->
throw new UnsupportedOperationException("Unrecognized condition operator: " + conditionOp);
};
}
private static int getConditionOp(String op) {
return switch (op.trim()) {
case "==" -> GeneratorAdapter.EQ;
case "!=" -> GeneratorAdapter.NE;
default ->
throw new UnsupportedOperationException("Unrecognized condition operator: " + op);
};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy