1
1
/*
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.
3
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
4
*
5
5
* This code is free software; you can redistribute it and/or modify it
27
27
28
28
#include " logging/log.hpp"
29
29
#include " memory/allocation.hpp"
30
+ #include " metaprogramming/enableIf.hpp"
30
31
#include " oops/array.hpp"
31
32
#include " utilities/globalDefinitions.hpp"
32
33
#include " utilities/growableArray.hpp"
33
34
#include " utilities/hashtable.inline.hpp"
35
+ #include " utilities/macros.hpp"
36
+ #include < type_traits>
34
37
35
38
// The metadata hierarchy is separate from the oop hierarchy
36
39
class MetaspaceObj ; // no C++ vtable
@@ -92,8 +95,8 @@ class MetaspaceClosure {
92
95
// However, to save space, MetaspaceObj has NO vtable. The vtable is introduced
93
96
// only in the Metadata class.
94
97
//
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 )
97
100
// so that we can statically discover the type of a object. The use of Ref
98
101
// depends on the fact that:
99
102
//
@@ -155,8 +158,8 @@ class MetaspaceClosure {
155
158
};
156
159
157
160
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 {
160
163
T** _mpp;
161
164
T* dereference () const {
162
165
return *_mpp;
@@ -167,7 +170,7 @@ class MetaspaceClosure {
167
170
}
168
171
169
172
public:
170
- ObjectRef (T** mpp, Writability w) : Ref(w), _mpp(mpp) {}
173
+ MSORef (T** mpp, Writability w) : Ref(w), _mpp(mpp) {}
171
174
172
175
virtual bool is_read_only_by_default () const { return T::is_read_only_by_default (); }
173
176
virtual bool not_null () const { return dereference () != NULL ; }
@@ -182,65 +185,80 @@ class MetaspaceClosure {
182
185
}
183
186
};
184
187
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 {
187
190
Array<T>** _mpp;
191
+ protected:
188
192
Array<T>* dereference () const {
189
193
return *_mpp;
190
194
}
191
- protected:
192
195
virtual void ** mpp () const {
193
196
return (void **)_mpp;
194
197
}
195
198
196
- public:
197
- PrimitiveArrayRef (Array<T>** mpp, Writability w) : Ref(w), _mpp(mpp) {}
199
+ ArrayRef (Array<T>** mpp, Writability w) : Ref(w), _mpp(mpp) {}
198
200
199
201
// all Arrays are read-only by default
200
202
virtual bool is_read_only_by_default () const { return true ; }
201
203
virtual bool not_null () const { return dereference () != NULL ; }
202
204
virtual int size () const { return dereference ()->size (); }
203
205
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) {}
204
214
205
215
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 ());
208
218
}
209
219
virtual void metaspace_pointers_do_at (MetaspaceClosure *it, address new_loc) const {
210
220
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 ());
212
222
}
213
223
};
214
224
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 ());
220
233
}
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
+ }
224
244
}
245
+ };
225
246
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*> {
226
250
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) {}
234
252
235
253
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 ());
237
255
}
238
256
virtual void metaspace_pointers_do_at (MetaspaceClosure *it, address new_loc) const {
239
257
metaspace_pointers_do_at_impl (it, (Array<T*>*)new_loc);
240
258
}
241
259
private:
242
260
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 ());
244
262
for (int i = 0 ; i < array->length (); i++) {
245
263
T** mpp = array->adr_at (i);
246
264
it->push (mpp);
@@ -288,31 +306,69 @@ class MetaspaceClosure {
288
306
// returns true if we want to keep iterating the pointers embedded inside <ref>
289
307
virtual bool do_ref (Ref* ref, bool read_only) = 0;
290
308
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):
295
328
//
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);
300
342
}
301
343
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);
306
347
}
307
348
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);
312
367
}
368
+ #endif
313
369
314
370
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);
316
372
push_special (_method_entry_ref, ref, (intptr_t *)p);
317
373
if (!ref->keep_after_pushing ()) {
318
374
delete ref;
0 commit comments