RFR: ShenandoahControlThread should not get GCLABs
Aleksey Shipilev
shade at redhat.com
Fri Aug 31 14:02:54 UTC 2018
This seems to be the bug that we are chasing now in sh/jdk. ShenandoahControlThread accidentally got
the GCLAB, but never retired it. When traversal-precleaning started, it walked some unmarked
objects, evacuating them, which happened in the ShControlThread context. Which means we have
allocated something in GCLAB we don't otherwise know about. And the whole thing crashes if we get to
recycle the region holding that GCLAB before it retires.
There are two possible fixes: let GCLAB retirement code know about ShControlThread, or make
ShControlThread go via shared allocs all the time, like other non-Java threads do. I think the
second option is better, because doing the GCLAB retirement from the ShControlThread might be prone
to other errors (e.g. deadlocks). We intended that only Java threads and GC worker threads have
GCLABs to simplify GCLAB tracking, but this is not true it current code, see below.
(Separately, we shall fix traversal-precleaning to get to worker threads so it can use GCLABs there).
Fix:
diff -r ea2f39b37e25 src/hotspot/share/gc/shenandoah/shenandoahHeap.cpp
--- a/src/hotspot/share/gc/shenandoah/shenandoahHeap.cpp Fri Aug 31 13:29:24 2018 +0200
+++ b/src/hotspot/share/gc/shenandoah/shenandoahHeap.cpp Fri Aug 31 15:49:05 2018 +0200
@@ -564,12 +564,11 @@
}
class ShenandoahInitGCLABClosure : public ThreadClosure {
public:
void do_thread(Thread* thread) {
- if (thread != NULL && (thread->is_Java_thread() || thread->is_Worker_thread() ||
- thread->is_ConcurrentGC_thread())) {
+ if (thread != NULL && (thread->is_Java_thread() || thread->is_Worker_thread())) {
ShenandoahThreadLocalData::initialize_gclab(thread);
}
}
};
diff -r ea2f39b37e25 src/hotspot/share/gc/shenandoah/shenandoahThreadLocalData.hpp
--- a/src/hotspot/share/gc/shenandoah/shenandoahThreadLocalData.hpp Fri Aug 31 13:29:24 2018 +0200
+++ b/src/hotspot/share/gc/shenandoah/shenandoahThreadLocalData.hpp Fri Aug 31 15:49:05 2018 +0200
@@ -120,10 +120,11 @@
static bool is_force_satb_flush(Thread* thread) {
return data(thread)->_force_satb_flush;
}
static void initialize_gclab(Thread* thread) {
+ assert (thread->is_Java_thread() || thread->is_Worker_thread(), "Only Java and GC worker
threads are allowed to get GCLABs");
data(thread)->_gclab = new PLAB(PLAB::min_size());
data(thread)->_gclab_size = 0;
}
static PLAB* gclab(Thread* thread) {
Testing: failing tests on my desktop and gotland, tier3_gc_shenandoah (running)
Thanks,
-Aleksey
More information about the shenandoah-dev
mailing list