langtools/src/share/classes/com/sun/tools/javac/comp/Flow.java
author jjg
Wed, 21 Sep 2011 21:56:53 -0700
changeset 10637 2ea5fbb913ac
parent 9601 a1b1716231ca
child 10950 e87b50888909
permissions -rw-r--r--
7092965: javac should not close processorClassLoader before end of compilation Reviewed-by: darcy
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
10
06bc494ca11e Initial load
duke
parents:
diff changeset
     1
/*
8031
d5fe2c1cecfc 6992999: fully remove JSR 308 from langtools
jjg
parents: 7681
diff changeset
     2
 * Copyright (c) 1999, 2011, Oracle and/or its affiliates. All rights reserved.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
06bc494ca11e Initial load
duke
parents:
diff changeset
     4
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
06bc494ca11e Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 5492
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
10
06bc494ca11e Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 5492
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    10
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
06bc494ca11e Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
06bc494ca11e Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
06bc494ca11e Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
06bc494ca11e Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
06bc494ca11e Initial load
duke
parents:
diff changeset
    16
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
06bc494ca11e Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
06bc494ca11e Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
06bc494ca11e Initial load
duke
parents:
diff changeset
    20
 *
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 5492
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 5492
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 5492
diff changeset
    23
 * questions.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    24
 */
06bc494ca11e Initial load
duke
parents:
diff changeset
    25
06bc494ca11e Initial load
duke
parents:
diff changeset
    26
//todo: one might eliminate uninits.andSets when monotonic
06bc494ca11e Initial load
duke
parents:
diff changeset
    27
06bc494ca11e Initial load
duke
parents:
diff changeset
    28
package com.sun.tools.javac.comp;
06bc494ca11e Initial load
duke
parents:
diff changeset
    29
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
    30
import java.util.HashMap;
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
    31
import java.util.Map;
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
    32
import java.util.LinkedHashMap;
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
    33
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    34
import com.sun.tools.javac.code.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    35
import com.sun.tools.javac.tree.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    36
import com.sun.tools.javac.util.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    37
import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition;
06bc494ca11e Initial load
duke
parents:
diff changeset
    38
06bc494ca11e Initial load
duke
parents:
diff changeset
    39
import com.sun.tools.javac.code.Symbol.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    40
import com.sun.tools.javac.tree.JCTree.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    41
06bc494ca11e Initial load
duke
parents:
diff changeset
    42
import static com.sun.tools.javac.code.Flags.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    43
import static com.sun.tools.javac.code.Kinds.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    44
import static com.sun.tools.javac.code.TypeTags.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    45
06bc494ca11e Initial load
duke
parents:
diff changeset
    46
/** This pass implements dataflow analysis for Java programs.
06bc494ca11e Initial load
duke
parents:
diff changeset
    47
 *  Liveness analysis checks that every statement is reachable.
06bc494ca11e Initial load
duke
parents:
diff changeset
    48
 *  Exception analysis ensures that every checked exception that is
06bc494ca11e Initial load
duke
parents:
diff changeset
    49
 *  thrown is declared or caught.  Definite assignment analysis
06bc494ca11e Initial load
duke
parents:
diff changeset
    50
 *  ensures that each variable is assigned when used.  Definite
06bc494ca11e Initial load
duke
parents:
diff changeset
    51
 *  unassignment analysis ensures that no final variable is assigned
06bc494ca11e Initial load
duke
parents:
diff changeset
    52
 *  more than once.
06bc494ca11e Initial load
duke
parents:
diff changeset
    53
 *
9303
eae35c201e19 7032975: API files in javax.annotation.processing need to be updated for references to JLS
jjh
parents: 9300
diff changeset
    54
 *  <p>The JLS has a number of problems in the
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    55
 *  specification of these flow analysis problems. This implementation
06bc494ca11e Initial load
duke
parents:
diff changeset
    56
 *  attempts to address those issues.
06bc494ca11e Initial load
duke
parents:
diff changeset
    57
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
    58
 *  <p>First, there is no accommodation for a finally clause that cannot
06bc494ca11e Initial load
duke
parents:
diff changeset
    59
 *  complete normally. For liveness analysis, an intervening finally
06bc494ca11e Initial load
duke
parents:
diff changeset
    60
 *  clause can cause a break, continue, or return not to reach its
06bc494ca11e Initial load
duke
parents:
diff changeset
    61
 *  target.  For exception analysis, an intervening finally clause can
06bc494ca11e Initial load
duke
parents:
diff changeset
    62
 *  cause any exception to be "caught".  For DA/DU analysis, the finally
06bc494ca11e Initial load
duke
parents:
diff changeset
    63
 *  clause can prevent a transfer of control from propagating DA/DU
06bc494ca11e Initial load
duke
parents:
diff changeset
    64
 *  state to the target.  In addition, code in the finally clause can
06bc494ca11e Initial load
duke
parents:
diff changeset
    65
 *  affect the DA/DU status of variables.
06bc494ca11e Initial load
duke
parents:
diff changeset
    66
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
    67
 *  <p>For try statements, we introduce the idea of a variable being
06bc494ca11e Initial load
duke
parents:
diff changeset
    68
 *  definitely unassigned "everywhere" in a block.  A variable V is
06bc494ca11e Initial load
duke
parents:
diff changeset
    69
 *  "unassigned everywhere" in a block iff it is unassigned at the
06bc494ca11e Initial load
duke
parents:
diff changeset
    70
 *  beginning of the block and there is no reachable assignment to V
06bc494ca11e Initial load
duke
parents:
diff changeset
    71
 *  in the block.  An assignment V=e is reachable iff V is not DA
06bc494ca11e Initial load
duke
parents:
diff changeset
    72
 *  after e.  Then we can say that V is DU at the beginning of the
06bc494ca11e Initial load
duke
parents:
diff changeset
    73
 *  catch block iff V is DU everywhere in the try block.  Similarly, V
06bc494ca11e Initial load
duke
parents:
diff changeset
    74
 *  is DU at the beginning of the finally block iff V is DU everywhere
06bc494ca11e Initial load
duke
parents:
diff changeset
    75
 *  in the try block and in every catch block.  Specifically, the
06bc494ca11e Initial load
duke
parents:
diff changeset
    76
 *  following bullet is added to 16.2.2
06bc494ca11e Initial load
duke
parents:
diff changeset
    77
 *  <pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    78
 *      V is <em>unassigned everywhere</em> in a block if it is
06bc494ca11e Initial load
duke
parents:
diff changeset
    79
 *      unassigned before the block and there is no reachable
06bc494ca11e Initial load
duke
parents:
diff changeset
    80
 *      assignment to V within the block.
06bc494ca11e Initial load
duke
parents:
diff changeset
    81
 *  </pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    82
 *  <p>In 16.2.15, the third bullet (and all of its sub-bullets) for all
06bc494ca11e Initial load
duke
parents:
diff changeset
    83
 *  try blocks is changed to
06bc494ca11e Initial load
duke
parents:
diff changeset
    84
 *  <pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    85
 *      V is definitely unassigned before a catch block iff V is
06bc494ca11e Initial load
duke
parents:
diff changeset
    86
 *      definitely unassigned everywhere in the try block.
06bc494ca11e Initial load
duke
parents:
diff changeset
    87
 *  </pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    88
 *  <p>The last bullet (and all of its sub-bullets) for try blocks that
06bc494ca11e Initial load
duke
parents:
diff changeset
    89
 *  have a finally block is changed to
06bc494ca11e Initial load
duke
parents:
diff changeset
    90
 *  <pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    91
 *      V is definitely unassigned before the finally block iff
06bc494ca11e Initial load
duke
parents:
diff changeset
    92
 *      V is definitely unassigned everywhere in the try block
06bc494ca11e Initial load
duke
parents:
diff changeset
    93
 *      and everywhere in each catch block of the try statement.
06bc494ca11e Initial load
duke
parents:
diff changeset
    94
 *  </pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    95
 *  <p>In addition,
06bc494ca11e Initial load
duke
parents:
diff changeset
    96
 *  <pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
    97
 *      V is definitely assigned at the end of a constructor iff
06bc494ca11e Initial load
duke
parents:
diff changeset
    98
 *      V is definitely assigned after the block that is the body
06bc494ca11e Initial load
duke
parents:
diff changeset
    99
 *      of the constructor and V is definitely assigned at every
06bc494ca11e Initial load
duke
parents:
diff changeset
   100
 *      return that can return from the constructor.
06bc494ca11e Initial load
duke
parents:
diff changeset
   101
 *  </pre>
06bc494ca11e Initial load
duke
parents:
diff changeset
   102
 *  <p>In addition, each continue statement with the loop as its target
06bc494ca11e Initial load
duke
parents:
diff changeset
   103
 *  is treated as a jump to the end of the loop body, and "intervening"
06bc494ca11e Initial load
duke
parents:
diff changeset
   104
 *  finally clauses are treated as follows: V is DA "due to the
06bc494ca11e Initial load
duke
parents:
diff changeset
   105
 *  continue" iff V is DA before the continue statement or V is DA at
06bc494ca11e Initial load
duke
parents:
diff changeset
   106
 *  the end of any intervening finally block.  V is DU "due to the
06bc494ca11e Initial load
duke
parents:
diff changeset
   107
 *  continue" iff any intervening finally cannot complete normally or V
06bc494ca11e Initial load
duke
parents:
diff changeset
   108
 *  is DU at the end of every intervening finally block.  This "due to
06bc494ca11e Initial load
duke
parents:
diff changeset
   109
 *  the continue" concept is then used in the spec for the loops.
06bc494ca11e Initial load
duke
parents:
diff changeset
   110
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
   111
 *  <p>Similarly, break statements must consider intervening finally
06bc494ca11e Initial load
duke
parents:
diff changeset
   112
 *  blocks.  For liveness analysis, a break statement for which any
06bc494ca11e Initial load
duke
parents:
diff changeset
   113
 *  intervening finally cannot complete normally is not considered to
06bc494ca11e Initial load
duke
parents:
diff changeset
   114
 *  cause the target statement to be able to complete normally. Then
06bc494ca11e Initial load
duke
parents:
diff changeset
   115
 *  we say V is DA "due to the break" iff V is DA before the break or
06bc494ca11e Initial load
duke
parents:
diff changeset
   116
 *  V is DA at the end of any intervening finally block.  V is DU "due
06bc494ca11e Initial load
duke
parents:
diff changeset
   117
 *  to the break" iff any intervening finally cannot complete normally
06bc494ca11e Initial load
duke
parents:
diff changeset
   118
 *  or V is DU at the break and at the end of every intervening
06bc494ca11e Initial load
duke
parents:
diff changeset
   119
 *  finally block.  (I suspect this latter condition can be
06bc494ca11e Initial load
duke
parents:
diff changeset
   120
 *  simplified.)  This "due to the break" is then used in the spec for
06bc494ca11e Initial load
duke
parents:
diff changeset
   121
 *  all statements that can be "broken".
06bc494ca11e Initial load
duke
parents:
diff changeset
   122
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
   123
 *  <p>The return statement is treated similarly.  V is DA "due to a
06bc494ca11e Initial load
duke
parents:
diff changeset
   124
 *  return statement" iff V is DA before the return statement or V is
06bc494ca11e Initial load
duke
parents:
diff changeset
   125
 *  DA at the end of any intervening finally block.  Note that we
06bc494ca11e Initial load
duke
parents:
diff changeset
   126
 *  don't have to worry about the return expression because this
06bc494ca11e Initial load
duke
parents:
diff changeset
   127
 *  concept is only used for construcrors.
06bc494ca11e Initial load
duke
parents:
diff changeset
   128
 *
9303
eae35c201e19 7032975: API files in javax.annotation.processing need to be updated for references to JLS
jjh
parents: 9300
diff changeset
   129
 *  <p>There is no spec in the JLS for when a variable is definitely
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   130
 *  assigned at the end of a constructor, which is needed for final
06bc494ca11e Initial load
duke
parents:
diff changeset
   131
 *  fields (8.3.1.2).  We implement the rule that V is DA at the end
06bc494ca11e Initial load
duke
parents:
diff changeset
   132
 *  of the constructor iff it is DA and the end of the body of the
06bc494ca11e Initial load
duke
parents:
diff changeset
   133
 *  constructor and V is DA "due to" every return of the constructor.
06bc494ca11e Initial load
duke
parents:
diff changeset
   134
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
   135
 *  <p>Intervening finally blocks similarly affect exception analysis.  An
06bc494ca11e Initial load
duke
parents:
diff changeset
   136
 *  intervening finally that cannot complete normally allows us to ignore
06bc494ca11e Initial load
duke
parents:
diff changeset
   137
 *  an otherwise uncaught exception.
06bc494ca11e Initial load
duke
parents:
diff changeset
   138
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
   139
 *  <p>To implement the semantics of intervening finally clauses, all
06bc494ca11e Initial load
duke
parents:
diff changeset
   140
 *  nonlocal transfers (break, continue, return, throw, method call that
06bc494ca11e Initial load
duke
parents:
diff changeset
   141
 *  can throw a checked exception, and a constructor invocation that can
06bc494ca11e Initial load
duke
parents:
diff changeset
   142
 *  thrown a checked exception) are recorded in a queue, and removed
06bc494ca11e Initial load
duke
parents:
diff changeset
   143
 *  from the queue when we complete processing the target of the
06bc494ca11e Initial load
duke
parents:
diff changeset
   144
 *  nonlocal transfer.  This allows us to modify the queue in accordance
06bc494ca11e Initial load
duke
parents:
diff changeset
   145
 *  with the above rules when we encounter a finally clause.  The only
06bc494ca11e Initial load
duke
parents:
diff changeset
   146
 *  exception to this [no pun intended] is that checked exceptions that
06bc494ca11e Initial load
duke
parents:
diff changeset
   147
 *  are known to be caught or declared to be caught in the enclosing
06bc494ca11e Initial load
duke
parents:
diff changeset
   148
 *  method are not recorded in the queue, but instead are recorded in a
06bc494ca11e Initial load
duke
parents:
diff changeset
   149
 *  global variable "Set<Type> thrown" that records the type of all
06bc494ca11e Initial load
duke
parents:
diff changeset
   150
 *  exceptions that can be thrown.
06bc494ca11e Initial load
duke
parents:
diff changeset
   151
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
   152
 *  <p>Other minor issues the treatment of members of other classes
06bc494ca11e Initial load
duke
parents:
diff changeset
   153
 *  (always considered DA except that within an anonymous class
06bc494ca11e Initial load
duke
parents:
diff changeset
   154
 *  constructor, where DA status from the enclosing scope is
06bc494ca11e Initial load
duke
parents:
diff changeset
   155
 *  preserved), treatment of the case expression (V is DA before the
06bc494ca11e Initial load
duke
parents:
diff changeset
   156
 *  case expression iff V is DA after the switch expression),
06bc494ca11e Initial load
duke
parents:
diff changeset
   157
 *  treatment of variables declared in a switch block (the implied
06bc494ca11e Initial load
duke
parents:
diff changeset
   158
 *  DA/DU status after the switch expression is DU and not DA for
06bc494ca11e Initial load
duke
parents:
diff changeset
   159
 *  variables defined in a switch block), the treatment of boolean ?:
06bc494ca11e Initial load
duke
parents:
diff changeset
   160
 *  expressions (The JLS rules only handle b and c non-boolean; the
06bc494ca11e Initial load
duke
parents:
diff changeset
   161
 *  new rule is that if b and c are boolean valued, then V is
06bc494ca11e Initial load
duke
parents:
diff changeset
   162
 *  (un)assigned after a?b:c when true/false iff V is (un)assigned
06bc494ca11e Initial load
duke
parents:
diff changeset
   163
 *  after b when true/false and V is (un)assigned after c when
06bc494ca11e Initial load
duke
parents:
diff changeset
   164
 *  true/false).
06bc494ca11e Initial load
duke
parents:
diff changeset
   165
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
   166
 *  <p>There is the remaining question of what syntactic forms constitute a
06bc494ca11e Initial load
duke
parents:
diff changeset
   167
 *  reference to a variable.  It is conventional to allow this.x on the
06bc494ca11e Initial load
duke
parents:
diff changeset
   168
 *  left-hand-side to initialize a final instance field named x, yet
06bc494ca11e Initial load
duke
parents:
diff changeset
   169
 *  this.x isn't considered a "use" when appearing on a right-hand-side
06bc494ca11e Initial load
duke
parents:
diff changeset
   170
 *  in most implementations.  Should parentheses affect what is
06bc494ca11e Initial load
duke
parents:
diff changeset
   171
 *  considered a variable reference?  The simplest rule would be to
06bc494ca11e Initial load
duke
parents:
diff changeset
   172
 *  allow unqualified forms only, parentheses optional, and phase out
06bc494ca11e Initial load
duke
parents:
diff changeset
   173
 *  support for assigning to a final field via this.x.
06bc494ca11e Initial load
duke
parents:
diff changeset
   174
 *
5847
1908176fd6e3 6944312: Potential rebranding issues in openjdk/langtools repository sources
jjg
parents: 5520
diff changeset
   175
 *  <p><b>This is NOT part of any supported API.
1908176fd6e3 6944312: Potential rebranding issues in openjdk/langtools repository sources
jjg
parents: 5520
diff changeset
   176
 *  If you write code that depends on this, you do so at your own risk.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   177
 *  This code and its internal interfaces are subject to change or
06bc494ca11e Initial load
duke
parents:
diff changeset
   178
 *  deletion without notice.</b>
06bc494ca11e Initial load
duke
parents:
diff changeset
   179
 */
06bc494ca11e Initial load
duke
parents:
diff changeset
   180
public class Flow extends TreeScanner {
06bc494ca11e Initial load
duke
parents:
diff changeset
   181
    protected static final Context.Key<Flow> flowKey =
06bc494ca11e Initial load
duke
parents:
diff changeset
   182
        new Context.Key<Flow>();
06bc494ca11e Initial load
duke
parents:
diff changeset
   183
1260
a772ba9ba43d 6574134: Allow for alternative implementation of Name Table with garbage collection of name bytes
jjg
parents: 10
diff changeset
   184
    private final Names names;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   185
    private final Log log;
06bc494ca11e Initial load
duke
parents:
diff changeset
   186
    private final Symtab syms;
06bc494ca11e Initial load
duke
parents:
diff changeset
   187
    private final Types types;
06bc494ca11e Initial load
duke
parents:
diff changeset
   188
    private final Check chk;
06bc494ca11e Initial load
duke
parents:
diff changeset
   189
    private       TreeMaker make;
6156
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
   190
    private final Resolve rs;
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
   191
    private Env<AttrContext> attrEnv;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   192
    private       Lint lint;
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
   193
    private final boolean allowImprovedRethrowAnalysis;
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
   194
    private final boolean allowImprovedCatchAnalysis;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   195
06bc494ca11e Initial load
duke
parents:
diff changeset
   196
    public static Flow instance(Context context) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   197
        Flow instance = context.get(flowKey);
06bc494ca11e Initial load
duke
parents:
diff changeset
   198
        if (instance == null)
06bc494ca11e Initial load
duke
parents:
diff changeset
   199
            instance = new Flow(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
   200
        return instance;
06bc494ca11e Initial load
duke
parents:
diff changeset
   201
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   202
06bc494ca11e Initial load
duke
parents:
diff changeset
   203
    protected Flow(Context context) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   204
        context.put(flowKey, this);
1260
a772ba9ba43d 6574134: Allow for alternative implementation of Name Table with garbage collection of name bytes
jjg
parents: 10
diff changeset
   205
        names = Names.instance(context);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   206
        log = Log.instance(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
   207
        syms = Symtab.instance(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
   208
        types = Types.instance(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
   209
        chk = Check.instance(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
   210
        lint = Lint.instance(context);
6156
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
   211
        rs = Resolve.instance(context);
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   212
        Source source = Source.instance(context);
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
   213
        allowImprovedRethrowAnalysis = source.allowImprovedRethrowAnalysis();
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
   214
        allowImprovedCatchAnalysis = source.allowImprovedCatchAnalysis();
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   215
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   216
06bc494ca11e Initial load
duke
parents:
diff changeset
   217
    /** A flag that indicates whether the last statement could
06bc494ca11e Initial load
duke
parents:
diff changeset
   218
     *  complete normally.
06bc494ca11e Initial load
duke
parents:
diff changeset
   219
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   220
    private boolean alive;
06bc494ca11e Initial load
duke
parents:
diff changeset
   221
06bc494ca11e Initial load
duke
parents:
diff changeset
   222
    /** The set of definitely assigned variables.
06bc494ca11e Initial load
duke
parents:
diff changeset
   223
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   224
    Bits inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   225
06bc494ca11e Initial load
duke
parents:
diff changeset
   226
    /** The set of definitely unassigned variables.
06bc494ca11e Initial load
duke
parents:
diff changeset
   227
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   228
    Bits uninits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   229
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
   230
    HashMap<Symbol, List<Type>> preciseRethrowTypes;
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   231
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   232
    /** The set of variables that are definitely unassigned everywhere
06bc494ca11e Initial load
duke
parents:
diff changeset
   233
     *  in current try block. This variable is maintained lazily; it is
06bc494ca11e Initial load
duke
parents:
diff changeset
   234
     *  updated only when something gets removed from uninits,
06bc494ca11e Initial load
duke
parents:
diff changeset
   235
     *  typically by being assigned in reachable code.  To obtain the
06bc494ca11e Initial load
duke
parents:
diff changeset
   236
     *  correct set of variables which are definitely unassigned
06bc494ca11e Initial load
duke
parents:
diff changeset
   237
     *  anywhere in current try block, intersect uninitsTry and
06bc494ca11e Initial load
duke
parents:
diff changeset
   238
     *  uninits.
06bc494ca11e Initial load
duke
parents:
diff changeset
   239
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   240
    Bits uninitsTry;
06bc494ca11e Initial load
duke
parents:
diff changeset
   241
06bc494ca11e Initial load
duke
parents:
diff changeset
   242
    /** When analyzing a condition, inits and uninits are null.
06bc494ca11e Initial load
duke
parents:
diff changeset
   243
     *  Instead we have:
06bc494ca11e Initial load
duke
parents:
diff changeset
   244
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   245
    Bits initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   246
    Bits initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   247
    Bits uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   248
    Bits uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   249
06bc494ca11e Initial load
duke
parents:
diff changeset
   250
    /** A mapping from addresses to variable symbols.
06bc494ca11e Initial load
duke
parents:
diff changeset
   251
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   252
    VarSymbol[] vars;
06bc494ca11e Initial load
duke
parents:
diff changeset
   253
06bc494ca11e Initial load
duke
parents:
diff changeset
   254
    /** The current class being defined.
06bc494ca11e Initial load
duke
parents:
diff changeset
   255
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   256
    JCClassDecl classDef;
06bc494ca11e Initial load
duke
parents:
diff changeset
   257
06bc494ca11e Initial load
duke
parents:
diff changeset
   258
    /** The first variable sequence number in this class definition.
06bc494ca11e Initial load
duke
parents:
diff changeset
   259
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   260
    int firstadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   261
06bc494ca11e Initial load
duke
parents:
diff changeset
   262
    /** The next available variable sequence number.
06bc494ca11e Initial load
duke
parents:
diff changeset
   263
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   264
    int nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   265
06bc494ca11e Initial load
duke
parents:
diff changeset
   266
    /** The list of possibly thrown declarable exceptions.
06bc494ca11e Initial load
duke
parents:
diff changeset
   267
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   268
    List<Type> thrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
   269
06bc494ca11e Initial load
duke
parents:
diff changeset
   270
    /** The list of exceptions that are either caught or declared to be
06bc494ca11e Initial load
duke
parents:
diff changeset
   271
     *  thrown.
06bc494ca11e Initial load
duke
parents:
diff changeset
   272
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   273
    List<Type> caught;
06bc494ca11e Initial load
duke
parents:
diff changeset
   274
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
   275
    /** The list of unreferenced automatic resources.
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
   276
     */
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
   277
    Scope unrefdResources;
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
   278
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   279
    /** Set when processing a loop body the second time for DU analysis. */
06bc494ca11e Initial load
duke
parents:
diff changeset
   280
    boolean loopPassTwo = false;
06bc494ca11e Initial load
duke
parents:
diff changeset
   281
06bc494ca11e Initial load
duke
parents:
diff changeset
   282
    /*-------------------- Environments ----------------------*/
06bc494ca11e Initial load
duke
parents:
diff changeset
   283
06bc494ca11e Initial load
duke
parents:
diff changeset
   284
    /** A pending exit.  These are the statements return, break, and
06bc494ca11e Initial load
duke
parents:
diff changeset
   285
     *  continue.  In addition, exception-throwing expressions or
06bc494ca11e Initial load
duke
parents:
diff changeset
   286
     *  statements are put here when not known to be caught.  This
06bc494ca11e Initial load
duke
parents:
diff changeset
   287
     *  will typically result in an error unless it is within a
06bc494ca11e Initial load
duke
parents:
diff changeset
   288
     *  try-finally whose finally block cannot complete normally.
06bc494ca11e Initial load
duke
parents:
diff changeset
   289
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   290
    static class PendingExit {
06bc494ca11e Initial load
duke
parents:
diff changeset
   291
        JCTree tree;
06bc494ca11e Initial load
duke
parents:
diff changeset
   292
        Bits inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   293
        Bits uninits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   294
        Type thrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
   295
        PendingExit(JCTree tree, Bits inits, Bits uninits) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   296
            this.tree = tree;
06bc494ca11e Initial load
duke
parents:
diff changeset
   297
            this.inits = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   298
            this.uninits = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   299
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   300
        PendingExit(JCTree tree, Type thrown) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   301
            this.tree = tree;
06bc494ca11e Initial load
duke
parents:
diff changeset
   302
            this.thrown = thrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
   303
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   304
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   305
06bc494ca11e Initial load
duke
parents:
diff changeset
   306
    /** The currently pending exits that go from current inner blocks
06bc494ca11e Initial load
duke
parents:
diff changeset
   307
     *  to an enclosing block, in source order.
06bc494ca11e Initial load
duke
parents:
diff changeset
   308
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   309
    ListBuffer<PendingExit> pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   310
06bc494ca11e Initial load
duke
parents:
diff changeset
   311
    /*-------------------- Exceptions ----------------------*/
06bc494ca11e Initial load
duke
parents:
diff changeset
   312
06bc494ca11e Initial load
duke
parents:
diff changeset
   313
    /** Complain that pending exceptions are not caught.
06bc494ca11e Initial load
duke
parents:
diff changeset
   314
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   315
    void errorUncaught() {
06bc494ca11e Initial load
duke
parents:
diff changeset
   316
        for (PendingExit exit = pendingExits.next();
06bc494ca11e Initial load
duke
parents:
diff changeset
   317
             exit != null;
06bc494ca11e Initial load
duke
parents:
diff changeset
   318
             exit = pendingExits.next()) {
8428
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   319
            if (classDef != null &&
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   320
                classDef.pos == exit.tree.pos) {
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   321
                log.error(exit.tree.pos(),
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   322
                        "unreported.exception.default.constructor",
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   323
                        exit.thrown);
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   324
            } else if (exit.tree.getTag() == JCTree.VARDEF &&
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   325
                    ((JCVariableDecl)exit.tree).sym.isResourceVariable()) {
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   326
                log.error(exit.tree.pos(),
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   327
                        "unreported.exception.implicit.close",
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   328
                        exit.thrown,
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   329
                        ((JCVariableDecl)exit.tree).sym.name);
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   330
            } else {
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   331
                log.error(exit.tree.pos(),
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   332
                        "unreported.exception.need.to.catch.or.throw",
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   333
                        exit.thrown);
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
   334
            }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   335
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   336
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   337
06bc494ca11e Initial load
duke
parents:
diff changeset
   338
    /** Record that exception is potentially thrown and check that it
06bc494ca11e Initial load
duke
parents:
diff changeset
   339
     *  is caught.
06bc494ca11e Initial load
duke
parents:
diff changeset
   340
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   341
    void markThrown(JCTree tree, Type exc) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   342
        if (!chk.isUnchecked(tree.pos(), exc)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   343
            if (!chk.isHandled(exc, caught))
06bc494ca11e Initial load
duke
parents:
diff changeset
   344
                pendingExits.append(new PendingExit(tree, exc));
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
   345
                thrown = chk.incl(exc, thrown);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   346
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   347
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   348
06bc494ca11e Initial load
duke
parents:
diff changeset
   349
    /*-------------- Processing variables ----------------------*/
06bc494ca11e Initial load
duke
parents:
diff changeset
   350
06bc494ca11e Initial load
duke
parents:
diff changeset
   351
    /** Do we need to track init/uninit state of this symbol?
06bc494ca11e Initial load
duke
parents:
diff changeset
   352
     *  I.e. is symbol either a local or a blank final variable?
06bc494ca11e Initial load
duke
parents:
diff changeset
   353
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   354
    boolean trackable(VarSymbol sym) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   355
        return
06bc494ca11e Initial load
duke
parents:
diff changeset
   356
            (sym.owner.kind == MTH ||
06bc494ca11e Initial load
duke
parents:
diff changeset
   357
             ((sym.flags() & (FINAL | HASINIT | PARAMETER)) == FINAL &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   358
              classDef.sym.isEnclosedBy((ClassSymbol)sym.owner)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   359
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   360
06bc494ca11e Initial load
duke
parents:
diff changeset
   361
    /** Initialize new trackable variable by setting its address field
06bc494ca11e Initial load
duke
parents:
diff changeset
   362
     *  to the next available sequence number and entering it under that
06bc494ca11e Initial load
duke
parents:
diff changeset
   363
     *  index into the vars array.
06bc494ca11e Initial load
duke
parents:
diff changeset
   364
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   365
    void newVar(VarSymbol sym) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   366
        if (nextadr == vars.length) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   367
            VarSymbol[] newvars = new VarSymbol[nextadr * 2];
06bc494ca11e Initial load
duke
parents:
diff changeset
   368
            System.arraycopy(vars, 0, newvars, 0, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   369
            vars = newvars;
06bc494ca11e Initial load
duke
parents:
diff changeset
   370
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   371
        sym.adr = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   372
        vars[nextadr] = sym;
06bc494ca11e Initial load
duke
parents:
diff changeset
   373
        inits.excl(nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   374
        uninits.incl(nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   375
        nextadr++;
06bc494ca11e Initial load
duke
parents:
diff changeset
   376
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   377
06bc494ca11e Initial load
duke
parents:
diff changeset
   378
    /** Record an initialization of a trackable variable.
06bc494ca11e Initial load
duke
parents:
diff changeset
   379
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   380
    void letInit(DiagnosticPosition pos, VarSymbol sym) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   381
        if (sym.adr >= firstadr && trackable(sym)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   382
            if ((sym.flags() & FINAL) != 0) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   383
                if ((sym.flags() & PARAMETER) != 0) {
9300
c2de4dd9853b 7033809: Rename "disjunctive" to "union" in javax.lang.model
darcy
parents: 8849
diff changeset
   384
                    if ((sym.flags() & UNION) != 0) { //multi-catch parameter
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   385
                        log.error(pos, "multicatch.parameter.may.not.be.assigned",
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   386
                                  sym);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   387
                    }
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   388
                    else {
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   389
                        log.error(pos, "final.parameter.may.not.be.assigned",
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   390
                              sym);
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
   391
                    }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   392
                } else if (!uninits.isMember(sym.adr)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   393
                    log.error(pos,
06bc494ca11e Initial load
duke
parents:
diff changeset
   394
                              loopPassTwo
06bc494ca11e Initial load
duke
parents:
diff changeset
   395
                              ? "var.might.be.assigned.in.loop"
06bc494ca11e Initial load
duke
parents:
diff changeset
   396
                              : "var.might.already.be.assigned",
06bc494ca11e Initial load
duke
parents:
diff changeset
   397
                              sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   398
                } else if (!inits.isMember(sym.adr)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   399
                    // reachable assignment
06bc494ca11e Initial load
duke
parents:
diff changeset
   400
                    uninits.excl(sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   401
                    uninitsTry.excl(sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   402
                } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   403
                    //log.rawWarning(pos, "unreachable assignment");//DEBUG
06bc494ca11e Initial load
duke
parents:
diff changeset
   404
                    uninits.excl(sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   405
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   406
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   407
            inits.incl(sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   408
        } else if ((sym.flags() & FINAL) != 0) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   409
            log.error(pos, "var.might.already.be.assigned", sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   410
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   411
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   412
06bc494ca11e Initial load
duke
parents:
diff changeset
   413
    /** If tree is either a simple name or of the form this.name or
06bc494ca11e Initial load
duke
parents:
diff changeset
   414
     *  C.this.name, and tree represents a trackable variable,
06bc494ca11e Initial load
duke
parents:
diff changeset
   415
     *  record an initialization of the variable.
06bc494ca11e Initial load
duke
parents:
diff changeset
   416
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   417
    void letInit(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   418
        tree = TreeInfo.skipParens(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   419
        if (tree.getTag() == JCTree.IDENT || tree.getTag() == JCTree.SELECT) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   420
            Symbol sym = TreeInfo.symbol(tree);
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   421
            if (sym.kind == VAR) {
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   422
                letInit(tree.pos(), (VarSymbol)sym);
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   423
            }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   424
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   425
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   426
06bc494ca11e Initial load
duke
parents:
diff changeset
   427
    /** Check that trackable variable is initialized.
06bc494ca11e Initial load
duke
parents:
diff changeset
   428
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   429
    void checkInit(DiagnosticPosition pos, VarSymbol sym) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   430
        if ((sym.adr >= firstadr || sym.owner.kind != TYP) &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   431
            trackable(sym) &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   432
            !inits.isMember(sym.adr)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   433
            log.error(pos, "var.might.not.have.been.initialized",
06bc494ca11e Initial load
duke
parents:
diff changeset
   434
                      sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   435
            inits.incl(sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   436
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   437
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   438
06bc494ca11e Initial load
duke
parents:
diff changeset
   439
    /*-------------------- Handling jumps ----------------------*/
06bc494ca11e Initial load
duke
parents:
diff changeset
   440
06bc494ca11e Initial load
duke
parents:
diff changeset
   441
    /** Record an outward transfer of control. */
06bc494ca11e Initial load
duke
parents:
diff changeset
   442
    void recordExit(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   443
        pendingExits.append(new PendingExit(tree, inits, uninits));
06bc494ca11e Initial load
duke
parents:
diff changeset
   444
        markDead();
06bc494ca11e Initial load
duke
parents:
diff changeset
   445
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   446
06bc494ca11e Initial load
duke
parents:
diff changeset
   447
    /** Resolve all breaks of this statement. */
06bc494ca11e Initial load
duke
parents:
diff changeset
   448
    boolean resolveBreaks(JCTree tree,
06bc494ca11e Initial load
duke
parents:
diff changeset
   449
                          ListBuffer<PendingExit> oldPendingExits) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   450
        boolean result = false;
06bc494ca11e Initial load
duke
parents:
diff changeset
   451
        List<PendingExit> exits = pendingExits.toList();
06bc494ca11e Initial load
duke
parents:
diff changeset
   452
        pendingExits = oldPendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   453
        for (; exits.nonEmpty(); exits = exits.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   454
            PendingExit exit = exits.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   455
            if (exit.tree.getTag() == JCTree.BREAK &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   456
                ((JCBreak) exit.tree).target == tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   457
                inits.andSet(exit.inits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   458
                uninits.andSet(exit.uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   459
                result = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   460
            } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   461
                pendingExits.append(exit);
06bc494ca11e Initial load
duke
parents:
diff changeset
   462
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   463
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   464
        return result;
06bc494ca11e Initial load
duke
parents:
diff changeset
   465
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   466
06bc494ca11e Initial load
duke
parents:
diff changeset
   467
    /** Resolve all continues of this statement. */
06bc494ca11e Initial load
duke
parents:
diff changeset
   468
    boolean resolveContinues(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   469
        boolean result = false;
06bc494ca11e Initial load
duke
parents:
diff changeset
   470
        List<PendingExit> exits = pendingExits.toList();
06bc494ca11e Initial load
duke
parents:
diff changeset
   471
        pendingExits = new ListBuffer<PendingExit>();
06bc494ca11e Initial load
duke
parents:
diff changeset
   472
        for (; exits.nonEmpty(); exits = exits.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   473
            PendingExit exit = exits.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   474
            if (exit.tree.getTag() == JCTree.CONTINUE &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   475
                ((JCContinue) exit.tree).target == tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   476
                inits.andSet(exit.inits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   477
                uninits.andSet(exit.uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   478
                result = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   479
            } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   480
                pendingExits.append(exit);
06bc494ca11e Initial load
duke
parents:
diff changeset
   481
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   482
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   483
        return result;
06bc494ca11e Initial load
duke
parents:
diff changeset
   484
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   485
06bc494ca11e Initial load
duke
parents:
diff changeset
   486
    /** Record that statement is unreachable.
06bc494ca11e Initial load
duke
parents:
diff changeset
   487
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   488
    void markDead() {
06bc494ca11e Initial load
duke
parents:
diff changeset
   489
        inits.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   490
        uninits.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   491
        alive = false;
06bc494ca11e Initial load
duke
parents:
diff changeset
   492
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   493
06bc494ca11e Initial load
duke
parents:
diff changeset
   494
    /** Split (duplicate) inits/uninits into WhenTrue/WhenFalse sets
06bc494ca11e Initial load
duke
parents:
diff changeset
   495
     */
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   496
    void split(boolean setToNull) {
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   497
        initsWhenFalse = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   498
        uninitsWhenFalse = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   499
        initsWhenTrue = inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   500
        uninitsWhenTrue = uninits;
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   501
        if (setToNull)
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   502
            inits = uninits = null;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   503
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   504
06bc494ca11e Initial load
duke
parents:
diff changeset
   505
    /** Merge (intersect) inits/uninits from WhenTrue/WhenFalse sets.
06bc494ca11e Initial load
duke
parents:
diff changeset
   506
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   507
    void merge() {
06bc494ca11e Initial load
duke
parents:
diff changeset
   508
        inits = initsWhenFalse.andSet(initsWhenTrue);
06bc494ca11e Initial load
duke
parents:
diff changeset
   509
        uninits = uninitsWhenFalse.andSet(uninitsWhenTrue);
06bc494ca11e Initial load
duke
parents:
diff changeset
   510
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   511
06bc494ca11e Initial load
duke
parents:
diff changeset
   512
/* ************************************************************************
06bc494ca11e Initial load
duke
parents:
diff changeset
   513
 * Visitor methods for statements and definitions
06bc494ca11e Initial load
duke
parents:
diff changeset
   514
 *************************************************************************/
06bc494ca11e Initial load
duke
parents:
diff changeset
   515
06bc494ca11e Initial load
duke
parents:
diff changeset
   516
    /** Analyze a definition.
06bc494ca11e Initial load
duke
parents:
diff changeset
   517
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   518
    void scanDef(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   519
        scanStat(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   520
        if (tree != null && tree.getTag() == JCTree.BLOCK && !alive) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   521
            log.error(tree.pos(),
06bc494ca11e Initial load
duke
parents:
diff changeset
   522
                      "initializer.must.be.able.to.complete.normally");
06bc494ca11e Initial load
duke
parents:
diff changeset
   523
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   524
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   525
06bc494ca11e Initial load
duke
parents:
diff changeset
   526
    /** Analyze a statement. Check that statement is reachable.
06bc494ca11e Initial load
duke
parents:
diff changeset
   527
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   528
    void scanStat(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   529
        if (!alive && tree != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   530
            log.error(tree.pos(), "unreachable.stmt");
06bc494ca11e Initial load
duke
parents:
diff changeset
   531
            if (tree.getTag() != JCTree.SKIP) alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   532
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   533
        scan(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   534
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   535
06bc494ca11e Initial load
duke
parents:
diff changeset
   536
    /** Analyze list of statements.
06bc494ca11e Initial load
duke
parents:
diff changeset
   537
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   538
    void scanStats(List<? extends JCStatement> trees) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   539
        if (trees != null)
06bc494ca11e Initial load
duke
parents:
diff changeset
   540
            for (List<? extends JCStatement> l = trees; l.nonEmpty(); l = l.tail)
06bc494ca11e Initial load
duke
parents:
diff changeset
   541
                scanStat(l.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
   542
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   543
06bc494ca11e Initial load
duke
parents:
diff changeset
   544
    /** Analyze an expression. Make sure to set (un)inits rather than
06bc494ca11e Initial load
duke
parents:
diff changeset
   545
     *  (un)initsWhenTrue(WhenFalse) on exit.
06bc494ca11e Initial load
duke
parents:
diff changeset
   546
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   547
    void scanExpr(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   548
        if (tree != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   549
            scan(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   550
            if (inits == null) merge();
06bc494ca11e Initial load
duke
parents:
diff changeset
   551
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   552
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   553
06bc494ca11e Initial load
duke
parents:
diff changeset
   554
    /** Analyze a list of expressions.
06bc494ca11e Initial load
duke
parents:
diff changeset
   555
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   556
    void scanExprs(List<? extends JCExpression> trees) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   557
        if (trees != null)
06bc494ca11e Initial load
duke
parents:
diff changeset
   558
            for (List<? extends JCExpression> l = trees; l.nonEmpty(); l = l.tail)
06bc494ca11e Initial load
duke
parents:
diff changeset
   559
                scanExpr(l.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
   560
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   561
06bc494ca11e Initial load
duke
parents:
diff changeset
   562
    /** Analyze a condition. Make sure to set (un)initsWhenTrue(WhenFalse)
06bc494ca11e Initial load
duke
parents:
diff changeset
   563
     *  rather than (un)inits on exit.
06bc494ca11e Initial load
duke
parents:
diff changeset
   564
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   565
    void scanCond(JCTree tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   566
        if (tree.type.isFalse()) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   567
            if (inits == null) merge();
06bc494ca11e Initial load
duke
parents:
diff changeset
   568
            initsWhenTrue = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   569
            initsWhenTrue.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   570
            uninitsWhenTrue = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   571
            uninitsWhenTrue.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   572
            initsWhenFalse = inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   573
            uninitsWhenFalse = uninits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   574
        } else if (tree.type.isTrue()) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   575
            if (inits == null) merge();
06bc494ca11e Initial load
duke
parents:
diff changeset
   576
            initsWhenFalse = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   577
            initsWhenFalse.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   578
            uninitsWhenFalse = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   579
            uninitsWhenFalse.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   580
            initsWhenTrue = inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   581
            uninitsWhenTrue = uninits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   582
        } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   583
            scan(tree);
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   584
            if (inits != null)
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   585
                split(tree.type != syms.unknownType);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   586
        }
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   587
        if (tree.type != syms.unknownType)
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
   588
            inits = uninits = null;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   589
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   590
06bc494ca11e Initial load
duke
parents:
diff changeset
   591
    /* ------------ Visitor methods for various sorts of trees -------------*/
06bc494ca11e Initial load
duke
parents:
diff changeset
   592
06bc494ca11e Initial load
duke
parents:
diff changeset
   593
    public void visitClassDef(JCClassDecl tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   594
        if (tree.sym == null) return;
06bc494ca11e Initial load
duke
parents:
diff changeset
   595
06bc494ca11e Initial load
duke
parents:
diff changeset
   596
        JCClassDecl classDefPrev = classDef;
06bc494ca11e Initial load
duke
parents:
diff changeset
   597
        List<Type> thrownPrev = thrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
   598
        List<Type> caughtPrev = caught;
06bc494ca11e Initial load
duke
parents:
diff changeset
   599
        boolean alivePrev = alive;
06bc494ca11e Initial load
duke
parents:
diff changeset
   600
        int firstadrPrev = firstadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   601
        int nextadrPrev = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   602
        ListBuffer<PendingExit> pendingExitsPrev = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   603
        Lint lintPrev = lint;
06bc494ca11e Initial load
duke
parents:
diff changeset
   604
06bc494ca11e Initial load
duke
parents:
diff changeset
   605
        pendingExits = new ListBuffer<PendingExit>();
06bc494ca11e Initial load
duke
parents:
diff changeset
   606
        if (tree.name != names.empty) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   607
            caught = List.nil();
06bc494ca11e Initial load
duke
parents:
diff changeset
   608
            firstadr = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   609
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   610
        classDef = tree;
06bc494ca11e Initial load
duke
parents:
diff changeset
   611
        thrown = List.nil();
06bc494ca11e Initial load
duke
parents:
diff changeset
   612
        lint = lint.augment(tree.sym.attributes_field);
06bc494ca11e Initial load
duke
parents:
diff changeset
   613
06bc494ca11e Initial load
duke
parents:
diff changeset
   614
        try {
06bc494ca11e Initial load
duke
parents:
diff changeset
   615
            // define all the static fields
06bc494ca11e Initial load
duke
parents:
diff changeset
   616
            for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   617
                if (l.head.getTag() == JCTree.VARDEF) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   618
                    JCVariableDecl def = (JCVariableDecl)l.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   619
                    if ((def.mods.flags & STATIC) != 0) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   620
                        VarSymbol sym = def.sym;
06bc494ca11e Initial load
duke
parents:
diff changeset
   621
                        if (trackable(sym))
06bc494ca11e Initial load
duke
parents:
diff changeset
   622
                            newVar(sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   623
                    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   624
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   625
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   626
06bc494ca11e Initial load
duke
parents:
diff changeset
   627
            // process all the static initializers
06bc494ca11e Initial load
duke
parents:
diff changeset
   628
            for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   629
                if (l.head.getTag() != JCTree.METHODDEF &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   630
                    (TreeInfo.flags(l.head) & STATIC) != 0) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   631
                    scanDef(l.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
   632
                    errorUncaught();
06bc494ca11e Initial load
duke
parents:
diff changeset
   633
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   634
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   635
06bc494ca11e Initial load
duke
parents:
diff changeset
   636
            // add intersection of all thrown clauses of initial constructors
06bc494ca11e Initial load
duke
parents:
diff changeset
   637
            // to set of caught exceptions, unless class is anonymous.
06bc494ca11e Initial load
duke
parents:
diff changeset
   638
            if (tree.name != names.empty) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   639
                boolean firstConstructor = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   640
                for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   641
                    if (TreeInfo.isInitialConstructor(l.head)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   642
                        List<Type> mthrown =
06bc494ca11e Initial load
duke
parents:
diff changeset
   643
                            ((JCMethodDecl) l.head).sym.type.getThrownTypes();
06bc494ca11e Initial load
duke
parents:
diff changeset
   644
                        if (firstConstructor) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   645
                            caught = mthrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
   646
                            firstConstructor = false;
06bc494ca11e Initial load
duke
parents:
diff changeset
   647
                        } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   648
                            caught = chk.intersect(mthrown, caught);
06bc494ca11e Initial load
duke
parents:
diff changeset
   649
                        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   650
                    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   651
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   652
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   653
06bc494ca11e Initial load
duke
parents:
diff changeset
   654
            // define all the instance fields
06bc494ca11e Initial load
duke
parents:
diff changeset
   655
            for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   656
                if (l.head.getTag() == JCTree.VARDEF) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   657
                    JCVariableDecl def = (JCVariableDecl)l.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   658
                    if ((def.mods.flags & STATIC) == 0) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   659
                        VarSymbol sym = def.sym;
06bc494ca11e Initial load
duke
parents:
diff changeset
   660
                        if (trackable(sym))
06bc494ca11e Initial load
duke
parents:
diff changeset
   661
                            newVar(sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   662
                    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   663
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   664
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   665
06bc494ca11e Initial load
duke
parents:
diff changeset
   666
            // process all the instance initializers
06bc494ca11e Initial load
duke
parents:
diff changeset
   667
            for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   668
                if (l.head.getTag() != JCTree.METHODDEF &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   669
                    (TreeInfo.flags(l.head) & STATIC) == 0) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   670
                    scanDef(l.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
   671
                    errorUncaught();
06bc494ca11e Initial load
duke
parents:
diff changeset
   672
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   673
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   674
06bc494ca11e Initial load
duke
parents:
diff changeset
   675
            // in an anonymous class, add the set of thrown exceptions to
06bc494ca11e Initial load
duke
parents:
diff changeset
   676
            // the throws clause of the synthetic constructor and propagate
06bc494ca11e Initial load
duke
parents:
diff changeset
   677
            // outwards.
8430
be3e5581ea25 6990136: Cleanup use of Type.clone()
dlsmith
parents: 8428
diff changeset
   678
            // Changing the throws clause on the fly is okay here because
be3e5581ea25 6990136: Cleanup use of Type.clone()
dlsmith
parents: 8428
diff changeset
   679
            // the anonymous constructor can't be invoked anywhere else,
be3e5581ea25 6990136: Cleanup use of Type.clone()
dlsmith
parents: 8428
diff changeset
   680
            // and its type hasn't been cached.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   681
            if (tree.name == names.empty) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   682
                for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   683
                    if (TreeInfo.isInitialConstructor(l.head)) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   684
                        JCMethodDecl mdef = (JCMethodDecl)l.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   685
                        mdef.thrown = make.Types(thrown);
8430
be3e5581ea25 6990136: Cleanup use of Type.clone()
dlsmith
parents: 8428
diff changeset
   686
                        mdef.sym.type = types.createMethodTypeWithThrown(mdef.sym.type, thrown);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   687
                    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   688
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   689
                thrownPrev = chk.union(thrown, thrownPrev);
06bc494ca11e Initial load
duke
parents:
diff changeset
   690
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   691
06bc494ca11e Initial load
duke
parents:
diff changeset
   692
            // process all the methods
06bc494ca11e Initial load
duke
parents:
diff changeset
   693
            for (List<JCTree> l = tree.defs; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   694
                if (l.head.getTag() == JCTree.METHODDEF) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   695
                    scan(l.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
   696
                    errorUncaught();
06bc494ca11e Initial load
duke
parents:
diff changeset
   697
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   698
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   699
06bc494ca11e Initial load
duke
parents:
diff changeset
   700
            thrown = thrownPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   701
        } finally {
06bc494ca11e Initial load
duke
parents:
diff changeset
   702
            pendingExits = pendingExitsPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   703
            alive = alivePrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   704
            nextadr = nextadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   705
            firstadr = firstadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   706
            caught = caughtPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   707
            classDef = classDefPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   708
            lint = lintPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   709
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   710
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   711
06bc494ca11e Initial load
duke
parents:
diff changeset
   712
    public void visitMethodDef(JCMethodDecl tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   713
        if (tree.body == null) return;
06bc494ca11e Initial load
duke
parents:
diff changeset
   714
06bc494ca11e Initial load
duke
parents:
diff changeset
   715
        List<Type> caughtPrev = caught;
06bc494ca11e Initial load
duke
parents:
diff changeset
   716
        List<Type> mthrown = tree.sym.type.getThrownTypes();
06bc494ca11e Initial load
duke
parents:
diff changeset
   717
        Bits initsPrev = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   718
        Bits uninitsPrev = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   719
        int nextadrPrev = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   720
        int firstadrPrev = firstadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   721
        Lint lintPrev = lint;
06bc494ca11e Initial load
duke
parents:
diff changeset
   722
06bc494ca11e Initial load
duke
parents:
diff changeset
   723
        lint = lint.augment(tree.sym.attributes_field);
06bc494ca11e Initial load
duke
parents:
diff changeset
   724
8032
e1aa25ccdabb 6396503: javac should not require assertions enabled
jjg
parents: 8031
diff changeset
   725
        Assert.check(pendingExits.isEmpty());
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   726
06bc494ca11e Initial load
duke
parents:
diff changeset
   727
        try {
06bc494ca11e Initial load
duke
parents:
diff changeset
   728
            boolean isInitialConstructor =
06bc494ca11e Initial load
duke
parents:
diff changeset
   729
                TreeInfo.isInitialConstructor(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   730
06bc494ca11e Initial load
duke
parents:
diff changeset
   731
            if (!isInitialConstructor)
06bc494ca11e Initial load
duke
parents:
diff changeset
   732
                firstadr = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   733
            for (List<JCVariableDecl> l = tree.params; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   734
                JCVariableDecl def = l.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   735
                scan(def);
06bc494ca11e Initial load
duke
parents:
diff changeset
   736
                inits.incl(def.sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   737
                uninits.excl(def.sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   738
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   739
            if (isInitialConstructor)
06bc494ca11e Initial load
duke
parents:
diff changeset
   740
                caught = chk.union(caught, mthrown);
06bc494ca11e Initial load
duke
parents:
diff changeset
   741
            else if ((tree.sym.flags() & (BLOCK | STATIC)) != BLOCK)
06bc494ca11e Initial load
duke
parents:
diff changeset
   742
                caught = mthrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
   743
            // else we are in an instance initializer block;
06bc494ca11e Initial load
duke
parents:
diff changeset
   744
            // leave caught unchanged.
06bc494ca11e Initial load
duke
parents:
diff changeset
   745
06bc494ca11e Initial load
duke
parents:
diff changeset
   746
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   747
            scanStat(tree.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
   748
06bc494ca11e Initial load
duke
parents:
diff changeset
   749
            if (alive && tree.sym.type.getReturnType().tag != VOID)
06bc494ca11e Initial load
duke
parents:
diff changeset
   750
                log.error(TreeInfo.diagEndPos(tree.body), "missing.ret.stmt");
06bc494ca11e Initial load
duke
parents:
diff changeset
   751
06bc494ca11e Initial load
duke
parents:
diff changeset
   752
            if (isInitialConstructor) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   753
                for (int i = firstadr; i < nextadr; i++)
06bc494ca11e Initial load
duke
parents:
diff changeset
   754
                    if (vars[i].owner == classDef.sym)
06bc494ca11e Initial load
duke
parents:
diff changeset
   755
                        checkInit(TreeInfo.diagEndPos(tree.body), vars[i]);
06bc494ca11e Initial load
duke
parents:
diff changeset
   756
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   757
            List<PendingExit> exits = pendingExits.toList();
06bc494ca11e Initial load
duke
parents:
diff changeset
   758
            pendingExits = new ListBuffer<PendingExit>();
06bc494ca11e Initial load
duke
parents:
diff changeset
   759
            while (exits.nonEmpty()) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   760
                PendingExit exit = exits.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   761
                exits = exits.tail;
06bc494ca11e Initial load
duke
parents:
diff changeset
   762
                if (exit.thrown == null) {
8032
e1aa25ccdabb 6396503: javac should not require assertions enabled
jjg
parents: 8031
diff changeset
   763
                    Assert.check(exit.tree.getTag() == JCTree.RETURN);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   764
                    if (isInitialConstructor) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   765
                        inits = exit.inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   766
                        for (int i = firstadr; i < nextadr; i++)
06bc494ca11e Initial load
duke
parents:
diff changeset
   767
                            checkInit(exit.tree.pos(), vars[i]);
06bc494ca11e Initial load
duke
parents:
diff changeset
   768
                    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   769
                } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   770
                    // uncaught throws will be reported later
06bc494ca11e Initial load
duke
parents:
diff changeset
   771
                    pendingExits.append(exit);
06bc494ca11e Initial load
duke
parents:
diff changeset
   772
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   773
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   774
        } finally {
06bc494ca11e Initial load
duke
parents:
diff changeset
   775
            inits = initsPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   776
            uninits = uninitsPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   777
            nextadr = nextadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   778
            firstadr = firstadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   779
            caught = caughtPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   780
            lint = lintPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   781
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   782
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   783
06bc494ca11e Initial load
duke
parents:
diff changeset
   784
    public void visitVarDef(JCVariableDecl tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   785
        boolean track = trackable(tree.sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   786
        if (track && tree.sym.owner.kind == MTH) newVar(tree.sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   787
        if (tree.init != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   788
            Lint lintPrev = lint;
06bc494ca11e Initial load
duke
parents:
diff changeset
   789
            lint = lint.augment(tree.sym.attributes_field);
06bc494ca11e Initial load
duke
parents:
diff changeset
   790
            try{
06bc494ca11e Initial load
duke
parents:
diff changeset
   791
                scanExpr(tree.init);
06bc494ca11e Initial load
duke
parents:
diff changeset
   792
                if (track) letInit(tree.pos(), tree.sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   793
            } finally {
06bc494ca11e Initial load
duke
parents:
diff changeset
   794
                lint = lintPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   795
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   796
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   797
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   798
06bc494ca11e Initial load
duke
parents:
diff changeset
   799
    public void visitBlock(JCBlock tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   800
        int nextadrPrev = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   801
        scanStats(tree.stats);
06bc494ca11e Initial load
duke
parents:
diff changeset
   802
        nextadr = nextadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   803
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   804
06bc494ca11e Initial load
duke
parents:
diff changeset
   805
    public void visitDoLoop(JCDoWhileLoop tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   806
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   807
        boolean prevLoopPassTwo = loopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   808
        pendingExits = new ListBuffer<PendingExit>();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   809
        int prevErrors = log.nerrors;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   810
        do {
06bc494ca11e Initial load
duke
parents:
diff changeset
   811
            Bits uninitsEntry = uninits.dup();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   812
            uninitsEntry.excludeFrom(nextadr);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   813
            scanStat(tree.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
   814
            alive |= resolveContinues(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   815
            scanCond(tree.cond);
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   816
            if (log.nerrors !=  prevErrors ||
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   817
                loopPassTwo ||
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   818
                uninitsEntry.dup().diffSet(uninitsWhenTrue).nextBit(firstadr)==-1)
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   819
                break;
06bc494ca11e Initial load
duke
parents:
diff changeset
   820
            inits = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   821
            uninits = uninitsEntry.andSet(uninitsWhenTrue);
06bc494ca11e Initial load
duke
parents:
diff changeset
   822
            loopPassTwo = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   823
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   824
        } while (true);
06bc494ca11e Initial load
duke
parents:
diff changeset
   825
        loopPassTwo = prevLoopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   826
        inits = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   827
        uninits = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   828
        alive = alive && !tree.cond.type.isTrue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   829
        alive |= resolveBreaks(tree, prevPendingExits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   830
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   831
06bc494ca11e Initial load
duke
parents:
diff changeset
   832
    public void visitWhileLoop(JCWhileLoop tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   833
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   834
        boolean prevLoopPassTwo = loopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   835
        Bits initsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   836
        Bits uninitsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   837
        pendingExits = new ListBuffer<PendingExit>();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   838
        int prevErrors = log.nerrors;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   839
        do {
06bc494ca11e Initial load
duke
parents:
diff changeset
   840
            Bits uninitsEntry = uninits.dup();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   841
            uninitsEntry.excludeFrom(nextadr);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   842
            scanCond(tree.cond);
06bc494ca11e Initial load
duke
parents:
diff changeset
   843
            initsCond = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   844
            uninitsCond = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   845
            inits = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   846
            uninits = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   847
            alive = !tree.cond.type.isFalse();
06bc494ca11e Initial load
duke
parents:
diff changeset
   848
            scanStat(tree.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
   849
            alive |= resolveContinues(tree);
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   850
            if (log.nerrors != prevErrors ||
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   851
                loopPassTwo ||
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   852
                uninitsEntry.dup().diffSet(uninits).nextBit(firstadr) == -1)
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   853
                break;
06bc494ca11e Initial load
duke
parents:
diff changeset
   854
            uninits = uninitsEntry.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   855
            loopPassTwo = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   856
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   857
        } while (true);
06bc494ca11e Initial load
duke
parents:
diff changeset
   858
        loopPassTwo = prevLoopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   859
        inits = initsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   860
        uninits = uninitsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   861
        alive = resolveBreaks(tree, prevPendingExits) ||
06bc494ca11e Initial load
duke
parents:
diff changeset
   862
            !tree.cond.type.isTrue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   863
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   864
06bc494ca11e Initial load
duke
parents:
diff changeset
   865
    public void visitForLoop(JCForLoop tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   866
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   867
        boolean prevLoopPassTwo = loopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   868
        int nextadrPrev = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   869
        scanStats(tree.init);
06bc494ca11e Initial load
duke
parents:
diff changeset
   870
        Bits initsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   871
        Bits uninitsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   872
        pendingExits = new ListBuffer<PendingExit>();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   873
        int prevErrors = log.nerrors;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   874
        do {
06bc494ca11e Initial load
duke
parents:
diff changeset
   875
            Bits uninitsEntry = uninits.dup();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   876
            uninitsEntry.excludeFrom(nextadr);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   877
            if (tree.cond != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   878
                scanCond(tree.cond);
06bc494ca11e Initial load
duke
parents:
diff changeset
   879
                initsCond = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   880
                uninitsCond = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
   881
                inits = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   882
                uninits = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
   883
                alive = !tree.cond.type.isFalse();
06bc494ca11e Initial load
duke
parents:
diff changeset
   884
            } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   885
                initsCond = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   886
                initsCond.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   887
                uninitsCond = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   888
                uninitsCond.inclRange(firstadr, nextadr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   889
                alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   890
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   891
            scanStat(tree.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
   892
            alive |= resolveContinues(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
   893
            scan(tree.step);
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   894
            if (log.nerrors != prevErrors ||
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   895
                loopPassTwo ||
06bc494ca11e Initial load
duke
parents:
diff changeset
   896
                uninitsEntry.dup().diffSet(uninits).nextBit(firstadr) == -1)
06bc494ca11e Initial load
duke
parents:
diff changeset
   897
                break;
06bc494ca11e Initial load
duke
parents:
diff changeset
   898
            uninits = uninitsEntry.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   899
            loopPassTwo = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   900
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   901
        } while (true);
06bc494ca11e Initial load
duke
parents:
diff changeset
   902
        loopPassTwo = prevLoopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   903
        inits = initsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   904
        uninits = uninitsCond;
06bc494ca11e Initial load
duke
parents:
diff changeset
   905
        alive = resolveBreaks(tree, prevPendingExits) ||
06bc494ca11e Initial load
duke
parents:
diff changeset
   906
            tree.cond != null && !tree.cond.type.isTrue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   907
        nextadr = nextadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   908
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   909
06bc494ca11e Initial load
duke
parents:
diff changeset
   910
    public void visitForeachLoop(JCEnhancedForLoop tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   911
        visitVarDef(tree.var);
06bc494ca11e Initial load
duke
parents:
diff changeset
   912
06bc494ca11e Initial load
duke
parents:
diff changeset
   913
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   914
        boolean prevLoopPassTwo = loopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   915
        int nextadrPrev = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   916
        scan(tree.expr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   917
        Bits initsStart = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   918
        Bits uninitsStart = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   919
06bc494ca11e Initial load
duke
parents:
diff changeset
   920
        letInit(tree.pos(), tree.var.sym);
06bc494ca11e Initial load
duke
parents:
diff changeset
   921
        pendingExits = new ListBuffer<PendingExit>();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   922
        int prevErrors = log.nerrors;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   923
        do {
06bc494ca11e Initial load
duke
parents:
diff changeset
   924
            Bits uninitsEntry = uninits.dup();
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   925
            uninitsEntry.excludeFrom(nextadr);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   926
            scanStat(tree.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
   927
            alive |= resolveContinues(tree);
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   928
            if (log.nerrors != prevErrors ||
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   929
                loopPassTwo ||
8627
e60ca990602f 7023703: Valid code doesn't compile
mcimadamore
parents: 8626
diff changeset
   930
                uninitsEntry.dup().diffSet(uninits).nextBit(firstadr) == -1)
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   931
                break;
06bc494ca11e Initial load
duke
parents:
diff changeset
   932
            uninits = uninitsEntry.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   933
            loopPassTwo = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   934
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   935
        } while (true);
06bc494ca11e Initial load
duke
parents:
diff changeset
   936
        loopPassTwo = prevLoopPassTwo;
06bc494ca11e Initial load
duke
parents:
diff changeset
   937
        inits = initsStart;
06bc494ca11e Initial load
duke
parents:
diff changeset
   938
        uninits = uninitsStart.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   939
        resolveBreaks(tree, prevPendingExits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   940
        alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   941
        nextadr = nextadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   942
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   943
06bc494ca11e Initial load
duke
parents:
diff changeset
   944
    public void visitLabelled(JCLabeledStatement tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   945
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   946
        pendingExits = new ListBuffer<PendingExit>();
06bc494ca11e Initial load
duke
parents:
diff changeset
   947
        scanStat(tree.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
   948
        alive |= resolveBreaks(tree, prevPendingExits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   949
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   950
06bc494ca11e Initial load
duke
parents:
diff changeset
   951
    public void visitSwitch(JCSwitch tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   952
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   953
        pendingExits = new ListBuffer<PendingExit>();
06bc494ca11e Initial load
duke
parents:
diff changeset
   954
        int nextadrPrev = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   955
        scanExpr(tree.selector);
06bc494ca11e Initial load
duke
parents:
diff changeset
   956
        Bits initsSwitch = inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
   957
        Bits uninitsSwitch = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   958
        boolean hasDefault = false;
06bc494ca11e Initial load
duke
parents:
diff changeset
   959
        for (List<JCCase> l = tree.cases; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   960
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   961
            inits = initsSwitch.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
   962
            uninits = uninits.andSet(uninitsSwitch);
06bc494ca11e Initial load
duke
parents:
diff changeset
   963
            JCCase c = l.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   964
            if (c.pat == null)
06bc494ca11e Initial load
duke
parents:
diff changeset
   965
                hasDefault = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   966
            else
06bc494ca11e Initial load
duke
parents:
diff changeset
   967
                scanExpr(c.pat);
06bc494ca11e Initial load
duke
parents:
diff changeset
   968
            scanStats(c.stats);
06bc494ca11e Initial load
duke
parents:
diff changeset
   969
            addVars(c.stats, initsSwitch, uninitsSwitch);
06bc494ca11e Initial load
duke
parents:
diff changeset
   970
            // Warn about fall-through if lint switch fallthrough enabled.
06bc494ca11e Initial load
duke
parents:
diff changeset
   971
            if (!loopPassTwo &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   972
                alive &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   973
                lint.isEnabled(Lint.LintCategory.FALLTHROUGH) &&
06bc494ca11e Initial load
duke
parents:
diff changeset
   974
                c.stats.nonEmpty() && l.tail.nonEmpty())
6151
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
   975
                log.warning(Lint.LintCategory.FALLTHROUGH,
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
   976
                            l.tail.head.pos(),
10
06bc494ca11e Initial load
duke
parents:
diff changeset
   977
                            "possible.fall-through.into.case");
06bc494ca11e Initial load
duke
parents:
diff changeset
   978
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   979
        if (!hasDefault) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   980
            inits.andSet(initsSwitch);
06bc494ca11e Initial load
duke
parents:
diff changeset
   981
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
   982
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   983
        alive |= resolveBreaks(tree, prevPendingExits);
06bc494ca11e Initial load
duke
parents:
diff changeset
   984
        nextadr = nextadrPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
   985
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   986
    // where
06bc494ca11e Initial load
duke
parents:
diff changeset
   987
        /** Add any variables defined in stats to inits and uninits. */
06bc494ca11e Initial load
duke
parents:
diff changeset
   988
        private static void addVars(List<JCStatement> stats, Bits inits,
06bc494ca11e Initial load
duke
parents:
diff changeset
   989
                                    Bits uninits) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   990
            for (;stats.nonEmpty(); stats = stats.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   991
                JCTree stat = stats.head;
06bc494ca11e Initial load
duke
parents:
diff changeset
   992
                if (stat.getTag() == JCTree.VARDEF) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   993
                    int adr = ((JCVariableDecl) stat).sym.adr;
06bc494ca11e Initial load
duke
parents:
diff changeset
   994
                    inits.excl(adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   995
                    uninits.incl(adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
   996
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   997
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   998
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   999
06bc494ca11e Initial load
duke
parents:
diff changeset
  1000
    public void visitTry(JCTry tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1001
        List<Type> caughtPrev = caught;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1002
        List<Type> thrownPrev = thrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1003
        thrown = List.nil();
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1004
        for (List<JCCatch> l = tree.catchers; l.nonEmpty(); l = l.tail) {
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1005
            List<JCExpression> subClauses = TreeInfo.isMultiCatch(l.head) ?
9300
c2de4dd9853b 7033809: Rename "disjunctive" to "union" in javax.lang.model
darcy
parents: 8849
diff changeset
  1006
                    ((JCTypeUnion)l.head.param.vartype).alternatives :
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1007
                    List.of(l.head.param.vartype);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1008
            for (JCExpression ct : subClauses) {
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1009
                caught = chk.incl(ct.type, caught);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1010
            }
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1011
        }
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1012
        ListBuffer<JCVariableDecl> resourceVarDecls = ListBuffer.lb();
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1013
        Bits uninitsTryPrev = uninitsTry;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1014
        ListBuffer<PendingExit> prevPendingExits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1015
        pendingExits = new ListBuffer<PendingExit>();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1016
        Bits initsTry = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1017
        uninitsTry = uninits.dup();
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1018
        for (JCTree resource : tree.resources) {
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1019
            if (resource instanceof JCVariableDecl) {
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1020
                JCVariableDecl vdecl = (JCVariableDecl) resource;
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1021
                visitVarDef(vdecl);
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1022
                unrefdResources.enter(vdecl.sym);
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1023
                resourceVarDecls.append(vdecl);
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1024
            } else if (resource instanceof JCExpression) {
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1025
                scanExpr((JCExpression) resource);
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1026
            } else {
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1027
                throw new AssertionError(tree);  // parser error
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1028
            }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1029
        }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1030
        for (JCTree resource : tree.resources) {
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1031
            List<Type> closeableSupertypes = resource.type.isCompound() ?
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1032
                types.interfaces(resource.type).prepend(types.supertype(resource.type)) :
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1033
                List.of(resource.type);
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1034
            for (Type sup : closeableSupertypes) {
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1035
                if (types.asSuper(sup, syms.autoCloseableType.tsym) != null) {
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
  1036
                    Symbol closeMethod = rs.resolveQualifiedMethod(tree,
6156
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1037
                            attrEnv,
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1038
                            sup,
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1039
                            names.close,
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1040
                            List.<Type>nil(),
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1041
                            List.<Type>nil());
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1042
                    if (closeMethod.kind == MTH) {
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1043
                        for (Type t : ((MethodSymbol)closeMethod).getThrownTypes()) {
8428
1e9935b883cd 7017104: improve error reporting for uncaught/undeclared exceptions from try-with-resources
mcimadamore
parents: 8036
diff changeset
  1044
                            markThrown(resource, t);
6156
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1045
                        }
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1046
                    }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1047
                }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1048
            }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1049
        }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1050
        scanStat(tree.body);
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1051
        List<Type> thrownInTry = allowImprovedCatchAnalysis ?
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1052
            chk.union(thrown, List.of(syms.runtimeExceptionType, syms.errorType)) :
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1053
            thrown;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1054
        thrown = thrownPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1055
        caught = caughtPrev;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1056
        boolean aliveEnd = alive;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1057
        uninitsTry.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1058
        Bits initsEnd = inits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1059
        Bits uninitsEnd = uninits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1060
        int nextadrCatch = nextadr;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1061
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1062
        if (!resourceVarDecls.isEmpty() &&
7211
163fe60f63de 6970016: Clean up ARM/try-with-resources implementation
mcimadamore
parents: 7210
diff changeset
  1063
                lint.isEnabled(Lint.LintCategory.TRY)) {
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1064
            for (JCVariableDecl resVar : resourceVarDecls) {
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1065
                if (unrefdResources.includes(resVar.sym)) {
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1066
                    log.warning(Lint.LintCategory.TRY, resVar.pos(),
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1067
                                "try.resource.not.referenced", resVar.sym);
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1068
                    unrefdResources.remove(resVar.sym);
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1069
                }
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1070
            }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1071
        }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1072
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1073
        List<Type> caughtInTry = List.nil();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1074
        for (List<JCCatch> l = tree.catchers; l.nonEmpty(); l = l.tail) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1075
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1076
            JCVariableDecl param = l.head.param;
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1077
            List<JCExpression> subClauses = TreeInfo.isMultiCatch(l.head) ?
9300
c2de4dd9853b 7033809: Rename "disjunctive" to "union" in javax.lang.model
darcy
parents: 8849
diff changeset
  1078
                    ((JCTypeUnion)l.head.param.vartype).alternatives :
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1079
                    List.of(l.head.param.vartype);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1080
            List<Type> ctypes = List.nil();
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1081
            List<Type> rethrownTypes = chk.diff(thrownInTry, caughtInTry);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1082
            for (JCExpression ct : subClauses) {
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1083
                Type exc = ct.type;
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
  1084
                if (exc != syms.unknownType) {
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
  1085
                    ctypes = ctypes.append(exc);
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
  1086
                    if (types.isSameType(exc, syms.objectType))
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
  1087
                        continue;
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1088
                    checkCaughtType(l.head.pos(), exc, thrownInTry, caughtInTry);
6594
d43f068fba19 6970584: Flow.java should be more error-friendly
mcimadamore
parents: 6156
diff changeset
  1089
                    caughtInTry = chk.incl(exc, caughtInTry);
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1090
                }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1091
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1092
            inits = initsTry.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1093
            uninits = uninitsTry.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1094
            scan(param);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1095
            inits.incl(param.sym.adr);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1096
            uninits.excl(param.sym.adr);
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
  1097
            preciseRethrowTypes.put(param.sym, chk.intersect(ctypes, rethrownTypes));
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1098
            scanStat(l.head.body);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1099
            initsEnd.andSet(inits);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1100
            uninitsEnd.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1101
            nextadr = nextadrCatch;
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
  1102
            preciseRethrowTypes.remove(param.sym);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1103
            aliveEnd |= alive;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1104
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1105
        if (tree.finalizer != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1106
            List<Type> savedThrown = thrown;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1107
            thrown = List.nil();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1108
            inits = initsTry.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1109
            uninits = uninitsTry.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1110
            ListBuffer<PendingExit> exits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1111
            pendingExits = prevPendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1112
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1113
            scanStat(tree.finalizer);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1114
            if (!alive) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1115
                // discard exits and exceptions from try and finally
06bc494ca11e Initial load
duke
parents:
diff changeset
  1116
                thrown = chk.union(thrown, thrownPrev);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1117
                if (!loopPassTwo &&
06bc494ca11e Initial load
duke
parents:
diff changeset
  1118
                    lint.isEnabled(Lint.LintCategory.FINALLY)) {
6151
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
  1119
                    log.warning(Lint.LintCategory.FINALLY,
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
  1120
                            TreeInfo.diagEndPos(tree.finalizer),
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
  1121
                            "finally.cannot.complete");
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1122
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1123
            } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1124
                thrown = chk.union(thrown, chk.diff(thrownInTry, caughtInTry));
06bc494ca11e Initial load
duke
parents:
diff changeset
  1125
                thrown = chk.union(thrown, savedThrown);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1126
                uninits.andSet(uninitsEnd);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1127
                // FIX: this doesn't preserve source order of exits in catch
06bc494ca11e Initial load
duke
parents:
diff changeset
  1128
                // versus finally!
06bc494ca11e Initial load
duke
parents:
diff changeset
  1129
                while (exits.nonEmpty()) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1130
                    PendingExit exit = exits.next();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1131
                    if (exit.inits != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1132
                        exit.inits.orSet(inits);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1133
                        exit.uninits.andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1134
                    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1135
                    pendingExits.append(exit);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1136
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1137
                inits.orSet(initsEnd);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1138
                alive = aliveEnd;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1139
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1140
        } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1141
            thrown = chk.union(thrown, chk.diff(thrownInTry, caughtInTry));
06bc494ca11e Initial load
duke
parents:
diff changeset
  1142
            inits = initsEnd;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1143
            uninits = uninitsEnd;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1144
            alive = aliveEnd;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1145
            ListBuffer<PendingExit> exits = pendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1146
            pendingExits = prevPendingExits;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1147
            while (exits.nonEmpty()) pendingExits.append(exits.next());
06bc494ca11e Initial load
duke
parents:
diff changeset
  1148
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1149
        uninitsTry.andSet(uninitsTryPrev).andSet(uninits);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1150
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1151
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1152
    void checkCaughtType(DiagnosticPosition pos, Type exc, List<Type> thrownInTry, List<Type> caughtInTry) {
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1153
        if (chk.subset(exc, caughtInTry)) {
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1154
            log.error(pos, "except.already.caught", exc);
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1155
        } else if (!chk.isUnchecked(pos, exc) &&
9601
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1156
                !isExceptionOrThrowable(exc) &&
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1157
                !chk.intersects(exc, thrownInTry)) {
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1158
            log.error(pos, "except.never.thrown.in.try", exc);
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1159
        } else if (allowImprovedCatchAnalysis) {
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1160
            List<Type> catchableThrownTypes = chk.intersect(List.of(exc), thrownInTry);
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1161
            // 'catchableThrownTypes' cannnot possibly be empty - if 'exc' was an
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1162
            // unchecked exception, the result list would not be empty, as the augmented
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1163
            // thrown set includes { RuntimeException, Error }; if 'exc' was a checked
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1164
            // exception, that would have been covered in the branch above
9601
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1165
            if (chk.diff(catchableThrownTypes, caughtInTry).isEmpty() &&
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1166
                    !isExceptionOrThrowable(exc)) {
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1167
                String key = catchableThrownTypes.length() == 1 ?
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1168
                        "unreachable.catch" :
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1169
                        "unreachable.catch.1";
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1170
                log.warning(pos, key, catchableThrownTypes);
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1171
            }
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1172
        }
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1173
    }
9601
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1174
    //where
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1175
        private boolean isExceptionOrThrowable(Type exc) {
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1176
            return exc.tsym == syms.throwableType.tsym ||
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1177
                exc.tsym == syms.exceptionType.tsym;
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1178
        }
a1b1716231ca 7039937: Improved catch analysis fails to handle a common idiom in the libraries
mcimadamore
parents: 9303
diff changeset
  1179
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1180
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1181
    public void visitConditional(JCConditional tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1182
        scanCond(tree.cond);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1183
        Bits initsBeforeElse = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1184
        Bits uninitsBeforeElse = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1185
        inits = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1186
        uninits = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1187
        if (tree.truepart.type.tag == BOOLEAN &&
06bc494ca11e Initial load
duke
parents:
diff changeset
  1188
            tree.falsepart.type.tag == BOOLEAN) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1189
            // if b and c are boolean valued, then
06bc494ca11e Initial load
duke
parents:
diff changeset
  1190
            // v is (un)assigned after a?b:c when true iff
06bc494ca11e Initial load
duke
parents:
diff changeset
  1191
            //    v is (un)assigned after b when true and
06bc494ca11e Initial load
duke
parents:
diff changeset
  1192
            //    v is (un)assigned after c when true
06bc494ca11e Initial load
duke
parents:
diff changeset
  1193
            scanCond(tree.truepart);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1194
            Bits initsAfterThenWhenTrue = initsWhenTrue.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1195
            Bits initsAfterThenWhenFalse = initsWhenFalse.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1196
            Bits uninitsAfterThenWhenTrue = uninitsWhenTrue.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1197
            Bits uninitsAfterThenWhenFalse = uninitsWhenFalse.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1198
            inits = initsBeforeElse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1199
            uninits = uninitsBeforeElse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1200
            scanCond(tree.falsepart);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1201
            initsWhenTrue.andSet(initsAfterThenWhenTrue);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1202
            initsWhenFalse.andSet(initsAfterThenWhenFalse);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1203
            uninitsWhenTrue.andSet(uninitsAfterThenWhenTrue);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1204
            uninitsWhenFalse.andSet(uninitsAfterThenWhenFalse);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1205
        } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1206
            scanExpr(tree.truepart);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1207
            Bits initsAfterThen = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1208
            Bits uninitsAfterThen = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1209
            inits = initsBeforeElse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1210
            uninits = uninitsBeforeElse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1211
            scanExpr(tree.falsepart);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1212
            inits.andSet(initsAfterThen);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1213
            uninits.andSet(uninitsAfterThen);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1214
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1215
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1216
06bc494ca11e Initial load
duke
parents:
diff changeset
  1217
    public void visitIf(JCIf tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1218
        scanCond(tree.cond);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1219
        Bits initsBeforeElse = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1220
        Bits uninitsBeforeElse = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1221
        inits = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1222
        uninits = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1223
        scanStat(tree.thenpart);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1224
        if (tree.elsepart != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1225
            boolean aliveAfterThen = alive;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1226
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1227
            Bits initsAfterThen = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1228
            Bits uninitsAfterThen = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1229
            inits = initsBeforeElse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1230
            uninits = uninitsBeforeElse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1231
            scanStat(tree.elsepart);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1232
            inits.andSet(initsAfterThen);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1233
            uninits.andSet(uninitsAfterThen);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1234
            alive = alive | aliveAfterThen;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1235
        } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1236
            inits.andSet(initsBeforeElse);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1237
            uninits.andSet(uninitsBeforeElse);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1238
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1239
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1240
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1241
06bc494ca11e Initial load
duke
parents:
diff changeset
  1242
06bc494ca11e Initial load
duke
parents:
diff changeset
  1243
06bc494ca11e Initial load
duke
parents:
diff changeset
  1244
    public void visitBreak(JCBreak tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1245
        recordExit(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1246
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1247
06bc494ca11e Initial load
duke
parents:
diff changeset
  1248
    public void visitContinue(JCContinue tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1249
        recordExit(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1250
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1251
06bc494ca11e Initial load
duke
parents:
diff changeset
  1252
    public void visitReturn(JCReturn tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1253
        scanExpr(tree.expr);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1254
        // if not initial constructor, should markDead instead of recordExit
06bc494ca11e Initial load
duke
parents:
diff changeset
  1255
        recordExit(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1256
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1257
06bc494ca11e Initial load
duke
parents:
diff changeset
  1258
    public void visitThrow(JCThrow tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1259
        scanExpr(tree.expr);
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1260
        Symbol sym = TreeInfo.symbol(tree.expr);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1261
        if (sym != null &&
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1262
            sym.kind == VAR &&
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
  1263
            (sym.flags() & (FINAL | EFFECTIVELY_FINAL)) != 0 &&
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
  1264
            preciseRethrowTypes.get(sym) != null &&
8849
4189ac38ddc9 6558548: The compiler needs to be aligned with clarified specification of throws
mcimadamore
parents: 8627
diff changeset
  1265
            allowImprovedRethrowAnalysis) {
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
  1266
            for (Type t : preciseRethrowTypes.get(sym)) {
5492
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1267
                markThrown(tree, t);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1268
            }
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1269
        }
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1270
        else {
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1271
            markThrown(tree, tree.expr.type);
515e4b33b335 6943289: Project Coin: Improved Exception Handling for Java (aka 'multicatch')
mcimadamore
parents: 3149
diff changeset
  1272
        }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1273
        markDead();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1274
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1275
06bc494ca11e Initial load
duke
parents:
diff changeset
  1276
    public void visitApply(JCMethodInvocation tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1277
        scanExpr(tree.meth);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1278
        scanExprs(tree.args);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1279
        for (List<Type> l = tree.meth.type.getThrownTypes(); l.nonEmpty(); l = l.tail)
06bc494ca11e Initial load
duke
parents:
diff changeset
  1280
            markThrown(tree, l.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1281
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1282
06bc494ca11e Initial load
duke
parents:
diff changeset
  1283
    public void visitNewClass(JCNewClass tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1284
        scanExpr(tree.encl);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1285
        scanExprs(tree.args);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1286
       // scan(tree.def);
1791
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1287
        for (List<Type> l = tree.constructorType.getThrownTypes();
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1288
             l.nonEmpty();
1791
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1289
             l = l.tail) {
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1290
            markThrown(tree, l.head);
1791
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1291
        }
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1292
        List<Type> caughtPrev = caught;
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1293
        try {
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1294
            // If the new class expression defines an anonymous class,
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1295
            // analysis of the anonymous constructor may encounter thrown
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1296
            // types which are unsubstituted type variables.
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1297
            // However, since the constructor's actual thrown types have
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1298
            // already been marked as thrown, it is safe to simply include
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1299
            // each of the constructor's formal thrown types in the set of
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1300
            // 'caught/declared to be thrown' types, for the duration of
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1301
            // the class def analysis.
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1302
            if (tree.def != null)
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1303
                for (List<Type> l = tree.constructor.type.getThrownTypes();
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1304
                     l.nonEmpty();
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1305
                     l = l.tail) {
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1306
                    caught = chk.incl(l.head, caught);
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1307
                }
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1308
            scan(tree.def);
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1309
        }
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1310
        finally {
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1311
            caught = caughtPrev;
d378f023c36d 6723444: javac fails to substitute type variables into a constructor's throws clause
mcimadamore
parents: 1264
diff changeset
  1312
        }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1313
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1314
06bc494ca11e Initial load
duke
parents:
diff changeset
  1315
    public void visitNewArray(JCNewArray tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1316
        scanExprs(tree.dims);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1317
        scanExprs(tree.elems);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1318
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1319
06bc494ca11e Initial load
duke
parents:
diff changeset
  1320
    public void visitAssert(JCAssert tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1321
        Bits initsExit = inits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1322
        Bits uninitsExit = uninits.dup();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1323
        scanCond(tree.cond);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1324
        uninitsExit.andSet(uninitsWhenTrue);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1325
        if (tree.detail != null) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1326
            inits = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1327
            uninits = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1328
            scanExpr(tree.detail);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1329
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1330
        inits = initsExit;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1331
        uninits = uninitsExit;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1332
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1333
06bc494ca11e Initial load
duke
parents:
diff changeset
  1334
    public void visitAssign(JCAssign tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1335
        JCTree lhs = TreeInfo.skipParens(tree.lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1336
        if (!(lhs instanceof JCIdent)) scanExpr(lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1337
        scanExpr(tree.rhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1338
        letInit(lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1339
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1340
06bc494ca11e Initial load
duke
parents:
diff changeset
  1341
    public void visitAssignop(JCAssignOp tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1342
        scanExpr(tree.lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1343
        scanExpr(tree.rhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1344
        letInit(tree.lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1345
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1346
06bc494ca11e Initial load
duke
parents:
diff changeset
  1347
    public void visitUnary(JCUnary tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1348
        switch (tree.getTag()) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1349
        case JCTree.NOT:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1350
            scanCond(tree.arg);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1351
            Bits t = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1352
            initsWhenFalse = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1353
            initsWhenTrue = t;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1354
            t = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1355
            uninitsWhenFalse = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1356
            uninitsWhenTrue = t;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1357
            break;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1358
        case JCTree.PREINC: case JCTree.POSTINC:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1359
        case JCTree.PREDEC: case JCTree.POSTDEC:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1360
            scanExpr(tree.arg);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1361
            letInit(tree.arg);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1362
            break;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1363
        default:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1364
            scanExpr(tree.arg);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1365
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1366
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1367
06bc494ca11e Initial load
duke
parents:
diff changeset
  1368
    public void visitBinary(JCBinary tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1369
        switch (tree.getTag()) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1370
        case JCTree.AND:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1371
            scanCond(tree.lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1372
            Bits initsWhenFalseLeft = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1373
            Bits uninitsWhenFalseLeft = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1374
            inits = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1375
            uninits = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1376
            scanCond(tree.rhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1377
            initsWhenFalse.andSet(initsWhenFalseLeft);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1378
            uninitsWhenFalse.andSet(uninitsWhenFalseLeft);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1379
            break;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1380
        case JCTree.OR:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1381
            scanCond(tree.lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1382
            Bits initsWhenTrueLeft = initsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1383
            Bits uninitsWhenTrueLeft = uninitsWhenTrue;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1384
            inits = initsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1385
            uninits = uninitsWhenFalse;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1386
            scanCond(tree.rhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1387
            initsWhenTrue.andSet(initsWhenTrueLeft);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1388
            uninitsWhenTrue.andSet(uninitsWhenTrueLeft);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1389
            break;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1390
        default:
06bc494ca11e Initial load
duke
parents:
diff changeset
  1391
            scanExpr(tree.lhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1392
            scanExpr(tree.rhs);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1393
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1394
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1395
06bc494ca11e Initial load
duke
parents:
diff changeset
  1396
    public void visitIdent(JCIdent tree) {
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1397
        if (tree.sym.kind == VAR) {
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1398
            checkInit(tree.pos(), (VarSymbol)tree.sym);
6148
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1399
            referenced(tree.sym);
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1400
        }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1401
    }
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1402
3a8158299c51 6911256: Project Coin: Support Automatic Resource Management (ARM) blocks in the compiler
darcy
parents: 5847
diff changeset
  1403
    void referenced(Symbol sym) {
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1404
        unrefdResources.remove(sym);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1405
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1406
06bc494ca11e Initial load
duke
parents:
diff changeset
  1407
    public void visitTypeCast(JCTypeCast tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1408
        super.visitTypeCast(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1409
        if (!tree.type.isErroneous()
06bc494ca11e Initial load
duke
parents:
diff changeset
  1410
            && lint.isEnabled(Lint.LintCategory.CAST)
3149
0cd06d598d6f 6843077: JSR 308: Annotations on types
jjg
parents: 2212
diff changeset
  1411
            && types.isSameType(tree.expr.type, tree.clazz.type)
7210
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1412
            && !is292targetTypeCast(tree)) {
6151
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
  1413
            log.warning(Lint.LintCategory.CAST,
dd513881e71d 6957438: improve code for generating warning messages containing option names
jjg
parents: 6148
diff changeset
  1414
                    tree.pos(), "redundant.cast", tree.expr.type);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1415
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1416
    }
7210
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1417
    //where
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1418
        private boolean is292targetTypeCast(JCTypeCast tree) {
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1419
            boolean is292targetTypeCast = false;
8036
17b976649c61 6992698: JSR 292: remove support for transient syntax in polymorphic signature calls
mcimadamore
parents: 8032
diff changeset
  1420
            JCExpression expr = TreeInfo.skipParens(tree.expr);
17b976649c61 6992698: JSR 292: remove support for transient syntax in polymorphic signature calls
mcimadamore
parents: 8032
diff changeset
  1421
            if (expr.getTag() == JCTree.APPLY) {
17b976649c61 6992698: JSR 292: remove support for transient syntax in polymorphic signature calls
mcimadamore
parents: 8032
diff changeset
  1422
                JCMethodInvocation apply = (JCMethodInvocation)expr;
7210
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1423
                Symbol sym = TreeInfo.symbol(apply.meth);
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1424
                is292targetTypeCast = sym != null &&
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1425
                    sym.kind == MTH &&
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1426
                    (sym.flags() & POLYMORPHIC_SIGNATURE) != 0;
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1427
            }
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1428
            return is292targetTypeCast;
8dd5f907461e 6999067: cast for invokeExact call gets redundant cast to <type> warnings
mcimadamore
parents: 7203
diff changeset
  1429
        }
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1430
06bc494ca11e Initial load
duke
parents:
diff changeset
  1431
    public void visitTopLevel(JCCompilationUnit tree) {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1432
        // Do nothing for TopLevel since each class is visited individually
06bc494ca11e Initial load
duke
parents:
diff changeset
  1433
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1434
06bc494ca11e Initial load
duke
parents:
diff changeset
  1435
/**************************************************************************
06bc494ca11e Initial load
duke
parents:
diff changeset
  1436
 * main method
06bc494ca11e Initial load
duke
parents:
diff changeset
  1437
 *************************************************************************/
06bc494ca11e Initial load
duke
parents:
diff changeset
  1438
06bc494ca11e Initial load
duke
parents:
diff changeset
  1439
    /** Perform definite assignment/unassignment analysis on a tree.
06bc494ca11e Initial load
duke
parents:
diff changeset
  1440
     */
6156
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1441
    public void analyzeTree(Env<AttrContext> env, TreeMaker make) {
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1442
        try {
6156
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1443
            attrEnv = env;
e15c221efaac 6970833: Try-with-resource implementation throws an NPE during Flow analysis
mcimadamore
parents: 6151
diff changeset
  1444
            JCTree tree = env.tree;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1445
            this.make = make;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1446
            inits = new Bits();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1447
            uninits = new Bits();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1448
            uninitsTry = new Bits();
06bc494ca11e Initial load
duke
parents:
diff changeset
  1449
            initsWhenTrue = initsWhenFalse =
06bc494ca11e Initial load
duke
parents:
diff changeset
  1450
                uninitsWhenTrue = uninitsWhenFalse = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1451
            if (vars == null)
06bc494ca11e Initial load
duke
parents:
diff changeset
  1452
                vars = new VarSymbol[32];
06bc494ca11e Initial load
duke
parents:
diff changeset
  1453
            else
06bc494ca11e Initial load
duke
parents:
diff changeset
  1454
                for (int i=0; i<vars.length; i++)
06bc494ca11e Initial load
duke
parents:
diff changeset
  1455
                    vars[i] = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1456
            firstadr = 0;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1457
            nextadr = 0;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1458
            pendingExits = new ListBuffer<PendingExit>();
7203
1153590927f7 6993963: Project Coin: Use precise exception analysis for effectively final catch parameters
mcimadamore
parents: 7074
diff changeset
  1459
            preciseRethrowTypes = new HashMap<Symbol, List<Type>>();
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1460
            alive = true;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1461
            this.thrown = this.caught = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1462
            this.classDef = null;
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1463
            unrefdResources = new Scope(env.enclClass.sym);
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1464
            scan(tree);
06bc494ca11e Initial load
duke
parents:
diff changeset
  1465
        } finally {
06bc494ca11e Initial load
duke
parents:
diff changeset
  1466
            // note that recursive invocations of this method fail hard
06bc494ca11e Initial load
duke
parents:
diff changeset
  1467
            inits = uninits = uninitsTry = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1468
            initsWhenTrue = initsWhenFalse =
06bc494ca11e Initial load
duke
parents:
diff changeset
  1469
                uninitsWhenTrue = uninitsWhenFalse = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1470
            if (vars != null) for (int i=0; i<vars.length; i++)
06bc494ca11e Initial load
duke
parents:
diff changeset
  1471
                vars[i] = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1472
            firstadr = 0;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1473
            nextadr = 0;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1474
            pendingExits = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1475
            this.make = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1476
            this.thrown = this.caught = null;
06bc494ca11e Initial load
duke
parents:
diff changeset
  1477
            this.classDef = null;
8626
38b24af530bc 7023233: False positive for -Xlint:try with nested try with resources blocks
mcimadamore
parents: 8430
diff changeset
  1478
            unrefdResources = null;
10
06bc494ca11e Initial load
duke
parents:
diff changeset
  1479
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1480
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
  1481
}