1
1
/*
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.
3
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
4
*
5
5
* This code is free software; you can redistribute it and/or modify it
@@ -40,7 +40,7 @@ void G1SentinelTask::execute() {
40
40
41
41
G1ServiceThread::G1ServiceThread () :
42
42
ConcurrentGCThread(),
43
- _monitor(Mutex::nonleaf ,
43
+ _monitor(Mutex::leaf ,
44
44
" G1ServiceThread monitor" ,
45
45
true ,
46
46
Monitor::_safepoint_check_never),
@@ -71,100 +71,87 @@ void G1ServiceThread::register_task(G1ServiceTask* task, jlong delay_ms) {
71
71
schedule_task (task, delay_ms);
72
72
}
73
73
74
- void G1ServiceThread::schedule (G1ServiceTask* task, jlong delay_ms) {
74
+ void G1ServiceThread::schedule (G1ServiceTask* task, jlong delay_ms, bool notify ) {
75
75
guarantee (task->is_registered (), " Must be registered before scheduled" );
76
76
guarantee (task->next () == NULL , " Task already in queue" );
77
77
78
78
// Schedule task by setting the task time and adding it to queue.
79
79
jlong delay = TimeHelper::millis_to_counter (delay_ms);
80
80
task->set_time (os::elapsed_counter () + delay);
81
81
82
- MutexLocker ml (&_monitor, Mutex::_no_safepoint_check_flag);
82
+ MonitorLocker ml (&_monitor, Mutex::_no_safepoint_check_flag);
83
83
_task_queue.add_ordered (task);
84
+ if (notify) {
85
+ ml.notify ();
86
+ }
84
87
85
88
log_trace (gc, task)(" G1 Service Thread (%s) (schedule) @%1.3fs" ,
86
89
task->name (), TimeHelper::counter_to_seconds (task->time ()));
87
90
}
88
91
89
92
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 */ );
107
94
}
108
95
109
- void G1ServiceThread::notify () {
96
+ G1ServiceTask* G1ServiceThread::wait_for_task () {
110
97
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
+ }
127
118
}
128
119
}
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.
138
121
}
139
122
140
123
void G1ServiceThread::run_task (G1ServiceTask* task) {
141
- double start = os::elapsedTime ();
124
+ jlong start = os::elapsed_counter ();
142
125
double vstart = os::elapsedVTime ();
143
126
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
+
145
134
task->execute ();
146
135
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);
151
140
}
152
141
153
142
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);
161
147
}
162
-
148
+ assert ( should_terminate (), " invariant " );
163
149
log_debug (gc, task)(" G1 Service Thread (stopping)" );
164
150
}
165
151
166
152
void G1ServiceThread::stop_service () {
167
- notify ();
153
+ MonitorLocker ml (&_monitor, Mutex::_no_safepoint_check_flag);
154
+ ml.notify ();
168
155
}
169
156
170
157
G1ServiceTask::G1ServiceTask (const char * name) :
@@ -184,7 +171,8 @@ bool G1ServiceTask::is_registered() {
184
171
void G1ServiceTask::schedule (jlong delay_ms) {
185
172
assert (Thread::current () == _service_thread,
186
173
" 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 */ );
188
176
}
189
177
190
178
const char * G1ServiceTask::name () {
@@ -210,17 +198,15 @@ G1ServiceTask* G1ServiceTask::next() {
210
198
211
199
G1ServiceTaskQueue::G1ServiceTaskQueue () : _sentinel() { }
212
200
213
- G1ServiceTask* G1ServiceTaskQueue::pop () {
201
+ void G1ServiceTaskQueue::remove_front () {
214
202
verify_task_queue ();
215
203
216
204
G1ServiceTask* task = _sentinel.next ();
217
205
_sentinel.set_next (task->next ());
218
206
task->set_next (NULL );
219
-
220
- return task;
221
207
}
222
208
223
- G1ServiceTask* G1ServiceTaskQueue::peek () {
209
+ G1ServiceTask* G1ServiceTaskQueue::front () {
224
210
verify_task_queue ();
225
211
return _sentinel.next ();
226
212
}
0 commit comments