annotate src/share/vm/interpreter/linkResolver.hpp @ 1716:be3f9c242c9d

6948538: CMS: BOT walkers can fall into object allocation and initialization cracks Summary: GC workers now recognize an intermediate transient state of blocks which are allocated but have not yet completed initialization. blk_start() calls do not attempt to determine the size of a block in the transient state, rather waiting for the block to become initialized so that it is safe to query its size. Audited and ensured the order of initialization of object fields (klass, free bit and size) to respect block state transition protocol. Also included some new assertion checking code enabled in debug mode. Reviewed-by: chrisphi, johnc, poonam
author ysr
date Mon, 16 Aug 2010 15:58:42 -0700
parents 083fde3b838e
children f95d63e2154a
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1508
diff changeset
2 * Copyright (c) 1997, 2009, Oracle and/or its affiliates. All rights reserved.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1508
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1508
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: 1508
diff changeset
21 * questions.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24
a61af66fc99e Initial load
duke
parents:
diff changeset
25 // All the necessary definitions for run-time link resolution.
a61af66fc99e Initial load
duke
parents:
diff changeset
26
a61af66fc99e Initial load
duke
parents:
diff changeset
27 // LinkInfo & its subclasses provide all the information gathered
a61af66fc99e Initial load
duke
parents:
diff changeset
28 // for a particular link after resolving it. A link is any reference
a61af66fc99e Initial load
duke
parents:
diff changeset
29 // made from within the bytecodes of a method to an object outside of
a61af66fc99e Initial load
duke
parents:
diff changeset
30 // that method. If the info is invalid, the link has not been resolved
a61af66fc99e Initial load
duke
parents:
diff changeset
31 // successfully.
a61af66fc99e Initial load
duke
parents:
diff changeset
32
a61af66fc99e Initial load
duke
parents:
diff changeset
33 class LinkInfo VALUE_OBJ_CLASS_SPEC {
a61af66fc99e Initial load
duke
parents:
diff changeset
34 };
a61af66fc99e Initial load
duke
parents:
diff changeset
35
a61af66fc99e Initial load
duke
parents:
diff changeset
36
a61af66fc99e Initial load
duke
parents:
diff changeset
37 // Link information for getfield/putfield & getstatic/putstatic bytecodes.
a61af66fc99e Initial load
duke
parents:
diff changeset
38
a61af66fc99e Initial load
duke
parents:
diff changeset
39 class FieldAccessInfo: public LinkInfo {
a61af66fc99e Initial load
duke
parents:
diff changeset
40 protected:
a61af66fc99e Initial load
duke
parents:
diff changeset
41 KlassHandle _klass;
a61af66fc99e Initial load
duke
parents:
diff changeset
42 symbolHandle _name;
a61af66fc99e Initial load
duke
parents:
diff changeset
43 AccessFlags _access_flags;
a61af66fc99e Initial load
duke
parents:
diff changeset
44 int _field_index; // original index in the klass
a61af66fc99e Initial load
duke
parents:
diff changeset
45 int _field_offset;
a61af66fc99e Initial load
duke
parents:
diff changeset
46 BasicType _field_type;
a61af66fc99e Initial load
duke
parents:
diff changeset
47
a61af66fc99e Initial load
duke
parents:
diff changeset
48 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
49 void set(KlassHandle klass, symbolHandle name, int field_index, int field_offset,
a61af66fc99e Initial load
duke
parents:
diff changeset
50 BasicType field_type, AccessFlags access_flags);
a61af66fc99e Initial load
duke
parents:
diff changeset
51 KlassHandle klass() const { return _klass; }
a61af66fc99e Initial load
duke
parents:
diff changeset
52 symbolHandle name() const { return _name; }
a61af66fc99e Initial load
duke
parents:
diff changeset
53 int field_index() const { return _field_index; }
a61af66fc99e Initial load
duke
parents:
diff changeset
54 int field_offset() const { return _field_offset; }
a61af66fc99e Initial load
duke
parents:
diff changeset
55 BasicType field_type() const { return _field_type; }
a61af66fc99e Initial load
duke
parents:
diff changeset
56 AccessFlags access_flags() const { return _access_flags; }
a61af66fc99e Initial load
duke
parents:
diff changeset
57
a61af66fc99e Initial load
duke
parents:
diff changeset
58 // debugging
a61af66fc99e Initial load
duke
parents:
diff changeset
59 void print() PRODUCT_RETURN;
a61af66fc99e Initial load
duke
parents:
diff changeset
60 };
a61af66fc99e Initial load
duke
parents:
diff changeset
61
a61af66fc99e Initial load
duke
parents:
diff changeset
62
a61af66fc99e Initial load
duke
parents:
diff changeset
63 // Link information for all calls.
a61af66fc99e Initial load
duke
parents:
diff changeset
64
a61af66fc99e Initial load
duke
parents:
diff changeset
65 class CallInfo: public LinkInfo {
a61af66fc99e Initial load
duke
parents:
diff changeset
66 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
67 KlassHandle _resolved_klass; // static receiver klass
a61af66fc99e Initial load
duke
parents:
diff changeset
68 KlassHandle _selected_klass; // dynamic receiver class (same as static, or subklass)
a61af66fc99e Initial load
duke
parents:
diff changeset
69 methodHandle _resolved_method; // static target method
a61af66fc99e Initial load
duke
parents:
diff changeset
70 methodHandle _selected_method; // dynamic (actual) target method
a61af66fc99e Initial load
duke
parents:
diff changeset
71 int _vtable_index; // vtable index of selected method
a61af66fc99e Initial load
duke
parents:
diff changeset
72
a61af66fc99e Initial load
duke
parents:
diff changeset
73 void set_static( KlassHandle resolved_klass, methodHandle resolved_method , TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
74 void set_interface(KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method , TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
75 void set_virtual( KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, int vtable_index, TRAPS);
1660
083fde3b838e 6964498: JSR 292 invokedynamic sites need local bootstrap methods
jrose
parents: 1552
diff changeset
76 void set_dynamic( methodHandle resolved_method, TRAPS);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
77 void set_common( KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, int vtable_index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
78
a61af66fc99e Initial load
duke
parents:
diff changeset
79 friend class LinkResolver;
a61af66fc99e Initial load
duke
parents:
diff changeset
80
a61af66fc99e Initial load
duke
parents:
diff changeset
81 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
82 KlassHandle resolved_klass() const { return _resolved_klass; }
a61af66fc99e Initial load
duke
parents:
diff changeset
83 KlassHandle selected_klass() const { return _selected_klass; }
a61af66fc99e Initial load
duke
parents:
diff changeset
84 methodHandle resolved_method() const { return _resolved_method; }
a61af66fc99e Initial load
duke
parents:
diff changeset
85 methodHandle selected_method() const { return _selected_method; }
a61af66fc99e Initial load
duke
parents:
diff changeset
86
a61af66fc99e Initial load
duke
parents:
diff changeset
87 BasicType result_type() const { return selected_method()->result_type(); }
a61af66fc99e Initial load
duke
parents:
diff changeset
88 bool has_vtable_index() const { return _vtable_index >= 0; }
a61af66fc99e Initial load
duke
parents:
diff changeset
89 bool is_statically_bound() const { return _vtable_index == methodOopDesc::nonvirtual_vtable_index; }
a61af66fc99e Initial load
duke
parents:
diff changeset
90 int vtable_index() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
91 // Even for interface calls the vtable index could be non-negative.
a61af66fc99e Initial load
duke
parents:
diff changeset
92 // See CallInfo::set_interface.
a61af66fc99e Initial load
duke
parents:
diff changeset
93 assert(has_vtable_index() || is_statically_bound(), "");
a61af66fc99e Initial load
duke
parents:
diff changeset
94 return _vtable_index;
a61af66fc99e Initial load
duke
parents:
diff changeset
95 }
a61af66fc99e Initial load
duke
parents:
diff changeset
96 };
a61af66fc99e Initial load
duke
parents:
diff changeset
97
a61af66fc99e Initial load
duke
parents:
diff changeset
98
a61af66fc99e Initial load
duke
parents:
diff changeset
99 // The LinkResolver is used to resolve constant-pool references at run-time.
a61af66fc99e Initial load
duke
parents:
diff changeset
100 // It does all necessary link-time checks & throws exceptions if necessary.
a61af66fc99e Initial load
duke
parents:
diff changeset
101
a61af66fc99e Initial load
duke
parents:
diff changeset
102 class LinkResolver: AllStatic {
a61af66fc99e Initial load
duke
parents:
diff changeset
103 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
104 static void lookup_method_in_klasses (methodHandle& result, KlassHandle klass, symbolHandle name, symbolHandle signature, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
105 static void lookup_instance_method_in_klasses (methodHandle& result, KlassHandle klass, symbolHandle name, symbolHandle signature, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
106 static void lookup_method_in_interfaces (methodHandle& result, KlassHandle klass, symbolHandle name, symbolHandle signature, TRAPS);
1508
2ffde6cfe049 6939196: method handle signatures off the boot class path get linkage errors
jrose
parents: 1135
diff changeset
107 static void lookup_implicit_method (methodHandle& result, KlassHandle klass, symbolHandle name, symbolHandle signature,
2ffde6cfe049 6939196: method handle signatures off the boot class path get linkage errors
jrose
parents: 1135
diff changeset
108 KlassHandle current_klass, TRAPS);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
109
a61af66fc99e Initial load
duke
parents:
diff changeset
110 static int vtable_index_of_miranda_method(KlassHandle klass, symbolHandle name, symbolHandle signature, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
111
a61af66fc99e Initial load
duke
parents:
diff changeset
112 static void resolve_klass (KlassHandle& result, constantPoolHandle pool, int index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
113 static void resolve_klass_no_update (KlassHandle& result, constantPoolHandle pool, int index, TRAPS); // no update of constantPool entry
a61af66fc99e Initial load
duke
parents:
diff changeset
114
a61af66fc99e Initial load
duke
parents:
diff changeset
115 static void resolve_pool (KlassHandle& resolved_klass, symbolHandle& method_name, symbolHandle& method_signature, KlassHandle& current_klass, constantPoolHandle pool, int index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
116
a61af66fc99e Initial load
duke
parents:
diff changeset
117 static void resolve_interface_method(methodHandle& resolved_method, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
118 static void resolve_method (methodHandle& resolved_method, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
119
a61af66fc99e Initial load
duke
parents:
diff changeset
120 static void linktime_resolve_static_method (methodHandle& resolved_method, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
121 static void linktime_resolve_special_method (methodHandle& resolved_method, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
122 static void linktime_resolve_virtual_method (methodHandle &resolved_method, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature,KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
123 static void linktime_resolve_interface_method (methodHandle& resolved_method, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
124
a61af66fc99e Initial load
duke
parents:
diff changeset
125 static void runtime_resolve_special_method (CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
126 static void runtime_resolve_virtual_method (CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, Handle recv, KlassHandle recv_klass, bool check_null_and_abstract, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
127 static void runtime_resolve_interface_method (CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, Handle recv, KlassHandle recv_klass, bool check_null_and_abstract, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
128
a61af66fc99e Initial load
duke
parents:
diff changeset
129 static void check_field_accessability (KlassHandle ref_klass, KlassHandle resolved_klass, KlassHandle sel_klass, fieldDescriptor& fd, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
130 static void check_method_accessability (KlassHandle ref_klass, KlassHandle resolved_klass, KlassHandle sel_klass, methodHandle sel_method, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
131
a61af66fc99e Initial load
duke
parents:
diff changeset
132 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
133 // constant pool resolving
a61af66fc99e Initial load
duke
parents:
diff changeset
134 static void check_klass_accessability(KlassHandle ref_klass, KlassHandle sel_klass, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
135
a61af66fc99e Initial load
duke
parents:
diff changeset
136 // static resolving for all calls except interface calls
a61af66fc99e Initial load
duke
parents:
diff changeset
137 static void resolve_method (methodHandle& method_result, KlassHandle& klass_result, constantPoolHandle pool, int index, TRAPS);
1135
e66fd840cb6b 6893081: method handle & invokedynamic code needs additional cleanup (post 6815692, 6858164)
twisti
parents: 726
diff changeset
138 static void resolve_dynamic_method (methodHandle& resolved_method, KlassHandle& resolved_klass, constantPoolHandle pool, int index, TRAPS);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
139 static void resolve_interface_method(methodHandle& method_result, KlassHandle& klass_result, constantPoolHandle pool, int index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
140
a61af66fc99e Initial load
duke
parents:
diff changeset
141 // runtime/static resolving for fields
a61af66fc99e Initial load
duke
parents:
diff changeset
142 static void resolve_field(FieldAccessInfo& result, constantPoolHandle pool, int index, Bytecodes::Code byte, bool check_only, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
143 // takes an extra bool argument "update_pool" to decide whether to update the constantPool during klass resolution.
a61af66fc99e Initial load
duke
parents:
diff changeset
144 static void resolve_field(FieldAccessInfo& result, constantPoolHandle pool, int index, Bytecodes::Code byte, bool check_only, bool update_pool, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
145
a61af66fc99e Initial load
duke
parents:
diff changeset
146 // runtime resolving:
a61af66fc99e Initial load
duke
parents:
diff changeset
147 // resolved_klass = specified class (i.e., static receiver class)
a61af66fc99e Initial load
duke
parents:
diff changeset
148 // current_klass = sending method holder (i.e., class containing the method containing the call being resolved)
a61af66fc99e Initial load
duke
parents:
diff changeset
149 static void resolve_static_call (CallInfo& result, KlassHandle& resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, bool initialize_klass, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
150 static void resolve_special_call (CallInfo& result, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
151 static void resolve_virtual_call (CallInfo& result, Handle recv, KlassHandle recv_klass, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, bool check_null_and_abstract, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
152 static void resolve_interface_call(CallInfo& result, Handle recv, KlassHandle recv_klass, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access, bool check_null_and_abstract, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
153
a61af66fc99e Initial load
duke
parents:
diff changeset
154 // same as above for compile-time resolution; but returns null handle instead of throwing an exception on error
a61af66fc99e Initial load
duke
parents:
diff changeset
155 // also, does not initialize klass (i.e., no side effects)
a61af66fc99e Initial load
duke
parents:
diff changeset
156 static methodHandle resolve_virtual_call_or_null (KlassHandle receiver_klass, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
157 static methodHandle resolve_interface_call_or_null(KlassHandle receiver_klass, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
158 static methodHandle resolve_static_call_or_null (KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
159 static methodHandle resolve_special_call_or_null (KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
160
a61af66fc99e Initial load
duke
parents:
diff changeset
161 // same as above for compile-time resolution; returns vtable_index if current_klass if linked
a61af66fc99e Initial load
duke
parents:
diff changeset
162 static int resolve_virtual_vtable_index (KlassHandle receiver_klass, KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
163
a61af66fc99e Initial load
duke
parents:
diff changeset
164 // static resolving for compiler (does not throw exceptions, returns null handle if unsuccessful)
a61af66fc99e Initial load
duke
parents:
diff changeset
165 static methodHandle linktime_resolve_virtual_method_or_null (KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access);
a61af66fc99e Initial load
duke
parents:
diff changeset
166 static methodHandle linktime_resolve_interface_method_or_null(KlassHandle resolved_klass, symbolHandle method_name, symbolHandle method_signature, KlassHandle current_klass, bool check_access);
a61af66fc99e Initial load
duke
parents:
diff changeset
167
a61af66fc99e Initial load
duke
parents:
diff changeset
168 // runtime resolving from constant pool
a61af66fc99e Initial load
duke
parents:
diff changeset
169 static void resolve_invokestatic (CallInfo& result, constantPoolHandle pool, int index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
170 static void resolve_invokespecial (CallInfo& result, constantPoolHandle pool, int index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
171 static void resolve_invokevirtual (CallInfo& result, Handle recv, constantPoolHandle pool, int index, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
172 static void resolve_invokeinterface(CallInfo& result, Handle recv, constantPoolHandle pool, int index, TRAPS);
726
be93aad57795 6655646: dynamic languages need dynamically linked call sites
jrose
parents: 710
diff changeset
173 static void resolve_invokedynamic (CallInfo& result, constantPoolHandle pool, int index, TRAPS);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
174
a61af66fc99e Initial load
duke
parents:
diff changeset
175 static void resolve_invoke (CallInfo& result, Handle recv, constantPoolHandle pool, int index, Bytecodes::Code byte, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
176 };