Skip to content

Commit f84b5d2

Browse files
committedJul 27, 2020
8248362: JVMTI frame operations should use Thread-Local Handshake
Reviewed-by: sspitsyn, dholmes, dcubed
1 parent 3dba35d commit f84b5d2

File tree

5 files changed

+48
-48
lines changed

5 files changed

+48
-48
lines changed
 

‎src/hotspot/share/prims/jvmtiEnv.cpp

+10-10
Original file line numberDiff line numberDiff line change
@@ -1618,14 +1618,14 @@ JvmtiEnv::GetFrameCount(JavaThread* java_thread, jint* count_ptr) {
16181618
}
16191619

16201620
// It is only safe to perform the direct operation on the current
1621-
// thread. All other usage needs to use a vm-safepoint-op for safety.
1621+
// thread. All other usage needs to use a direct handshake for safety.
16221622
if (java_thread == JavaThread::current()) {
16231623
err = get_frame_count(state, count_ptr);
16241624
} else {
1625-
// get java stack frame count at safepoint.
1626-
VM_GetFrameCount op(this, state, count_ptr);
1627-
VMThread::execute(&op);
1628-
err = op.result();
1625+
// get java stack frame count with handshake.
1626+
GetFrameCountClosure op(this, state, count_ptr);
1627+
bool executed = Handshake::execute_direct(&op, java_thread);
1628+
err = executed ? op.result() : JVMTI_ERROR_THREAD_NOT_ALIVE;
16291629
}
16301630
return err;
16311631
} /* end GetFrameCount */
@@ -1748,14 +1748,14 @@ JvmtiEnv::GetFrameLocation(JavaThread* java_thread, jint depth, jmethodID* metho
17481748
jvmtiError err = JVMTI_ERROR_NONE;
17491749

17501750
// It is only safe to perform the direct operation on the current
1751-
// thread. All other usage needs to use a vm-safepoint-op for safety.
1751+
// thread. All other usage needs to use a direct handshake for safety.
17521752
if (java_thread == JavaThread::current()) {
17531753
err = get_frame_location(java_thread, depth, method_ptr, location_ptr);
17541754
} else {
1755-
// JVMTI get java stack frame location at safepoint.
1756-
VM_GetFrameLocation op(this, java_thread, depth, method_ptr, location_ptr);
1757-
VMThread::execute(&op);
1758-
err = op.result();
1755+
// JVMTI get java stack frame location via direct handshake.
1756+
GetFrameLocationClosure op(this, depth, method_ptr, location_ptr);
1757+
bool executed = Handshake::execute_direct(&op, java_thread);
1758+
err = executed ? op.result() : JVMTI_ERROR_THREAD_NOT_ALIVE;
17591759
}
17601760
return err;
17611761
} /* end GetFrameLocation */

‎src/hotspot/share/prims/jvmtiEnvBase.cpp

+11-13
Original file line numberDiff line numberDiff line change
@@ -900,10 +900,10 @@ JvmtiEnvBase::get_frame_location(JavaThread *java_thread, jint depth,
900900
#ifdef ASSERT
901901
uint32_t debug_bits = 0;
902902
#endif
903-
assert((SafepointSynchronize::is_at_safepoint() ||
904-
java_thread->is_thread_fully_suspended(false, &debug_bits)),
905-
"at safepoint or target thread is suspended");
906903
Thread* current_thread = Thread::current();
904+
assert(current_thread == java_thread ||
905+
current_thread == java_thread->active_handshaker(),
906+
"call by myself or at direct handshake");
907907
ResourceMark rm(current_thread);
908908

909909
vframe *vf = vframeFor(java_thread, depth);
@@ -1558,22 +1558,20 @@ GetStackTraceClosure::do_thread(Thread *target) {
15581558
}
15591559

15601560
void
1561-
VM_GetFrameCount::doit() {
1562-
_result = JVMTI_ERROR_THREAD_NOT_ALIVE;
1561+
GetFrameCountClosure::do_thread(Thread *target) {
15631562
JavaThread* jt = _state->get_thread();
1564-
ThreadsListHandle tlh;
1565-
if (jt != NULL && tlh.includes(jt) && !jt->is_exiting() && jt->threadObj() != NULL) {
1563+
assert(target == jt, "just checking");
1564+
if (!jt->is_exiting() && jt->threadObj() != NULL) {
15661565
_result = ((JvmtiEnvBase*)_env)->get_frame_count(_state, _count_ptr);
15671566
}
15681567
}
15691568

15701569
void
1571-
VM_GetFrameLocation::doit() {
1572-
_result = JVMTI_ERROR_THREAD_NOT_ALIVE;
1573-
ThreadsListHandle tlh;
1574-
if (_java_thread != NULL && tlh.includes(_java_thread)
1575-
&& !_java_thread->is_exiting() && _java_thread->threadObj() != NULL) {
1576-
_result = ((JvmtiEnvBase*)_env)->get_frame_location(_java_thread, _depth,
1570+
GetFrameLocationClosure::do_thread(Thread *target) {
1571+
assert(target->is_Java_thread(), "just checking");
1572+
JavaThread *jt = (JavaThread *)target;
1573+
if (!jt->is_exiting() && jt->threadObj() != NULL) {
1574+
_result = ((JvmtiEnvBase*)_env)->get_frame_location(jt, _depth,
15771575
_method_ptr, _location_ptr);
15781576
}
15791577
}

‎src/hotspot/share/prims/jvmtiEnvBase.hpp

+20-20
Original file line numberDiff line numberDiff line change
@@ -557,47 +557,47 @@ class GetSingleStackTraceClosure : public HandshakeClosure {
557557
jvmtiError result() { return _collector.result(); }
558558
};
559559

560-
// VM operation to count stack frames at safepoint.
561-
class VM_GetFrameCount : public VM_Operation {
560+
// HandshakeClosure to count stack frames.
561+
class GetFrameCountClosure : public HandshakeClosure {
562562
private:
563563
JvmtiEnv *_env;
564564
JvmtiThreadState *_state;
565565
jint *_count_ptr;
566566
jvmtiError _result;
567567

568568
public:
569-
VM_GetFrameCount(JvmtiEnv *env, JvmtiThreadState *state, jint *count_ptr) {
570-
_env = env;
571-
_state = state;
572-
_count_ptr = count_ptr;
569+
GetFrameCountClosure(JvmtiEnv *env, JvmtiThreadState *state, jint *count_ptr)
570+
: HandshakeClosure("GetFrameCount"),
571+
_env(env),
572+
_state(state),
573+
_count_ptr(count_ptr),
574+
_result(JVMTI_ERROR_THREAD_NOT_ALIVE) {
573575
}
574-
VMOp_Type type() const { return VMOp_GetFrameCount; }
575576
jvmtiError result() { return _result; }
576-
void doit();
577+
void do_thread(Thread *target);
577578
};
578579

579-
// VM operation to frame location at safepoint.
580-
class VM_GetFrameLocation : public VM_Operation {
580+
// HandshakeClosure to get frame location.
581+
class GetFrameLocationClosure : public HandshakeClosure {
581582
private:
582583
JvmtiEnv *_env;
583-
JavaThread* _java_thread;
584584
jint _depth;
585585
jmethodID* _method_ptr;
586586
jlocation* _location_ptr;
587587
jvmtiError _result;
588588

589589
public:
590-
VM_GetFrameLocation(JvmtiEnv *env, JavaThread* java_thread, jint depth,
591-
jmethodID* method_ptr, jlocation* location_ptr) {
592-
_env = env;
593-
_java_thread = java_thread;
594-
_depth = depth;
595-
_method_ptr = method_ptr;
596-
_location_ptr = location_ptr;
590+
GetFrameLocationClosure(JvmtiEnv *env, jint depth,
591+
jmethodID* method_ptr, jlocation* location_ptr)
592+
: HandshakeClosure("GetFrameLocation"),
593+
_env(env),
594+
_depth(depth),
595+
_method_ptr(method_ptr),
596+
_location_ptr(location_ptr),
597+
_result(JVMTI_ERROR_THREAD_NOT_ALIVE) {
597598
}
598-
VMOp_Type type() const { return VMOp_GetFrameLocation; }
599599
jvmtiError result() { return _result; }
600-
void doit();
600+
void do_thread(Thread *target);
601601
};
602602

603603

‎src/hotspot/share/prims/jvmtiThreadState.cpp

+7-3
Original file line numberDiff line numberDiff line change
@@ -216,9 +216,13 @@ void JvmtiThreadState::leave_interp_only_mode() {
216216

217217
// Helper routine used in several places
218218
int JvmtiThreadState::count_frames() {
219-
guarantee(SafepointSynchronize::is_at_safepoint() ||
220-
(JavaThread *)Thread::current() == get_thread(),
221-
"must be current thread or at safepoint");
219+
#ifdef ASSERT
220+
Thread *current_thread = Thread::current();
221+
#endif
222+
assert(current_thread == get_thread() ||
223+
SafepointSynchronize::is_at_safepoint() ||
224+
current_thread == get_thread()->active_handshaker(),
225+
"call by myself / at safepoint / at handshake");
222226

223227
if (!get_thread()->has_last_Java_frame()) return 0; // no Java frames
224228

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

-2
Original file line numberDiff line numberDiff line change
@@ -81,8 +81,6 @@
8181
template(GetObjectMonitorUsage) \
8282
template(GetAllStackTraces) \
8383
template(GetThreadListStackTraces) \
84-
template(GetFrameCount) \
85-
template(GetFrameLocation) \
8684
template(ChangeBreakpoints) \
8785
template(GetOrSetLocal) \
8886
template(GetCurrentLocation) \

0 commit comments

Comments
 (0)
Please sign in to comment.