blob: be83c3144c68d2dc5bf897fb0285fcf47d9a5d4d [file] [log] [blame]
// Copyright 2014 The Bazel Authors. All rights reserved.
//
// 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
//
// http://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 com.google.devtools.build.lib.syntax;
import static com.google.devtools.build.lib.syntax.compiler.ByteCodeUtils.append;
import com.google.devtools.build.lib.syntax.compiler.ByteCodeUtils;
import com.google.devtools.build.lib.syntax.compiler.DebugInfo;
import com.google.devtools.build.lib.syntax.compiler.Jump;
import com.google.devtools.build.lib.syntax.compiler.Jump.PrimitiveComparison;
import com.google.devtools.build.lib.syntax.compiler.LabelAdder;
import com.google.devtools.build.lib.syntax.compiler.VariableScope;
import net.bytebuddy.implementation.bytecode.ByteCodeAppender;
import java.util.ArrayList;
import java.util.List;
/**
* Syntax node for an if/else expression.
*/
public final class ConditionalExpression extends Expression {
// Python conditional expressions: $thenCase if $condition else $elseCase
// https://docs.python.org/3.5/reference/expressions.html#conditional-expressions
private final Expression thenCase;
private final Expression condition;
private final Expression elseCase;
public Expression getThenCase() { return thenCase; }
public Expression getCondition() { return condition; }
public Expression getElseCase() { return elseCase; }
/**
* Constructor for a conditional expression
*/
public ConditionalExpression(
Expression thenCase, Expression condition, Expression elseCase) {
this.thenCase = thenCase;
this.condition = condition;
this.elseCase = elseCase;
}
/**
* Constructs a string representation of the if expression
*/
@Override
public String toString() {
return thenCase + " if " + condition + " else " + elseCase;
}
@Override
Object doEval(Environment env) throws EvalException, InterruptedException {
if (EvalUtils.toBoolean(condition.eval(env))) {
return thenCase.eval(env);
} else {
return elseCase.eval(env);
}
}
@Override
public void accept(SyntaxTreeVisitor visitor) {
visitor.visit(this);
}
@Override
void validate(ValidationEnvironment env) throws EvalException {
condition.validate(env);
thenCase.validate(env);
elseCase.validate(env);
}
@Override
ByteCodeAppender compile(VariableScope scope, DebugInfo debugInfo) throws EvalException {
List<ByteCodeAppender> code = new ArrayList<>();
LabelAdder afterLabel = new LabelAdder();
LabelAdder elseLabel = new LabelAdder();
// compile condition and convert to boolean
code.add(condition.compile(scope, debugInfo));
append(
code,
EvalUtils.toBoolean,
// jump to else block if false
Jump.ifIntOperandToZero(PrimitiveComparison.EQUAL).to(elseLabel));
// otherwise evaluate the expression for "then" and jump to end
code.add(thenCase.compile(scope, debugInfo));
append(
code,
Jump.to(afterLabel),
// add label for "else" and evaluate the expression
elseLabel);
code.add(elseCase.compile(scope, debugInfo));
append(code, afterLabel);
return ByteCodeUtils.compoundAppender(code);
}
}