@@ -59,40 +59,51 @@ inline void StackWatermark::ensure_safe(frame f) {
59
59
return ;
60
60
}
61
61
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 ());
63
66
64
- if (is_above_watermark (reinterpret_cast < uintptr_t >(f. sp ()) , watermark ())) {
67
+ if (is_above_watermark (f_fp , watermark ())) {
65
68
process_one ();
66
69
}
67
70
68
71
assert (is_frame_safe (f), " frame should be safe after processing" );
69
- assert (!is_above_watermark (reinterpret_cast <uintptr_t >(f.sp ()), watermark ()), " " );
70
72
}
71
73
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 ();
75
76
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);
78
81
}
79
82
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" );
91
85
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" );
94
104
95
- ensure_safe (caller);
105
+ // after_unwind() potentially exposes the top frame.
106
+ ensure_safe (f);
96
107
}
97
108
98
109
inline void StackWatermark::on_iteration (frame f) {
0 commit comments