Skip to content

Commit 74851c5

Browse files
author
Kim Barrett
committedSep 15, 2020
8253169: [BACKOUT] Improve large object handling during evacuation
Revert "8027545: Improve object array chunking test in G1's copy_to_survivor_space" Revert "8158045: Improve large object handling during evacuation" Revert "8027761: Investigate fast-path for scanning only objects with references during gc" Reviewed-by: tschatzl, sjohanss
1 parent b5620a3 commit 74851c5

8 files changed

+62
-439
lines changed
 

‎src/hotspot/share/gc/g1/g1ParScanThreadState.cpp

+48-77
Original file line numberDiff line numberDiff line change
@@ -31,19 +31,15 @@
3131
#include "gc/g1/g1RootClosures.hpp"
3232
#include "gc/g1/g1StringDedup.hpp"
3333
#include "gc/g1/g1Trace.hpp"
34-
#include "gc/shared/partialArrayTaskStepper.inline.hpp"
3534
#include "gc/shared/taskqueue.inline.hpp"
3635
#include "memory/allocation.inline.hpp"
3736
#include "oops/access.inline.hpp"
3837
#include "oops/oop.inline.hpp"
39-
#include "runtime/atomic.hpp"
4038
#include "runtime/prefetch.inline.hpp"
41-
#include "utilities/globalDefinitions.hpp"
4239

4340
G1ParScanThreadState::G1ParScanThreadState(G1CollectedHeap* g1h,
4441
G1RedirtyCardsQueueSet* rdcqs,
4542
uint worker_id,
46-
uint n_workers,
4743
size_t young_cset_length,
4844
size_t optional_cset_length)
4945
: _g1h(g1h),
@@ -64,8 +60,6 @@ G1ParScanThreadState::G1ParScanThreadState(G1CollectedHeap* g1h,
6460
_surviving_young_words(NULL),
6561
_surviving_words_length(young_cset_length + 1),
6662
_old_gen_is_full(false),
67-
_partial_objarray_chunk_size(ParGCArrayScanChunk),
68-
_partial_array_stepper(n_workers),
6963
_num_optional_regions(optional_cset_length),
7064
_numa(g1h->numa()),
7165
_obj_alloc_stat(NULL)
@@ -205,59 +199,48 @@ void G1ParScanThreadState::do_partial_array(PartialArrayScanTask task) {
205199

206200
assert(_g1h->is_in_reserved(from_obj), "must be in heap.");
207201
assert(from_obj->is_objArray(), "must be obj array");
208-
assert(from_obj->is_forwarded(), "must be forwarded");
202+
objArrayOop from_obj_array = objArrayOop(from_obj);
203+
// The from-space object contains the real length.
204+
int length = from_obj_array->length();
209205

210-
oop to_obj = from_obj->forwardee();
206+
assert(from_obj->is_forwarded(), "must be forwarded");
207+
oop to_obj = from_obj->forwardee();
211208
assert(from_obj != to_obj, "should not be chunking self-forwarded objects");
212-
assert(to_obj->is_objArray(), "must be obj array");
213-
objArrayOop to_array = objArrayOop(to_obj);
214-
215-
PartialArrayTaskStepper::Step step
216-
= _partial_array_stepper.next(objArrayOop(from_obj),
217-
to_array,
218-
_partial_objarray_chunk_size);
219-
for (uint i = 0; i < step._ncreate; ++i) {
209+
objArrayOop to_obj_array = objArrayOop(to_obj);
210+
// We keep track of the next start index in the length field of the
211+
// to-space object.
212+
int next_index = to_obj_array->length();
213+
assert(0 <= next_index && next_index < length,
214+
"invariant, next index: %d, length: %d", next_index, length);
215+
216+
int start = next_index;
217+
int end = length;
218+
int remainder = end - start;
219+
// We'll try not to push a range that's smaller than ParGCArrayScanChunk.
220+
if (remainder > 2 * ParGCArrayScanChunk) {
221+
end = start + ParGCArrayScanChunk;
222+
to_obj_array->set_length(end);
223+
// Push the remainder before we process the range in case another
224+
// worker has run out of things to do and can steal it.
220225
push_on_queue(ScannerTask(PartialArrayScanTask(from_obj)));
226+
} else {
227+
assert(length == end, "sanity");
228+
// We'll process the final range for this object. Restore the length
229+
// so that the heap remains parsable in case of evacuation failure.
230+
to_obj_array->set_length(end);
221231
}
222232

223-
HeapRegion* hr = _g1h->heap_region_containing(to_array);
233+
HeapRegion* hr = _g1h->heap_region_containing(to_obj);
224234
G1ScanInYoungSetter x(&_scanner, hr->is_young());
225-
// Process claimed task. The length of to_array is not correct, but
226-
// fortunately the iteration ignores the length field and just relies
227-
// on start/end.
228-
to_array->oop_iterate_range(&_scanner,
229-
step._index,
230-
step._index + _partial_objarray_chunk_size);
231-
}
232-
233-
void G1ParScanThreadState::start_partial_objarray(G1HeapRegionAttr dest_attr,
234-
oop from_obj,
235-
oop to_obj) {
236-
assert(from_obj->is_objArray(), "precondition");
237-
assert(from_obj->is_forwarded(), "precondition");
238-
assert(from_obj->forwardee() == to_obj, "precondition");
239-
assert(from_obj != to_obj, "should not be scanning self-forwarded objects");
240-
assert(to_obj->is_objArray(), "precondition");
241-
242-
objArrayOop to_array = objArrayOop(to_obj);
243-
244-
PartialArrayTaskStepper::Step step
245-
= _partial_array_stepper.start(objArrayOop(from_obj),
246-
to_array,
247-
_partial_objarray_chunk_size);
248-
249-
// Push any needed partial scan tasks. Pushed before processing the
250-
// intitial chunk to allow other workers to steal while we're processing.
251-
for (uint i = 0; i < step._ncreate; ++i) {
252-
push_on_queue(ScannerTask(PartialArrayScanTask(from_obj)));
253-
}
254-
255-
G1ScanInYoungSetter x(&_scanner, dest_attr.is_young());
256-
// Process the initial chunk. No need to process the type in the
257-
// klass, as it will already be handled by processing the built-in
258-
// module. The length of to_array is not correct, but fortunately
259-
// the iteration ignores that length field and relies on start/end.
260-
to_array->oop_iterate_range(&_scanner, 0, step._index);
235+
// Process indexes [start,end). It will also process the header
236+
// along with the first chunk (i.e., the chunk with start == 0).
237+
// Note that at this point the length field of to_obj_array is not
238+
// correct given that we are using it to keep track of the next
239+
// start index. oop_iterate_range() (thankfully!) ignores the length
240+
// field and only relies on the start / end parameters. It does
241+
// however return the size of the object which will be incorrect. So
242+
// we have to ignore it even if we wanted to use it.
243+
to_obj_array->oop_iterate_range(&_scanner, start, end);
261244
}
262245

263246
void G1ParScanThreadState::dispatch_task(ScannerTask task) {
@@ -411,10 +394,7 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio
411394
assert(region_attr.is_in_cset(),
412395
"Unexpected region attr type: %s", region_attr.get_type_str());
413396

414-
// Get the klass once. We'll need it again later, and this avoids
415-
// re-decoding when it's compressed.
416-
Klass* klass = old->klass();
417-
const size_t word_sz = old->size_given_klass(klass);
397+
const size_t word_sz = old->size();
418398

419399
uint age = 0;
420400
G1HeapRegionAttr dest_attr = next_region_attr(region_attr, old_mark, age);
@@ -481,20 +461,6 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio
481461
obj->set_mark_raw(old_mark);
482462
}
483463

484-
// Most objects are not arrays, so do one array check rather than
485-
// checking for each array category for each object.
486-
if (klass->is_array_klass()) {
487-
if (klass->is_objArray_klass()) {
488-
start_partial_objarray(dest_attr, old, obj);
489-
} else {
490-
// Nothing needs to be done for typeArrays. Body doesn't contain
491-
// any oops to scan, and the type in the klass will already be handled
492-
// by processing the built-in module.
493-
assert(klass->is_typeArray_klass(), "invariant");
494-
}
495-
return obj;
496-
}
497-
498464
if (G1StringDedup::is_enabled()) {
499465
const bool is_from_young = region_attr.is_young();
500466
const bool is_to_young = dest_attr.is_young();
@@ -508,10 +474,17 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio
508474
obj);
509475
}
510476

511-
G1ScanInYoungSetter x(&_scanner, dest_attr.is_young());
512-
obj->oop_iterate_backwards(&_scanner);
477+
if (obj->is_objArray() && arrayOop(obj)->length() >= ParGCArrayScanChunk) {
478+
// We keep track of the next start index in the length field of
479+
// the to-space object. The actual length can be found in the
480+
// length field of the from-space object.
481+
arrayOop(obj)->set_length(0);
482+
do_partial_array(PartialArrayScanTask(old));
483+
} else {
484+
G1ScanInYoungSetter x(&_scanner, dest_attr.is_young());
485+
obj->oop_iterate_backwards(&_scanner);
486+
}
513487
return obj;
514-
515488
} else {
516489
_plab_allocator->undo_allocation(dest_attr, obj_ptr, word_sz, node_index);
517490
return forward_ptr;
@@ -530,9 +503,7 @@ G1ParScanThreadState* G1ParScanThreadStateSet::state_for_worker(uint worker_id)
530503
assert(worker_id < _n_workers, "out of bounds access");
531504
if (_states[worker_id] == NULL) {
532505
_states[worker_id] =
533-
new G1ParScanThreadState(_g1h, _rdcqs,
534-
worker_id, _n_workers,
535-
_young_cset_length, _optional_cset_length);
506+
new G1ParScanThreadState(_g1h, _rdcqs, worker_id, _young_cset_length, _optional_cset_length);
536507
}
537508
return _states[worker_id];
538509
}

‎src/hotspot/share/gc/g1/g1ParScanThreadState.hpp

+3-6
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,6 @@
3232
#include "gc/g1/g1RemSet.hpp"
3333
#include "gc/g1/heapRegionRemSet.hpp"
3434
#include "gc/shared/ageTable.hpp"
35-
#include "gc/shared/partialArrayTaskStepper.hpp"
3635
#include "gc/shared/taskqueue.hpp"
3736
#include "memory/allocation.hpp"
3837
#include "oops/oop.hpp"
@@ -80,9 +79,6 @@ class G1ParScanThreadState : public CHeapObj<mtGC> {
8079
// Indicates whether in the last generation (old) there is no more space
8180
// available for allocation.
8281
bool _old_gen_is_full;
83-
// Size (in elements) of a partial objArray task chunk.
84-
int _partial_objarray_chunk_size;
85-
PartialArrayTaskStepper _partial_array_stepper;
8682

8783
G1RedirtyCardsQueue& redirty_cards_queue() { return _rdcq; }
8884
G1CardTable* ct() { return _ct; }
@@ -109,7 +105,6 @@ class G1ParScanThreadState : public CHeapObj<mtGC> {
109105
G1ParScanThreadState(G1CollectedHeap* g1h,
110106
G1RedirtyCardsQueueSet* rdcqs,
111107
uint worker_id,
112-
uint n_workers,
113108
size_t young_cset_length,
114109
size_t optional_cset_length);
115110
virtual ~G1ParScanThreadState();
@@ -162,7 +157,6 @@ class G1ParScanThreadState : public CHeapObj<mtGC> {
162157

163158
private:
164159
inline void do_partial_array(PartialArrayScanTask task);
165-
inline void start_partial_objarray(G1HeapRegionAttr dest_dir, oop from, oop to);
166160

167161
HeapWord* allocate_copy_slow(G1HeapRegionAttr* dest_attr,
168162
oop old,
@@ -255,6 +249,9 @@ class G1ParScanThreadStateSet : public StackObj {
255249
G1ParScanThreadState* state_for_worker(uint worker_id);
256250

257251
const size_t* surviving_young_words() const;
252+
253+
private:
254+
G1ParScanThreadState* new_par_scan_state(uint worker_id, size_t young_cset_length);
258255
};
259256

260257
#endif // SHARE_GC_G1_G1PARSCANTHREADSTATE_HPP

‎src/hotspot/share/gc/shared/partialArrayTaskStepper.cpp

-53
This file was deleted.

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

-82
This file was deleted.

0 commit comments

Comments
 (0)
Please sign in to comment.