@@ -1044,28 +1044,79 @@ void ShenandoahHeap::trash_humongous_region_at(ShenandoahHeapRegion* start) {
1044
1044
}
1045
1045
}
1046
1046
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
+
1047
1057
class ShenandoahRetireGCLABClosure : public ThreadClosure {
1058
+ private:
1059
+ bool const _resize;
1048
1060
public:
1061
+ ShenandoahRetireGCLABClosure (bool resize) : _resize(resize) {}
1049
1062
void do_thread (Thread* thread) {
1050
1063
PLAB* gclab = ShenandoahThreadLocalData::gclab (thread);
1051
1064
assert (gclab != NULL , " GCLAB should be initialized for %s" , thread->name ());
1052
1065
gclab->retire ();
1066
+ if (_resize && ShenandoahThreadLocalData::gclab_size (thread) > 0 ) {
1067
+ ShenandoahThreadLocalData::set_gclab_size (thread, 0 );
1068
+ }
1053
1069
}
1054
1070
};
1055
1071
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);
1059
1081
}
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;
1061
1104
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next (); ) {
1062
1105
cl.do_thread (t);
1063
1106
}
1064
1107
workers ()->threads_do (&cl);
1108
+ #endif
1065
1109
}
1066
1110
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);
1069
1120
}
1070
1121
1071
1122
class ShenandoahEvacuateUpdateRootsTask : public AbstractGangTask {
@@ -1122,25 +1173,6 @@ size_t ShenandoahHeap::max_tlab_size() const {
1122
1173
return ShenandoahHeapRegion::max_tlab_size_words ();
1123
1174
}
1124
1175
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
-
1144
1176
void ShenandoahHeap::collect (GCCause::Cause cause) {
1145
1177
control_thread ()->request_gc (cause);
1146
1178
}
@@ -1173,8 +1205,8 @@ jlong ShenandoahHeap::millis_since_last_gc() {
1173
1205
}
1174
1206
1175
1207
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 ( );
1178
1210
}
1179
1211
}
1180
1212
@@ -1264,7 +1296,7 @@ class ObjectIterateScanRootClosure : public BasicOopIterateClosure {
1264
1296
* This is public API, used in preparation of object_iterate().
1265
1297
* Since we don't do linear scan of heap in object_iterate() (see comment below), we don't
1266
1298
* 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 .
1268
1300
*/
1269
1301
void ShenandoahHeap::ensure_parsability (bool retire_tlabs) {
1270
1302
// No-op.
@@ -1420,10 +1452,13 @@ void ShenandoahHeap::op_init_mark() {
1420
1452
}
1421
1453
1422
1454
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);
1427
1462
}
1428
1463
1429
1464
{
@@ -1437,11 +1472,6 @@ void ShenandoahHeap::op_init_mark() {
1437
1472
1438
1473
concurrent_mark ()->mark_roots (ShenandoahPhaseTimings::scan_roots);
1439
1474
1440
- if (UseTLAB) {
1441
- ShenandoahGCPhase phase (ShenandoahPhaseTimings::resize_tlabs);
1442
- resize_tlabs ();
1443
- }
1444
-
1445
1475
if (ShenandoahPacing) {
1446
1476
pacer ()->setup_for_mark ();
1447
1477
}
@@ -1539,9 +1569,9 @@ void ShenandoahHeap::op_final_mark() {
1539
1569
// which would be outside the collection set, so no cset writes would happen there.
1540
1570
// Weaker one: new allocations would happen past update watermark, and so less work would
1541
1571
// 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 );
1545
1575
}
1546
1576
1547
1577
{
@@ -1917,10 +1947,6 @@ void ShenandoahHeap::op_full(GCCause::Cause cause) {
1917
1947
metrics.snap_before ();
1918
1948
1919
1949
full_gc ()->do_it (cause);
1920
- if (UseTLAB) {
1921
- ShenandoahGCPhase phase (ShenandoahPhaseTimings::full_gc_resize_tlabs);
1922
- resize_all_tlabs ();
1923
- }
1924
1950
1925
1951
metrics.snap_after ();
1926
1952
@@ -2511,9 +2537,12 @@ void ShenandoahHeap::op_init_updaterefs() {
2511
2537
2512
2538
set_evacuation_in_progress (false );
2513
2539
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 );
2517
2546
}
2518
2547
2519
2548
if (ShenandoahVerify) {
0 commit comments