annotate src/share/vm/oops/klassVtable.hpp @ 452:00b023ae2d78

6722113: CMS: Incorrect overflow handling during precleaning of Reference lists Summary: When we encounter marking stack overflow during precleaning of Reference lists, we were using the overflow list mechanism, which can cause problems on account of mutating the mark word of the header because of conflicts with mutator accesses and updates of that field. Instead we should use the usual mechanism for overflow handling in concurrent phases, namely dirtying of the card on which the overflowed object lies. Since precleaning effectively does a form of discovered list processing, albeit with discovery enabled, we needed to adjust some code to be correct in the face of interleaved processing and discovery. Reviewed-by: apetrusenko, jcoomes
author ysr
date Thu, 20 Nov 2008 12:27:41 -0800
parents f8236e79048a
children 4aaa9f5e02a8 c89f86385056
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
a61af66fc99e Initial load
duke
parents:
diff changeset
2 * Copyright 1997-2006 Sun Microsystems, Inc. All Rights Reserved.
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 *
a61af66fc99e Initial load
duke
parents:
diff changeset
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
a61af66fc99e Initial load
duke
parents:
diff changeset
20 * CA 95054 USA or visit www.sun.com if you need additional information or
a61af66fc99e Initial load
duke
parents:
diff changeset
21 * have any questions.
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 // A klassVtable abstracts the variable-length vtable that is embedded in instanceKlass
a61af66fc99e Initial load
duke
parents:
diff changeset
26 // and arrayKlass. klassVtable objects are used just as convenient transient accessors to the vtable,
a61af66fc99e Initial load
duke
parents:
diff changeset
27 // not to actually hold the vtable data.
a61af66fc99e Initial load
duke
parents:
diff changeset
28 // Note: the klassVtable should not be accessed before the class has been verified
a61af66fc99e Initial load
duke
parents:
diff changeset
29 // (until that point, the vtable is uninitialized).
a61af66fc99e Initial load
duke
parents:
diff changeset
30
a61af66fc99e Initial load
duke
parents:
diff changeset
31 // Currently a klassVtable contains a direct reference to the vtable data, and is therefore
a61af66fc99e Initial load
duke
parents:
diff changeset
32 // not preserved across GCs.
a61af66fc99e Initial load
duke
parents:
diff changeset
33
a61af66fc99e Initial load
duke
parents:
diff changeset
34 class vtableEntry;
a61af66fc99e Initial load
duke
parents:
diff changeset
35
a61af66fc99e Initial load
duke
parents:
diff changeset
36 class klassVtable : public ResourceObj {
a61af66fc99e Initial load
duke
parents:
diff changeset
37 KlassHandle _klass; // my klass
a61af66fc99e Initial load
duke
parents:
diff changeset
38 int _tableOffset; // offset of start of vtable data within klass
a61af66fc99e Initial load
duke
parents:
diff changeset
39 int _length; // length of vtable (number of entries)
a61af66fc99e Initial load
duke
parents:
diff changeset
40 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
41 int _verify_count; // to make verify faster
a61af66fc99e Initial load
duke
parents:
diff changeset
42 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
43
a61af66fc99e Initial load
duke
parents:
diff changeset
44 // Ordering important, so greater_than (>) can be used as an merge operator.
a61af66fc99e Initial load
duke
parents:
diff changeset
45 enum AccessType {
a61af66fc99e Initial load
duke
parents:
diff changeset
46 acc_private = 0,
a61af66fc99e Initial load
duke
parents:
diff changeset
47 acc_package_private = 1,
a61af66fc99e Initial load
duke
parents:
diff changeset
48 acc_publicprotected = 2
a61af66fc99e Initial load
duke
parents:
diff changeset
49 };
a61af66fc99e Initial load
duke
parents:
diff changeset
50
a61af66fc99e Initial load
duke
parents:
diff changeset
51 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
52 klassVtable(KlassHandle h_klass, void* base, int length) : _klass(h_klass) {
a61af66fc99e Initial load
duke
parents:
diff changeset
53 _tableOffset = (address)base - (address)h_klass(); _length = length;
a61af66fc99e Initial load
duke
parents:
diff changeset
54 }
a61af66fc99e Initial load
duke
parents:
diff changeset
55
a61af66fc99e Initial load
duke
parents:
diff changeset
56 // accessors
a61af66fc99e Initial load
duke
parents:
diff changeset
57 vtableEntry* table() const { return (vtableEntry*)(address(_klass()) + _tableOffset); }
a61af66fc99e Initial load
duke
parents:
diff changeset
58 KlassHandle klass() const { return _klass; }
a61af66fc99e Initial load
duke
parents:
diff changeset
59 int length() const { return _length; }
a61af66fc99e Initial load
duke
parents:
diff changeset
60 inline methodOop method_at(int i) const;
a61af66fc99e Initial load
duke
parents:
diff changeset
61 inline methodOop unchecked_method_at(int i) const;
a61af66fc99e Initial load
duke
parents:
diff changeset
62 inline oop* adr_method_at(int i) const;
a61af66fc99e Initial load
duke
parents:
diff changeset
63
a61af66fc99e Initial load
duke
parents:
diff changeset
64 // searching; all methods return -1 if not found
a61af66fc99e Initial load
duke
parents:
diff changeset
65 int index_of(methodOop m) const { return index_of(m, _length); }
a61af66fc99e Initial load
duke
parents:
diff changeset
66 int index_of_miranda(symbolOop name, symbolOop signature);
a61af66fc99e Initial load
duke
parents:
diff changeset
67
a61af66fc99e Initial load
duke
parents:
diff changeset
68 void initialize_vtable(bool checkconstraints, TRAPS); // initialize vtable of a new klass
a61af66fc99e Initial load
duke
parents:
diff changeset
69
a61af66fc99e Initial load
duke
parents:
diff changeset
70 // conputes vtable length (in words) and the number of miranda methods
a61af66fc99e Initial load
duke
parents:
diff changeset
71 static void compute_vtable_size_and_num_mirandas(int &vtable_length, int &num_miranda_methods,
a61af66fc99e Initial load
duke
parents:
diff changeset
72 klassOop super, objArrayOop methods,
a61af66fc99e Initial load
duke
parents:
diff changeset
73 AccessFlags class_flags, oop classloader,
a61af66fc99e Initial load
duke
parents:
diff changeset
74 symbolOop classname, objArrayOop local_interfaces);
a61af66fc99e Initial load
duke
parents:
diff changeset
75
a61af66fc99e Initial load
duke
parents:
diff changeset
76 // RedefineClasses() API support:
a61af66fc99e Initial load
duke
parents:
diff changeset
77 // If any entry of this vtable points to any of old_methods,
a61af66fc99e Initial load
duke
parents:
diff changeset
78 // replace it with the corresponding new_method.
a61af66fc99e Initial load
duke
parents:
diff changeset
79 // trace_name_printed is set to true if the current call has
a61af66fc99e Initial load
duke
parents:
diff changeset
80 // printed the klass name so that other routines in the adjust_*
a61af66fc99e Initial load
duke
parents:
diff changeset
81 // group don't print the klass name.
a61af66fc99e Initial load
duke
parents:
diff changeset
82 void adjust_method_entries(methodOop* old_methods, methodOop* new_methods,
a61af66fc99e Initial load
duke
parents:
diff changeset
83 int methods_length, bool * trace_name_printed);
a61af66fc99e Initial load
duke
parents:
diff changeset
84
a61af66fc99e Initial load
duke
parents:
diff changeset
85 // Garbage collection
a61af66fc99e Initial load
duke
parents:
diff changeset
86 void oop_follow_contents();
a61af66fc99e Initial load
duke
parents:
diff changeset
87 void oop_adjust_pointers();
a61af66fc99e Initial load
duke
parents:
diff changeset
88
a61af66fc99e Initial load
duke
parents:
diff changeset
89 #ifndef SERIALGC
a61af66fc99e Initial load
duke
parents:
diff changeset
90 // Parallel Old
a61af66fc99e Initial load
duke
parents:
diff changeset
91 void oop_follow_contents(ParCompactionManager* cm);
a61af66fc99e Initial load
duke
parents:
diff changeset
92 void oop_update_pointers(ParCompactionManager* cm);
a61af66fc99e Initial load
duke
parents:
diff changeset
93 void oop_update_pointers(ParCompactionManager* cm,
a61af66fc99e Initial load
duke
parents:
diff changeset
94 HeapWord* beg_addr, HeapWord* end_addr);
a61af66fc99e Initial load
duke
parents:
diff changeset
95 #endif // SERIALGC
a61af66fc99e Initial load
duke
parents:
diff changeset
96
a61af66fc99e Initial load
duke
parents:
diff changeset
97 // Iterators
a61af66fc99e Initial load
duke
parents:
diff changeset
98 void oop_oop_iterate(OopClosure* blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
99 void oop_oop_iterate_m(OopClosure* blk, MemRegion mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
100
a61af66fc99e Initial load
duke
parents:
diff changeset
101 // Debugging code
a61af66fc99e Initial load
duke
parents:
diff changeset
102 void print() PRODUCT_RETURN;
a61af66fc99e Initial load
duke
parents:
diff changeset
103 void verify(outputStream* st, bool force = false);
a61af66fc99e Initial load
duke
parents:
diff changeset
104 static void print_statistics() PRODUCT_RETURN;
a61af66fc99e Initial load
duke
parents:
diff changeset
105
a61af66fc99e Initial load
duke
parents:
diff changeset
106 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
107 bool check_no_old_entries();
a61af66fc99e Initial load
duke
parents:
diff changeset
108 void dump_vtable();
a61af66fc99e Initial load
duke
parents:
diff changeset
109 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
110
a61af66fc99e Initial load
duke
parents:
diff changeset
111 protected:
a61af66fc99e Initial load
duke
parents:
diff changeset
112 friend class vtableEntry;
a61af66fc99e Initial load
duke
parents:
diff changeset
113 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
114 void copy_vtable_to(vtableEntry* start);
a61af66fc99e Initial load
duke
parents:
diff changeset
115 int initialize_from_super(KlassHandle super);
a61af66fc99e Initial load
duke
parents:
diff changeset
116 int index_of(methodOop m, int len) const; // same as index_of, but search only up to len
a61af66fc99e Initial load
duke
parents:
diff changeset
117 void put_method_at(methodOop m, int index);
a61af66fc99e Initial load
duke
parents:
diff changeset
118 static bool needs_new_vtable_entry(methodOop m, klassOop super, oop classloader, symbolOop classname, AccessFlags access_flags);
a61af66fc99e Initial load
duke
parents:
diff changeset
119 AccessType vtable_accessibility_at(int i);
a61af66fc99e Initial load
duke
parents:
diff changeset
120
a61af66fc99e Initial load
duke
parents:
diff changeset
121 bool update_super_vtable(instanceKlass* klass, methodHandle target_method, int super_vtable_len, bool checkconstraints, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
122
a61af66fc99e Initial load
duke
parents:
diff changeset
123 // support for miranda methods
a61af66fc99e Initial load
duke
parents:
diff changeset
124 bool is_miranda_entry_at(int i);
a61af66fc99e Initial load
duke
parents:
diff changeset
125 void fill_in_mirandas(int& initialized);
a61af66fc99e Initial load
duke
parents:
diff changeset
126 static bool is_miranda(methodOop m, objArrayOop class_methods, klassOop super);
a61af66fc99e Initial load
duke
parents:
diff changeset
127 static void add_new_mirandas_to_list(GrowableArray<methodOop>* list_of_current_mirandas, objArrayOop current_interface_methods, objArrayOop class_methods, klassOop super);
a61af66fc99e Initial load
duke
parents:
diff changeset
128 static void get_mirandas(GrowableArray<methodOop>* mirandas, klassOop super, objArrayOop class_methods, objArrayOop local_interfaces);
a61af66fc99e Initial load
duke
parents:
diff changeset
129 static int get_num_mirandas(klassOop super, objArrayOop class_methods, objArrayOop local_interfaces);
a61af66fc99e Initial load
duke
parents:
diff changeset
130
a61af66fc99e Initial load
duke
parents:
diff changeset
131
a61af66fc99e Initial load
duke
parents:
diff changeset
132 void verify_against(outputStream* st, klassVtable* vt, int index);
a61af66fc99e Initial load
duke
parents:
diff changeset
133 inline instanceKlass* ik() const;
a61af66fc99e Initial load
duke
parents:
diff changeset
134 };
a61af66fc99e Initial load
duke
parents:
diff changeset
135
a61af66fc99e Initial load
duke
parents:
diff changeset
136
a61af66fc99e Initial load
duke
parents:
diff changeset
137 // private helper class for klassVtable
a61af66fc99e Initial load
duke
parents:
diff changeset
138 // description of entry points:
a61af66fc99e Initial load
duke
parents:
diff changeset
139 // destination is interpreted:
a61af66fc99e Initial load
duke
parents:
diff changeset
140 // from_compiled_code_entry_point -> c2iadapter
a61af66fc99e Initial load
duke
parents:
diff changeset
141 // from_interpreter_entry_point -> interpreter entry point
a61af66fc99e Initial load
duke
parents:
diff changeset
142 // destination is compiled:
a61af66fc99e Initial load
duke
parents:
diff changeset
143 // from_compiled_code_entry_point -> nmethod entry point
a61af66fc99e Initial load
duke
parents:
diff changeset
144 // from_interpreter_entry_point -> i2cadapter
a61af66fc99e Initial load
duke
parents:
diff changeset
145 class vtableEntry VALUE_OBJ_CLASS_SPEC {
a61af66fc99e Initial load
duke
parents:
diff changeset
146 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
147 // size in words
a61af66fc99e Initial load
duke
parents:
diff changeset
148 static int size() {
a61af66fc99e Initial load
duke
parents:
diff changeset
149 return sizeof(vtableEntry) / sizeof(HeapWord);
a61af66fc99e Initial load
duke
parents:
diff changeset
150 }
a61af66fc99e Initial load
duke
parents:
diff changeset
151 static int method_offset_in_bytes() { return offset_of(vtableEntry, _method); }
a61af66fc99e Initial load
duke
parents:
diff changeset
152 methodOop method() const { return _method; }
a61af66fc99e Initial load
duke
parents:
diff changeset
153
a61af66fc99e Initial load
duke
parents:
diff changeset
154 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
155 methodOop _method;
a61af66fc99e Initial load
duke
parents:
diff changeset
156 void set(methodOop method) { assert(method != NULL, "use clear"); _method = method; }
a61af66fc99e Initial load
duke
parents:
diff changeset
157 void clear() { _method = NULL; }
a61af66fc99e Initial load
duke
parents:
diff changeset
158 void print() PRODUCT_RETURN;
a61af66fc99e Initial load
duke
parents:
diff changeset
159 void verify(klassVtable* vt, outputStream* st);
a61af66fc99e Initial load
duke
parents:
diff changeset
160
a61af66fc99e Initial load
duke
parents:
diff changeset
161 friend class klassVtable;
a61af66fc99e Initial load
duke
parents:
diff changeset
162 };
a61af66fc99e Initial load
duke
parents:
diff changeset
163
a61af66fc99e Initial load
duke
parents:
diff changeset
164
a61af66fc99e Initial load
duke
parents:
diff changeset
165 inline methodOop klassVtable::method_at(int i) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
166 assert(i >= 0 && i < _length, "index out of bounds");
a61af66fc99e Initial load
duke
parents:
diff changeset
167 assert(table()[i].method() != NULL, "should not be null");
a61af66fc99e Initial load
duke
parents:
diff changeset
168 assert(oop(table()[i].method())->is_method(), "should be method");
a61af66fc99e Initial load
duke
parents:
diff changeset
169 return table()[i].method();
a61af66fc99e Initial load
duke
parents:
diff changeset
170 }
a61af66fc99e Initial load
duke
parents:
diff changeset
171
a61af66fc99e Initial load
duke
parents:
diff changeset
172 inline methodOop klassVtable::unchecked_method_at(int i) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
173 assert(i >= 0 && i < _length, "index out of bounds");
a61af66fc99e Initial load
duke
parents:
diff changeset
174 return table()[i].method();
a61af66fc99e Initial load
duke
parents:
diff changeset
175 }
a61af66fc99e Initial load
duke
parents:
diff changeset
176
a61af66fc99e Initial load
duke
parents:
diff changeset
177 inline oop* klassVtable::adr_method_at(int i) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
178 // Allow one past the last entry to be referenced; useful for loop bounds.
a61af66fc99e Initial load
duke
parents:
diff changeset
179 assert(i >= 0 && i <= _length, "index out of bounds");
a61af66fc99e Initial load
duke
parents:
diff changeset
180 return (oop*)(address(table() + i) + vtableEntry::method_offset_in_bytes());
a61af66fc99e Initial load
duke
parents:
diff changeset
181 }
a61af66fc99e Initial load
duke
parents:
diff changeset
182
a61af66fc99e Initial load
duke
parents:
diff changeset
183 // --------------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
184 class klassItable;
a61af66fc99e Initial load
duke
parents:
diff changeset
185 class itableMethodEntry;
a61af66fc99e Initial load
duke
parents:
diff changeset
186
a61af66fc99e Initial load
duke
parents:
diff changeset
187 class itableOffsetEntry VALUE_OBJ_CLASS_SPEC {
a61af66fc99e Initial load
duke
parents:
diff changeset
188 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
189 klassOop _interface;
a61af66fc99e Initial load
duke
parents:
diff changeset
190 int _offset;
a61af66fc99e Initial load
duke
parents:
diff changeset
191 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
192 klassOop interface_klass() const { return _interface; }
a61af66fc99e Initial load
duke
parents:
diff changeset
193 int offset() const { return _offset; }
a61af66fc99e Initial load
duke
parents:
diff changeset
194
a61af66fc99e Initial load
duke
parents:
diff changeset
195 static itableMethodEntry* method_entry(klassOop k, int offset) { return (itableMethodEntry*)(((address)k) + offset); }
a61af66fc99e Initial load
duke
parents:
diff changeset
196 itableMethodEntry* first_method_entry(klassOop k) { return method_entry(k, _offset); }
a61af66fc99e Initial load
duke
parents:
diff changeset
197
a61af66fc99e Initial load
duke
parents:
diff changeset
198 void initialize(klassOop interf, int offset) { _interface = interf; _offset = offset; }
a61af66fc99e Initial load
duke
parents:
diff changeset
199
a61af66fc99e Initial load
duke
parents:
diff changeset
200 // Static size and offset accessors
a61af66fc99e Initial load
duke
parents:
diff changeset
201 static int size() { return sizeof(itableOffsetEntry) / HeapWordSize; } // size in words
a61af66fc99e Initial load
duke
parents:
diff changeset
202 static int interface_offset_in_bytes() { return offset_of(itableOffsetEntry, _interface); }
a61af66fc99e Initial load
duke
parents:
diff changeset
203 static int offset_offset_in_bytes() { return offset_of(itableOffsetEntry, _offset); }
a61af66fc99e Initial load
duke
parents:
diff changeset
204
a61af66fc99e Initial load
duke
parents:
diff changeset
205 friend class klassItable;
a61af66fc99e Initial load
duke
parents:
diff changeset
206 };
a61af66fc99e Initial load
duke
parents:
diff changeset
207
a61af66fc99e Initial load
duke
parents:
diff changeset
208
a61af66fc99e Initial load
duke
parents:
diff changeset
209 class itableMethodEntry VALUE_OBJ_CLASS_SPEC {
a61af66fc99e Initial load
duke
parents:
diff changeset
210 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
211 methodOop _method;
a61af66fc99e Initial load
duke
parents:
diff changeset
212
a61af66fc99e Initial load
duke
parents:
diff changeset
213 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
214 methodOop method() const { return _method; }
a61af66fc99e Initial load
duke
parents:
diff changeset
215
a61af66fc99e Initial load
duke
parents:
diff changeset
216 void clear() { _method = NULL; }
a61af66fc99e Initial load
duke
parents:
diff changeset
217
a61af66fc99e Initial load
duke
parents:
diff changeset
218 void initialize(methodOop method);
a61af66fc99e Initial load
duke
parents:
diff changeset
219
a61af66fc99e Initial load
duke
parents:
diff changeset
220 // Static size and offset accessors
a61af66fc99e Initial load
duke
parents:
diff changeset
221 static int size() { return sizeof(itableMethodEntry) / HeapWordSize; } // size in words
a61af66fc99e Initial load
duke
parents:
diff changeset
222 static int method_offset_in_bytes() { return offset_of(itableMethodEntry, _method); }
a61af66fc99e Initial load
duke
parents:
diff changeset
223
a61af66fc99e Initial load
duke
parents:
diff changeset
224 friend class klassItable;
a61af66fc99e Initial load
duke
parents:
diff changeset
225 };
a61af66fc99e Initial load
duke
parents:
diff changeset
226
a61af66fc99e Initial load
duke
parents:
diff changeset
227 //
a61af66fc99e Initial load
duke
parents:
diff changeset
228 // Format of an itable
a61af66fc99e Initial load
duke
parents:
diff changeset
229 //
a61af66fc99e Initial load
duke
parents:
diff changeset
230 // ---- offset table ---
a61af66fc99e Initial load
duke
parents:
diff changeset
231 // klassOop of interface 1 \
a61af66fc99e Initial load
duke
parents:
diff changeset
232 // offset to vtable from start of oop / offset table entry
a61af66fc99e Initial load
duke
parents:
diff changeset
233 // ...
a61af66fc99e Initial load
duke
parents:
diff changeset
234 // klassOop of interface n \
a61af66fc99e Initial load
duke
parents:
diff changeset
235 // offset to vtable from start of oop / offset table entry
a61af66fc99e Initial load
duke
parents:
diff changeset
236 // --- vtable for interface 1 ---
a61af66fc99e Initial load
duke
parents:
diff changeset
237 // methodOop \
a61af66fc99e Initial load
duke
parents:
diff changeset
238 // compiler entry point / method table entry
a61af66fc99e Initial load
duke
parents:
diff changeset
239 // ...
a61af66fc99e Initial load
duke
parents:
diff changeset
240 // methodOop \
a61af66fc99e Initial load
duke
parents:
diff changeset
241 // compiler entry point / method table entry
a61af66fc99e Initial load
duke
parents:
diff changeset
242 // -- vtable for interface 2 ---
a61af66fc99e Initial load
duke
parents:
diff changeset
243 // ...
a61af66fc99e Initial load
duke
parents:
diff changeset
244 //
a61af66fc99e Initial load
duke
parents:
diff changeset
245 class klassItable : public ResourceObj {
a61af66fc99e Initial load
duke
parents:
diff changeset
246 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
247 instanceKlassHandle _klass; // my klass
a61af66fc99e Initial load
duke
parents:
diff changeset
248 int _table_offset; // offset of start of itable data within klass (in words)
a61af66fc99e Initial load
duke
parents:
diff changeset
249 int _size_offset_table; // size of offset table (in itableOffset entries)
a61af66fc99e Initial load
duke
parents:
diff changeset
250 int _size_method_table; // size of methodtable (in itableMethodEntry entries)
a61af66fc99e Initial load
duke
parents:
diff changeset
251
a61af66fc99e Initial load
duke
parents:
diff changeset
252 void initialize_itable_for_interface(int method_table_offset, KlassHandle interf_h, bool checkconstraints, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
253 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
254 klassItable(instanceKlassHandle klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
255
a61af66fc99e Initial load
duke
parents:
diff changeset
256 itableOffsetEntry* offset_entry(int i) { assert(0 <= i && i <= _size_offset_table, "index out of bounds");
a61af66fc99e Initial load
duke
parents:
diff changeset
257 return &((itableOffsetEntry*)vtable_start())[i]; }
a61af66fc99e Initial load
duke
parents:
diff changeset
258
a61af66fc99e Initial load
duke
parents:
diff changeset
259 itableMethodEntry* method_entry(int i) { assert(0 <= i && i <= _size_method_table, "index out of bounds");
a61af66fc99e Initial load
duke
parents:
diff changeset
260 return &((itableMethodEntry*)method_start())[i]; }
a61af66fc99e Initial load
duke
parents:
diff changeset
261
16
f8236e79048a 6664627: Merge changes made only in hotspot 11 forward to jdk 7
dcubed
parents: 0
diff changeset
262 int size_offset_table() { return _size_offset_table; }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
263
a61af66fc99e Initial load
duke
parents:
diff changeset
264 // Initialization
a61af66fc99e Initial load
duke
parents:
diff changeset
265 void initialize_itable(bool checkconstraints, TRAPS);
a61af66fc99e Initial load
duke
parents:
diff changeset
266
a61af66fc99e Initial load
duke
parents:
diff changeset
267 // Updates
a61af66fc99e Initial load
duke
parents:
diff changeset
268 void initialize_with_method(methodOop m);
a61af66fc99e Initial load
duke
parents:
diff changeset
269
a61af66fc99e Initial load
duke
parents:
diff changeset
270 // RedefineClasses() API support:
a61af66fc99e Initial load
duke
parents:
diff changeset
271 // if any entry of this itable points to any of old_methods,
a61af66fc99e Initial load
duke
parents:
diff changeset
272 // replace it with the corresponding new_method.
a61af66fc99e Initial load
duke
parents:
diff changeset
273 // trace_name_printed is set to true if the current call has
a61af66fc99e Initial load
duke
parents:
diff changeset
274 // printed the klass name so that other routines in the adjust_*
a61af66fc99e Initial load
duke
parents:
diff changeset
275 // group don't print the klass name.
a61af66fc99e Initial load
duke
parents:
diff changeset
276 void adjust_method_entries(methodOop* old_methods, methodOop* new_methods,
a61af66fc99e Initial load
duke
parents:
diff changeset
277 int methods_length, bool * trace_name_printed);
a61af66fc99e Initial load
duke
parents:
diff changeset
278
a61af66fc99e Initial load
duke
parents:
diff changeset
279 // Garbage collection
a61af66fc99e Initial load
duke
parents:
diff changeset
280 void oop_follow_contents();
a61af66fc99e Initial load
duke
parents:
diff changeset
281 void oop_adjust_pointers();
a61af66fc99e Initial load
duke
parents:
diff changeset
282
a61af66fc99e Initial load
duke
parents:
diff changeset
283 #ifndef SERIALGC
a61af66fc99e Initial load
duke
parents:
diff changeset
284 // Parallel Old
a61af66fc99e Initial load
duke
parents:
diff changeset
285 void oop_follow_contents(ParCompactionManager* cm);
a61af66fc99e Initial load
duke
parents:
diff changeset
286 void oop_update_pointers(ParCompactionManager* cm);
a61af66fc99e Initial load
duke
parents:
diff changeset
287 void oop_update_pointers(ParCompactionManager* cm,
a61af66fc99e Initial load
duke
parents:
diff changeset
288 HeapWord* beg_addr, HeapWord* end_addr);
a61af66fc99e Initial load
duke
parents:
diff changeset
289 #endif // SERIALGC
a61af66fc99e Initial load
duke
parents:
diff changeset
290
a61af66fc99e Initial load
duke
parents:
diff changeset
291 // Iterators
a61af66fc99e Initial load
duke
parents:
diff changeset
292 void oop_oop_iterate(OopClosure* blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
293 void oop_oop_iterate_m(OopClosure* blk, MemRegion mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
294
a61af66fc99e Initial load
duke
parents:
diff changeset
295 // Setup of itable
a61af66fc99e Initial load
duke
parents:
diff changeset
296 static int compute_itable_size(objArrayHandle transitive_interfaces);
a61af66fc99e Initial load
duke
parents:
diff changeset
297 static void setup_itable_offset_table(instanceKlassHandle klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
298
a61af66fc99e Initial load
duke
parents:
diff changeset
299 // Resolving of method to index
a61af66fc99e Initial load
duke
parents:
diff changeset
300 static int compute_itable_index(methodOop m);
a61af66fc99e Initial load
duke
parents:
diff changeset
301
a61af66fc99e Initial load
duke
parents:
diff changeset
302 // Debugging/Statistics
a61af66fc99e Initial load
duke
parents:
diff changeset
303 static void print_statistics() PRODUCT_RETURN;
a61af66fc99e Initial load
duke
parents:
diff changeset
304 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
305 intptr_t* vtable_start() const { return ((intptr_t*)_klass()) + _table_offset; }
a61af66fc99e Initial load
duke
parents:
diff changeset
306 intptr_t* method_start() const { return vtable_start() + _size_offset_table * itableOffsetEntry::size(); }
a61af66fc99e Initial load
duke
parents:
diff changeset
307
a61af66fc99e Initial load
duke
parents:
diff changeset
308 // Helper methods
a61af66fc99e Initial load
duke
parents:
diff changeset
309 static int calc_itable_size(int num_interfaces, int num_methods) { return (num_interfaces * itableOffsetEntry::size()) + (num_methods * itableMethodEntry::size()); }
a61af66fc99e Initial load
duke
parents:
diff changeset
310
a61af66fc99e Initial load
duke
parents:
diff changeset
311 // Statistics
a61af66fc99e Initial load
duke
parents:
diff changeset
312 NOT_PRODUCT(static int _total_classes;) // Total no. of classes with itables
a61af66fc99e Initial load
duke
parents:
diff changeset
313 NOT_PRODUCT(static long _total_size;) // Total no. of bytes used for itables
a61af66fc99e Initial load
duke
parents:
diff changeset
314
a61af66fc99e Initial load
duke
parents:
diff changeset
315 static void update_stats(int size) PRODUCT_RETURN NOT_PRODUCT({ _total_classes++; _total_size += size; })
a61af66fc99e Initial load
duke
parents:
diff changeset
316 };