Skip to content

Commit a793293

Browse files
committedJun 29, 2020
8247845: Shenandoah: refactor TLAB/GCLAB retirement code
Reviewed-by: rkennke
1 parent 5ad963c commit a793293

File tree

5 files changed

+91
-58
lines changed

5 files changed

+91
-58
lines changed
 

‎src/hotspot/share/gc/shenandoah/shenandoahHeap.cpp

+76-47
Original file line numberDiff line numberDiff line change
@@ -1044,28 +1044,79 @@ void ShenandoahHeap::trash_humongous_region_at(ShenandoahHeapRegion* start) {
10441044
}
10451045
}
10461046

1047+
class ShenandoahCheckCleanGCLABClosure : public ThreadClosure {
1048+
public:
1049+
ShenandoahCheckCleanGCLABClosure() {}
1050+
void do_thread(Thread* thread) {
1051+
PLAB* gclab = ShenandoahThreadLocalData::gclab(thread);
1052+
assert(gclab != NULL, "GCLAB should be initialized for %s", thread->name());
1053+
assert(gclab->words_remaining() == 0, "GCLAB should not need retirement");
1054+
}
1055+
};
1056+
10471057
class ShenandoahRetireGCLABClosure : public ThreadClosure {
1058+
private:
1059+
bool const _resize;
10481060
public:
1061+
ShenandoahRetireGCLABClosure(bool resize) : _resize(resize) {}
10491062
void do_thread(Thread* thread) {
10501063
PLAB* gclab = ShenandoahThreadLocalData::gclab(thread);
10511064
assert(gclab != NULL, "GCLAB should be initialized for %s", thread->name());
10521065
gclab->retire();
1066+
if (_resize && ShenandoahThreadLocalData::gclab_size(thread) > 0) {
1067+
ShenandoahThreadLocalData::set_gclab_size(thread, 0);
1068+
}
10531069
}
10541070
};
10551071

1056-
void ShenandoahHeap::make_parsable(bool retire_tlabs) {
1057-
if (UseTLAB) {
1058-
CollectedHeap::ensure_parsability(retire_tlabs);
1072+
void ShenandoahHeap::labs_make_parsable() {
1073+
assert(UseTLAB, "Only call with UseTLAB");
1074+
1075+
ShenandoahRetireGCLABClosure cl(false);
1076+
1077+
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
1078+
ThreadLocalAllocBuffer& tlab = t->tlab();
1079+
tlab.make_parsable();
1080+
cl.do_thread(t);
10591081
}
1060-
ShenandoahRetireGCLABClosure cl;
1082+
1083+
workers()->threads_do(&cl);
1084+
}
1085+
1086+
void ShenandoahHeap::tlabs_retire(bool resize) {
1087+
assert(UseTLAB, "Only call with UseTLAB");
1088+
assert(!resize || ResizeTLAB, "Only call for resize when ResizeTLAB is enabled");
1089+
1090+
ThreadLocalAllocStats stats;
1091+
1092+
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
1093+
ThreadLocalAllocBuffer& tlab = t->tlab();
1094+
tlab.retire(&stats);
1095+
if (resize) {
1096+
tlab.resize();
1097+
}
1098+
}
1099+
1100+
stats.publish();
1101+
1102+
#ifdef ASSERT
1103+
ShenandoahCheckCleanGCLABClosure cl;
10611104
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
10621105
cl.do_thread(t);
10631106
}
10641107
workers()->threads_do(&cl);
1108+
#endif
10651109
}
10661110

1067-
void ShenandoahHeap::resize_tlabs() {
1068-
CollectedHeap::resize_all_tlabs();
1111+
void ShenandoahHeap::gclabs_retire(bool resize) {
1112+
assert(UseTLAB, "Only call with UseTLAB");
1113+
assert(!resize || ResizeTLAB, "Only call for resize when ResizeTLAB is enabled");
1114+
1115+
ShenandoahRetireGCLABClosure cl(resize);
1116+
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
1117+
cl.do_thread(t);
1118+
}
1119+
workers()->threads_do(&cl);
10691120
}
10701121

10711122
class ShenandoahEvacuateUpdateRootsTask : public AbstractGangTask {
@@ -1122,25 +1173,6 @@ size_t ShenandoahHeap::max_tlab_size() const {
11221173
return ShenandoahHeapRegion::max_tlab_size_words();
11231174
}
11241175

1125-
class ShenandoahRetireAndResetGCLABClosure : public ThreadClosure {
1126-
public:
1127-
void do_thread(Thread* thread) {
1128-
PLAB* gclab = ShenandoahThreadLocalData::gclab(thread);
1129-
gclab->retire();
1130-
if (ShenandoahThreadLocalData::gclab_size(thread) > 0) {
1131-
ShenandoahThreadLocalData::set_gclab_size(thread, 0);
1132-
}
1133-
}
1134-
};
1135-
1136-
void ShenandoahHeap::retire_and_reset_gclabs() {
1137-
ShenandoahRetireAndResetGCLABClosure cl;
1138-
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
1139-
cl.do_thread(t);
1140-
}
1141-
workers()->threads_do(&cl);
1142-
}
1143-
11441176
void ShenandoahHeap::collect(GCCause::Cause cause) {
11451177
control_thread()->request_gc(cause);
11461178
}
@@ -1173,8 +1205,8 @@ jlong ShenandoahHeap::millis_since_last_gc() {
11731205
}
11741206

11751207
void ShenandoahHeap::prepare_for_verify() {
1176-
if (SafepointSynchronize::is_at_safepoint() || ! UseTLAB) {
1177-
make_parsable(false);
1208+
if (SafepointSynchronize::is_at_safepoint() && UseTLAB) {
1209+
labs_make_parsable();
11781210
}
11791211
}
11801212

@@ -1264,7 +1296,7 @@ class ObjectIterateScanRootClosure : public BasicOopIterateClosure {
12641296
* This is public API, used in preparation of object_iterate().
12651297
* Since we don't do linear scan of heap in object_iterate() (see comment below), we don't
12661298
* need to make the heap parsable. For Shenandoah-internal linear heap scans that we can
1267-
* control, we call SH::make_tlabs_parsable().
1299+
* control, we call SH::tlabs_retire, SH::gclabs_retire.
12681300
*/
12691301
void ShenandoahHeap::ensure_parsability(bool retire_tlabs) {
12701302
// No-op.
@@ -1420,10 +1452,13 @@ void ShenandoahHeap::op_init_mark() {
14201452
}
14211453

14221454
set_concurrent_mark_in_progress(true);
1423-
// We need to reset all TLABs because we'd lose marks on all objects allocated in them.
1424-
{
1425-
ShenandoahGCPhase phase(ShenandoahPhaseTimings::make_parsable);
1426-
make_parsable(true);
1455+
1456+
// We need to reset all TLABs because they might be below the TAMS, and we need to mark
1457+
// the objects in them. Do not let mutators allocate any new objects in their current TLABs.
1458+
// It is also a good place to resize the TLAB sizes for future allocations.
1459+
if (UseTLAB) {
1460+
ShenandoahGCPhase phase(ShenandoahPhaseTimings::init_manage_tlabs);
1461+
tlabs_retire(ResizeTLAB);
14271462
}
14281463

14291464
{
@@ -1437,11 +1472,6 @@ void ShenandoahHeap::op_init_mark() {
14371472

14381473
concurrent_mark()->mark_roots(ShenandoahPhaseTimings::scan_roots);
14391474

1440-
if (UseTLAB) {
1441-
ShenandoahGCPhase phase(ShenandoahPhaseTimings::resize_tlabs);
1442-
resize_tlabs();
1443-
}
1444-
14451475
if (ShenandoahPacing) {
14461476
pacer()->setup_for_mark();
14471477
}
@@ -1539,9 +1569,9 @@ void ShenandoahHeap::op_final_mark() {
15391569
// which would be outside the collection set, so no cset writes would happen there.
15401570
// Weaker one: new allocations would happen past update watermark, and so less work would
15411571
// be needed for reference updates (would update the large filler instead).
1542-
{
1543-
ShenandoahGCPhase phase(ShenandoahPhaseTimings::retire_tlabs);
1544-
make_parsable(true);
1572+
if (UseTLAB) {
1573+
ShenandoahGCPhase phase(ShenandoahPhaseTimings::final_manage_labs);
1574+
tlabs_retire(false);
15451575
}
15461576

15471577
{
@@ -1917,10 +1947,6 @@ void ShenandoahHeap::op_full(GCCause::Cause cause) {
19171947
metrics.snap_before();
19181948

19191949
full_gc()->do_it(cause);
1920-
if (UseTLAB) {
1921-
ShenandoahGCPhase phase(ShenandoahPhaseTimings::full_gc_resize_tlabs);
1922-
resize_all_tlabs();
1923-
}
19241950

19251951
metrics.snap_after();
19261952

@@ -2511,9 +2537,12 @@ void ShenandoahHeap::op_init_updaterefs() {
25112537

25122538
set_evacuation_in_progress(false);
25132539

2514-
{
2515-
ShenandoahGCPhase phase(ShenandoahPhaseTimings::init_update_refs_retire_gclabs);
2516-
retire_and_reset_gclabs();
2540+
// Evacuation is over, no GCLABs are needed anymore. GCLABs are under URWM, so we need to
2541+
// make them parsable for update code to work correctly. Plus, we can compute new sizes
2542+
// for future GCLABs here.
2543+
if (UseTLAB) {
2544+
ShenandoahGCPhase phase(ShenandoahPhaseTimings::init_update_refs_manage_gclabs);
2545+
gclabs_retire(true);
25172546
}
25182547

25192548
if (ShenandoahVerify) {

‎src/hotspot/share/gc/shenandoah/shenandoahHeap.hpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -595,7 +595,6 @@ class ShenandoahHeap : public CollectedHeap {
595595
inline HeapWord* allocate_from_gclab(Thread* thread, size_t size);
596596
HeapWord* allocate_from_gclab_slow(Thread* thread, size_t size);
597597
HeapWord* allocate_new_gclab(size_t min_size, size_t word_size, size_t* actual_size);
598-
void retire_and_reset_gclabs();
599598

600599
public:
601600
HeapWord* allocate_memory(ShenandoahAllocRequest& request);
@@ -615,10 +614,11 @@ class ShenandoahHeap : public CollectedHeap {
615614
size_t max_tlab_size() const;
616615
size_t tlab_used(Thread* ignored) const;
617616

618-
void resize_tlabs();
617+
void ensure_parsability(bool retire_labs);
619618

620-
void ensure_parsability(bool retire_tlabs);
621-
void make_parsable(bool retire_tlabs);
619+
void labs_make_parsable();
620+
void tlabs_retire(bool resize);
621+
void gclabs_retire(bool resize);
622622

623623
// ---------- Marking support
624624
//

‎src/hotspot/share/gc/shenandoah/shenandoahMarkCompact.cpp

+4-1
Original file line numberDiff line numberDiff line change
@@ -145,7 +145,10 @@ void ShenandoahMarkCompact::do_it(GCCause::Cause gc_cause) {
145145
_preserved_marks->init(heap->workers()->active_workers());
146146
}
147147

148-
heap->make_parsable(true);
148+
if (UseTLAB) {
149+
heap->gclabs_retire(true);
150+
heap->tlabs_retire(true);
151+
}
149152

150153
OrderAccess::fence();
151154

‎src/hotspot/share/gc/shenandoah/shenandoahPhaseTimings.hpp

+3-5
Original file line numberDiff line numberDiff line change
@@ -60,11 +60,10 @@ class outputStream;
6060
\
6161
f(init_mark_gross, "Pause Init Mark (G)") \
6262
f(init_mark, "Pause Init Mark (N)") \
63-
f(make_parsable, " Make Parsable") \
63+
f(init_manage_tlabs, " Manage TLABs") \
6464
f(init_update_region_states, " Update Region States") \
6565
f(scan_roots, " Scan Roots") \
6666
SHENANDOAH_PAR_PHASE_DO(scan_, " S: ", f) \
67-
f(resize_tlabs, " Resize TLABs") \
6867
\
6968
f(conc_mark, "Concurrent Marking") \
7069
f(conc_mark_roots, " Roots ") \
@@ -86,7 +85,7 @@ class outputStream;
8685
SHENANDOAH_PAR_PHASE_DO(purge_weak_par_, " WR: ", f) \
8786
f(purge_cldg, " CLDG") \
8887
f(final_update_region_states, " Update Region States") \
89-
f(retire_tlabs, " Retire TLABs") \
88+
f(final_manage_labs, " Manage GC/TLABs") \
9089
f(choose_cset, " Choose Collection Set") \
9190
f(final_rebuild_freeset, " Rebuild Free Set") \
9291
f(init_evac, " Initial Evacuation") \
@@ -113,7 +112,7 @@ class outputStream;
113112
\
114113
f(init_update_refs_gross, "Pause Init Update Refs (G)") \
115114
f(init_update_refs, "Pause Init Update Refs (N)") \
116-
f(init_update_refs_retire_gclabs, " Retire GCLABs") \
115+
f(init_update_refs_manage_gclabs, " Manage GCLABs") \
117116
\
118117
f(conc_update_refs, "Concurrent Update Refs") \
119118
\
@@ -166,7 +165,6 @@ class outputStream;
166165
f(full_gc_copy_objects_humong, " Humongous Objects") \
167166
f(full_gc_copy_objects_reset_complete, " Reset Complete Bitmap") \
168167
f(full_gc_copy_objects_rebuild, " Rebuild Region Sets") \
169-
f(full_gc_resize_tlabs, " Resize TLABs") \
170168
f(full_gc_heapdump_post, " Post Heap Dump") \
171169
\
172170
f(conc_uncommit, "Concurrent Uncommit") \

‎src/hotspot/share/gc/shenandoah/shenandoahVerifier.cpp

+4-1
Original file line numberDiff line numberDiff line change
@@ -690,7 +690,10 @@ void ShenandoahVerifier::verify_at_safepoint(const char *label,
690690
}
691691

692692
OrderAccess::fence();
693-
_heap->make_parsable(false);
693+
694+
if (UseTLAB) {
695+
_heap->labs_make_parsable();
696+
}
694697

695698
// Allocate temporary bitmap for storing marking wavefront:
696699
_verification_bit_map->clear();

0 commit comments

Comments
 (0)
Please sign in to comment.