/* * Licensed to Elasticsearch under one or more contributor * license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright * ownership. Elasticsearch licenses this file to you 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 org.elasticsearch.painless.node; import org.elasticsearch.painless.Definition; import org.elasticsearch.painless.Globals; import org.elasticsearch.painless.Locals; import org.elasticsearch.painless.Location; import org.elasticsearch.painless.MethodWriter; import org.objectweb.asm.Label; import org.objectweb.asm.Opcodes; import java.util.Arrays; import java.util.Objects; import java.util.Set; import static java.util.Collections.singleton; /** * Represents an if/else block. */ public final class SIfElse extends AStatement { private AExpression condition; private final SBlock ifblock; private final SBlock elseblock; public SIfElse(Location location, AExpression condition, SBlock ifblock, SBlock elseblock) { super(location); this.condition = Objects.requireNonNull(condition); this.ifblock = ifblock; this.elseblock = elseblock; } @Override void extractVariables(Set<String> variables) { condition.extractVariables(variables); if (ifblock != null) { ifblock.extractVariables(variables); } if (elseblock != null) { elseblock.extractVariables(variables); } } @Override void analyze(Locals locals) { condition.expected = Definition.BOOLEAN_TYPE; condition.analyze(locals); condition = condition.cast(locals); if (condition.constant != null) { throw createError(new IllegalArgumentException("Extraneous if statement.")); } if (ifblock == null) { throw createError(new IllegalArgumentException("Extraneous if statement.")); } ifblock.lastSource = lastSource; ifblock.inLoop = inLoop; ifblock.lastLoop = lastLoop; ifblock.analyze(Locals.newLocalScope(locals)); anyContinue = ifblock.anyContinue; anyBreak = ifblock.anyBreak; statementCount = ifblock.statementCount; if (elseblock == null) { throw createError(new IllegalArgumentException("Extraneous else statement.")); } elseblock.lastSource = lastSource; elseblock.inLoop = inLoop; elseblock.lastLoop = lastLoop; elseblock.analyze(Locals.newLocalScope(locals)); methodEscape = ifblock.methodEscape && elseblock.methodEscape; loopEscape = ifblock.loopEscape && elseblock.loopEscape; allEscape = ifblock.allEscape && elseblock.allEscape; anyContinue |= elseblock.anyContinue; anyBreak |= elseblock.anyBreak; statementCount = Math.max(ifblock.statementCount, elseblock.statementCount); } @Override void write(MethodWriter writer, Globals globals) { writer.writeStatementOffset(location); Label fals = new Label(); Label end = new Label(); condition.write(writer, globals); writer.ifZCmp(Opcodes.IFEQ, fals); ifblock.continu = continu; ifblock.brake = brake; ifblock.write(writer, globals); if (!ifblock.allEscape) { writer.goTo(end); } writer.mark(fals); elseblock.continu = continu; elseblock.brake = brake; elseblock.write(writer, globals); writer.mark(end); } @Override public String toString() { return multilineToString(singleton(condition), Arrays.asList(ifblock, elseblock)); } }