Skip to content

Commit 316d52c

Browse files
committedJan 28, 2021
8260497: Shenandoah: Improve SATB flushing
Reviewed-by: shade, zgu
1 parent 11a70d1 commit 316d52c

10 files changed

+35
-64
lines changed
 

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,7 @@ class SATBMarkQueueSet: public PtrQueueSet {
111111

112112
// Return true if the queue's buffer should be enqueued, even if not full.
113113
// The default method uses the buffer enqueue threshold.
114-
virtual bool should_enqueue_buffer(SATBMarkQueue& queue);
114+
bool should_enqueue_buffer(SATBMarkQueue& queue);
115115

116116
template<typename Filter>
117117
void apply_filter(Filter filter, SATBMarkQueue& queue);

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

+31-2
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
#include "gc/shared/weakProcessor.inline.hpp"
3232
#include "gc/shared/gcTimer.hpp"
3333
#include "gc/shared/gcTrace.hpp"
34+
#include "gc/shared/satbMarkQueue.hpp"
3435
#include "gc/shared/strongRootsScope.hpp"
3536

3637
#include "gc/shenandoah/shenandoahBarrierSet.inline.hpp"
@@ -232,18 +233,46 @@ void ShenandoahConcurrentMark::mark_concurrent_roots() {
232233
workers->run_task(&task);
233234
}
234235

236+
class ShenandoahFlushSATBHandshakeClosure : public HandshakeClosure {
237+
private:
238+
SATBMarkQueueSet& _qset;
239+
public:
240+
ShenandoahFlushSATBHandshakeClosure(SATBMarkQueueSet& qset) :
241+
HandshakeClosure("Shenandoah Flush SATB Handshake"),
242+
_qset(qset) {}
243+
244+
void do_thread(Thread* thread) {
245+
_qset.flush_queue(ShenandoahThreadLocalData::satb_mark_queue(thread));
246+
}
247+
};
248+
235249
void ShenandoahConcurrentMark::concurrent_mark() {
236250
ShenandoahHeap* const heap = ShenandoahHeap::heap();
237251
WorkGang* workers = heap->workers();
238252
uint nworkers = workers->active_workers();
239253
task_queues()->reserve(nworkers);
240254

241-
{
255+
ShenandoahSATBMarkQueueSet& qset = ShenandoahBarrierSet::satb_mark_queue_set();
256+
ShenandoahFlushSATBHandshakeClosure flush_satb(qset);
257+
for (uint flushes = 0; flushes < ShenandoahMaxSATBBufferFlushes; flushes++) {
242258
TaskTerminator terminator(nworkers, task_queues());
243259
ShenandoahConcurrentMarkingTask task(this, &terminator);
244260
workers->run_task(&task);
245-
}
246261

262+
if (heap->cancelled_gc()) {
263+
// GC is cancelled, break out.
264+
break;
265+
}
266+
267+
size_t before = qset.completed_buffers_num();
268+
Handshake::execute(&flush_satb);
269+
size_t after = qset.completed_buffers_num();
270+
271+
if (before == after) {
272+
// No more retries needed, break out.
273+
break;
274+
}
275+
}
247276
assert(task_queues()->is_empty() || heap->cancelled_gc(), "Should be empty when not cancelled");
248277
}
249278

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

-5
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,6 @@ ShenandoahControlThread::ShenandoahControlThread() :
5656
reset_gc_id();
5757
create_and_start();
5858
_periodic_task.enroll();
59-
_periodic_satb_flush_task.enroll();
6059
if (ShenandoahPacing) {
6160
_periodic_pacer_notify_task.enroll();
6261
}
@@ -71,10 +70,6 @@ void ShenandoahPeriodicTask::task() {
7170
_thread->handle_counters_update();
7271
}
7372

74-
void ShenandoahPeriodicSATBFlushTask::task() {
75-
ShenandoahHeap::heap()->force_satb_flush_all_threads();
76-
}
77-
7873
void ShenandoahPeriodicPacerNotify::task() {
7974
assert(ShenandoahPacing, "Should not be here otherwise");
8075
ShenandoahHeap::heap()->pacer()->notify_waiters();

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

-8
Original file line numberDiff line numberDiff line change
@@ -46,13 +46,6 @@ class ShenandoahPeriodicTask : public PeriodicTask {
4646
virtual void task();
4747
};
4848

49-
// Periodic task to flush SATB buffers periodically.
50-
class ShenandoahPeriodicSATBFlushTask : public PeriodicTask {
51-
public:
52-
ShenandoahPeriodicSATBFlushTask() : PeriodicTask(ShenandoahSATBBufferFlushInterval) {}
53-
virtual void task();
54-
};
55-
5649
// Periodic task to notify blocked paced waiters.
5750
class ShenandoahPeriodicPacerNotify : public PeriodicTask {
5851
public:
@@ -77,7 +70,6 @@ class ShenandoahControlThread: public ConcurrentGCThread {
7770
Monitor _alloc_failure_waiters_lock;
7871
Monitor _gc_waiters_lock;
7972
ShenandoahPeriodicTask _periodic_task;
80-
ShenandoahPeriodicSATBFlushTask _periodic_satb_flush_task;
8173
ShenandoahPeriodicPacerNotify _periodic_pacer_notify_task;
8274

8375
public:

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

-14
Original file line numberDiff line numberDiff line change
@@ -1684,20 +1684,6 @@ void ShenandoahHeap::prepare_update_heap_references(bool concurrent) {
16841684
_update_refs_iterator.reset();
16851685
}
16861686

1687-
void ShenandoahHeap::force_satb_flush_all_threads() {
1688-
if (!is_concurrent_mark_in_progress()) {
1689-
// No need to flush SATBs
1690-
return;
1691-
}
1692-
1693-
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
1694-
ShenandoahThreadLocalData::set_force_satb_flush(t, true);
1695-
}
1696-
// The threads are not "acquiring" their thread-local data, but it does not
1697-
// hurt to "release" the updates here anyway.
1698-
OrderAccess::fence();
1699-
}
1700-
17011687
void ShenandoahHeap::set_gc_state_all_threads(char state) {
17021688
for (JavaThreadIteratorWithHandle jtiwh; JavaThread *t = jtiwh.next(); ) {
17031689
ShenandoahThreadLocalData::set_gc_state(t, state);

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

-1
Original file line numberDiff line numberDiff line change
@@ -580,7 +580,6 @@ class ShenandoahHeap : public CollectedHeap {
580580

581581
// SATB barriers hooks
582582
inline bool requires_marking(const void* entry) const;
583-
void force_satb_flush_all_threads();
584583

585584
// Support for bitmap uncommits
586585
bool commit_bitmap_slice(ShenandoahHeapRegion *r);

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

-17
Original file line numberDiff line numberDiff line change
@@ -53,20 +53,3 @@ void ShenandoahSATBMarkQueueSet::filter(SATBMarkQueue& queue) {
5353
ShenandoahHeap* heap = ShenandoahHeap::heap();
5454
apply_filter(ShenandoahSATBMarkQueueFilterFn(heap), queue);
5555
}
56-
57-
bool ShenandoahSATBMarkQueueSet::should_enqueue_buffer(SATBMarkQueue& queue) {
58-
if (SATBMarkQueueSet::should_enqueue_buffer(queue)) {
59-
return true;
60-
} else if (queue.index() < buffer_size()) { // Is buffer not empty?
61-
Thread* t = Thread::current();
62-
if (ShenandoahThreadLocalData::is_force_satb_flush(t)) {
63-
// Non-empty buffer is compacted, and we decided not to enqueue it.
64-
// We still want to know about leftover work in that buffer eventually.
65-
// This avoid dealing with these leftovers during the final-mark, after
66-
// the buffers are drained completely. See JDK-8205353 for more discussion.
67-
ShenandoahThreadLocalData::set_force_satb_flush(t, false);
68-
return true;
69-
}
70-
}
71-
return false;
72-
}

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

-3
Original file line numberDiff line numberDiff line change
@@ -31,9 +31,6 @@
3131
#include "runtime/thread.hpp"
3232

3333
class ShenandoahSATBMarkQueueSet : public SATBMarkQueueSet {
34-
protected:
35-
virtual bool should_enqueue_buffer(SATBMarkQueue& queue);
36-
3734
public:
3835
ShenandoahSATBMarkQueueSet(BufferNode::Allocator* allocator);
3936

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

-10
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,6 @@ class ShenandoahThreadLocalData {
4848
PLAB* _gclab;
4949
size_t _gclab_size;
5050
uint _worker_id;
51-
bool _force_satb_flush;
5251
int _disarmed_value;
5352
double _paced_time;
5453

@@ -60,7 +59,6 @@ class ShenandoahThreadLocalData {
6059
_gclab(NULL),
6160
_gclab_size(0),
6261
_worker_id(INVALID_WORKER_ID),
63-
_force_satb_flush(false),
6462
_disarmed_value(0),
6563
_paced_time(0) {
6664

@@ -115,14 +113,6 @@ class ShenandoahThreadLocalData {
115113
return data(thread)->_worker_id;
116114
}
117115

118-
static void set_force_satb_flush(Thread* thread, bool v) {
119-
data(thread)->_force_satb_flush = v;
120-
}
121-
122-
static bool is_force_satb_flush(Thread* thread) {
123-
return data(thread)->_force_satb_flush;
124-
}
125-
126116
static void initialize_gclab(Thread* thread) {
127117
assert (thread->is_Java_thread() || thread->is_Worker_thread(), "Only Java and GC worker threads are allowed to get GCLABs");
128118
assert(data(thread)->_gclab == NULL, "Only initialize once");

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

+3-3
Original file line numberDiff line numberDiff line change
@@ -330,9 +330,9 @@
330330
"Number of entries in an SATB log buffer.") \
331331
range(1, max_uintx) \
332332
\
333-
product(uintx, ShenandoahSATBBufferFlushInterval, 100, EXPERIMENTAL, \
334-
"Forcefully flush non-empty SATB buffers at this interval. " \
335-
"Time is in milliseconds.") \
333+
product(uintx, ShenandoahMaxSATBBufferFlushes, 5, EXPERIMENTAL, \
334+
"How many times to maximum attempt to flush SATB buffers at the " \
335+
"end of concurrent marking.") \
336336
\
337337
product(bool, ShenandoahSuspendibleWorkers, false, EXPERIMENTAL, \
338338
"Suspend concurrent GC worker threads at safepoints") \

0 commit comments

Comments
 (0)
Please sign in to comment.