1 /* 2 * Licensed to the Apache Software Foundation (ASF) under one or more 3 * contributor license agreements. See the NOTICE file distributed with 4 * this work for additional information regarding copyright ownership. 5 * The ASF licenses this file to You under the Apache License, Version 2.0 6 * (the "License"); you may not use this file except in compliance with 7 * the License. You may obtain a copy of the License at 8 * 9 * http://www.apache.org/licenses/LICENSE-2.0 10 * 11 * Unless required by applicable law or agreed to in writing, software 12 * distributed under the License is distributed on an "AS IS" BASIS, 13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 * See the License for the specific language governing permissions and 15 * limitations under the License. 16 * 17 */ 18 /* Generated By:JJTree: Do not edit this line. ASTExpr.java */ 19 /* JJT: 0.3pre1 */ 20 21 package Mini; 22 import org.apache.bcel.generic.BranchHandle; 23 import org.apache.bcel.generic.ConstantPoolGen; 24 import org.apache.bcel.generic.GOTO; 25 import org.apache.bcel.generic.IF_ICMPEQ; 26 import org.apache.bcel.generic.IF_ICMPGE; 27 import org.apache.bcel.generic.IF_ICMPGT; 28 import org.apache.bcel.generic.IF_ICMPLE; 29 import org.apache.bcel.generic.IF_ICMPLT; 30 import org.apache.bcel.generic.IF_ICMPNE; 31 import org.apache.bcel.generic.InstructionConstants; 32 import org.apache.bcel.generic.InstructionList; 33 import org.apache.bcel.generic.MethodGen; 34 import org.apache.bcel.generic.PUSH; 35 36 /** 37 * Represents arithmetic expressions such as `(a + 12 == b) OR c'. 38 * The parse tree is built initially by the parser and modified, i.e. 39 * compacted with `traverse()'. Each (Expr, Term, Factor) node 40 * with kind == -1 is replaced which its successor node, which is 41 * converted to type `Expr' 42 * 43 * A node with kind == -1 denotes the fact that this expression 44 * node has just one child branch and thus may be replaced by this 45 * branch (or leaf) directly without altering the expression 46 * semantics. Term and Factor nodes are used only to build the parse tree 47 * obeying the aritmetical precedences (* stronger than +, etc.) and 48 * are discarded in the first pass. 49 */ 50 public class ASTExpr extends SimpleNode 51 implements MiniParserConstants, MiniParserTreeConstants, org.apache.bcel.Constants { 52 protected int kind=-1; // Single twig to leaf? 53 private int unop=-1; // Special case: Unary operand applied 54 protected ASTExpr[] exprs; // Sub expressions 55 protected Environment env; // Needed in all passes 56 protected int line, column; 57 protected boolean is_simple; // true, if simple expression like `12 + f(a)' 58 59 /* Not all children shall inherit this, exceptions are ASTIdent and ASTFunAppl, which 60 * look up the type in the corresponding environment entry. 61 */ 62 protected int type = T_UNKNOWN; 63 64 // Generated methods ASTExpr(int id)65 ASTExpr(int id) { 66 super(id); 67 } 68 ASTExpr(MiniParser p, int id)69 ASTExpr(MiniParser p, int id) { 70 super(p, id); 71 } 72 jjtCreate(MiniParser p, int id)73 public static Node jjtCreate(MiniParser p, int id) { 74 return new ASTExpr(p, id); 75 } 76 ASTExpr(int line, int column, int id)77 ASTExpr(int line, int column, int id) { 78 super(id); 79 this.line = line; 80 this.column = column; 81 } 82 ASTExpr(int line, int column, int kind, int id)83 ASTExpr(int line, int column, int kind, int id) { 84 this(line, column, id); 85 this.kind = kind; 86 } 87 88 /* Special constructor, called from ASTTerm.traverse() and 89 * ASTFactor.traverse(), when traverse()ing the parse tree replace 90 * themselves with Expr nodes. 91 */ ASTExpr(ASTExpr[] children, int kind, int line, int column)92 ASTExpr(ASTExpr[] children, int kind, int line, int column) { 93 this(line, column, kind, JJTEXPR); 94 exprs = children; 95 } 96 97 /** 98 * @return name of node, its kind and the number of children. 99 */ 100 @Override toString()101 public String toString() { 102 String op=""; 103 int len = (children != null)? children.length : 0; 104 if(unop != -1) { 105 op = tokenImage[unop]; 106 } else if(kind != -1) { 107 op = tokenImage[kind]; 108 } 109 110 return jjtNodeName[id] + "(" + op + ")[" + len + "]<" + 111 TYPE_NAMES[type] + "> @" + line + ", " + column; 112 } 113 114 /** 115 * Overrides SimpleNode.closeNode(). Overridden by some subclasses. 116 * 117 * Called by the parser when the construction of this node is finished. 118 * Casts children Node[] to precise ASTExpr[] type. 119 */ 120 @Override closeNode()121 public void closeNode() { 122 if(children != null) { 123 exprs = new ASTExpr[children.length]; 124 System.arraycopy(children, 0, exprs, 0, children.length); 125 children=null; // Throw away old reference 126 } 127 } 128 129 /** 130 * First pass 131 * Overridden by subclasses. Traverse the whole parse tree recursively and 132 * drop redundant nodes. 133 */ traverse(Environment env)134 public ASTExpr traverse(Environment env) { 135 this.env = env; 136 137 if((kind == -1) && (unop == -1)) { 138 return exprs[0].traverse(env); // --> Replaced by successor 139 } else { 140 for(int i=0; i < exprs.length; i++) { 141 exprs[i] = exprs[i].traverse(env); // References may change 142 } 143 144 return this; 145 } 146 } 147 148 /** 149 * Second and third pass 150 * @return type of expression 151 * @param expected type 152 */ eval(int expected)153 public int eval(int expected) { 154 int child_type = T_UNKNOWN, t; 155 156 is_simple = true; 157 158 // Determine expected node type depending on used operator. 159 if(unop != -1) { 160 if(unop == MINUS) { 161 child_type = type = T_INT; // - 162 } else { 163 child_type = type = T_BOOLEAN; // ! 164 } 165 } 166 else { 167 // Compute expected type 168 if((kind == PLUS) || (kind == MINUS) || (kind == MULT) || 169 (kind == MOD) || (kind == DIV)) { 170 child_type = type = T_INT; 171 } else if((kind == AND) || (kind == OR)) { 172 child_type = type = T_BOOLEAN; 173 } else { // LEQ, GT, etc. 174 child_type = T_INT; 175 type = T_BOOLEAN; 176 } 177 } 178 179 // Get type of subexpressions 180 for(int i=0; i < exprs.length; i++) { 181 t = exprs[i].eval(child_type); 182 183 if(t != child_type) { 184 MiniC.addError(exprs[i].getLine(), exprs[i].getColumn(), 185 "Expression has not expected type " + TYPE_NAMES[child_type] + 186 " but " + TYPE_NAMES[t] + "."); 187 } 188 189 is_simple = is_simple && exprs[i].isSimple(); 190 } 191 192 return type; 193 } 194 toBool(String i)195 private static String toBool(String i) { 196 return "(" + i + " != 0)"; 197 } 198 toInt(String i)199 private static String toInt(String i) { 200 return "((" + i + ")? 1 : 0)"; 201 } 202 203 /** 204 * Fourth pass, produce Java code. 205 */ code(StringBuffer buf)206 public void code(StringBuffer buf) { 207 if(unop != -1) { 208 exprs[0].code(buf); 209 String top = ASTFunDecl.pop(); 210 if(unop == MINUS) { 211 ASTFunDecl.push(buf, "-" + top); 212 } else { 213 ASTFunDecl.push(buf, "(" + top + " == 1)? 0 : 1)"); 214 } 215 } 216 else { 217 exprs[0].code(buf); 218 exprs[1].code(buf); 219 String _body_int2 = ASTFunDecl.pop(); 220 String _body_int = ASTFunDecl.pop(); 221 222 switch(kind) { 223 case PLUS: ASTFunDecl.push(buf, _body_int + " + " + _body_int2); break; 224 case MINUS: ASTFunDecl.push(buf, _body_int + " - " + _body_int2); break; 225 case MULT: ASTFunDecl.push(buf, _body_int + " * " + _body_int2); break; 226 case DIV: ASTFunDecl.push(buf, _body_int + " / " + _body_int2); break; 227 228 case AND: ASTFunDecl.push(buf, toInt(toBool(_body_int) + " && " + 229 toBool(_body_int2))); break; 230 case OR: ASTFunDecl.push(buf, toInt(toBool(_body_int) + " || " + 231 toBool(_body_int2))); break; 232 233 case EQ: ASTFunDecl.push(buf, toInt(_body_int + " == " + _body_int2)); 234 break; 235 case LEQ: ASTFunDecl.push(buf, toInt(_body_int + " <= " + _body_int2)); 236 break; 237 case GEQ: ASTFunDecl.push(buf, toInt(_body_int + " >= " + _body_int2)); 238 break; 239 case NEQ: ASTFunDecl.push(buf, toInt(_body_int + " != " + _body_int2)); 240 break; 241 case LT: ASTFunDecl.push(buf, toInt(_body_int + " < " + _body_int2)); 242 break; 243 case GT: ASTFunDecl.push(buf, toInt(_body_int + " > " + _body_int2)); 244 break; 245 246 default: System.err.println("Ooops"); 247 } 248 } 249 } 250 251 /** 252 * Fifth pass, produce Java byte code. 253 */ byte_code(InstructionList il, MethodGen method, ConstantPoolGen cp)254 public void byte_code(InstructionList il, MethodGen method, ConstantPoolGen cp) { 255 exprs[0].byte_code(il, method, cp); 256 257 if(unop != -1) { // Apply unary operand 258 if(unop == MINUS) { 259 il.append(InstructionConstants.INEG); 260 } else { // == NOT 261 il.append(new PUSH(cp, 1)); ASTFunDecl.push(); // Push TRUE 262 il.append(InstructionConstants.IXOR); ASTFunDecl.pop(); 263 } 264 } 265 else { // Apply binary operand 266 BranchHandle bh=null; 267 268 exprs[1].byte_code(il, method, cp); 269 270 switch(kind) { 271 case PLUS: il.append(InstructionConstants.IADD); ASTFunDecl.pop(); break; 272 case MINUS: il.append(InstructionConstants.ISUB); ASTFunDecl.pop(); break; 273 case MULT: il.append(InstructionConstants.IMUL); ASTFunDecl.pop(); break; 274 case DIV: il.append(InstructionConstants.IDIV); ASTFunDecl.pop(); break; 275 case AND: il.append(InstructionConstants.IAND); ASTFunDecl.pop(); break; 276 case OR: il.append(InstructionConstants.IOR); ASTFunDecl.pop(); break; 277 278 /* Use negated operands */ 279 case EQ: bh = il.append(new IF_ICMPNE(null)); ASTFunDecl.pop(2); break; 280 case LEQ: bh = il.append(new IF_ICMPGT(null)); ASTFunDecl.pop(2); break; 281 case GEQ: bh = il.append(new IF_ICMPLT(null)); ASTFunDecl.pop(2); break; 282 case NEQ: bh = il.append(new IF_ICMPEQ(null)); ASTFunDecl.pop(2); break; 283 case LT: bh = il.append(new IF_ICMPGE(null)); ASTFunDecl.pop(2); break; 284 case GT: bh = il.append(new IF_ICMPLE(null)); ASTFunDecl.pop(2); break; 285 default: System.err.println("Ooops"); 286 } 287 288 switch(kind) { 289 case EQ: case LEQ: case GEQ: case NEQ: case LT: case GT: 290 BranchHandle g; 291 292 il.append(new PUSH(cp, 1)); 293 g = il.append(new GOTO(null)); 294 bh.setTarget(il.append(new PUSH(cp, 0))); 295 g.setTarget(il.append(InstructionConstants.NOP)); // May be optimized away later 296 ASTFunDecl.push(); 297 break; 298 299 default: break; 300 } 301 } 302 } 303 isSimple()304 public boolean isSimple() { return is_simple; } setType(int type)305 public void setType(int type) { this.type = type; } getType()306 public int getType() { return type; } setKind(int kind)307 public void setKind(int kind) { this.kind = kind; } getKind()308 public int getKind() { return kind; } setUnOp(int unop)309 public void setUnOp(int unop) { this.unop = unop; } getUnOp()310 public int getUnOp() { return unop; } setLine(int line)311 public void setLine(int line) { this.line = line; } getLine()312 public int getLine() { return line; } setColumn(int column)313 public void setColumn(int column) { this.column = column; } getColumn()314 public int getColumn() { return column; } setPosition(int line, int column)315 public void setPosition(int line, int column) { 316 this.line = line; 317 this.column = column; 318 } 319 320 @Override dump(String prefix)321 public void dump(String prefix) { 322 System.out.println(toString(prefix)); 323 324 if(exprs != null) { 325 for(int i=0; i < exprs.length; ++i) { 326 exprs[i].dump(prefix + " "); 327 } 328 } 329 } 330 } 331