Skip to content

Commit 7ac0816

Browse files
committedJul 13, 2021
8269897: Shenandoah: Resolve UNKNOWN access strength, where possible
Reviewed-by: shade
1 parent 460c4bb commit 7ac0816

File tree

3 files changed

+83
-67
lines changed

3 files changed

+83
-67
lines changed
 

‎src/hotspot/share/gc/shenandoah/shenandoahBarrierSet.hpp

+13-6
Original file line numberDiff line numberDiff line change
@@ -54,11 +54,11 @@ class ShenandoahBarrierSet: public BarrierSet {
5454
static bool need_keep_alive_barrier(DecoratorSet decorators, BasicType type);
5555

5656
static bool is_strong_access(DecoratorSet decorators) {
57-
return (decorators & (ON_WEAK_OOP_REF | ON_PHANTOM_OOP_REF | ON_UNKNOWN_OOP_REF)) == 0;
57+
return (decorators & (ON_WEAK_OOP_REF | ON_PHANTOM_OOP_REF)) == 0;
5858
}
5959

6060
static bool is_weak_access(DecoratorSet decorators) {
61-
return (decorators & (ON_WEAK_OOP_REF | ON_UNKNOWN_OOP_REF)) != 0;
61+
return (decorators & ON_WEAK_OOP_REF) != 0;
6262
}
6363

6464
static bool is_phantom_access(DecoratorSet decorators) {
@@ -90,8 +90,6 @@ class ShenandoahBarrierSet: public BarrierSet {
9090
inline void satb_enqueue(oop value);
9191
inline void iu_barrier(oop obj);
9292

93-
template <DecoratorSet decorators>
94-
inline void keep_alive_if_weak(oop value);
9593
inline void keep_alive_if_weak(DecoratorSet decorators, oop value);
9694

9795
inline void enqueue(oop obj);
@@ -101,8 +99,17 @@ class ShenandoahBarrierSet: public BarrierSet {
10199
template <class T>
102100
inline oop load_reference_barrier_mutator(oop obj, T* load_addr);
103101

104-
template <DecoratorSet decorators, class T>
105-
inline oop load_reference_barrier(oop obj, T* load_addr);
102+
template <class T>
103+
inline oop load_reference_barrier(DecoratorSet decorators, oop obj, T* load_addr);
104+
105+
template <typename T>
106+
inline oop oop_load(DecoratorSet decorators, T* addr);
107+
108+
template <typename T>
109+
inline oop oop_cmpxchg(DecoratorSet decorators, T* addr, oop compare_value, oop new_value);
110+
111+
template <typename T>
112+
inline oop oop_xchg(DecoratorSet decorators, T* addr, oop new_value);
106113

107114
private:
108115
template <class T>

‎src/hotspot/share/gc/shenandoah/shenandoahBarrierSet.inline.hpp

+66-57
Original file line numberDiff line numberDiff line change
@@ -99,29 +99,29 @@ inline oop ShenandoahBarrierSet::load_reference_barrier(oop obj) {
9999
return obj;
100100
}
101101

102-
template <DecoratorSet decorators, class T>
103-
inline oop ShenandoahBarrierSet::load_reference_barrier(oop obj, T* load_addr) {
102+
template <class T>
103+
inline oop ShenandoahBarrierSet::load_reference_barrier(DecoratorSet decorators, oop obj, T* load_addr) {
104104
if (obj == NULL) {
105105
return NULL;
106106
}
107107

108108
// Prevent resurrection of unreachable phantom (i.e. weak-native) references.
109-
if (HasDecorator<decorators, ON_PHANTOM_OOP_REF>::value &&
109+
if ((decorators & ON_PHANTOM_OOP_REF) != 0 &&
110110
_heap->is_concurrent_weak_root_in_progress() &&
111111
!_heap->marking_context()->is_marked(obj)) {
112112
return NULL;
113113
}
114114

115115
// Prevent resurrection of unreachable weak references.
116-
if ((HasDecorator<decorators, ON_WEAK_OOP_REF>::value || HasDecorator<decorators, ON_UNKNOWN_OOP_REF>::value) &&
116+
if ((decorators & ON_WEAK_OOP_REF) != 0 &&
117117
_heap->is_concurrent_weak_root_in_progress() &&
118118
!_heap->marking_context()->is_marked_strong(obj)) {
119119
return NULL;
120120
}
121121

122122
// Prevent resurrection of unreachable objects that are visited during
123123
// concurrent class-unloading.
124-
if (HasDecorator<decorators, AS_NO_KEEPALIVE>::value &&
124+
if ((decorators & AS_NO_KEEPALIVE) != 0 &&
125125
_heap->is_evacuation_in_progress() &&
126126
!_heap->marking_context()->is_marked(obj)) {
127127
return obj;
@@ -184,45 +184,64 @@ inline void ShenandoahBarrierSet::keep_alive_if_weak(DecoratorSet decorators, oo
184184
}
185185
}
186186

187-
template <DecoratorSet decorators>
188-
inline void ShenandoahBarrierSet::keep_alive_if_weak(oop value) {
189-
assert((decorators & ON_UNKNOWN_OOP_REF) == 0, "Reference strength must be known");
190-
if (!HasDecorator<decorators, ON_STRONG_OOP_REF>::value &&
191-
!HasDecorator<decorators, AS_NO_KEEPALIVE>::value) {
192-
satb_enqueue(value);
193-
}
187+
template <typename T>
188+
inline oop ShenandoahBarrierSet::oop_load(DecoratorSet decorators, T* addr) {
189+
oop value = RawAccess<>::oop_load(addr);
190+
value = load_reference_barrier(decorators, value, addr);
191+
keep_alive_if_weak(decorators, value);
192+
return value;
193+
}
194+
195+
template <typename T>
196+
inline oop ShenandoahBarrierSet::oop_cmpxchg(DecoratorSet decorators, T* addr, oop compare_value, oop new_value) {
197+
iu_barrier(new_value);
198+
oop res;
199+
oop expected = compare_value;
200+
do {
201+
compare_value = expected;
202+
res = RawAccess<>::oop_atomic_cmpxchg(addr, compare_value, new_value);
203+
expected = res;
204+
} while ((compare_value != expected) && (resolve_forwarded(compare_value) == resolve_forwarded(expected)));
205+
206+
// Note: We don't need a keep-alive-barrier here. We already enqueue any loaded reference for SATB anyway,
207+
// because it must be the previous value.
208+
res = load_reference_barrier(decorators, res, reinterpret_cast<T*>(NULL));
209+
satb_enqueue(res);
210+
return res;
211+
}
212+
213+
template <typename T>
214+
inline oop ShenandoahBarrierSet::oop_xchg(DecoratorSet decorators, T* addr, oop new_value) {
215+
iu_barrier(new_value);
216+
oop previous = RawAccess<>::oop_atomic_xchg(addr, new_value);
217+
// Note: We don't need a keep-alive-barrier here. We already enqueue any loaded reference for SATB anyway,
218+
// because it must be the previous value.
219+
previous = load_reference_barrier<T>(decorators, previous, reinterpret_cast<T*>(NULL));
220+
satb_enqueue(previous);
221+
return previous;
194222
}
195223

196224
template <DecoratorSet decorators, typename BarrierSetT>
197225
template <typename T>
198226
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_load_not_in_heap(T* addr) {
199-
oop value = Raw::oop_load_not_in_heap(addr);
200-
if (value != NULL) {
201-
ShenandoahBarrierSet *const bs = ShenandoahBarrierSet::barrier_set();
202-
value = bs->load_reference_barrier<decorators, T>(value, addr);
203-
bs->keep_alive_if_weak<decorators>(value);
204-
}
205-
return value;
227+
assert((decorators & ON_UNKNOWN_OOP_REF) == 0, "must be absent");
228+
ShenandoahBarrierSet* const bs = ShenandoahBarrierSet::barrier_set();
229+
return bs->oop_load(decorators, addr);
206230
}
207231

208232
template <DecoratorSet decorators, typename BarrierSetT>
209233
template <typename T>
210234
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_load_in_heap(T* addr) {
211-
oop value = Raw::oop_load_in_heap(addr);
212-
ShenandoahBarrierSet *const bs = ShenandoahBarrierSet::barrier_set();
213-
value = bs->load_reference_barrier<decorators, T>(value, addr);
214-
bs->keep_alive_if_weak<decorators>(value);
215-
return value;
235+
assert((decorators & ON_UNKNOWN_OOP_REF) == 0, "must be absent");
236+
ShenandoahBarrierSet* const bs = ShenandoahBarrierSet::barrier_set();
237+
return bs->oop_load(decorators, addr);
216238
}
217239

218240
template <DecoratorSet decorators, typename BarrierSetT>
219241
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_load_in_heap_at(oop base, ptrdiff_t offset) {
220-
oop value = Raw::oop_load_in_heap_at(base, offset);
221-
ShenandoahBarrierSet *const bs = ShenandoahBarrierSet::barrier_set();
242+
ShenandoahBarrierSet* const bs = ShenandoahBarrierSet::barrier_set();
222243
DecoratorSet resolved_decorators = AccessBarrierSupport::resolve_possibly_unknown_oop_ref_strength<decorators>(base, offset);
223-
value = bs->load_reference_barrier<decorators>(value, AccessInternal::oop_field_addr<decorators>(base, offset));
224-
bs->keep_alive_if_weak(resolved_decorators, value);
225-
return value;
244+
return bs->oop_load(resolved_decorators, AccessInternal::oop_field_addr<decorators>(base, offset));
226245
}
227246

228247
template <DecoratorSet decorators, typename BarrierSetT>
@@ -254,59 +273,49 @@ inline void ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_st
254273
template <DecoratorSet decorators, typename BarrierSetT>
255274
template <typename T>
256275
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_atomic_cmpxchg_not_in_heap(T* addr, oop compare_value, oop new_value) {
276+
assert((decorators & (AS_NO_KEEPALIVE | ON_UNKNOWN_OOP_REF)) == 0, "must be absent");
257277
ShenandoahBarrierSet* bs = ShenandoahBarrierSet::barrier_set();
258-
bs->iu_barrier(new_value);
259-
260-
oop res;
261-
oop expected = compare_value;
262-
do {
263-
compare_value = expected;
264-
res = Raw::oop_atomic_cmpxchg(addr, compare_value, new_value);
265-
expected = res;
266-
} while ((compare_value != expected) && (resolve_forwarded(compare_value) == resolve_forwarded(expected)));
267-
268-
// Note: We don't need a keep-alive-barrier here. We already enqueue any loaded reference for SATB anyway,
269-
// because it must be the previous value.
270-
res = ShenandoahBarrierSet::barrier_set()->load_reference_barrier<decorators, T>(res, NULL);
271-
bs->satb_enqueue(res);
272-
return res;
278+
return bs->oop_cmpxchg(decorators, addr, compare_value, new_value);
273279
}
274280

275281
template <DecoratorSet decorators, typename BarrierSetT>
276282
template <typename T>
277283
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_atomic_cmpxchg_in_heap(T* addr, oop compare_value, oop new_value) {
278-
return oop_atomic_cmpxchg_not_in_heap(addr, compare_value, new_value);
284+
assert((decorators & (AS_NO_KEEPALIVE | ON_UNKNOWN_OOP_REF)) == 0, "must be absent");
285+
ShenandoahBarrierSet* bs = ShenandoahBarrierSet::barrier_set();
286+
return bs->oop_cmpxchg(decorators, addr, compare_value, new_value);
279287
}
280288

281289
template <DecoratorSet decorators, typename BarrierSetT>
282290
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_atomic_cmpxchg_in_heap_at(oop base, ptrdiff_t offset, oop compare_value, oop new_value) {
283-
return oop_atomic_cmpxchg_in_heap(AccessInternal::oop_field_addr<decorators>(base, offset), compare_value, new_value);
291+
assert((decorators & AS_NO_KEEPALIVE) == 0, "must be absent");
292+
ShenandoahBarrierSet* bs = ShenandoahBarrierSet::barrier_set();
293+
DecoratorSet resolved_decorators = AccessBarrierSupport::resolve_possibly_unknown_oop_ref_strength<decorators>(base, offset);
294+
return bs->oop_cmpxchg(resolved_decorators, AccessInternal::oop_field_addr<decorators>(base, offset), compare_value, new_value);
284295
}
285296

286297
template <DecoratorSet decorators, typename BarrierSetT>
287298
template <typename T>
288299
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_atomic_xchg_not_in_heap(T* addr, oop new_value) {
300+
assert((decorators & (AS_NO_KEEPALIVE | ON_UNKNOWN_OOP_REF)) == 0, "must be absent");
289301
ShenandoahBarrierSet* bs = ShenandoahBarrierSet::barrier_set();
290-
bs->iu_barrier(new_value);
291-
292-
oop previous = Raw::oop_atomic_xchg(addr, new_value);
293-
294-
// Note: We don't need a keep-alive-barrier here. We already enqueue any loaded reference for SATB anyway,
295-
// because it must be the previous value.
296-
previous = ShenandoahBarrierSet::barrier_set()->load_reference_barrier<decorators, T>(previous, NULL);
297-
bs->satb_enqueue(previous);
298-
return previous;
302+
return bs->oop_xchg(decorators, addr, new_value);
299303
}
300304

301305
template <DecoratorSet decorators, typename BarrierSetT>
302306
template <typename T>
303307
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_atomic_xchg_in_heap(T* addr, oop new_value) {
304-
return oop_atomic_xchg_not_in_heap(addr, new_value);
308+
assert((decorators & (AS_NO_KEEPALIVE | ON_UNKNOWN_OOP_REF)) == 0, "must be absent");
309+
ShenandoahBarrierSet* bs = ShenandoahBarrierSet::barrier_set();
310+
return bs->oop_xchg(decorators, addr, new_value);
305311
}
306312

307313
template <DecoratorSet decorators, typename BarrierSetT>
308314
inline oop ShenandoahBarrierSet::AccessBarrier<decorators, BarrierSetT>::oop_atomic_xchg_in_heap_at(oop base, ptrdiff_t offset, oop new_value) {
309-
return oop_atomic_xchg_in_heap(AccessInternal::oop_field_addr<decorators>(base, offset), new_value);
315+
assert((decorators & AS_NO_KEEPALIVE) == 0, "must be absent");
316+
ShenandoahBarrierSet* bs = ShenandoahBarrierSet::barrier_set();
317+
DecoratorSet resolved_decorators = AccessBarrierSupport::resolve_possibly_unknown_oop_ref_strength<decorators>(base, offset);
318+
return bs->oop_xchg(resolved_decorators, AccessInternal::oop_field_addr<decorators>(base, offset), new_value);
310319
}
311320

312321
// Clone barrier support

‎src/hotspot/share/gc/shenandoah/shenandoahRuntime.cpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -68,17 +68,17 @@ JRT_LEAF(void, ShenandoahRuntime::shenandoah_clone_barrier(oopDesc* src))
6868
JRT_END
6969

7070
JRT_LEAF(oopDesc*, ShenandoahRuntime::load_reference_barrier_weak(oopDesc * src, oop* load_addr))
71-
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<ON_WEAK_OOP_REF, oop>(oop(src), load_addr);
71+
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<oop>(ON_WEAK_OOP_REF, oop(src), load_addr);
7272
JRT_END
7373

7474
JRT_LEAF(oopDesc*, ShenandoahRuntime::load_reference_barrier_weak_narrow(oopDesc * src, narrowOop* load_addr))
75-
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<ON_WEAK_OOP_REF, narrowOop>(oop(src), load_addr);
75+
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<narrowOop>(ON_WEAK_OOP_REF, oop(src), load_addr);
7676
JRT_END
7777

7878
JRT_LEAF(oopDesc*, ShenandoahRuntime::load_reference_barrier_phantom(oopDesc * src, oop* load_addr))
79-
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<ON_PHANTOM_OOP_REF, oop>(oop(src), load_addr);
79+
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<oop>(ON_PHANTOM_OOP_REF, oop(src), load_addr);
8080
JRT_END
8181

8282
JRT_LEAF(oopDesc*, ShenandoahRuntime::load_reference_barrier_phantom_narrow(oopDesc * src, narrowOop* load_addr))
83-
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<ON_PHANTOM_OOP_REF, narrowOop>(oop(src), load_addr);
83+
return (oopDesc*) ShenandoahBarrierSet::barrier_set()->load_reference_barrier<narrowOop>(ON_PHANTOM_OOP_REF, oop(src), load_addr);
8484
JRT_END

0 commit comments

Comments
 (0)
Please sign in to comment.