Skip to content

Commit acec564

Browse files
committedSep 10, 2020
ZGC: Review 6
1 parent d73b749 commit acec564

12 files changed

+77
-55
lines changed
 

‎src/hotspot/share/c1/c1_Runtime1.cpp

+10-1
Original file line numberDiff line numberDiff line change
@@ -512,7 +512,16 @@ JRT_ENTRY_NO_ASYNC(static address, exception_handler_for_pc_helper(JavaThread* t
512512
// This function is called when we are about to throw an exception. Therefore,
513513
// we have to poll the stack watermark barrier to make sure that not yet safe
514514
// stack frames are made safe before returning into them.
515-
StackWatermarkSet::on_unwind(thread);
515+
if (thread->last_frame().is_runtime_frame()) {
516+
// The Runtime1::handle_exception_from_callee_id handler is invoked after the
517+
// frame has been unwinded. It instead builds its own stub frame, to call the
518+
// runtime. But the throwing frame has already been unwinded here.
519+
StackWatermarkSet::after_unwind(thread);
520+
} else {
521+
// The other C1 exception handler call in here before the throwing frame has
522+
// been unwinded.
523+
StackWatermarkSet::before_unwind(thread);
524+
}
516525

517526
nm = CodeCache::find_nmethod(pc);
518527
assert(nm != NULL, "this is not an nmethod");

‎src/hotspot/share/interpreter/interpreterRuntime.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -1163,7 +1163,7 @@ JRT_ENTRY(void, InterpreterRuntime::at_safepoint(JavaThread* thread))
11631163
// This function is called by the interpreter when single stepping. Such single
11641164
// stepping could unwind a frame. Then, it is important that we process any frames
11651165
// that we might return into.
1166-
StackWatermarkSet::on_unwind(thread);
1166+
StackWatermarkSet::before_unwind(thread);
11671167

11681168
// We are called during regular safepoints and when the VM is
11691169
// single stepping. If any thread is marked for single stepping,
@@ -1184,7 +1184,7 @@ JRT_ENTRY(void, InterpreterRuntime::at_unwind(JavaThread* thread))
11841184
// to single step when unwinding frames for an exception being thrown. Instead,
11851185
// such single stepping code will use the safepoint table, which will use the
11861186
// InterpreterRuntime::at_safepoint callback.
1187-
StackWatermarkSet::on_unwind(thread);
1187+
StackWatermarkSet::before_unwind(thread);
11881188
JRT_END
11891189

11901190
JRT_ENTRY(void, InterpreterRuntime::post_field_access(JavaThread *thread, oopDesc* obj,

‎src/hotspot/share/opto/runtime.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -1290,7 +1290,7 @@ JRT_ENTRY_NO_ASYNC(address, OptoRuntime::handle_exception_C_helper(JavaThread* t
12901290
// We get here when we are about to throw an exception. The exception could
12911291
// return into a not yet safe to use frame. We catch such conditions in the
12921292
// following stack watermark barrier poll.
1293-
StackWatermarkSet::on_unwind(thread);
1293+
StackWatermarkSet::before_unwind(thread);
12941294

12951295
// Do not confuse exception_oop with pending_exception. The exception_oop
12961296
// is only used to pass arguments into the method. Not for general
@@ -1472,7 +1472,7 @@ address OptoRuntime::rethrow_C(oopDesc* exception, JavaThread* thread, address r
14721472
// The frame we rethrow the exception to might not have been processed by the GC yet.
14731473
// The stack watermark barrier takes care of detecting that and ensuring the frame
14741474
// has updated oops.
1475-
StackWatermarkSet::on_unwind(thread);
1475+
StackWatermarkSet::after_unwind(thread);
14761476

14771477
#ifndef PRODUCT
14781478
SharedRuntime::_rethrow_ctr++; // count rethrows

‎src/hotspot/share/runtime/deoptimization.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -258,7 +258,7 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread
258258
// When we get here we are about to unwind a frame. In order to catch not yet
259259
// safe to use frames, the following stack watermark barrier poll will make
260260
// such frames safe to use.
261-
StackWatermarkSet::on_unwind(thread);
261+
StackWatermarkSet::before_unwind(thread);
262262

263263
// Note: there is a safepoint safety issue here. No matter whether we enter
264264
// via vanilla deopt or uncommon trap we MUST NOT stop at a safepoint once

‎src/hotspot/share/runtime/safepoint.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -995,9 +995,9 @@ void ThreadSafepointState::handle_polling_page_exception() {
995995
}
996996

997997
// We get here if compiled return polls found a reason to call into the VM.
998-
// One condition for that is that a frame above is not yet safe to use.
998+
// One condition for that is that the top frame is not yet safe to use.
999999
// The following stack watermark barrier poll will catch such situations.
1000-
StackWatermarkSet::on_unwind(thread());
1000+
StackWatermarkSet::after_unwind(thread());
10011001

10021002
// Process pending operation
10031003
SafepointMechanism::process_operation_if_requested_slow(thread());

‎src/hotspot/share/runtime/sharedRuntime.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -522,7 +522,7 @@ JRT_LEAF(address, SharedRuntime::exception_handler_for_return_address(JavaThread
522522
// This is called when we are about to throw an exception. We must make sure the
523523
// frame we are unwinding to is safe to be access w.r.t. concurrent stack processing.
524524
// The stack watermark code will take care of ensuring that.
525-
StackWatermarkSet::on_unwind(thread);
525+
StackWatermarkSet::after_unwind(thread);
526526
return raw_exception_handler_for_return_address(thread, return_address);
527527
JRT_END
528528

@@ -3032,7 +3032,7 @@ JRT_LEAF(intptr_t*, SharedRuntime::OSR_migration_begin( JavaThread *thread) )
30323032
// before it gets unwinded. This is helpful as the size of the compiled frame could be
30333033
// larger than the interpreted frame, which could result in the new frame not being
30343034
// processed correctly.
3035-
StackWatermarkSet::on_unwind(thread);
3035+
StackWatermarkSet::before_unwind(thread);
30363036

30373037
//
30383038
// This code is dependent on the memory layout of the interpreter local

‎src/hotspot/share/runtime/stackWatermark.cpp

+2-17
Original file line numberDiff line numberDiff line change
@@ -169,24 +169,9 @@ StackWatermark::~StackWatermark() {
169169
delete _iterator;
170170
}
171171

172-
bool StackWatermark::is_frame_processed(frame fr) {
173-
MutexLocker ml(&_lock, Mutex::_no_safepoint_check_flag);
174-
uint32_t state = Atomic::load(&_state);
175-
if (StackWatermarkState::epoch(state) != epoch_id()) {
176-
return false;
177-
}
178-
if (StackWatermarkState::is_done(state)) {
179-
return true;
180-
}
181-
if (_iterator != NULL) {
182-
return reinterpret_cast<uintptr_t>(fr.sp()) <= _iterator->caller();
183-
}
184-
return true;
185-
}
186-
187-
// A frame is "safe" it it *and* its caller has been processed. This is the invariant
172+
// A frame is "safe" if it *and* its caller have been processed. This is the invariant
188173
// that allows exposing a frame, and for that frame to directly access its caller frame
189-
// without going through any hooks. This is a stricter guarantee than is_frame_processed().
174+
// without going through any hooks.
190175
bool StackWatermark::is_frame_safe(frame f) {
191176
MutexLocker ml(&_lock, Mutex::_no_safepoint_check_flag);
192177
uint32_t state = Atomic::load(&_state);

‎src/hotspot/share/runtime/stackWatermark.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,6 @@ class StackWatermark : public CHeapObj<mtInternal> {
6464
void yield_processing();
6565
static bool has_barrier(frame& f);
6666
void ensure_safe(frame f);
67-
bool is_frame_processed(frame f);
6867
bool is_frame_safe(frame f);
6968

7069
// API for consumers of the stack watermark barrier.
@@ -94,7 +93,8 @@ class StackWatermark : public CHeapObj<mtInternal> {
9493

9594
uintptr_t last_processed();
9695

97-
void on_unwind();
96+
void before_unwind();
97+
void after_unwind();
9898
void on_iteration(frame fr);
9999
void start_iteration();
100100
void finish_iteration(void* context);

‎src/hotspot/share/runtime/stackWatermark.inline.hpp

+33-22
Original file line numberDiff line numberDiff line change
@@ -59,40 +59,51 @@ inline void StackWatermark::ensure_safe(frame f) {
5959
return;
6060
}
6161

62-
assert(is_frame_processed(f), "frame should be safe before processing");
62+
// Get caller sp to get the callee fp.
63+
RegisterMap map(_jt, false /* update_map */, false /* process_frames */);
64+
frame f_caller = f.sender(&map);
65+
uintptr_t f_fp = reinterpret_cast<uintptr_t>(f_caller.sp());
6366

64-
if (is_above_watermark(reinterpret_cast<uintptr_t>(f.sp()), watermark())) {
67+
if (is_above_watermark(f_fp, watermark())) {
6568
process_one();
6669
}
6770

6871
assert(is_frame_safe(f), "frame should be safe after processing");
69-
assert(!is_above_watermark(reinterpret_cast<uintptr_t>(f.sp()), watermark()), "");
7072
}
7173

72-
inline void StackWatermark::on_unwind() {
73-
const frame& f = _jt->last_frame();
74-
assert(is_frame_safe(f), "frame should be safe after processing");
74+
inline void StackWatermark::before_unwind() {
75+
frame f = _jt->last_frame();
7576

76-
if (f.is_first_frame()) {
77-
return;
77+
// Skip any stub frames etc up until the frame that triggered before_unwind().
78+
RegisterMap map(_jt, false /* update_map */, false /* process_frames */);
79+
if (f.is_safepoint_blob_frame() || f.is_runtime_frame()) {
80+
f = f.sender(&map);
7881
}
7982

80-
// on_unwind() potentially exposes a new frame. The new exposed frame is
81-
// always the caller of the top frame, but for two different reasons.
82-
//
83-
// 1) Return sites in nmethods unwind the frame *before* polling. In other
84-
// words, the frame of the nmethod performing the poll, will not be
85-
// on-stack when it gets to the runtime. However, it trampolines into the
86-
// runtime with a safepoint blob, which will be the top frame. Therefore,
87-
// the caller of the safepoint blob, will be the new exposed frame.
88-
//
89-
// 2) All other calls to on_unwind() perform the unwinding *after* polling.
90-
// Therefore, the caller of the top frame will be the new exposed frame.
83+
assert(is_frame_safe(f), "frame should be safe before processing");
84+
assert(!f.is_runtime_frame(), "should have skipped all runtime stubs");
9185

92-
RegisterMap map(_jt, false /* update_map */, false /* process_frames */);
93-
const frame& caller = f.sender(&map);
86+
// before_unwind() potentially exposes a new frame. The new exposed frame is
87+
// always the caller of the top frame.
88+
if (!f.is_first_frame()) {
89+
f = f.sender(&map);
90+
ensure_safe(f);
91+
}
92+
}
93+
94+
inline void StackWatermark::after_unwind() {
95+
frame f = _jt->last_frame();
96+
97+
if (f.is_safepoint_blob_frame() || f.is_runtime_frame()) {
98+
// Skip safepoint blob.
99+
RegisterMap map(_jt, false /* update_map */, false /* process_frames */);
100+
f = f.sender(&map);
101+
}
102+
103+
assert(!f.is_runtime_frame(), "should have skipped all runtime stubs");
94104

95-
ensure_safe(caller);
105+
// after_unwind() potentially exposes the top frame.
106+
ensure_safe(f);
96107
}
97108

98109
inline void StackWatermark::on_iteration(frame f) {

‎src/hotspot/share/runtime/stackWatermarkSet.cpp

+16-2
Original file line numberDiff line numberDiff line change
@@ -63,15 +63,29 @@ static void verify_poll_context() {
6363
#endif
6464
}
6565

66-
void StackWatermarkSet::on_unwind(JavaThread* jt) {
66+
void StackWatermarkSet::before_unwind(JavaThread* jt) {
6767
verify_poll_context();
6868
if (!jt->has_last_Java_frame()) {
6969
// Sometimes we throw exceptions and use native transitions on threads that
7070
// do not have any Java threads. Skip those callsites.
7171
return;
7272
}
7373
for (StackWatermark* current = jt->stack_watermark_set()->_head; current != NULL; current = current->next()) {
74-
current->on_unwind();
74+
current->before_unwind();
75+
}
76+
SafepointMechanism::update_poll_values(jt);
77+
}
78+
79+
80+
void StackWatermarkSet::after_unwind(JavaThread* jt) {
81+
verify_poll_context();
82+
if (!jt->has_last_Java_frame()) {
83+
// Sometimes we throw exceptions and use native transitions on threads that
84+
// do not have any Java threads. Skip those callsites.
85+
return;
86+
}
87+
for (StackWatermark* current = jt->stack_watermark_set()->_head; current != NULL; current = current->next()) {
88+
current->after_unwind();
7589
}
7690
SafepointMechanism::update_poll_values(jt);
7791
}

‎src/hotspot/share/runtime/stackWatermarkSet.hpp

+4-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,10 @@ class StackWatermarkSet {
4242

4343
public:
4444
// Called when a thread is about to unwind a frame
45-
static void on_unwind(JavaThread* jt);
45+
static void before_unwind(JavaThread* jt);
46+
47+
// Called when a thread just unwinded a frame
48+
static void after_unwind(JavaThread* jt);
4649

4750
// Called by stack walkers when walking into a frame
4851
static void on_iteration(JavaThread* jt, frame fr);

‎src/hotspot/share/runtime/thread.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -2687,7 +2687,7 @@ void JavaThread::check_safepoint_and_suspend_for_native_trans(JavaThread *thread
26872687
// After returning from native, it could be that the stack frames are not
26882688
// yet safe to use. We catch such situations in the subsequent stack watermark
26892689
// barrier, which will trap unsafe stack frames.
2690-
StackWatermarkSet::on_unwind(thread);
2690+
StackWatermarkSet::before_unwind(thread);
26912691

26922692
JFR_ONLY(SUSPEND_THREAD_CONDITIONAL(thread);)
26932693
}

0 commit comments

Comments
 (0)
Please sign in to comment.