/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF 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.
*
*/
/* Generated By:JJTree: Do not edit this line. ASTIfExpr.java */
/* JJT: 0.3pre1 */
package Mini;
import org.apache.bcel.generic.BranchHandle;
import org.apache.bcel.generic.ConstantPoolGen;
import org.apache.bcel.generic.GOTO;
import org.apache.bcel.generic.IFEQ;
import org.apache.bcel.generic.InstructionConstants;
import org.apache.bcel.generic.InstructionList;
import org.apache.bcel.generic.MethodGen;
/**
*
* @version $Id$
*/
public class ASTIfExpr extends ASTExpr implements org.apache.bcel.Constants {
private ASTExpr if_expr, then_expr, else_expr;
// Generated methods
ASTIfExpr(int id) {
super(id);
}
ASTIfExpr(MiniParser p, int id) {
super(p, id);
}
public static Node jjtCreate(MiniParser p, int id) {
return new ASTIfExpr(p, id);
}
/**
* Overrides ASTExpr.closeNode()
* Cast children nodes Node[] to appropiate type ASTExpr[]
*/
@Override
public void closeNode() {
if_expr = (ASTExpr)children[0];
then_expr = (ASTExpr)children[1];
if(children.length == 3) {
else_expr = (ASTExpr)children[2];
} else {
MiniC.addError(if_expr.getLine(), if_expr.getColumn(),
"IF expression has no ELSE branch");
}
children=null; // Throw away
}
/**
* Overrides ASTExpr.traverse()
*/
@Override
public ASTExpr traverse(Environment env) {
this.env = env;
if_expr = if_expr.traverse(env);
then_expr = then_expr.traverse(env);
if(else_expr != null) {
else_expr = else_expr.traverse(env);
}
return this;
}
/**
* Second pass
* Overrides AstExpr.eval()
* @return type of expression
* @param expected type
*/
@Override
public int eval(int expected) {
int then_type, else_type, if_type;
if((if_type=if_expr.eval(T_BOOLEAN)) != T_BOOLEAN) {
MiniC.addError(if_expr.getLine(), if_expr.getColumn(),
"IF expression is not of type boolean, but " +
TYPE_NAMES[if_type] + ".");
}
then_type=then_expr.eval(expected);
if((expected != T_UNKNOWN) && (then_type != expected)) {
MiniC.addError(then_expr.getLine(), then_expr.getColumn(),
"THEN expression is not of expected type " +
TYPE_NAMES[expected] + " but " + TYPE_NAMES[then_type] + ".");
}
if(else_expr != null) {
else_type = else_expr.eval(expected);
if((expected != T_UNKNOWN) && (else_type != expected)) {
MiniC.addError(else_expr.getLine(), else_expr.getColumn(),
"ELSE expression is not of expected type " +
TYPE_NAMES[expected] + " but " + TYPE_NAMES[else_type] + ".");
} else if(then_type == T_UNKNOWN) {
then_type = else_type;
then_expr.setType(else_type);
}
}
else {
else_type = then_type;
else_expr = then_expr;
}
if(then_type != else_type) {
MiniC.addError(line, column,
"Type mismatch in THEN-ELSE: " +
TYPE_NAMES[then_type] + " vs. " + TYPE_NAMES[else_type] + ".");
}
type = then_type;
is_simple = if_expr.isSimple() && then_expr.isSimple() && else_expr.isSimple();
return type;
}
/**
* Fourth pass, produce Java code.
*/
@Override
public void code(StringBuffer buf) {
if_expr.code(buf);
buf.append(" if(" + ASTFunDecl.pop() + " == 1) {\n");
int size = ASTFunDecl.size;
then_expr.code(buf);
ASTFunDecl.size = size; // reset stack
buf.append(" } else {\n");
else_expr.code(buf);
buf.append(" }\n");
}
/**
* Fifth pass, produce Java byte code.
*/
@Override
public void byte_code(InstructionList il, MethodGen method, ConstantPoolGen cp) {
if_expr.byte_code(il, method, cp);
InstructionList then_code = new InstructionList();
InstructionList else_code = new InstructionList();
then_expr.byte_code(then_code, method, cp);
else_expr.byte_code(else_code, method, cp);
BranchHandle i, g;
i = il.append(new IFEQ(null)); // If POP() == FALSE(i.e. 0) then branch to ELSE
ASTFunDecl.pop();
il.append(then_code);
g = il.append(new GOTO(null));
i.setTarget(il.append(else_code));
g.setTarget(il.append(InstructionConstants.NOP)); // May be optimized away later
}
@Override
public void dump(String prefix) {
System.out.println(toString(prefix));
if_expr.dump(prefix + " ");
then_expr.dump(prefix + " ");
if(else_expr != null) {
else_expr.dump(prefix + " ");
}
}
}