TryStatement.java :  » 6.0-JDK-Modules-sun » tools » sun » tools » tree » Java Open Source

Java Open Source » 6.0 JDK Modules sun » tools 
tools » sun » tools » tree » TryStatement.java
/*
 * Copyright 1994-2003 Sun Microsystems, Inc.  All Rights Reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Sun designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Sun in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
 * CA 95054 USA or visit www.sun.com if you need additional information or
 * have any questions.
 */

package sun.tools.tree;

import sun.tools.java.*;
import sun.tools.asm.Assembler;
import sun.tools.asm.Label;
import sun.tools.asm.TryData;
import sun.tools.asm.CatchData;
import java.io.PrintStream;
import java.util.Enumeration;
import java.util.Hashtable;

/**
 * WARNING: The contents of this source file are not part of any
 * supported API.  Code that depends on them does so at its own risk:
 * they are subject to change or removal without notice.
 */
public
class TryStatement extends Statement {
    Statement body;
    Statement args[];
    long arrayCloneWhere;  // private note posted from MethodExpression

    /**
     * Constructor
     */
    public TryStatement(long where, Statement body, Statement args[]) {
  super(TRY, where);
  this.body = body;
  this.args = args;
    }

    /**
     * Check statement
     */
    Vset check(Environment env, Context ctx, Vset vset, Hashtable exp) {
  checkLabel(env, ctx);
  try {
      vset = reach(env, vset);
      Hashtable newexp = new Hashtable();
      CheckContext newctx =  new CheckContext(ctx, this);

      // Check 'try' block.  A variable is DA (DU) before the try
      // block if it is DA (DU) before the try statement.
      Vset vs = body.check(env, newctx, vset.copy(), newexp);

      // A variable is DA before a catch block if it is DA before the
      // try statement.  A variable is DU before a catch block if it
      // is DU after the try block and before any 'break', 'continue',
      // 'throw', or 'return' contained therein. That is, the variable
      // is DU upon entry to the try-statement and is not assigned to
      // anywhere within the try block.
      Vset cvs = Vset.firstDAandSecondDU(vset, vs.copy().join(newctx.vsTryExit));
      
      for (int i = 0 ; i < args.length ; i++) {
    // A variable is DA (DU) after a try statement if 
    // it is DA (DU) after every catch block.
    vs = vs.join(args[i].check(env, newctx, cvs.copy(), exp));
      }
      
      // Check that catch statements are actually reached
      for (int i = 1 ; i < args.length ; i++) {
    CatchStatement cs = (CatchStatement)args[i];
    if (cs.field == null) {
        continue;
    }
    Type type = cs.field.getType();
    ClassDefinition def = env.getClassDefinition(type);

    for (int j = 0 ; j < i ; j++) {
        CatchStatement cs2 = (CatchStatement)args[j];
        if (cs2.field == null) {
      continue;
        }
        Type t = cs2.field.getType();
        ClassDeclaration c = env.getClassDeclaration(t);
        if (def.subClassOf(env, c)) {
      env.error(args[i].where, "catch.not.reached");
      break;
        }
    }
      }

      ClassDeclaration ignore1 = env.getClassDeclaration(idJavaLangError);
      ClassDeclaration ignore2 = env.getClassDeclaration(idJavaLangRuntimeException);

      // Make sure the exception is actually throw in that part of the code
      for (int i = 0 ; i < args.length ; i++) {
    CatchStatement cs = (CatchStatement)args[i];
    if (cs.field == null) {
        continue;
    }
    Type type = cs.field.getType();
    if (!type.isType(TC_CLASS)) {
        // CatchStatement.checkValue() will have already printed
        // an error message
        continue;
    }

    ClassDefinition def = env.getClassDefinition(type);

    // Anyone can throw these!
    if (def.subClassOf(env, ignore1) || def.superClassOf(env, ignore1) || 
        def.subClassOf(env, ignore2) || def.superClassOf(env, ignore2)) {
        continue;
    }

    // Make sure the exception is actually throw in that part of the code
    boolean ok = false;
    for (Enumeration e = newexp.keys() ; e.hasMoreElements() ; ) {
        ClassDeclaration c = (ClassDeclaration)e.nextElement();
        if (def.superClassOf(env, c) || def.subClassOf(env, c)) {
      ok = true;
      break;
        }
    }
    if (!ok && arrayCloneWhere != 0
        && def.getName().toString().equals("java.lang.CloneNotSupportedException")) {
        env.error(arrayCloneWhere, "warn.array.clone.supported", def.getName());
    }

    if (!ok) {
        env.error(cs.where, "catch.not.thrown", def.getName());
    }
      }

      // Only carry over exceptions that are not caught
      for (Enumeration e = newexp.keys() ; e.hasMoreElements() ; ) {
    ClassDeclaration c = (ClassDeclaration)e.nextElement();
    ClassDefinition def = c.getClassDefinition(env);
    boolean add = true;
    for (int i = 0 ; i < args.length ; i++) {
        CatchStatement cs = (CatchStatement)args[i];
        if (cs.field == null) {
      continue;
        }
        Type type = cs.field.getType();
        if (type.isType(TC_ERROR))
      continue;
        if (def.subClassOf(env, env.getClassDeclaration(type))) {
      add = false;
      break;
        }
    }
    if (add) {
        exp.put(c, newexp.get(c));
    }
      }
      // A variable is DA (DU) after a try statement if it is DA (DU)
      // after the try block and after every catch block. These variables
      // are represented by 'vs'.  If the try statement is labelled, we
      // may also exit from it (including from within a catch block) via
      // a break statement.
      // If there is a finally block, the Vset returned here is further
      // adjusted. Note that this 'TryStatement' node will be a child of
      // a 'FinallyStatement' node in that case.
      return ctx.removeAdditionalVars(vs.join(newctx.vsBreak));
  } catch (ClassNotFound e) {
      env.error(where, "class.not.found", e.name, opNames[op]);
      return vset;
  }
    }

    /**
     * Inline
     */
    public Statement inline(Environment env, Context ctx) {
  if (body != null) {
      body = body.inline(env, new Context(ctx, this));
  }
  if (body == null) {
      return null;
  }
  for (int i = 0 ; i < args.length ; i++) {
      if (args[i] != null) {
    args[i] = args[i].inline(env, new Context(ctx, this));
      }
  }
  return (args.length == 0) ? eliminate(env, body) : this;
    }

    /**
     * Create a copy of the statement for method inlining
     */
    public Statement copyInline(Context ctx, boolean valNeeded) {
  TryStatement s = (TryStatement)clone();
  if (body != null) {
      s.body = body.copyInline(ctx, valNeeded);
  }
  s.args = new Statement[args.length];
  for (int i = 0 ; i < args.length ; i++) {
      if (args[i] != null) {
    s.args[i] = args[i].copyInline(ctx, valNeeded);
      }
  }
  return s;
    }

    /**
     * Compute cost of inlining this statement
     */
    public int costInline(int thresh, Environment env, Context ctx){

  // Don't inline methods containing try statements.
  // If the try statement is being inlined in order to
  // inline a method that returns a value which is
  // a subexpression of an expression involving the
  // operand stack, then the early operands may get lost.
  // This shows up as a verifier error.  For example,
  // in the following:
        //
  //    public static int test() {
  //       try { return 2; } catch (Exception e)  { return 0; }
  //    }
  //
  //    System.out.println(test());
        //
        // an inlined call to test() might look like this:
        //
  //     0 getstatic <Field java.io.PrintStream out>
  //     3 iconst_2
  //     4 goto 9
  //     7 pop
  //     8 iconst_0
  //     9 invokevirtual <Method void println(int)>
  //    12 return
  //  Exception table:
  //     from   to  target type
  //   3     7     7   <Class java.lang.Exception> 
        //
        // This fails to verify because the operand stored
        // for System.out gets axed at an exception, leading to
  // an inconsistent stack depth at pc=7.  
  //
        // Note that although all code must be able to be inlined
  // to implement initializers, this problem doesn't come up,
  // as try statements themselves can never be expressions.
  // It suffices here to make sure they are never inlined as part
  // of optimization.

  return thresh;
    }

    /**
     * Code
     */
    public void code(Environment env, Context ctx, Assembler asm) {
  CodeContext newctx = new CodeContext(ctx, this);
  
  TryData td = new TryData();
  for (int i = 0 ; i < args.length ; i++) {
      Type t = ((CatchStatement)args[i]).field.getType();
      if (t.isType(TC_CLASS)) {
    td.add(env.getClassDeclaration(t));
      } else {
    td.add(t);
      }
  }
  asm.add(where, opc_try, td);
  if (body != null) {
      body.code(env, newctx, asm);
  }

  asm.add(td.getEndLabel());
  asm.add(where, opc_goto, newctx.breakLabel);
  
  for (int i = 0 ; i < args.length ; i++) {
      CatchData cd = td.getCatch(i);
      asm.add(cd.getLabel());
      args[i].code(env, newctx, asm);
      asm.add(where, opc_goto, newctx.breakLabel);
  }
  
  asm.add(newctx.breakLabel);
    }
    
    /**
     * Print
     */
    public void print(PrintStream out, int indent) {
  super.print(out, indent);
  out.print("try ");
  if (body != null) {
      body.print(out, indent);
  } else {
      out.print("<empty>");
  }
  for (int i = 0 ; i < args.length ; i++) {
      out.print(" ");
      args[i].print(out, indent);
  }
    }
}
java2s.com  | Contact Us | Privacy Policy
Copyright 2009 - 12 Demo Source and Support. All rights reserved.
All other trademarks are property of their respective owners.