Skip to content

Commit 0cc4bb7

Browse files
author
Kim Barrett
committedJul 23, 2021
8270870: Simplify G1ServiceThread
Reviewed-by: tschatzl, iwalulya
1 parent 8c8e3a0 commit 0cc4bb7

File tree

3 files changed

+73
-89
lines changed

3 files changed

+73
-89
lines changed
 

‎src/hotspot/share/gc/g1/g1ServiceThread.cpp

+53-67
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -40,7 +40,7 @@ void G1SentinelTask::execute() {
4040

4141
G1ServiceThread::G1ServiceThread() :
4242
ConcurrentGCThread(),
43-
_monitor(Mutex::nonleaf,
43+
_monitor(Mutex::leaf,
4444
"G1ServiceThread monitor",
4545
true,
4646
Monitor::_safepoint_check_never),
@@ -71,100 +71,87 @@ void G1ServiceThread::register_task(G1ServiceTask* task, jlong delay_ms) {
7171
schedule_task(task, delay_ms);
7272
}
7373

74-
void G1ServiceThread::schedule(G1ServiceTask* task, jlong delay_ms) {
74+
void G1ServiceThread::schedule(G1ServiceTask* task, jlong delay_ms, bool notify) {
7575
guarantee(task->is_registered(), "Must be registered before scheduled");
7676
guarantee(task->next() == NULL, "Task already in queue");
7777

7878
// Schedule task by setting the task time and adding it to queue.
7979
jlong delay = TimeHelper::millis_to_counter(delay_ms);
8080
task->set_time(os::elapsed_counter() + delay);
8181

82-
MutexLocker ml(&_monitor, Mutex::_no_safepoint_check_flag);
82+
MonitorLocker ml(&_monitor, Mutex::_no_safepoint_check_flag);
8383
_task_queue.add_ordered(task);
84+
if (notify) {
85+
ml.notify();
86+
}
8487

8588
log_trace(gc, task)("G1 Service Thread (%s) (schedule) @%1.3fs",
8689
task->name(), TimeHelper::counter_to_seconds(task->time()));
8790
}
8891

8992
void G1ServiceThread::schedule_task(G1ServiceTask* task, jlong delay_ms) {
90-
schedule(task, delay_ms);
91-
notify();
92-
}
93-
94-
int64_t G1ServiceThread::time_to_next_task_ms() {
95-
assert(_monitor.owned_by_self(), "Must be owner of lock");
96-
assert(!_task_queue.is_empty(), "Should not be called for empty list");
97-
98-
jlong time_diff = _task_queue.peek()->time() - os::elapsed_counter();
99-
if (time_diff < 0) {
100-
// Run without sleeping.
101-
return 0;
102-
}
103-
104-
// Return sleep time in milliseconds. Using ceil to make sure we never
105-
// schedule a task too early.
106-
return (int64_t) ceil(TimeHelper::counter_to_millis(time_diff));
93+
schedule(task, delay_ms, true /* notify */);
10794
}
10895

109-
void G1ServiceThread::notify() {
96+
G1ServiceTask* G1ServiceThread::wait_for_task() {
11097
MonitorLocker ml(&_monitor, Mutex::_no_safepoint_check_flag);
111-
ml.notify();
112-
}
113-
114-
void G1ServiceThread::sleep_before_next_cycle() {
115-
MonitorLocker ml(&_monitor, Mutex::_no_safepoint_check_flag);
116-
if (should_terminate()) {
117-
return;
118-
} else if (_task_queue.is_empty()) {
119-
// Sleep until new task is registered if no tasks available.
120-
log_trace(gc, task)("G1 Service Thread (wait for new tasks)");
121-
ml.wait(0);
122-
} else {
123-
int64_t sleep_ms = time_to_next_task_ms();
124-
if (sleep_ms > 0) {
125-
log_trace(gc, task)("G1 Service Thread (wait) %1.3fs", sleep_ms / 1000.0);
126-
ml.wait(sleep_ms);
98+
while (!should_terminate()) {
99+
if (_task_queue.is_empty()) {
100+
log_trace(gc, task)("G1 Service Thread (wait for new tasks)");
101+
ml.wait();
102+
} else {
103+
G1ServiceTask* task = _task_queue.front();
104+
jlong scheduled = task->time();
105+
jlong now = os::elapsed_counter();
106+
if (scheduled <= now) {
107+
_task_queue.remove_front();
108+
return task;
109+
} else {
110+
// Round up to try not to wake up early, and to avoid round down to
111+
// zero (which has special meaning of wait forever) by conversion.
112+
double delay = ceil(TimeHelper::counter_to_millis(scheduled - now));
113+
log_trace(gc, task)("G1 Service Thread (wait %1.3fs)", (delay / 1000.0));
114+
int64_t delay_ms = static_cast<int64_t>(delay);
115+
assert(delay_ms > 0, "invariant");
116+
ml.wait(delay_ms);
117+
}
127118
}
128119
}
129-
}
130-
131-
G1ServiceTask* G1ServiceThread::pop_due_task() {
132-
MutexLocker ml(&_monitor, Mutex::_no_safepoint_check_flag);
133-
if (_task_queue.is_empty() || time_to_next_task_ms() != 0) {
134-
return NULL;
135-
}
136-
137-
return _task_queue.pop();
120+
return nullptr; // Return nullptr when terminating.
138121
}
139122

140123
void G1ServiceThread::run_task(G1ServiceTask* task) {
141-
double start = os::elapsedTime();
124+
jlong start = os::elapsed_counter();
142125
double vstart = os::elapsedVTime();
143126

144-
log_debug(gc, task, start)("G1 Service Thread (%s) (run)", task->name());
127+
assert(task->time() <= start,
128+
"task run early: " JLONG_FORMAT " > " JLONG_FORMAT,
129+
task->time(), start);
130+
log_debug(gc, task, start)("G1 Service Thread (%s) (run %1.3fms after schedule)",
131+
task->name(),
132+
TimeHelper::counter_to_millis(start - task->time()));
133+
145134
task->execute();
146135

147-
double duration = os::elapsedTime() - start;
148-
double vduration = os::elapsedVTime() - vstart;
149-
log_debug(gc, task)("G1 Service Thread (%s) (run) %1.3fms (cpu: %1.3fms)",
150-
task->name(), duration * MILLIUNITS, vduration * MILLIUNITS);
136+
log_debug(gc, task)("G1 Service Thread (%s) (run: %1.3fms) (cpu: %1.3fms)",
137+
task->name(),
138+
TimeHelper::counter_to_millis(os::elapsed_counter() - start),
139+
(os::elapsedVTime() - vstart) * MILLIUNITS);
151140
}
152141

153142
void G1ServiceThread::run_service() {
154-
while (!should_terminate()) {
155-
G1ServiceTask* task = pop_due_task();
156-
if (task != NULL) {
157-
run_task(task);
158-
}
159-
160-
sleep_before_next_cycle();
143+
while (true) {
144+
G1ServiceTask* task = wait_for_task();
145+
if (task == nullptr) break;
146+
run_task(task);
161147
}
162-
148+
assert(should_terminate(), "invariant");
163149
log_debug(gc, task)("G1 Service Thread (stopping)");
164150
}
165151

166152
void G1ServiceThread::stop_service() {
167-
notify();
153+
MonitorLocker ml(&_monitor, Mutex::_no_safepoint_check_flag);
154+
ml.notify();
168155
}
169156

170157
G1ServiceTask::G1ServiceTask(const char* name) :
@@ -184,7 +171,8 @@ bool G1ServiceTask::is_registered() {
184171
void G1ServiceTask::schedule(jlong delay_ms) {
185172
assert(Thread::current() == _service_thread,
186173
"Can only be used when already running on the service thread");
187-
_service_thread->schedule(this, delay_ms);
174+
// No need to notify, since we *are* the service thread.
175+
_service_thread->schedule(this, delay_ms, false /* notify */);
188176
}
189177

190178
const char* G1ServiceTask::name() {
@@ -210,17 +198,15 @@ G1ServiceTask* G1ServiceTask::next() {
210198

211199
G1ServiceTaskQueue::G1ServiceTaskQueue() : _sentinel() { }
212200

213-
G1ServiceTask* G1ServiceTaskQueue::pop() {
201+
void G1ServiceTaskQueue::remove_front() {
214202
verify_task_queue();
215203

216204
G1ServiceTask* task = _sentinel.next();
217205
_sentinel.set_next(task->next());
218206
task->set_next(NULL);
219-
220-
return task;
221207
}
222208

223-
G1ServiceTask* G1ServiceTaskQueue::peek() {
209+
G1ServiceTask* G1ServiceTaskQueue::front() {
224210
verify_task_queue();
225211
return _sentinel.next();
226212
}

‎src/hotspot/share/gc/g1/g1ServiceThread.hpp

+10-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -87,8 +87,11 @@ class G1ServiceTaskQueue {
8787
void verify_task_queue() NOT_DEBUG_RETURN;
8888
public:
8989
G1ServiceTaskQueue();
90-
G1ServiceTask* pop();
91-
G1ServiceTask* peek();
90+
91+
// precondition: !is_empty().
92+
G1ServiceTask* front();
93+
// precondition: !is_empty().
94+
void remove_front();
9295
void add_ordered(G1ServiceTask* task);
9396
bool is_empty();
9497
};
@@ -107,22 +110,15 @@ class G1ServiceThread: public ConcurrentGCThread {
107110
void run_service();
108111
void stop_service();
109112

110-
// Returns the time in milliseconds until the next task is due.
111-
// Used both to determine if there are tasks ready to run and
112-
// how long to sleep when nothing is ready.
113-
int64_t time_to_next_task_ms();
114-
void sleep_before_next_cycle();
113+
// Return the next ready task, waiting until a task is ready.
114+
// Instead returns nullptr if termination requested.
115+
G1ServiceTask* wait_for_task();
115116

116-
G1ServiceTask* pop_due_task();
117117
void run_task(G1ServiceTask* task);
118118

119119
// Helper used by both schedule_task() and G1ServiceTask::schedule()
120120
// to schedule a registered task to run after the given delay.
121-
void schedule(G1ServiceTask* task, jlong delay);
122-
123-
// Notify a change to the service thread. Used to either stop
124-
// the service or to force check for new tasks.
125-
void notify();
121+
void schedule(G1ServiceTask* task, jlong delay, bool notify);
126122

127123
public:
128124
G1ServiceThread();

‎test/hotspot/gtest/gc/g1/test_g1ServiceThread.cpp

+10-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -142,8 +142,9 @@ TEST_VM(G1ServiceTaskQueue, add_ordered) {
142142
for (jlong now = 0; now < 1000000; now++) {
143143
// Random multiplier is at least 1 to ensure progress.
144144
int multiplier = 1 + os::random() % 10;
145-
while (queue.peek()->time() < now) {
146-
TestTask* task = (TestTask*) queue.pop();
145+
while (queue.front()->time() < now) {
146+
TestTask* task = (TestTask*) queue.front();
147+
queue.remove_front();
147148
// Update delay multiplier.
148149
task->execute();
149150
task->update_time(now, multiplier);
@@ -153,22 +154,23 @@ TEST_VM(G1ServiceTaskQueue, add_ordered) {
153154
}
154155

155156
while (!queue.is_empty()) {
156-
G1ServiceTask* task = queue.pop();
157+
G1ServiceTask* task = queue.front();
158+
queue.remove_front();
157159
delete task;
158160
}
159161
}
160162

161163
#ifdef ASSERT
162-
TEST_VM_ASSERT_MSG(G1ServiceTaskQueue, pop_empty,
164+
TEST_VM_ASSERT_MSG(G1ServiceTaskQueue, remove_from_empty,
163165
".*Should never try to verify empty queue") {
164166
G1ServiceTaskQueue queue;
165-
queue.pop();
167+
queue.remove_front();
166168
}
167169

168-
TEST_VM_ASSERT_MSG(G1ServiceTaskQueue, peek_empty,
170+
TEST_VM_ASSERT_MSG(G1ServiceTaskQueue, get_from_empty,
169171
".*Should never try to verify empty queue") {
170172
G1ServiceTaskQueue queue;
171-
queue.peek();
173+
queue.front();
172174
}
173175

174176
TEST_VM_ASSERT_MSG(G1ServiceTaskQueue, set_time_in_queue,

0 commit comments

Comments
 (0)
Please sign in to comment.