Skip to content

Commit 3fa6778

Browse files
committedSep 22, 2022
8292296: Use multiple threads to process ParallelGC deferred updates
Reviewed-by: tschatzl, ayang
1 parent 800e68d commit 3fa6778

File tree

4 files changed

+45
-50
lines changed

4 files changed

+45
-50
lines changed
 

‎src/hotspot/share/gc/parallel/psCompactionManager.cpp

+16-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -59,6 +59,8 @@ ParCompactionManager::ParCompactionManager() {
5959
_start_array = old_gen()->start_array();
6060

6161
reset_bitmap_query_cache();
62+
63+
_deferred_obj_array = new (ResourceObj::C_HEAP, mtGC) GrowableArray<HeapWord*>(10, mtGC);
6264
}
6365

6466
void ParCompactionManager::initialize(ParMarkBitMap* mbm) {
@@ -165,6 +167,15 @@ void ParCompactionManager::drain_region_stacks() {
165167
} while (!region_stack()->is_empty());
166168
}
167169

170+
void ParCompactionManager::drain_deferred_objects() {
171+
while (!_deferred_obj_array->is_empty()) {
172+
HeapWord* addr = _deferred_obj_array->pop();
173+
assert(addr != NULL, "expected a deferred object");
174+
PSParallelCompact::update_deferred_object(this, addr);
175+
}
176+
_deferred_obj_array->clear_and_deallocate();
177+
}
178+
168179
size_t ParCompactionManager::pop_shadow_region_mt_safe(PSParallelCompact::RegionData* region_ptr) {
169180
MonitorLocker ml(_shadow_region_monitor, Mutex::_no_safepoint_check_flag);
170181
while (true) {
@@ -195,6 +206,10 @@ void ParCompactionManager::remove_all_shadow_regions() {
195206
_shadow_region_array->clear();
196207
}
197208

209+
void ParCompactionManager::push_deferred_object(HeapWord* addr) {
210+
_deferred_obj_array->push(addr);
211+
}
212+
198213
#ifdef ASSERT
199214
void ParCompactionManager::verify_all_marking_stack_empty() {
200215
uint parallel_gc_threads = ParallelGCThreads;

‎src/hotspot/share/gc/parallel/psCompactionManager.hpp

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -75,6 +75,8 @@ class ParCompactionManager : public CHeapObj<mtGC> {
7575
// type of TaskQueue.
7676
RegionTaskQueue _region_stack;
7777

78+
GrowableArray<HeapWord*>* _deferred_obj_array;
79+
7880
static ParMarkBitMap* _mark_bitmap;
7981

8082
// Contains currently free shadow regions. We use it in
@@ -128,6 +130,8 @@ class ParCompactionManager : public CHeapObj<mtGC> {
128130
return next_shadow_region();
129131
}
130132

133+
void push_deferred_object(HeapWord* addr);
134+
131135
void reset_bitmap_query_cache() {
132136
_last_query_beg = NULL;
133137
_last_query_obj = NULL;
@@ -195,6 +199,7 @@ class ParCompactionManager : public CHeapObj<mtGC> {
195199

196200
// Process tasks remaining on any stack
197201
void drain_region_stacks();
202+
void drain_deferred_objects();
198203

199204
void follow_contents(oop obj);
200205
void follow_array(objArrayOop array, int index);

‎src/hotspot/share/gc/parallel/psParallelCompact.cpp

+19-36
Original file line numberDiff line numberDiff line change
@@ -2427,6 +2427,10 @@ class UpdateDensePrefixAndCompactionTask: public WorkerTask {
24272427
// Once a thread has drained it's stack, it should try to steal regions from
24282428
// other threads.
24292429
compaction_with_stealing_work(&_terminator, worker_id);
2430+
2431+
// At this point all regions have been compacted, so it's now safe
2432+
// to update the deferred objects that cross region boundaries.
2433+
cm->drain_deferred_objects();
24302434
}
24312435
};
24322436

@@ -2456,22 +2460,13 @@ void PSParallelCompact::compact() {
24562460
ParallelScavengeHeap::heap()->workers().run_task(&task);
24572461

24582462
#ifdef ASSERT
2459-
// Verify that all regions have been processed before the deferred updates.
2463+
// Verify that all regions have been processed.
24602464
for (unsigned int id = old_space_id; id < last_space_id; ++id) {
24612465
verify_complete(SpaceId(id));
24622466
}
24632467
#endif
24642468
}
24652469

2466-
{
2467-
GCTraceTime(Trace, gc, phases) tm("Deferred Updates", &_gc_timer);
2468-
// Update the deferred objects, if any.
2469-
ParCompactionManager* cm = ParCompactionManager::get_vmthread_cm();
2470-
for (unsigned int id = old_space_id; id < last_space_id; ++id) {
2471-
update_deferred_objects(cm, SpaceId(id));
2472-
}
2473-
}
2474-
24752470
DEBUG_ONLY(write_block_fill_histogram());
24762471
}
24772472

@@ -2598,32 +2593,22 @@ PSParallelCompact::SpaceId PSParallelCompact::space_id(HeapWord* addr) {
25982593
return last_space_id;
25992594
}
26002595

2601-
void PSParallelCompact::update_deferred_objects(ParCompactionManager* cm,
2602-
SpaceId id) {
2603-
assert(id < last_space_id, "bad space id");
2604-
2596+
void PSParallelCompact::update_deferred_object(ParCompactionManager* cm, HeapWord *addr) {
2597+
#ifdef ASSERT
26052598
ParallelCompactData& sd = summary_data();
2606-
const SpaceInfo* const space_info = _space_info + id;
2607-
ObjectStartArray* const start_array = space_info->start_array();
2599+
size_t region_idx = sd.addr_to_region_idx(addr);
2600+
assert(sd.region(region_idx)->completed(), "first region must be completed before deferred updates");
2601+
assert(sd.region(region_idx + 1)->completed(), "second region must be completed before deferred updates");
2602+
#endif
26082603

2609-
const MutableSpace* const space = space_info->space();
2610-
assert(space_info->dense_prefix() >= space->bottom(), "dense_prefix not set");
2611-
HeapWord* const beg_addr = space_info->dense_prefix();
2612-
HeapWord* const end_addr = sd.region_align_up(space_info->new_top());
2613-
2614-
const RegionData* const beg_region = sd.addr_to_region_ptr(beg_addr);
2615-
const RegionData* const end_region = sd.addr_to_region_ptr(end_addr);
2616-
const RegionData* cur_region;
2617-
for (cur_region = beg_region; cur_region < end_region; ++cur_region) {
2618-
HeapWord* const addr = cur_region->deferred_obj_addr();
2619-
if (addr != NULL) {
2620-
if (start_array != NULL) {
2621-
start_array->allocate_block(addr);
2622-
}
2623-
cm->update_contents(cast_to_oop(addr));
2624-
assert(oopDesc::is_oop_or_null(cast_to_oop(addr)), "Expected an oop or NULL at " PTR_FORMAT, p2i(cast_to_oop(addr)));
2625-
}
2604+
const SpaceInfo* const space_info = _space_info + space_id(addr);
2605+
ObjectStartArray* const start_array = space_info->start_array();
2606+
if (start_array != NULL) {
2607+
start_array->allocate_block(addr);
26262608
}
2609+
2610+
cm->update_contents(cast_to_oop(addr));
2611+
assert(oopDesc::is_oop(cast_to_oop(addr)), "Expected an oop at " PTR_FORMAT, p2i(cast_to_oop(addr)));
26272612
}
26282613

26292614
// Skip over count live words starting from beg, and return the address of the
@@ -2870,7 +2855,6 @@ void PSParallelCompact::fill_region(ParCompactionManager* cm, MoveAndUpdateClosu
28702855
if (closure.is_full()) {
28712856
decrement_destination_counts(cm, src_space_id, src_region_idx,
28722857
closure.source());
2873-
region_ptr->set_deferred_obj_addr(NULL);
28742858
closure.complete_region(cm, dest_addr, region_ptr);
28752859
return;
28762860
}
@@ -2915,7 +2899,7 @@ void PSParallelCompact::fill_region(ParCompactionManager* cm, MoveAndUpdateClosu
29152899
if (status == ParMarkBitMap::would_overflow) {
29162900
// The last object did not fit. Note that interior oop updates were
29172901
// deferred, then copy enough of the object to fill the region.
2918-
region_ptr->set_deferred_obj_addr(closure.destination());
2902+
cm->push_deferred_object(closure.destination());
29192903
status = closure.copy_until_full(); // copies from closure.source()
29202904

29212905
decrement_destination_counts(cm, src_space_id, src_region_idx,
@@ -2927,7 +2911,6 @@ void PSParallelCompact::fill_region(ParCompactionManager* cm, MoveAndUpdateClosu
29272911
if (status == ParMarkBitMap::full) {
29282912
decrement_destination_counts(cm, src_space_id, src_region_idx,
29292913
closure.source());
2930-
region_ptr->set_deferred_obj_addr(NULL);
29312914
closure.complete_region(cm, dest_addr, region_ptr);
29322915
return;
29332916
}

‎src/hotspot/share/gc/parallel/psParallelCompact.hpp

+4-12
Original file line numberDiff line numberDiff line change
@@ -243,13 +243,6 @@ class ParallelCompactData
243243
// Reuse _source_region to store the corresponding shadow region index
244244
size_t shadow_region() const { return _source_region; }
245245

246-
// The object (if any) starting in this region and ending in a different
247-
// region that could not be updated during the main (parallel) compaction
248-
// phase. This is different from _partial_obj_addr, which is an object that
249-
// extends onto a source region. However, the two uses do not overlap in
250-
// time, so the same field is used to save space.
251-
HeapWord* deferred_obj_addr() const { return _partial_obj_addr; }
252-
253246
// The starting address of the partial object extending onto the region.
254247
HeapWord* partial_obj_addr() const { return _partial_obj_addr; }
255248

@@ -312,7 +305,6 @@ class ParallelCompactData
312305
void set_destination(HeapWord* addr) { _destination = addr; }
313306
void set_source_region(size_t region) { _source_region = region; }
314307
void set_shadow_region(size_t region) { _source_region = region; }
315-
void set_deferred_obj_addr(HeapWord* addr) { _partial_obj_addr = addr; }
316308
void set_partial_obj_addr(HeapWord* addr) { _partial_obj_addr = addr; }
317309
void set_partial_obj_size(size_t words) {
318310
_partial_obj_size = (region_sz_t) words;
@@ -948,8 +940,8 @@ inline void ParMarkBitMapClosure::decrement_words_remaining(size_t words) {
948940
// but do not have their references updated. References are not updated because
949941
// it cannot easily be determined if the klass pointer KKK for the object AAA
950942
// has been updated. KKK likely resides in a region to the left of the region
951-
// containing AAA. These AAA's have there references updated at the end in a
952-
// clean up phase. See the method PSParallelCompact::update_deferred_objects().
943+
// containing AAA. These AAA's have their references updated at the end in a
944+
// clean up phase. See the method PSParallelCompact::update_deferred_object().
953945
//
954946
// Compaction is done on a region basis. A region that is ready to be filled is
955947
// put on a ready list and GC threads take region off the list and fill them. A
@@ -1248,8 +1240,8 @@ class PSParallelCompact : AllStatic {
12481240
// Fill in the block table for the specified region.
12491241
static void fill_blocks(size_t region_idx);
12501242

1251-
// Update the deferred objects in the space.
1252-
static void update_deferred_objects(ParCompactionManager* cm, SpaceId id);
1243+
// Update a single deferred object.
1244+
static void update_deferred_object(ParCompactionManager* cm, HeapWord* addr);
12531245

12541246
static ParMarkBitMap* mark_bitmap() { return &_mark_bitmap; }
12551247
static ParallelCompactData& summary_data() { return _summary_data; }

0 commit comments

Comments
 (0)
Please sign in to comment.