|
31 | 31 | #include "gc/shared/weakProcessor.inline.hpp"
|
32 | 32 | #include "gc/shared/gcTimer.hpp"
|
33 | 33 | #include "gc/shared/gcTrace.hpp"
|
| 34 | +#include "gc/shared/satbMarkQueue.hpp" |
34 | 35 | #include "gc/shared/strongRootsScope.hpp"
|
35 | 36 |
|
36 | 37 | #include "gc/shenandoah/shenandoahBarrierSet.inline.hpp"
|
@@ -232,18 +233,46 @@ void ShenandoahConcurrentMark::mark_concurrent_roots() {
|
232 | 233 | workers->run_task(&task);
|
233 | 234 | }
|
234 | 235 |
|
| 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 | + |
235 | 249 | void ShenandoahConcurrentMark::concurrent_mark() {
|
236 | 250 | ShenandoahHeap* const heap = ShenandoahHeap::heap();
|
237 | 251 | WorkGang* workers = heap->workers();
|
238 | 252 | uint nworkers = workers->active_workers();
|
239 | 253 | task_queues()->reserve(nworkers);
|
240 | 254 |
|
241 |
| - { |
| 255 | + ShenandoahSATBMarkQueueSet& qset = ShenandoahBarrierSet::satb_mark_queue_set(); |
| 256 | + ShenandoahFlushSATBHandshakeClosure flush_satb(qset); |
| 257 | + for (uint flushes = 0; flushes < ShenandoahMaxSATBBufferFlushes; flushes++) { |
242 | 258 | TaskTerminator terminator(nworkers, task_queues());
|
243 | 259 | ShenandoahConcurrentMarkingTask task(this, &terminator);
|
244 | 260 | workers->run_task(&task);
|
245 |
| - } |
246 | 261 |
|
| 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 | + } |
247 | 276 | assert(task_queues()->is_empty() || heap->cancelled_gc(), "Should be empty when not cancelled");
|
248 | 277 | }
|
249 | 278 |
|
|
0 commit comments