Skip to content

Commit aa57d07

Browse files
committedJan 22, 2021
8259214: MetaspaceClosure support for Arrays of MetaspaceObj
Reviewed-by: fparain, ccheung
1 parent bdc305e commit aa57d07

File tree

2 files changed

+242
-48
lines changed

2 files changed

+242
-48
lines changed
 

‎src/hotspot/share/memory/metaspaceClosure.hpp

+104-48
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2017, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -27,10 +27,13 @@
2727

2828
#include "logging/log.hpp"
2929
#include "memory/allocation.hpp"
30+
#include "metaprogramming/enableIf.hpp"
3031
#include "oops/array.hpp"
3132
#include "utilities/globalDefinitions.hpp"
3233
#include "utilities/growableArray.hpp"
3334
#include "utilities/hashtable.inline.hpp"
35+
#include "utilities/macros.hpp"
36+
#include <type_traits>
3437

3538
// The metadata hierarchy is separate from the oop hierarchy
3639
class MetaspaceObj; // no C++ vtable
@@ -92,8 +95,8 @@ class MetaspaceClosure {
9295
// However, to save space, MetaspaceObj has NO vtable. The vtable is introduced
9396
// only in the Metadata class.
9497
//
95-
// To work around the lack of a vtable, we use Ref class with templates
96-
// (see ObjectRef, PrimitiveArrayRef and PointerArrayRef)
98+
// To work around the lack of a vtable, we use the Ref class with templates
99+
// (see MSORef, OtherArrayRef, MSOArrayRef, and MSOPointerArrayRef)
97100
// so that we can statically discover the type of a object. The use of Ref
98101
// depends on the fact that:
99102
//
@@ -155,8 +158,8 @@ class MetaspaceClosure {
155158
};
156159

157160
private:
158-
// -------------------------------------------------- ObjectRef
159-
template <class T> class ObjectRef : public Ref {
161+
// MSORef -- iterate an instance of MetaspaceObj
162+
template <class T> class MSORef : public Ref {
160163
T** _mpp;
161164
T* dereference() const {
162165
return *_mpp;
@@ -167,7 +170,7 @@ class MetaspaceClosure {
167170
}
168171

169172
public:
170-
ObjectRef(T** mpp, Writability w) : Ref(w), _mpp(mpp) {}
173+
MSORef(T** mpp, Writability w) : Ref(w), _mpp(mpp) {}
171174

172175
virtual bool is_read_only_by_default() const { return T::is_read_only_by_default(); }
173176
virtual bool not_null() const { return dereference() != NULL; }
@@ -182,65 +185,80 @@ class MetaspaceClosure {
182185
}
183186
};
184187

185-
// -------------------------------------------------- PrimitiveArrayRef
186-
template <class T> class PrimitiveArrayRef : public Ref {
188+
// abstract base class for MSOArrayRef, MSOPointerArrayRef and OtherArrayRef
189+
template <class T> class ArrayRef : public Ref {
187190
Array<T>** _mpp;
191+
protected:
188192
Array<T>* dereference() const {
189193
return *_mpp;
190194
}
191-
protected:
192195
virtual void** mpp() const {
193196
return (void**)_mpp;
194197
}
195198

196-
public:
197-
PrimitiveArrayRef(Array<T>** mpp, Writability w) : Ref(w), _mpp(mpp) {}
199+
ArrayRef(Array<T>** mpp, Writability w) : Ref(w), _mpp(mpp) {}
198200

199201
// all Arrays are read-only by default
200202
virtual bool is_read_only_by_default() const { return true; }
201203
virtual bool not_null() const { return dereference() != NULL; }
202204
virtual int size() const { return dereference()->size(); }
203205
virtual MetaspaceObj::Type msotype() const { return MetaspaceObj::array_type(sizeof(T)); }
206+
};
207+
208+
// OtherArrayRef -- iterate an instance of Array<T>, where T is NOT a subtype of MetaspaceObj.
209+
// T can be a primitive type, such as int, or a structure. However, we do not scan
210+
// the fields inside T, so you should not embed any pointers inside T.
211+
template <class T> class OtherArrayRef : public ArrayRef<T> {
212+
public:
213+
OtherArrayRef(Array<T>** mpp, Writability w) : ArrayRef<T>(mpp, w) {}
204214

205215
virtual void metaspace_pointers_do(MetaspaceClosure *it) const {
206-
Array<T>* array = dereference();
207-
log_trace(cds)("Iter(PrimitiveArray): %p [%d]", array, array->length());
216+
Array<T>* array = ArrayRef<T>::dereference();
217+
log_trace(cds)("Iter(OtherArray): %p [%d]", array, array->length());
208218
}
209219
virtual void metaspace_pointers_do_at(MetaspaceClosure *it, address new_loc) const {
210220
Array<T>* array = (Array<T>*)new_loc;
211-
log_trace(cds)("Iter(PrimitiveArray): %p [%d]", array, array->length());
221+
log_trace(cds)("Iter(OtherArray): %p [%d]", array, array->length());
212222
}
213223
};
214224

215-
// -------------------------------------------------- PointerArrayRef
216-
template <class T> class PointerArrayRef : public Ref {
217-
Array<T*>** _mpp;
218-
Array<T*>* dereference() const {
219-
return *_mpp;
225+
// MSOArrayRef -- iterate an instance of Array<T>, where T is a subtype of MetaspaceObj.
226+
// We recursively call T::metaspace_pointers_do() for each element in this array.
227+
template <class T> class MSOArrayRef : public ArrayRef<T> {
228+
public:
229+
MSOArrayRef(Array<T>** mpp, Writability w) : ArrayRef<T>(mpp, w) {}
230+
231+
virtual void metaspace_pointers_do(MetaspaceClosure *it) const {
232+
metaspace_pointers_do_at_impl(it, ArrayRef<T>::dereference());
220233
}
221-
protected:
222-
virtual void** mpp() const {
223-
return (void**)_mpp;
234+
virtual void metaspace_pointers_do_at(MetaspaceClosure *it, address new_loc) const {
235+
metaspace_pointers_do_at_impl(it, (Array<T>*)new_loc);
236+
}
237+
private:
238+
void metaspace_pointers_do_at_impl(MetaspaceClosure *it, Array<T>* array) const {
239+
log_trace(cds)("Iter(MSOArray): %p [%d]", array, array->length());
240+
for (int i = 0; i < array->length(); i++) {
241+
T* elm = array->adr_at(i);
242+
elm->metaspace_pointers_do(it);
243+
}
224244
}
245+
};
225246

247+
// MSOPointerArrayRef -- iterate an instance of Array<T*>, where T is a subtype of MetaspaceObj.
248+
// We recursively call MetaspaceClosure::push() for each pointer in this array.
249+
template <class T> class MSOPointerArrayRef : public ArrayRef<T*> {
226250
public:
227-
PointerArrayRef(Array<T*>** mpp, Writability w) : Ref(w), _mpp(mpp) {}
228-
229-
// all Arrays are read-only by default
230-
virtual bool is_read_only_by_default() const { return true; }
231-
virtual bool not_null() const { return dereference() != NULL; }
232-
virtual int size() const { return dereference()->size(); }
233-
virtual MetaspaceObj::Type msotype() const { return MetaspaceObj::array_type(sizeof(T*)); }
251+
MSOPointerArrayRef(Array<T*>** mpp, Writability w) : ArrayRef<T*>(mpp, w) {}
234252

235253
virtual void metaspace_pointers_do(MetaspaceClosure *it) const {
236-
metaspace_pointers_do_at_impl(it, dereference());
254+
metaspace_pointers_do_at_impl(it, ArrayRef<T*>::dereference());
237255
}
238256
virtual void metaspace_pointers_do_at(MetaspaceClosure *it, address new_loc) const {
239257
metaspace_pointers_do_at_impl(it, (Array<T*>*)new_loc);
240258
}
241259
private:
242260
void metaspace_pointers_do_at_impl(MetaspaceClosure *it, Array<T*>* array) const {
243-
log_trace(cds)("Iter(ObjectArray): %p [%d]", array, array->length());
261+
log_trace(cds)("Iter(MSOPointerArray): %p [%d]", array, array->length());
244262
for (int i = 0; i < array->length(); i++) {
245263
T** mpp = array->adr_at(i);
246264
it->push(mpp);
@@ -288,31 +306,69 @@ class MetaspaceClosure {
288306
// returns true if we want to keep iterating the pointers embedded inside <ref>
289307
virtual bool do_ref(Ref* ref, bool read_only) = 0;
290308

291-
// When you do:
292-
// void MyType::metaspace_pointers_do(MetaspaceClosure* it) {
293-
// it->push(_my_field)
294-
// }
309+
private:
310+
template <class REF_TYPE, typename T>
311+
void push_with_ref(T** mpp, Writability w) {
312+
push_impl(new REF_TYPE(mpp, w));
313+
}
314+
315+
public:
316+
// When MetaspaceClosure::push(...) is called, pick the correct Ref subtype to handle it:
317+
//
318+
// MetaspaceClosure* it = ...;
319+
// Klass* o = ...; it->push(&o); => MSORef
320+
// Array<int>* a1 = ...; it->push(&a1); => OtherArrayRef
321+
// Array<Annotation>* a2 = ...; it->push(&a2); => MSOArrayRef
322+
// Array<Klass*>* a3 = ...; it->push(&a3); => MSOPointerArrayRef
323+
// Array<Array<Klass*>*>* a4 = ...; it->push(&a4); => MSOPointerArrayRef
324+
// Array<Annotation*>* a5 = ...; it->push(&a5); => MSOPointerArrayRef
325+
//
326+
// Note that the following will fail to compile (to prevent you from adding new fields
327+
// into the MetaspaceObj subtypes that cannot be properly copied by CDS):
295328
//
296-
// C++ will try to match the "most specific" template function. This one will
297-
// will be matched if possible (if mpp is an Array<> of any pointer type).
298-
template <typename T> void push(Array<T*>** mpp, Writability w = _default) {
299-
push_impl(new PointerArrayRef<T>(mpp, w));
329+
// Hashtable* h = ...; it->push(&h); => Hashtable is not a subclass of MetaspaceObj
330+
// Array<Hashtable*>* a6 = ...; it->push(&a6); => Hashtable is not a subclass of MetaspaceObj
331+
// Array<int*>* a7 = ...; it->push(&a7); => int is not a subclass of MetaspaceObj
332+
333+
template <typename T>
334+
void push(T** mpp, Writability w = _default) {
335+
static_assert(std::is_base_of<MetaspaceObj, T>::value, "Do not push pointers of arbitrary types");
336+
push_with_ref<MSORef<T>>(mpp, w);
337+
}
338+
339+
template <typename T, ENABLE_IF(!std::is_base_of<MetaspaceObj, T>::value)>
340+
void push(Array<T>** mpp, Writability w = _default) {
341+
push_with_ref<OtherArrayRef<T>>(mpp, w);
300342
}
301343

302-
// If the above function doesn't match (mpp is an Array<>, but T is not a pointer type), then
303-
// this is the second choice.
304-
template <typename T> void push(Array<T>** mpp, Writability w = _default) {
305-
push_impl(new PrimitiveArrayRef<T>(mpp, w));
344+
template <typename T, ENABLE_IF(std::is_base_of<MetaspaceObj, T>::value)>
345+
void push(Array<T>** mpp, Writability w = _default) {
346+
push_with_ref<MSOArrayRef<T>>(mpp, w);
306347
}
307348

308-
// If the above function doesn't match (mpp is not an Array<> type), then
309-
// this will be matched by default.
310-
template <class T> void push(T** mpp, Writability w = _default) {
311-
push_impl(new ObjectRef<T>(mpp, w));
349+
template <typename T>
350+
void push(Array<T*>** mpp, Writability w = _default) {
351+
static_assert(std::is_base_of<MetaspaceObj, T>::value, "Do not push Arrays of arbitrary pointer types");
352+
push_with_ref<MSOPointerArrayRef<T>>(mpp, w);
353+
}
354+
355+
#if 0
356+
// Enable this block if you're changing the push(...) methods, to test for types that should be
357+
// disallowed. Each of the following "push" calls should result in a compile-time error.
358+
void test_disallowed_types(MetaspaceClosure* it) {
359+
Hashtable<bool, mtInternal>* h = NULL;
360+
it->push(&h);
361+
362+
Array<Hashtable<bool, mtInternal>*>* a6 = NULL;
363+
it->push(&a6);
364+
365+
Array<int*>* a7 = NULL;
366+
it->push(&a7);
312367
}
368+
#endif
313369

314370
template <class T> void push_method_entry(T** mpp, intptr_t* p) {
315-
Ref* ref = new ObjectRef<T>(mpp, _default);
371+
Ref* ref = new MSORef<T>(mpp, _default);
316372
push_special(_method_entry_ref, ref, (intptr_t*)p);
317373
if (!ref->keep_after_pushing()) {
318374
delete ref;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,138 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
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+
*
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+
#include "precompiled.hpp"
25+
#include "memory/allocation.hpp"
26+
#include "memory/metadataFactory.hpp"
27+
#include "memory/metaspaceClosure.hpp"
28+
#include "oops/array.hpp"
29+
#include "oops/metadata.hpp"
30+
#include "runtime/thread.hpp"
31+
#include "unittest.hpp"
32+
33+
class MyMetaData : public MetaspaceObj {
34+
public:
35+
MyMetaData* _a;
36+
MyMetaData* _b;
37+
38+
MyMetaData() : _a(NULL), _b(NULL) {}
39+
40+
MetaspaceObj::Type type() const {
41+
return MetaspaceObj::SymbolType; // Just lie. It doesn't matter in this test
42+
}
43+
const char* internal_name() const {
44+
return "MyMetaData";
45+
}
46+
int size() const {
47+
return align_up((int)sizeof(MyMetaData), wordSize) / wordSize;
48+
};
49+
50+
static bool is_read_only_by_default() {
51+
return true;
52+
}
53+
54+
void metaspace_pointers_do(MetaspaceClosure* it) {
55+
it->push(&_a);
56+
it->push(&_b);
57+
}
58+
};
59+
60+
class MyUniqueMetaspaceClosure : public MetaspaceClosure {
61+
static constexpr int SIZE = 10;
62+
MyMetaData* _visited[SIZE];
63+
int _count;
64+
public:
65+
MyUniqueMetaspaceClosure() {
66+
for (int i = 0; i < SIZE; i++) {
67+
_visited[i] = NULL;
68+
}
69+
_count = 0;
70+
}
71+
72+
virtual bool do_ref(Ref* ref, bool read_only) {
73+
MyMetaData* ptr = (MyMetaData*)ref->obj();
74+
assert(_count < SIZE, "out of bounds");
75+
_visited[_count++] = ptr;
76+
return true; // recurse
77+
}
78+
79+
bool has_visited(MyMetaData* p) {
80+
for (int i = 0; i < SIZE; i++) {
81+
if (_visited[i] == p) {
82+
return true;
83+
}
84+
}
85+
return false;
86+
}
87+
};
88+
89+
// iterate an Array<MyMetaData*>
90+
TEST_VM(MetaspaceClosure, MSOPointerArrayRef) {
91+
Thread* THREAD = Thread::current();
92+
ClassLoaderData* cld = ClassLoaderData::the_null_class_loader_data();
93+
Array<MyMetaData*>* array = MetadataFactory::new_array<MyMetaData*>(cld, 4, THREAD);
94+
for (int i = 0; i < array->length(); i++) {
95+
EXPECT_TRUE(array->at(i) == NULL) << "should be initialized to null";
96+
}
97+
98+
MyMetaData x;
99+
MyMetaData y;
100+
MyMetaData z;
101+
102+
array->at_put(0, &x);
103+
array->at_put(2, &y);
104+
y._a = &z;
105+
106+
MyUniqueMetaspaceClosure closure;
107+
closure.push(&array);
108+
109+
EXPECT_TRUE(closure.has_visited(&x)) << "must be";
110+
EXPECT_TRUE(closure.has_visited(&y)) << "must be";
111+
EXPECT_TRUE(closure.has_visited(&z)) << "must be";
112+
}
113+
114+
// iterate an Array<MyMetaData>
115+
TEST_VM(MetaspaceClosure, MSOArrayRef) {
116+
Thread* THREAD = Thread::current();
117+
ClassLoaderData* cld = ClassLoaderData::the_null_class_loader_data();
118+
Array<MyMetaData>* array = MetadataFactory::new_array<MyMetaData>(cld, 4, THREAD);
119+
for (int i = 0; i < array->length(); i++) {
120+
EXPECT_TRUE(array->at(i)._a == NULL) << "should be initialized to null";
121+
EXPECT_TRUE(array->at(i)._b == NULL) << "should be initialized to null";
122+
}
123+
124+
MyMetaData x;
125+
MyMetaData y;
126+
MyMetaData z;
127+
128+
array->adr_at(0)->_a = &x;
129+
array->adr_at(2)->_b = &y;
130+
y._a = &z;
131+
132+
MyUniqueMetaspaceClosure closure;
133+
closure.push(&array);
134+
135+
EXPECT_TRUE(closure.has_visited(&x)) << "must be";
136+
EXPECT_TRUE(closure.has_visited(&y)) << "must be";
137+
EXPECT_TRUE(closure.has_visited(&z)) << "must be";
138+
}

0 commit comments

Comments
 (0)
Please sign in to comment.