Skip to content

Commit 98d4101

Browse files
author
Kim Barrett
committedMay 6, 2020
8244307: Improve assertions against taskqueue underflow
Added assert_not_underflow. Reviewed-by: tschatzl, sjohanss
1 parent 7ae3bea commit 98d4101

File tree

2 files changed

+21
-8
lines changed

2 files changed

+21
-8
lines changed
 

‎src/hotspot/share/gc/shared/taskqueue.hpp

+16-3
Original file line numberDiff line numberDiff line change
@@ -202,12 +202,24 @@ class TaskQueueSuper: public CHeapObj<F> {
202202
// threads attempting to perform the pop_global will all perform the same
203203
// CAS, and only one can succeed.) Any stealing thread that reads after
204204
// either the increment or decrement will see an empty queue, and will not
205-
// join the competitors. The "sz == -1 || sz == N-1" state will not be
206-
// modified by concurrent queues, so the owner thread can reset the state to
207-
// _bottom == top so subsequent pushes will be performed normally.
205+
// join the competitors. The "sz == -1" / "sz == N-1" state will not be
206+
// modified by concurrent threads, so the owner thread can reset the state
207+
// to _bottom == top so subsequent pushes will be performed normally.
208208
return (sz == N - 1) ? 0 : sz;
209209
}
210210

211+
// Assert that we're not in the underflow state where bottom has
212+
// been decremented past top, so that _bottom+1 mod N == top. See
213+
// the discussion in clean_size.
214+
215+
void assert_not_underflow(uint bot, uint top) const {
216+
assert_not_underflow(dirty_size(bot, top));
217+
}
218+
219+
void assert_not_underflow(uint dirty_size) const {
220+
assert(dirty_size != N - 1, "invariant");
221+
}
222+
211223
private:
212224
DEFINE_PAD_MINUS_SIZE(0, DEFAULT_CACHE_LINE_SIZE, 0);
213225

@@ -313,6 +325,7 @@ class GenericTaskQueue: public TaskQueueSuper<N, F> {
313325
using TaskQueueSuper<N, F>::decrement_index;
314326
using TaskQueueSuper<N, F>::dirty_size;
315327
using TaskQueueSuper<N, F>::clean_size;
328+
using TaskQueueSuper<N, F>::assert_not_underflow;
316329

317330
public:
318331
using TaskQueueSuper<N, F>::max_elems;

‎src/hotspot/share/gc/shared/taskqueue.inline.hpp

+5-5
Original file line numberDiff line numberDiff line change
@@ -123,7 +123,7 @@ bool GenericTaskQueue<E, F, N>::pop_local_slow(uint localBot, Age oldAge) {
123123
Age tempAge = cmpxchg_age(oldAge, newAge);
124124
if (tempAge == oldAge) {
125125
// We win.
126-
assert(dirty_size(localBot, age_top_relaxed()) != N - 1, "sanity");
126+
assert_not_underflow(localBot, age_top_relaxed());
127127
TASKQUEUE_STATS_ONLY(stats.record_pop_slow());
128128
return true;
129129
}
@@ -132,7 +132,7 @@ bool GenericTaskQueue<E, F, N>::pop_local_slow(uint localBot, Age oldAge) {
132132
// and top is greater than bottom. Fix this representation of the empty queue
133133
// to become the canonical one.
134134
set_age_relaxed(newAge);
135-
assert(dirty_size(localBot, age_top_relaxed()) != N - 1, "sanity");
135+
assert_not_underflow(localBot, age_top_relaxed());
136136
return false;
137137
}
138138

@@ -144,7 +144,7 @@ GenericTaskQueue<E, F, N>::pop_local(E& t, uint threshold) {
144144
// resets the size to 0 before the next call (which is sequential,
145145
// since this is pop_local.)
146146
uint dirty_n_elems = dirty_size(localBot, age_top_relaxed());
147-
assert(dirty_n_elems != N - 1, "Shouldn't be possible...");
147+
assert_not_underflow(dirty_n_elems);
148148
if (dirty_n_elems <= threshold) return false;
149149
localBot = decrement_index(localBot);
150150
set_bottom_relaxed(localBot);
@@ -158,7 +158,7 @@ GenericTaskQueue<E, F, N>::pop_local(E& t, uint threshold) {
158158
// a "pop_global" operation, and we're done.
159159
idx_t tp = age_top_relaxed();
160160
if (clean_size(localBot, tp) > 0) {
161-
assert(dirty_size(localBot, tp) != N - 1, "sanity");
161+
assert_not_underflow(localBot, tp);
162162
TASKQUEUE_STATS_ONLY(stats.record_pop());
163163
return true;
164164
} else {
@@ -241,7 +241,7 @@ bool GenericTaskQueue<E, F, N>::pop_global(E& t) {
241241

242242
// Note that using "bottom" here might fail, since a pop_local might
243243
// have decremented it.
244-
assert(dirty_size(localBot, newAge.top()) != N - 1, "sanity");
244+
assert_not_underflow(localBot, newAge.top());
245245
return resAge == oldAge;
246246
}
247247

0 commit comments

Comments
 (0)
Please sign in to comment.