Mercurial > hg > truffle
annotate src/share/vm/ci/ciMethod.hpp @ 9126:bc26f978b0ce
HotSpotResolvedObjectType: implement hasFinalizeSubclass() correctly
don't use the (wrong) cached value, but ask the runtime on each request.
Fixes regression on xml.* benchmarks @ specjvm2008. The problem was:
After the constructor of Object was deoptimized due to an assumption violation,
it was recompiled again after some time. However, on recompilation, the value
of hasFinalizeSubclass for the class was not updated and it was compiled again
with a, now wrong, assumption, which then triggers deoptimization again.
This was repeated until it hit the recompilation limit (defined by
PerMethodRecompilationCutoff), and therefore only executed by the interpreter
from now on, causing the performance regression.
author | Bernhard Urban <bernhard.urban@jku.at> |
---|---|
date | Mon, 15 Apr 2013 19:54:58 +0200 |
parents | 16885e702c88 |
children | aeaca88565e6 |
rev | line source |
---|---|
0 | 1 /* |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6266
diff
changeset
|
2 * Copyright (c) 1999, 2012, Oracle and/or its affiliates. All rights reserved. |
0 | 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
4 * | |
5 * This code is free software; you can redistribute it and/or modify it | |
6 * under the terms of the GNU General Public License version 2 only, as | |
7 * published by the Free Software Foundation. | |
8 * | |
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 * | |
1552
c18cbe5936b8
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
1152
diff
changeset
|
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
c18cbe5936b8
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
1152
diff
changeset
|
20 * or visit www.oracle.com if you need additional information or have any |
c18cbe5936b8
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
1152
diff
changeset
|
21 * questions. |
0 | 22 * |
23 */ | |
24 | |
1972 | 25 #ifndef SHARE_VM_CI_CIMETHOD_HPP |
26 #define SHARE_VM_CI_CIMETHOD_HPP | |
27 | |
28 #include "ci/ciFlags.hpp" | |
29 #include "ci/ciInstanceKlass.hpp" | |
30 #include "ci/ciObject.hpp" | |
31 #include "ci/ciSignature.hpp" | |
32 #include "compiler/methodLiveness.hpp" | |
33 #include "prims/methodHandles.hpp" | |
34 #include "utilities/bitMap.hpp" | |
35 | |
0 | 36 class ciMethodBlocks; |
37 class MethodLiveness; | |
38 class BitMap; | |
39 class Arena; | |
40 class BCEscapeAnalyzer; | |
41 | |
42 | |
43 // ciMethod | |
44 // | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6266
diff
changeset
|
45 // This class represents a Method* in the HotSpot virtual |
0 | 46 // machine. |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6266
diff
changeset
|
47 class ciMethod : public ciMetadata { |
0 | 48 friend class CompileBroker; |
49 CI_PACKAGE_ACCESS | |
50 friend class ciEnv; | |
51 friend class ciExceptionHandlerStream; | |
1138
dd57230ba8fe
6893268: additional dynamic language related optimizations in C2
twisti
parents:
1137
diff
changeset
|
52 friend class ciBytecodeStream; |
dd57230ba8fe
6893268: additional dynamic language related optimizations in C2
twisti
parents:
1137
diff
changeset
|
53 friend class ciMethodHandle; |
6972
bd7a7ce2e264
6830717: replay of compilations would help with debugging
minqi
parents:
6725
diff
changeset
|
54 friend class ciReplay; |
0 | 55 |
56 private: | |
57 // General method information. | |
58 ciFlags _flags; | |
59 ciSymbol* _name; | |
60 ciInstanceKlass* _holder; | |
61 ciSignature* _signature; | |
62 ciMethodData* _method_data; | |
63 ciMethodBlocks* _method_blocks; | |
64 | |
65 // Code attributes. | |
66 int _code_size; | |
67 int _max_stack; | |
68 int _max_locals; | |
69 vmIntrinsics::ID _intrinsic_id; | |
70 int _handler_count; | |
71 int _interpreter_invocation_count; | |
72 int _interpreter_throwout_count; | |
6972
bd7a7ce2e264
6830717: replay of compilations would help with debugging
minqi
parents:
6725
diff
changeset
|
73 int _instructions_size; |
0 | 74 |
75 bool _uses_monitors; | |
76 bool _balanced_monitors; | |
1783 | 77 bool _is_c1_compilable; |
78 bool _is_c2_compilable; | |
0 | 79 bool _can_be_statically_bound; |
80 | |
81 // Lazy fields, filled in on demand | |
82 address _code; | |
83 ciExceptionHandler** _exception_handlers; | |
84 | |
85 // Optional liveness analyzer. | |
86 MethodLiveness* _liveness; | |
1692 | 87 #if defined(COMPILER2) || defined(SHARK) |
1648
8099e71601df
6968368: SIGSEGV in the BCEscapeAnalyzer::copy_dependencies
kvn
parents:
1552
diff
changeset
|
88 ciTypeFlow* _flow; |
8099e71601df
6968368: SIGSEGV in the BCEscapeAnalyzer::copy_dependencies
kvn
parents:
1552
diff
changeset
|
89 BCEscapeAnalyzer* _bcea; |
0 | 90 #endif |
91 | |
92 ciMethod(methodHandle h_m); | |
4001
5eb9169b1a14
7092712: JSR 292: unloaded invokedynamic call sites can lead to a crash with signature types not on BCP
twisti
parents:
3897
diff
changeset
|
93 ciMethod(ciInstanceKlass* holder, ciSymbol* name, ciSymbol* signature, ciInstanceKlass* accessor); |
0 | 94 |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6266
diff
changeset
|
95 Method* get_Method() const { |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6266
diff
changeset
|
96 Method* m = (Method*)_metadata; |
0 | 97 assert(m != NULL, "illegal use of unloaded method"); |
98 return m; | |
99 } | |
100 | |
101 oop loader() const { return _holder->loader(); } | |
102 | |
103 const char* type_string() { return "ciMethod"; } | |
104 | |
105 void print_impl(outputStream* st); | |
106 | |
107 void load_code(); | |
108 | |
109 void check_is_loaded() const { assert(is_loaded(), "not loaded"); } | |
110 | |
2007
5ddfcf4b079e
7003554: (tiered) assert(is_null_object() || handle() != NULL) failed: cannot embed null pointer
iveresov
parents:
1972
diff
changeset
|
111 bool ensure_method_data(methodHandle h_m); |
0 | 112 |
113 void code_at_put(int bci, Bytecodes::Code code) { | |
114 Bytecodes::check(code); | |
115 assert(0 <= bci && bci < code_size(), "valid bci"); | |
116 address bcp = _code + bci; | |
117 *bcp = code; | |
118 } | |
119 | |
120 public: | |
121 // Basic method information. | |
122 ciFlags flags() const { check_is_loaded(); return _flags; } | |
123 ciSymbol* name() const { return _name; } | |
124 ciInstanceKlass* holder() const { return _holder; } | |
125 ciMethodData* method_data(); | |
2007
5ddfcf4b079e
7003554: (tiered) assert(is_null_object() || handle() != NULL) failed: cannot embed null pointer
iveresov
parents:
1972
diff
changeset
|
126 ciMethodData* method_data_or_null(); |
0 | 127 |
128 // Signature information. | |
129 ciSignature* signature() const { return _signature; } | |
130 ciType* return_type() const { return _signature->return_type(); } | |
131 int arg_size_no_receiver() const { return _signature->size(); } | |
3280
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
132 // Can only be used on loaded ciMethods |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
133 int arg_size() const { |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
134 check_is_loaded(); |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
135 return _signature->size() + (_flags.is_static() ? 0 : 1); |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
136 } |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
137 // Report the number of elements on stack when invoking this method. |
6266
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
138 // This is different than the regular arg_size because invokedynamic |
3280
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
139 // has an implicit receiver. |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
140 int invoke_arg_size(Bytecodes::Code code) const { |
6266
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
141 if (is_loaded()) { |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
142 return arg_size(); |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
143 } else { |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
144 int arg_size = _signature->size(); |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
145 // Add a receiver argument, maybe: |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
146 if (code != Bytecodes::_invokestatic && |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
147 code != Bytecodes::_invokedynamic) { |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
148 arg_size++; |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
149 } |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
150 return arg_size; |
3280
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
151 } |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
152 } |
548597e74aa4
7030715: JSR 292 JRuby test/test_super_call_site_caching.rb asserts with +DoEscapeAnalysis
never
parents:
3274
diff
changeset
|
153 |
0 | 154 |
155 // Method code and related information. | |
156 address code() { if (_code == NULL) load_code(); return _code; } | |
157 int code_size() const { check_is_loaded(); return _code_size; } | |
158 int max_stack() const { check_is_loaded(); return _max_stack; } | |
159 int max_locals() const { check_is_loaded(); return _max_locals; } | |
160 vmIntrinsics::ID intrinsic_id() const { check_is_loaded(); return _intrinsic_id; } | |
161 bool has_exception_handlers() const { check_is_loaded(); return _handler_count > 0; } | |
162 int exception_table_length() const { check_is_loaded(); return _handler_count; } | |
163 int interpreter_invocation_count() const { check_is_loaded(); return _interpreter_invocation_count; } | |
164 int interpreter_throwout_count() const { check_is_loaded(); return _interpreter_throwout_count; } | |
165 | |
3897
de847cac9235
7078382: JSR 292: don't count method handle adapters against inlining budgets
twisti
parents:
3791
diff
changeset
|
166 // Code size for inlining decisions. |
de847cac9235
7078382: JSR 292: don't count method handle adapters against inlining budgets
twisti
parents:
3791
diff
changeset
|
167 int code_size_for_inlining(); |
de847cac9235
7078382: JSR 292: don't count method handle adapters against inlining budgets
twisti
parents:
3791
diff
changeset
|
168 |
8866
16885e702c88
7198429: need checked categorization of caller-sensitive methods in the JDK
twisti
parents:
7998
diff
changeset
|
169 bool caller_sensitive() { return get_Method()->caller_sensitive(); } |
16885e702c88
7198429: need checked categorization of caller-sensitive methods in the JDK
twisti
parents:
7998
diff
changeset
|
170 bool force_inline() { return get_Method()->force_inline(); } |
16885e702c88
7198429: need checked categorization of caller-sensitive methods in the JDK
twisti
parents:
7998
diff
changeset
|
171 bool dont_inline() { return get_Method()->dont_inline(); } |
6222
6d8f36bcef55
6711908: JVM needs direct access to some annotations
jrose
parents:
4117
diff
changeset
|
172 |
1783 | 173 int comp_level(); |
3791
2c359f27615c
7057120: Tiered: Allow C1 to inline methods with loops
iveresov
parents:
3280
diff
changeset
|
174 int highest_osr_comp_level(); |
1783 | 175 |
0 | 176 Bytecodes::Code java_code_at_bci(int bci) { |
177 address bcp = code() + bci; | |
2142 | 178 return Bytecodes::java_code_at(NULL, bcp); |
0 | 179 } |
180 BCEscapeAnalyzer *get_bcea(); | |
181 ciMethodBlocks *get_method_blocks(); | |
182 | |
183 bool has_linenumber_table() const; // length unknown until decompression | |
184 u_char* compressed_linenumber_table() const; // not preserved by gc | |
185 | |
186 int line_number_from_bci(int bci) const; | |
187 | |
188 // Runtime information. | |
189 int vtable_index(); | |
1692 | 190 #ifdef SHARK |
191 int itable_index(); | |
192 #endif // SHARK | |
0 | 193 address native_entry(); |
194 address interpreter_entry(); | |
195 | |
196 // Analysis and profiling. | |
197 // | |
198 // Usage note: liveness_at_bci and init_vars should be wrapped in ResourceMarks. | |
199 bool uses_monitors() const { return _uses_monitors; } // this one should go away, it has a misleading name | |
200 bool has_monitor_bytecodes() const { return _uses_monitors; } | |
201 bool has_balanced_monitors(); | |
202 | |
991 | 203 // Returns a bitmap indicating which locals are required to be |
204 // maintained as live for deopt. raw_liveness_at_bci is always the | |
205 // direct output of the liveness computation while liveness_at_bci | |
206 // may mark all locals as live to improve support for debugging Java | |
207 // code by maintaining the state of as many locals as possible. | |
208 MethodLivenessResult raw_liveness_at_bci(int bci); | |
0 | 209 MethodLivenessResult liveness_at_bci(int bci); |
210 | |
211 // Get the interpreters viewpoint on oop liveness. MethodLiveness is | |
212 // conservative in the sense that it may consider locals to be live which | |
213 // cannot be live, like in the case where a local could contain an oop or | |
214 // a primitive along different paths. In that case the local must be | |
215 // dead when those paths merge. Since the interpreter's viewpoint is | |
216 // used when gc'ing an interpreter frame we need to use its viewpoint | |
217 // during OSR when loading the locals. | |
218 | |
219 BitMap live_local_oops_at_bci(int bci); | |
220 | |
221 #ifdef COMPILER1 | |
222 const BitMap bci_block_start(); | |
223 #endif | |
224 | |
225 ciTypeFlow* get_flow_analysis(); | |
226 ciTypeFlow* get_osr_flow_analysis(int osr_bci); // alternate entry point | |
227 ciCallProfile call_profile_at_bci(int bci); | |
228 int interpreter_call_site_count(int bci); | |
229 | |
7194
beebba0acc11
7172640: C2: instrinsic implementations in LibraryCallKit should use argument() instead of pop()
twisti
parents:
6973
diff
changeset
|
230 ciField* get_field_at_bci( int bci, bool &will_link); |
beebba0acc11
7172640: C2: instrinsic implementations in LibraryCallKit should use argument() instead of pop()
twisti
parents:
6973
diff
changeset
|
231 ciMethod* get_method_at_bci(int bci, bool &will_link, ciSignature* *declared_signature); |
beebba0acc11
7172640: C2: instrinsic implementations in LibraryCallKit should use argument() instead of pop()
twisti
parents:
6973
diff
changeset
|
232 |
0 | 233 // Given a certain calling environment, find the monomorphic target |
234 // for the call. Return NULL if the call is not monomorphic in | |
235 // its calling environment. | |
236 ciMethod* find_monomorphic_target(ciInstanceKlass* caller, | |
237 ciInstanceKlass* callee_holder, | |
238 ciInstanceKlass* actual_receiver); | |
239 | |
240 // Given a known receiver klass, find the target for the call. | |
241 // Return NULL if the call has no target or is abstract. | |
242 ciMethod* resolve_invoke(ciKlass* caller, ciKlass* exact_receiver); | |
243 | |
244 // Find the proper vtable index to invoke this method. | |
245 int resolve_vtable_index(ciKlass* caller, ciKlass* receiver); | |
246 | |
247 // Compilation directives | |
248 bool should_exclude(); | |
249 bool should_inline(); | |
250 bool should_not_inline(); | |
251 bool should_print_assembly(); | |
252 bool break_at_execute(); | |
253 bool has_option(const char *option); | |
254 bool can_be_compiled(); | |
255 bool can_be_osr_compiled(int entry_bci); | |
7998 | 256 void set_not_compilable(const char* reason = NULL); |
0 | 257 bool has_compiled_code(); |
258 void log_nmethod_identity(xmlStream* log); | |
259 bool is_not_reached(int bci); | |
260 bool was_executed_more_than(int times); | |
261 bool has_unloaded_classes_in_signature(); | |
262 bool is_klass_loaded(int refinfo_index, bool must_be_resolved) const; | |
263 bool check_call(int refinfo_index, bool is_static) const; | |
2007
5ddfcf4b079e
7003554: (tiered) assert(is_null_object() || handle() != NULL) failed: cannot embed null pointer
iveresov
parents:
1972
diff
changeset
|
264 bool ensure_method_data(); // make sure it exists in the VM also |
6972
bd7a7ce2e264
6830717: replay of compilations would help with debugging
minqi
parents:
6725
diff
changeset
|
265 int instructions_size(); |
0 | 266 int scale_count(int count, float prof_factor = 1.); // make MDO count commensurate with IIC |
1152
cd37471eaecc
6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents:
1138
diff
changeset
|
267 |
8866
16885e702c88
7198429: need checked categorization of caller-sensitive methods in the JDK
twisti
parents:
7998
diff
changeset
|
268 // Stack walking support |
16885e702c88
7198429: need checked categorization of caller-sensitive methods in the JDK
twisti
parents:
7998
diff
changeset
|
269 bool is_ignored_by_security_stack_walk() const; |
16885e702c88
7198429: need checked categorization of caller-sensitive methods in the JDK
twisti
parents:
7998
diff
changeset
|
270 |
1152
cd37471eaecc
6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents:
1138
diff
changeset
|
271 // JSR 292 support |
6266
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
272 bool is_method_handle_intrinsic() const; |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
273 bool is_compiled_lambda_form() const; |
1d7922586cf6
7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents:
6222
diff
changeset
|
274 bool has_member_arg() const; |
0 | 275 |
276 // What kind of ciObject is this? | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6266
diff
changeset
|
277 bool is_method() const { return true; } |
0 | 278 |
279 // Java access flags | |
280 bool is_public () const { return flags().is_public(); } | |
281 bool is_private () const { return flags().is_private(); } | |
282 bool is_protected () const { return flags().is_protected(); } | |
283 bool is_static () const { return flags().is_static(); } | |
284 bool is_final () const { return flags().is_final(); } | |
285 bool is_synchronized() const { return flags().is_synchronized(); } | |
286 bool is_native () const { return flags().is_native(); } | |
287 bool is_interface () const { return flags().is_interface(); } | |
288 bool is_abstract () const { return flags().is_abstract(); } | |
289 bool is_strict () const { return flags().is_strict(); } | |
290 | |
291 // Other flags | |
292 bool is_empty_method() const; | |
293 bool is_vanilla_constructor() const; | |
294 bool is_final_method() const { return is_final() || holder()->is_final(); } | |
295 bool has_loops () const; | |
296 bool has_jsrs () const; | |
297 bool is_accessor () const; | |
298 bool is_initializer () const; | |
299 bool can_be_statically_bound() const { return _can_be_statically_bound; } | |
6972
bd7a7ce2e264
6830717: replay of compilations would help with debugging
minqi
parents:
6725
diff
changeset
|
300 void dump_replay_data(outputStream* st); |
0 | 301 |
302 // Print the bytecodes of this method. | |
303 void print_codes_on(outputStream* st); | |
304 void print_codes() { | |
305 print_codes_on(tty); | |
306 } | |
307 void print_codes_on(int from, int to, outputStream* st); | |
308 | |
309 // Print the name of this method in various incarnations. | |
310 void print_name(outputStream* st = tty); | |
311 void print_short_name(outputStream* st = tty); | |
312 }; | |
1972 | 313 |
314 #endif // SHARE_VM_CI_CIMETHOD_HPP |