src/share/vm/c1/c1_Compilation.hpp

Print this page
rev 4136 : 7153771: array bound check elimination for c1
Summary: when possible optimize out array bound checks, inserting predicates when needed.
Reviewed-by:


   9  * This code is distributed in the hope that it will be useful, but WITHOUT
  10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  12  * version 2 for more details (a copy is included in the LICENSE file that
  13  * accompanied this code).
  14  *
  15  * You should have received a copy of the GNU General Public License version
  16  * 2 along with this work; if not, write to the Free Software Foundation,
  17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  18  *
  19  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  20  * or visit www.oracle.com if you need additional information or have any
  21  * questions.
  22  *
  23  */
  24 
  25 #ifndef SHARE_VM_C1_C1_COMPILATION_HPP
  26 #define SHARE_VM_C1_C1_COMPILATION_HPP
  27 
  28 #include "ci/ciEnv.hpp"

  29 #include "code/exceptionHandlerTable.hpp"
  30 #include "memory/resourceArea.hpp"

  31 
  32 class CompilationResourceObj;
  33 class XHandlers;
  34 class ExceptionInfo;
  35 class DebugInformationRecorder;
  36 class FrameMap;
  37 class IR;
  38 class IRScope;
  39 class Instruction;
  40 class LinearScan;
  41 class OopMap;
  42 class LIR_Emitter;
  43 class LIR_Assembler;
  44 class CodeEmitInfo;
  45 class ciEnv;
  46 class ciMethod;
  47 class ValueStack;
  48 class LIR_OprDesc;
  49 class C1_MacroAssembler;
  50 class CFGPrinter;


  68   ciEnv*             _env;
  69   CompileLog*        _log;
  70   ciMethod*          _method;
  71   int                _osr_bci;
  72   IR*                _hir;
  73   int                _max_spills;
  74   FrameMap*          _frame_map;
  75   C1_MacroAssembler* _masm;
  76   bool               _has_exception_handlers;
  77   bool               _has_fpu_code;
  78   bool               _has_unsafe_access;
  79   bool               _would_profile;
  80   bool               _has_method_handle_invokes;  // True if this method has MethodHandle invokes.
  81   const char*        _bailout_msg;
  82   ExceptionInfoList* _exception_info_list;
  83   ExceptionHandlerTable _exception_handler_table;
  84   ImplicitExceptionTable _implicit_exception_table;
  85   LinearScan*        _allocator;
  86   CodeOffsets        _offsets;
  87   CodeBuffer         _code;

  88 
  89   // compilation helpers
  90   void initialize();
  91   void build_hir();
  92   void emit_lir();
  93 
  94   void emit_code_epilog(LIR_Assembler* assembler);
  95   int  emit_code_body();
  96 
  97   int  compile_java_method();
  98   void install_code(int frame_size);
  99   void compile_method();
 100 
 101   void generate_exception_handler_table();
 102 
 103   ExceptionInfoList* exception_info_list() const { return _exception_info_list; }
 104   ExceptionHandlerTable* exception_handler_table() { return &_exception_handler_table; }
 105 
 106   LinearScan* allocator()                          { return _allocator;      }
 107   void        set_allocator(LinearScan* allocator) { _allocator = allocator; }


 123   }
 124 
 125   // accessors
 126   ciEnv* env() const                             { return _env; }
 127   CompileLog* log() const                        { return _log; }
 128   AbstractCompiler* compiler() const             { return _compiler; }
 129   bool has_exception_handlers() const            { return _has_exception_handlers; }
 130   bool has_fpu_code() const                      { return _has_fpu_code; }
 131   bool has_unsafe_access() const                 { return _has_unsafe_access; }
 132   int max_vector_size() const                    { return 0; }
 133   ciMethod* method() const                       { return _method; }
 134   int osr_bci() const                            { return _osr_bci; }
 135   bool is_osr_compile() const                    { return osr_bci() >= 0; }
 136   IR* hir() const                                { return _hir; }
 137   int max_spills() const                         { return _max_spills; }
 138   FrameMap* frame_map() const                    { return _frame_map; }
 139   CodeBuffer* code()                             { return &_code; }
 140   C1_MacroAssembler* masm() const                { return _masm; }
 141   CodeOffsets* offsets()                         { return &_offsets; }
 142   Arena* arena()                                 { return _arena; }

 143 
 144   // Instruction ids
 145   int get_next_id()                              { return _next_id++; }
 146   int number_of_instructions() const             { return _next_id; }
 147 
 148   // BlockBegin ids
 149   int get_next_block_id()                        { return _next_block_id++; }
 150   int number_of_blocks() const                   { return _next_block_id; }
 151 
 152   // setters
 153   void set_has_exception_handlers(bool f)        { _has_exception_handlers = f; }
 154   void set_has_fpu_code(bool f)                  { _has_fpu_code = f; }
 155   void set_has_unsafe_access(bool f)             { _has_unsafe_access = f; }
 156   void set_would_profile(bool f)                 { _would_profile = f; }

 157   // Add a set of exception handlers covering the given PC offset
 158   void add_exception_handlers_for_pco(int pco, XHandlers* exception_handlers);
 159   // Statistics gathering
 160   void notice_inlined_method(ciMethod* method);
 161 
 162   // JSR 292
 163   bool     has_method_handle_invokes() const { return _has_method_handle_invokes;     }
 164   void set_has_method_handle_invokes(bool z) {        _has_method_handle_invokes = z; }
 165 
 166   DebugInformationRecorder* debug_info_recorder() const; // = _env->debug_info();
 167   Dependencies* dependency_recorder() const; // = _env->dependencies()
 168   ImplicitExceptionTable* implicit_exception_table()     { return &_implicit_exception_table; }
 169 
 170   Instruction* current_instruction() const       { return _current_instruction; }
 171   Instruction* set_current_instruction(Instruction* instr) {
 172     Instruction* previous = _current_instruction;
 173     _current_instruction = instr;
 174     return previous;
 175   }
 176 


 215            env()->comp_level() == CompLevel_limited_profile;
 216   }
 217   bool count_invocations() { return is_profiling(); }
 218   bool count_backedges()   { return is_profiling(); }
 219 
 220   // Helpers for generation of profile information
 221   bool profile_branches() {
 222     return env()->comp_level() == CompLevel_full_profile &&
 223       C1UpdateMethodData && C1ProfileBranches;
 224   }
 225   bool profile_calls() {
 226     return env()->comp_level() == CompLevel_full_profile &&
 227       C1UpdateMethodData && C1ProfileCalls;
 228   }
 229   bool profile_inlined_calls() {
 230     return profile_calls() && C1ProfileInlinedCalls;
 231   }
 232   bool profile_checkcasts() {
 233     return env()->comp_level() == CompLevel_full_profile &&
 234       C1UpdateMethodData && C1ProfileCheckcasts;








 235   }
 236 };
 237 
 238 
 239 // Macro definitions for unified bailout-support
 240 // The methods bailout() and bailed_out() are present in all classes
 241 // that might bailout, but forward all calls to Compilation
 242 #define BAILOUT(msg)               { bailout(msg); return;              }
 243 #define BAILOUT_(msg, res)         { bailout(msg); return res;          }
 244 
 245 #define CHECK_BAILOUT()            { if (bailed_out()) return;          }
 246 #define CHECK_BAILOUT_(res)        { if (bailed_out()) return res;      }
 247 
 248 
 249 class InstructionMark: public StackObj {
 250  private:
 251   Compilation* _compilation;
 252   Instruction*  _previous;
 253 
 254  public:




   9  * This code is distributed in the hope that it will be useful, but WITHOUT
  10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  12  * version 2 for more details (a copy is included in the LICENSE file that
  13  * accompanied this code).
  14  *
  15  * You should have received a copy of the GNU General Public License version
  16  * 2 along with this work; if not, write to the Free Software Foundation,
  17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  18  *
  19  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  20  * or visit www.oracle.com if you need additional information or have any
  21  * questions.
  22  *
  23  */
  24 
  25 #ifndef SHARE_VM_C1_C1_COMPILATION_HPP
  26 #define SHARE_VM_C1_C1_COMPILATION_HPP
  27 
  28 #include "ci/ciEnv.hpp"
  29 #include "ci/ciMethodData.hpp"
  30 #include "code/exceptionHandlerTable.hpp"
  31 #include "memory/resourceArea.hpp"
  32 #include "runtime/deoptimization.hpp"
  33 
  34 class CompilationResourceObj;
  35 class XHandlers;
  36 class ExceptionInfo;
  37 class DebugInformationRecorder;
  38 class FrameMap;
  39 class IR;
  40 class IRScope;
  41 class Instruction;
  42 class LinearScan;
  43 class OopMap;
  44 class LIR_Emitter;
  45 class LIR_Assembler;
  46 class CodeEmitInfo;
  47 class ciEnv;
  48 class ciMethod;
  49 class ValueStack;
  50 class LIR_OprDesc;
  51 class C1_MacroAssembler;
  52 class CFGPrinter;


  70   ciEnv*             _env;
  71   CompileLog*        _log;
  72   ciMethod*          _method;
  73   int                _osr_bci;
  74   IR*                _hir;
  75   int                _max_spills;
  76   FrameMap*          _frame_map;
  77   C1_MacroAssembler* _masm;
  78   bool               _has_exception_handlers;
  79   bool               _has_fpu_code;
  80   bool               _has_unsafe_access;
  81   bool               _would_profile;
  82   bool               _has_method_handle_invokes;  // True if this method has MethodHandle invokes.
  83   const char*        _bailout_msg;
  84   ExceptionInfoList* _exception_info_list;
  85   ExceptionHandlerTable _exception_handler_table;
  86   ImplicitExceptionTable _implicit_exception_table;
  87   LinearScan*        _allocator;
  88   CodeOffsets        _offsets;
  89   CodeBuffer         _code;
  90   bool               _has_access_indexed;
  91 
  92   // compilation helpers
  93   void initialize();
  94   void build_hir();
  95   void emit_lir();
  96 
  97   void emit_code_epilog(LIR_Assembler* assembler);
  98   int  emit_code_body();
  99 
 100   int  compile_java_method();
 101   void install_code(int frame_size);
 102   void compile_method();
 103 
 104   void generate_exception_handler_table();
 105 
 106   ExceptionInfoList* exception_info_list() const { return _exception_info_list; }
 107   ExceptionHandlerTable* exception_handler_table() { return &_exception_handler_table; }
 108 
 109   LinearScan* allocator()                          { return _allocator;      }
 110   void        set_allocator(LinearScan* allocator) { _allocator = allocator; }


 126   }
 127 
 128   // accessors
 129   ciEnv* env() const                             { return _env; }
 130   CompileLog* log() const                        { return _log; }
 131   AbstractCompiler* compiler() const             { return _compiler; }
 132   bool has_exception_handlers() const            { return _has_exception_handlers; }
 133   bool has_fpu_code() const                      { return _has_fpu_code; }
 134   bool has_unsafe_access() const                 { return _has_unsafe_access; }
 135   int max_vector_size() const                    { return 0; }
 136   ciMethod* method() const                       { return _method; }
 137   int osr_bci() const                            { return _osr_bci; }
 138   bool is_osr_compile() const                    { return osr_bci() >= 0; }
 139   IR* hir() const                                { return _hir; }
 140   int max_spills() const                         { return _max_spills; }
 141   FrameMap* frame_map() const                    { return _frame_map; }
 142   CodeBuffer* code()                             { return &_code; }
 143   C1_MacroAssembler* masm() const                { return _masm; }
 144   CodeOffsets* offsets()                         { return &_offsets; }
 145   Arena* arena()                                 { return _arena; }
 146   bool has_access_indexed()                      { return _has_access_indexed; }
 147 
 148   // Instruction ids
 149   int get_next_id()                              { return _next_id++; }
 150   int number_of_instructions() const             { return _next_id; }
 151 
 152   // BlockBegin ids
 153   int get_next_block_id()                        { return _next_block_id++; }
 154   int number_of_blocks() const                   { return _next_block_id; }
 155 
 156   // setters
 157   void set_has_exception_handlers(bool f)        { _has_exception_handlers = f; }
 158   void set_has_fpu_code(bool f)                  { _has_fpu_code = f; }
 159   void set_has_unsafe_access(bool f)             { _has_unsafe_access = f; }
 160   void set_would_profile(bool f)                 { _would_profile = f; }
 161   void set_has_access_indexed(bool f)            { _has_access_indexed = f; }
 162   // Add a set of exception handlers covering the given PC offset
 163   void add_exception_handlers_for_pco(int pco, XHandlers* exception_handlers);
 164   // Statistics gathering
 165   void notice_inlined_method(ciMethod* method);
 166 
 167   // JSR 292
 168   bool     has_method_handle_invokes() const { return _has_method_handle_invokes;     }
 169   void set_has_method_handle_invokes(bool z) {        _has_method_handle_invokes = z; }
 170 
 171   DebugInformationRecorder* debug_info_recorder() const; // = _env->debug_info();
 172   Dependencies* dependency_recorder() const; // = _env->dependencies()
 173   ImplicitExceptionTable* implicit_exception_table()     { return &_implicit_exception_table; }
 174 
 175   Instruction* current_instruction() const       { return _current_instruction; }
 176   Instruction* set_current_instruction(Instruction* instr) {
 177     Instruction* previous = _current_instruction;
 178     _current_instruction = instr;
 179     return previous;
 180   }
 181 


 220            env()->comp_level() == CompLevel_limited_profile;
 221   }
 222   bool count_invocations() { return is_profiling(); }
 223   bool count_backedges()   { return is_profiling(); }
 224 
 225   // Helpers for generation of profile information
 226   bool profile_branches() {
 227     return env()->comp_level() == CompLevel_full_profile &&
 228       C1UpdateMethodData && C1ProfileBranches;
 229   }
 230   bool profile_calls() {
 231     return env()->comp_level() == CompLevel_full_profile &&
 232       C1UpdateMethodData && C1ProfileCalls;
 233   }
 234   bool profile_inlined_calls() {
 235     return profile_calls() && C1ProfileInlinedCalls;
 236   }
 237   bool profile_checkcasts() {
 238     return env()->comp_level() == CompLevel_full_profile &&
 239       C1UpdateMethodData && C1ProfileCheckcasts;
 240   }
 241 
 242   // will compilation make optimistic assumptions that might lead to
 243   // deoptimization and that the runtime will account for?
 244   bool is_optimistic() const                             {
 245     return !TieredCompilation && 
 246       (RangeCheckElimination || UseLoopInvariantCodeMotion) && 
 247       method()->method_data()->trap_count(Deoptimization::Reason_none) == 0;
 248   }
 249 };
 250 
 251 
 252 // Macro definitions for unified bailout-support
 253 // The methods bailout() and bailed_out() are present in all classes
 254 // that might bailout, but forward all calls to Compilation
 255 #define BAILOUT(msg)               { bailout(msg); return;              }
 256 #define BAILOUT_(msg, res)         { bailout(msg); return res;          }
 257 
 258 #define CHECK_BAILOUT()            { if (bailed_out()) return;          }
 259 #define CHECK_BAILOUT_(res)        { if (bailed_out()) return res;      }
 260 
 261 
 262 class InstructionMark: public StackObj {
 263  private:
 264   Compilation* _compilation;
 265   Instruction*  _previous;
 266 
 267  public: