Skip to content

Commit adaff7d

Browse files
committedOct 12, 2022
8294900: Refactor ZObjArrayAllocator
Reviewed-by: eosterlund
1 parent 86ec158 commit adaff7d

File tree

4 files changed

+63
-31
lines changed

4 files changed

+63
-31
lines changed
 

‎src/hotspot/share/gc/shared/memAllocator.hpp

+9-4
Original file line numberDiff line numberDiff line change
@@ -56,12 +56,16 @@ class MemAllocator: StackObj {
5656
_word_size(word_size)
5757
{ }
5858

59-
// This function clears the memory of the object
59+
// Initialization provided by subclasses.
60+
virtual oop initialize(HeapWord* mem) const = 0;
61+
62+
// This function clears the memory of the object.
6063
void mem_clear(HeapWord* mem) const;
64+
6165
// This finish constructing an oop by installing the mark word and the Klass* pointer
6266
// last. At the point when the Klass pointer is initialized, this is a constructed object
6367
// that must be parseable as an oop by concurrent collectors.
64-
virtual oop finish(HeapWord* mem) const;
68+
oop finish(HeapWord* mem) const;
6569

6670
// Raw memory allocation. This will try to do a TLAB allocation, and otherwise fall
6771
// back to calling CollectedHeap::mem_allocate().
@@ -72,9 +76,9 @@ class MemAllocator: StackObj {
7276
}
7377

7478
public:
79+
// Allocate and fully construct the object, and perform various instrumentation. Could safepoint.
7580
oop allocate() const;
7681
oop try_allocate_in_existing_tlab();
77-
virtual oop initialize(HeapWord* mem) const = 0;
7882
};
7983

8084
class ObjAllocator: public MemAllocator {
@@ -85,9 +89,10 @@ class ObjAllocator: public MemAllocator {
8589
};
8690

8791
class ObjArrayAllocator: public MemAllocator {
92+
protected:
8893
const int _length;
8994
const bool _do_zero;
90-
protected:
95+
9196
virtual MemRegion obj_memory_range(oop obj) const;
9297

9398
public:

‎src/hotspot/share/gc/z/zCollectedHeap.cpp

+1-5
Original file line numberDiff line numberDiff line change
@@ -161,11 +161,7 @@ HeapWord* ZCollectedHeap::allocate_new_tlab(size_t min_size, size_t requested_si
161161
}
162162

163163
oop ZCollectedHeap::array_allocate(Klass* klass, size_t size, int length, bool do_zero, TRAPS) {
164-
if (!do_zero) {
165-
return CollectedHeap::array_allocate(klass, size, length, false /* do_zero */, THREAD);
166-
}
167-
168-
ZObjArrayAllocator allocator(klass, size, length, THREAD);
164+
ZObjArrayAllocator allocator(klass, size, length, do_zero, THREAD);
169165
return allocator.allocate();
170166
}
171167

‎src/hotspot/share/gc/z/zObjArrayAllocator.cpp

+47-19
Original file line numberDiff line numberDiff line change
@@ -27,36 +27,64 @@
2727
#include "gc/z/zUtils.inline.hpp"
2828
#include "oops/arrayKlass.hpp"
2929
#include "runtime/interfaceSupport.inline.hpp"
30+
#include "utilities/debug.hpp"
3031

31-
ZObjArrayAllocator::ZObjArrayAllocator(Klass* klass, size_t word_size, int length, Thread* thread) :
32-
ObjArrayAllocator(klass, word_size, length, false /* do_zero */, thread) {}
32+
ZObjArrayAllocator::ZObjArrayAllocator(Klass* klass, size_t word_size, int length, bool do_zero, Thread* thread) :
33+
ObjArrayAllocator(klass, word_size, length, do_zero, thread) {}
3334

34-
oop ZObjArrayAllocator::finish(HeapWord* mem) const {
35-
// Initialize object header and length field
36-
ObjArrayAllocator::finish(mem);
35+
void ZObjArrayAllocator::yield_for_safepoint() const {
36+
ThreadBlockInVM tbivm(JavaThread::cast(_thread));
37+
}
3738

38-
// Keep the array alive across safepoints through an invisible
39-
// root. Invisible roots are not visited by the heap itarator
40-
// and the marking logic will not attempt to follow its elements.
41-
ZThreadLocalData::set_invisible_root(_thread, (oop*)&mem);
39+
oop ZObjArrayAllocator::initialize(HeapWord* mem) const {
40+
// ZGC specializes the initialization by performing segmented clearing
41+
// to allow shorter time-to-safepoints.
42+
43+
if (!_do_zero) {
44+
// No need for ZGC specialization
45+
return ObjArrayAllocator::initialize(mem);
46+
}
4247

4348
// A max segment size of 64K was chosen because microbenchmarking
4449
// suggested that it offered a good trade-off between allocation
4550
// time and time-to-safepoint
4651
const size_t segment_max = ZUtils::bytes_to_words(64 * K);
47-
const size_t skip = arrayOopDesc::header_size(ArrayKlass::cast(_klass)->element_type());
48-
size_t remaining = _word_size - skip;
52+
const BasicType element_type = ArrayKlass::cast(_klass)->element_type();
53+
const size_t header = arrayOopDesc::header_size(element_type);
54+
const size_t payload_size = _word_size - header;
55+
56+
if (payload_size <= segment_max) {
57+
// To small to use segmented clearing
58+
return ObjArrayAllocator::initialize(mem);
59+
}
60+
61+
// Segmented clearing
62+
63+
// The array is going to be exposed before it has been completely
64+
// cleared, therefore we can't expose the header at the end of this
65+
// function. Instead explicitly initialize it according to our needs.
66+
arrayOopDesc::set_mark(mem, markWord::prototype());
67+
arrayOopDesc::release_set_klass(mem, _klass);
68+
assert(_length >= 0, "length should be non-negative");
69+
arrayOopDesc::set_length(mem, _length);
70+
71+
// Keep the array alive across safepoints through an invisible
72+
// root. Invisible roots are not visited by the heap itarator
73+
// and the marking logic will not attempt to follow its elements.
74+
// Relocation knows how to dodge iterating over such objects.
75+
ZThreadLocalData::set_invisible_root(_thread, (oop*)&mem);
76+
77+
for (size_t processed = 0; processed < payload_size; processed += segment_max) {
78+
// Calculate segment
79+
HeapWord* const start = (HeapWord*)(mem + header + processed);
80+
const size_t remaining = payload_size - processed;
81+
const size_t segment_size = MIN2(remaining, segment_max);
4982

50-
while (remaining > 0) {
5183
// Clear segment
52-
const size_t segment = MIN2(remaining, segment_max);
53-
Copy::zero_to_words(mem + (_word_size - remaining), segment);
54-
remaining -= segment;
84+
Copy::zero_to_words(start, segment_size);
5585

56-
if (remaining > 0) {
57-
// Safepoint
58-
ThreadBlockInVM tbivm(JavaThread::cast(_thread));
59-
}
86+
// Safepoint
87+
yield_for_safepoint();
6088
}
6189

6290
ZThreadLocalData::clear_invisible_root(_thread);

‎src/hotspot/share/gc/z/zObjArrayAllocator.hpp

+6-3
Original file line numberDiff line numberDiff line change
@@ -27,10 +27,13 @@
2727
#include "gc/shared/memAllocator.hpp"
2828

2929
class ZObjArrayAllocator : public ObjArrayAllocator {
30-
public:
31-
ZObjArrayAllocator(Klass* klass, size_t word_size, int length, Thread* thread);
30+
private:
31+
virtual oop initialize(HeapWord* mem) const override;
32+
33+
void yield_for_safepoint() const;
3234

33-
virtual oop finish(HeapWord* mem) const;
35+
public:
36+
ZObjArrayAllocator(Klass* klass, size_t word_size, int length, bool do_zero, Thread* thread);
3437
};
3538

3639
#endif // SHARE_GC_Z_ZOBJARRAYALLOCATOR_HPP

0 commit comments

Comments
 (0)
Please sign in to comment.