|
27 | 27 | #include "gc/z/zUtils.inline.hpp"
|
28 | 28 | #include "oops/arrayKlass.hpp"
|
29 | 29 | #include "runtime/interfaceSupport.inline.hpp"
|
| 30 | +#include "utilities/debug.hpp" |
30 | 31 |
|
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) {} |
33 | 34 |
|
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 | +} |
37 | 38 |
|
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 | + } |
42 | 47 |
|
43 | 48 | // A max segment size of 64K was chosen because microbenchmarking
|
44 | 49 | // suggested that it offered a good trade-off between allocation
|
45 | 50 | // time and time-to-safepoint
|
46 | 51 | 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); |
49 | 82 |
|
50 |
| - while (remaining > 0) { |
51 | 83 | // 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); |
55 | 85 |
|
56 |
| - if (remaining > 0) { |
57 |
| - // Safepoint |
58 |
| - ThreadBlockInVM tbivm(JavaThread::cast(_thread)); |
59 |
| - } |
| 86 | + // Safepoint |
| 87 | + yield_for_safepoint(); |
60 | 88 | }
|
61 | 89 |
|
62 | 90 | ZThreadLocalData::clear_invisible_root(_thread);
|
|
0 commit comments