8138863: Refactor WaitForBarrierGCTask
Reviewed-by: mgerdin, jwilhelm
This commit is contained in:
parent
ef3a3a84bc
commit
48bbdafeed
@ -395,7 +395,6 @@ void GCTaskManager::initialize() {
|
|||||||
GCTaskQueue* unsynchronized_queue = GCTaskQueue::create_on_c_heap();
|
GCTaskQueue* unsynchronized_queue = GCTaskQueue::create_on_c_heap();
|
||||||
_queue = SynchronizedGCTaskQueue::create(unsynchronized_queue, lock());
|
_queue = SynchronizedGCTaskQueue::create(unsynchronized_queue, lock());
|
||||||
_noop_task = NoopGCTask::create_on_c_heap();
|
_noop_task = NoopGCTask::create_on_c_heap();
|
||||||
_idle_inactive_task = WaitForBarrierGCTask::create_on_c_heap();
|
|
||||||
_resource_flag = NEW_C_HEAP_ARRAY(bool, workers(), mtGC);
|
_resource_flag = NEW_C_HEAP_ARRAY(bool, workers(), mtGC);
|
||||||
{
|
{
|
||||||
// Set up worker threads.
|
// Set up worker threads.
|
||||||
@ -440,8 +439,6 @@ GCTaskManager::~GCTaskManager() {
|
|||||||
assert(queue()->is_empty(), "still have queued work");
|
assert(queue()->is_empty(), "still have queued work");
|
||||||
NoopGCTask::destroy(_noop_task);
|
NoopGCTask::destroy(_noop_task);
|
||||||
_noop_task = NULL;
|
_noop_task = NULL;
|
||||||
WaitForBarrierGCTask::destroy(_idle_inactive_task);
|
|
||||||
_idle_inactive_task = NULL;
|
|
||||||
if (_thread != NULL) {
|
if (_thread != NULL) {
|
||||||
for (uint i = 0; i < workers(); i += 1) {
|
for (uint i = 0; i < workers(); i += 1) {
|
||||||
GCTaskThread::destroy(thread(i));
|
GCTaskThread::destroy(thread(i));
|
||||||
@ -497,7 +494,7 @@ void GCTaskManager::task_idle_workers() {
|
|||||||
// the GCTaskManager's monitor so that the "more_inactive_workers"
|
// the GCTaskManager's monitor so that the "more_inactive_workers"
|
||||||
// count is correct.
|
// count is correct.
|
||||||
MutexLockerEx ml(monitor(), Mutex::_no_safepoint_check_flag);
|
MutexLockerEx ml(monitor(), Mutex::_no_safepoint_check_flag);
|
||||||
_idle_inactive_task->set_should_wait(true);
|
_wait_helper.set_should_wait(true);
|
||||||
// active_workers are a number being requested. idle_workers
|
// active_workers are a number being requested. idle_workers
|
||||||
// are the number currently idle. If all the workers are being
|
// are the number currently idle. If all the workers are being
|
||||||
// requested to be active but some are already idle, reduce
|
// requested to be active but some are already idle, reduce
|
||||||
@ -540,7 +537,7 @@ void GCTaskManager::release_idle_workers() {
|
|||||||
{
|
{
|
||||||
MutexLockerEx ml(monitor(),
|
MutexLockerEx ml(monitor(),
|
||||||
Mutex::_no_safepoint_check_flag);
|
Mutex::_no_safepoint_check_flag);
|
||||||
_idle_inactive_task->set_should_wait(false);
|
_wait_helper.set_should_wait(false);
|
||||||
monitor()->notify_all();
|
monitor()->notify_all();
|
||||||
// Release monitor
|
// Release monitor
|
||||||
}
|
}
|
||||||
@ -834,12 +831,12 @@ IdleGCTask* IdleGCTask::create_on_c_heap() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void IdleGCTask::do_it(GCTaskManager* manager, uint which) {
|
void IdleGCTask::do_it(GCTaskManager* manager, uint which) {
|
||||||
WaitForBarrierGCTask* wait_for_task = manager->idle_inactive_task();
|
WaitHelper* wait_helper = manager->wait_helper();
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
" IdleGCTask:::do_it()"
|
" IdleGCTask:::do_it()"
|
||||||
" should_wait: %s",
|
" should_wait: %s",
|
||||||
p2i(this), wait_for_task->should_wait() ? "true" : "false");
|
p2i(this), wait_helper->should_wait() ? "true" : "false");
|
||||||
}
|
}
|
||||||
MutexLockerEx ml(manager->monitor(), Mutex::_no_safepoint_check_flag);
|
MutexLockerEx ml(manager->monitor(), Mutex::_no_safepoint_check_flag);
|
||||||
if (TraceDynamicGCThreads) {
|
if (TraceDynamicGCThreads) {
|
||||||
@ -848,7 +845,7 @@ void IdleGCTask::do_it(GCTaskManager* manager, uint which) {
|
|||||||
// Increment has to be done when the idle tasks are created.
|
// Increment has to be done when the idle tasks are created.
|
||||||
// manager->increment_idle_workers();
|
// manager->increment_idle_workers();
|
||||||
manager->monitor()->notify_all();
|
manager->monitor()->notify_all();
|
||||||
while (wait_for_task->should_wait()) {
|
while (wait_helper->should_wait()) {
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
" IdleGCTask::do_it()"
|
" IdleGCTask::do_it()"
|
||||||
@ -865,7 +862,7 @@ void IdleGCTask::do_it(GCTaskManager* manager, uint which) {
|
|||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
" IdleGCTask::do_it() returns"
|
" IdleGCTask::do_it() returns"
|
||||||
" should_wait: %s",
|
" should_wait: %s",
|
||||||
p2i(this), wait_for_task->should_wait() ? "true" : "false");
|
p2i(this), wait_helper->should_wait() ? "true" : "false");
|
||||||
}
|
}
|
||||||
// Release monitor().
|
// Release monitor().
|
||||||
}
|
}
|
||||||
@ -889,62 +886,29 @@ void IdleGCTask::destruct() {
|
|||||||
// WaitForBarrierGCTask
|
// WaitForBarrierGCTask
|
||||||
//
|
//
|
||||||
WaitForBarrierGCTask* WaitForBarrierGCTask::create() {
|
WaitForBarrierGCTask* WaitForBarrierGCTask::create() {
|
||||||
WaitForBarrierGCTask* result = new WaitForBarrierGCTask(false);
|
WaitForBarrierGCTask* result = new WaitForBarrierGCTask();
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
WaitForBarrierGCTask* WaitForBarrierGCTask::create_on_c_heap() {
|
WaitForBarrierGCTask::WaitForBarrierGCTask() : GCTask(GCTask::Kind::wait_for_barrier_task) { }
|
||||||
WaitForBarrierGCTask* result =
|
|
||||||
new (ResourceObj::C_HEAP, mtGC) WaitForBarrierGCTask(true);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
WaitForBarrierGCTask::WaitForBarrierGCTask(bool on_c_heap) :
|
|
||||||
GCTask(GCTask::Kind::wait_for_barrier_task),
|
|
||||||
_is_c_heap_obj(on_c_heap) {
|
|
||||||
_monitor = MonitorSupply::reserve();
|
|
||||||
set_should_wait(true);
|
|
||||||
if (TraceGCTaskManager) {
|
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
|
||||||
" WaitForBarrierGCTask::WaitForBarrierGCTask()"
|
|
||||||
" monitor: " INTPTR_FORMAT,
|
|
||||||
p2i(this), p2i(monitor()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void WaitForBarrierGCTask::destroy(WaitForBarrierGCTask* that) {
|
void WaitForBarrierGCTask::destroy(WaitForBarrierGCTask* that) {
|
||||||
if (that != NULL) {
|
if (that != NULL) {
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "] WaitForBarrierGCTask::destroy()", p2i(that));
|
||||||
" WaitForBarrierGCTask::destroy()"
|
|
||||||
" is_c_heap_obj: %s"
|
|
||||||
" monitor: " INTPTR_FORMAT,
|
|
||||||
p2i(that),
|
|
||||||
that->is_c_heap_obj() ? "true" : "false",
|
|
||||||
p2i(that->monitor()));
|
|
||||||
}
|
}
|
||||||
that->destruct();
|
that->destruct();
|
||||||
if (that->is_c_heap_obj()) {
|
|
||||||
FreeHeap(that);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void WaitForBarrierGCTask::destruct() {
|
void WaitForBarrierGCTask::destruct() {
|
||||||
assert(monitor() != NULL, "monitor should not be NULL");
|
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "] WaitForBarrierGCTask::destruct()", p2i(this));
|
||||||
" WaitForBarrierGCTask::destruct()"
|
|
||||||
" monitor: " INTPTR_FORMAT,
|
|
||||||
p2i(this), p2i(monitor()));
|
|
||||||
}
|
}
|
||||||
this->GCTask::destruct();
|
this->GCTask::destruct();
|
||||||
// Clean up that should be in the destructor,
|
// Clean up that should be in the destructor,
|
||||||
// except that ResourceMarks don't call destructors.
|
// except that ResourceMarks don't call destructors.
|
||||||
if (monitor() != NULL) {
|
_wait_helper.release_monitor();
|
||||||
MonitorSupply::release(monitor());
|
|
||||||
}
|
|
||||||
_monitor = (Monitor*) 0xDEAD000F;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void WaitForBarrierGCTask::do_it_internal(GCTaskManager* manager, uint which) {
|
void WaitForBarrierGCTask::do_it_internal(GCTaskManager* manager, uint which) {
|
||||||
@ -963,9 +927,8 @@ void WaitForBarrierGCTask::do_it_internal(GCTaskManager* manager, uint which) {
|
|||||||
void WaitForBarrierGCTask::do_it(GCTaskManager* manager, uint which) {
|
void WaitForBarrierGCTask::do_it(GCTaskManager* manager, uint which) {
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
" WaitForBarrierGCTask::do_it() waiting for idle"
|
" WaitForBarrierGCTask::do_it() waiting for idle",
|
||||||
" monitor: " INTPTR_FORMAT,
|
p2i(this));
|
||||||
p2i(this), p2i(monitor()));
|
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
// First, wait for the barrier to arrive.
|
// First, wait for the barrier to arrive.
|
||||||
@ -973,24 +936,30 @@ void WaitForBarrierGCTask::do_it(GCTaskManager* manager, uint which) {
|
|||||||
do_it_internal(manager, which);
|
do_it_internal(manager, which);
|
||||||
// Release manager->lock().
|
// Release manager->lock().
|
||||||
}
|
}
|
||||||
{
|
|
||||||
// Then notify the waiter.
|
// Then notify the waiter.
|
||||||
MutexLockerEx ml(monitor(), Mutex::_no_safepoint_check_flag);
|
_wait_helper.notify();
|
||||||
set_should_wait(false);
|
}
|
||||||
// Waiter doesn't miss the notify in the wait_for method
|
|
||||||
// since it checks the flag after grabbing the monitor.
|
WaitHelper::WaitHelper() : _should_wait(true), _monitor(MonitorSupply::reserve()) {
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
" WaitForBarrierGCTask::do_it()"
|
" WaitHelper::WaitHelper()"
|
||||||
" [" INTPTR_FORMAT "] (%s)->notify_all()",
|
" monitor: " INTPTR_FORMAT,
|
||||||
p2i(this), p2i(monitor()), monitor()->name());
|
p2i(this), p2i(monitor()));
|
||||||
}
|
|
||||||
monitor()->notify_all();
|
|
||||||
// Release monitor().
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void WaitForBarrierGCTask::wait_for(bool reset) {
|
void WaitHelper::release_monitor() {
|
||||||
|
assert(_monitor != NULL, "");
|
||||||
|
MonitorSupply::release(_monitor);
|
||||||
|
_monitor = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
WaitHelper::~WaitHelper() {
|
||||||
|
release_monitor();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WaitHelper::wait_for(bool reset) {
|
||||||
if (TraceGCTaskManager) {
|
if (TraceGCTaskManager) {
|
||||||
tty->print_cr("[" INTPTR_FORMAT "]"
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
" WaitForBarrierGCTask::wait_for()"
|
" WaitForBarrierGCTask::wait_for()"
|
||||||
@ -1023,6 +992,20 @@ void WaitForBarrierGCTask::wait_for(bool reset) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void WaitHelper::notify() {
|
||||||
|
MutexLockerEx ml(monitor(), Mutex::_no_safepoint_check_flag);
|
||||||
|
set_should_wait(false);
|
||||||
|
// Waiter doesn't miss the notify in the wait_for method
|
||||||
|
// since it checks the flag after grabbing the monitor.
|
||||||
|
if (TraceGCTaskManager) {
|
||||||
|
tty->print_cr("[" INTPTR_FORMAT "]"
|
||||||
|
" WaitForBarrierGCTask::do_it()"
|
||||||
|
" [" INTPTR_FORMAT "] (%s)->notify_all()",
|
||||||
|
p2i(this), p2i(monitor()), monitor()->name());
|
||||||
|
}
|
||||||
|
monitor()->notify_all();
|
||||||
|
}
|
||||||
|
|
||||||
Mutex* MonitorSupply::_lock = NULL;
|
Mutex* MonitorSupply::_lock = NULL;
|
||||||
GrowableArray<Monitor*>* MonitorSupply::_freelist = NULL;
|
GrowableArray<Monitor*>* MonitorSupply::_freelist = NULL;
|
||||||
|
|
||||||
|
@ -272,6 +272,28 @@ protected:
|
|||||||
~SynchronizedGCTaskQueue();
|
~SynchronizedGCTaskQueue();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class WaitHelper VALUE_OBJ_CLASS_SPEC {
|
||||||
|
private:
|
||||||
|
Monitor* _monitor;
|
||||||
|
volatile bool _should_wait;
|
||||||
|
public:
|
||||||
|
WaitHelper();
|
||||||
|
~WaitHelper();
|
||||||
|
void wait_for(bool reset);
|
||||||
|
void notify();
|
||||||
|
void set_should_wait(bool value) {
|
||||||
|
_should_wait = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
Monitor* monitor() const {
|
||||||
|
return _monitor;
|
||||||
|
}
|
||||||
|
bool should_wait() const {
|
||||||
|
return _should_wait;
|
||||||
|
}
|
||||||
|
void release_monitor();
|
||||||
|
};
|
||||||
|
|
||||||
// Dynamic number of GC threads
|
// Dynamic number of GC threads
|
||||||
//
|
//
|
||||||
// GC threads wait in get_task() for work (i.e., a task) to perform.
|
// GC threads wait in get_task() for work (i.e., a task) to perform.
|
||||||
@ -357,7 +379,7 @@ private:
|
|||||||
uint _barriers; // Count of barrier tasks.
|
uint _barriers; // Count of barrier tasks.
|
||||||
uint _emptied_queue; // Times we emptied the queue.
|
uint _emptied_queue; // Times we emptied the queue.
|
||||||
NoopGCTask* _noop_task; // The NoopGCTask instance.
|
NoopGCTask* _noop_task; // The NoopGCTask instance.
|
||||||
WaitForBarrierGCTask* _idle_inactive_task;// Task for inactive workers
|
WaitHelper _wait_helper; // Used by inactive worker
|
||||||
volatile uint _idle_workers; // Number of idled workers
|
volatile uint _idle_workers; // Number of idled workers
|
||||||
public:
|
public:
|
||||||
// Factory create and destroy methods.
|
// Factory create and destroy methods.
|
||||||
@ -383,8 +405,8 @@ public:
|
|||||||
Monitor * lock() const {
|
Monitor * lock() const {
|
||||||
return _monitor;
|
return _monitor;
|
||||||
}
|
}
|
||||||
WaitForBarrierGCTask* idle_inactive_task() {
|
WaitHelper* wait_helper() {
|
||||||
return _idle_inactive_task;
|
return &_wait_helper;
|
||||||
}
|
}
|
||||||
// Methods.
|
// Methods.
|
||||||
// Add the argument task to be run.
|
// Add the argument task to be run.
|
||||||
@ -559,25 +581,17 @@ class WaitForBarrierGCTask : public GCTask {
|
|||||||
friend class IdleGCTask;
|
friend class IdleGCTask;
|
||||||
private:
|
private:
|
||||||
// Instance state.
|
// Instance state.
|
||||||
Monitor* _monitor; // Guard and notify changes.
|
WaitHelper _wait_helper;
|
||||||
volatile bool _should_wait; // true=>wait, false=>proceed.
|
WaitForBarrierGCTask();
|
||||||
const bool _is_c_heap_obj; // Was allocated on the heap.
|
|
||||||
public:
|
public:
|
||||||
virtual char* name() { return (char *) "waitfor-barrier-task"; }
|
virtual char* name() { return (char *) "waitfor-barrier-task"; }
|
||||||
|
|
||||||
// Factory create and destroy methods.
|
// Factory create and destroy methods.
|
||||||
static WaitForBarrierGCTask* create();
|
static WaitForBarrierGCTask* create();
|
||||||
static WaitForBarrierGCTask* create_on_c_heap();
|
|
||||||
static void destroy(WaitForBarrierGCTask* that);
|
static void destroy(WaitForBarrierGCTask* that);
|
||||||
// Methods.
|
// Methods.
|
||||||
void do_it(GCTaskManager* manager, uint which);
|
void do_it(GCTaskManager* manager, uint which);
|
||||||
void wait_for(bool reset);
|
|
||||||
void set_should_wait(bool value) {
|
|
||||||
_should_wait = value;
|
|
||||||
}
|
|
||||||
protected:
|
protected:
|
||||||
// Constructor. Clients use factory, but there might be subclasses.
|
|
||||||
WaitForBarrierGCTask(bool on_c_heap);
|
|
||||||
// Destructor-like method.
|
// Destructor-like method.
|
||||||
void destruct();
|
void destruct();
|
||||||
|
|
||||||
@ -585,15 +599,8 @@ protected:
|
|||||||
// Wait for this to be the only task running.
|
// Wait for this to be the only task running.
|
||||||
void do_it_internal(GCTaskManager* manager, uint which);
|
void do_it_internal(GCTaskManager* manager, uint which);
|
||||||
|
|
||||||
// Accessors.
|
void wait_for(bool reset) {
|
||||||
Monitor* monitor() const {
|
_wait_helper.wait_for(reset);
|
||||||
return _monitor;
|
|
||||||
}
|
|
||||||
bool should_wait() const {
|
|
||||||
return _should_wait;
|
|
||||||
}
|
|
||||||
bool is_c_heap_obj() {
|
|
||||||
return _is_c_heap_obj;
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user