Skip to content

Commit 4e20a03

Browse files
committedApr 4, 2022
8283044: Use asynchronous handshakes to deliver asynchronous exceptions
Reviewed-by: dcubed, dholmes, rehn
1 parent 9d200d6 commit 4e20a03

20 files changed

+763
-253
lines changed
 

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

+3-1
Original file line numberDiff line numberDiff line change
@@ -549,7 +549,9 @@ JNI_END
549549

550550
static void jni_check_async_exceptions(JavaThread *thread) {
551551
assert(thread == Thread::current(), "must be itself");
552-
thread->check_and_handle_async_exceptions();
552+
if (thread->has_async_exception_condition()) {
553+
SafepointMechanism::process_if_requested_with_exit_check(thread, true /* check asyncs */);
554+
}
553555
}
554556

555557
JNI_ENTRY_NO_PRESERVE(jthrowable, jni_ExceptionOccurred(JNIEnv *env))

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

+72-23
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,8 @@
2424

2525
#include "precompiled.hpp"
2626
#include "jvm_io.h"
27+
#include "classfile/javaClasses.hpp"
28+
#include "classfile/vmSymbols.hpp"
2729
#include "logging/log.hpp"
2830
#include "logging/logStream.hpp"
2931
#include "memory/resourceArea.hpp"
@@ -78,6 +80,8 @@ class HandshakeOperation : public CHeapObj<mtThread> {
7880
const char* name() { return _handshake_cl->name(); }
7981
bool is_async() { return _handshake_cl->is_async(); }
8082
bool is_suspend() { return _handshake_cl->is_suspend(); }
83+
bool is_async_exception() { return _handshake_cl->is_async_exception(); }
84+
bool is_ThreadDeath() { return _handshake_cl->is_ThreadDeath(); }
8185
};
8286

8387
class AsyncHandshakeOperation : public HandshakeOperation {
@@ -313,7 +317,6 @@ void HandshakeOperation::do_handshake(JavaThread* thread) {
313317

314318
// Only actually execute the operation for non terminated threads.
315319
if (!thread->is_terminated()) {
316-
NoSafepointVerifier nsv;
317320
_handshake_cl->do_thread(thread);
318321
}
319322

@@ -426,6 +429,7 @@ HandshakeState::HandshakeState(JavaThread* target) :
426429
_queue(),
427430
_lock(Monitor::nosafepoint, "HandshakeState_lock"),
428431
_active_handshaker(),
432+
_async_exceptions_blocked(false),
429433
_suspended(false),
430434
_async_suspend_handshake(false)
431435
{
@@ -443,39 +447,61 @@ bool HandshakeState::operation_pending(HandshakeOperation* op) {
443447
return _queue.contains(mo);
444448
}
445449

446-
static bool no_suspend_filter(HandshakeOperation* op) {
447-
return !op->is_suspend();
450+
// Filters
451+
static bool non_self_executable_filter(HandshakeOperation* op) {
452+
return !op->is_async();
453+
}
454+
static bool no_async_exception_filter(HandshakeOperation* op) {
455+
return !op->is_async_exception();
456+
}
457+
static bool async_exception_filter(HandshakeOperation* op) {
458+
return op->is_async_exception();
459+
}
460+
static bool is_ThreadDeath_filter(HandshakeOperation* op) {
461+
return op->is_ThreadDeath();
462+
}
463+
static bool no_suspend_no_async_exception_filter(HandshakeOperation* op) {
464+
return !op->is_suspend() && !op->is_async_exception();
448465
}
449466

450-
HandshakeOperation* HandshakeState::get_op_for_self(bool allow_suspend) {
467+
HandshakeOperation* HandshakeState::get_op_for_self(bool allow_suspend, bool check_async_exception) {
451468
assert(_handshakee == Thread::current(), "Must be called by self");
452469
assert(_lock.owned_by_self(), "Lock must be held");
453-
if (allow_suspend) {
470+
assert(allow_suspend || !check_async_exception, "invalid case");
471+
if (!allow_suspend) {
472+
return _queue.peek(no_suspend_no_async_exception_filter);
473+
} else if (check_async_exception && !_async_exceptions_blocked) {
454474
return _queue.peek();
455475
} else {
456-
return _queue.peek(no_suspend_filter);
476+
return _queue.peek(no_async_exception_filter);
457477
}
458478
}
459479

460-
static bool non_self_queue_filter(HandshakeOperation* op) {
461-
return !op->is_async();
480+
bool HandshakeState::has_operation(bool allow_suspend, bool check_async_exception) {
481+
MutexLocker ml(&_lock, Mutex::_no_safepoint_check_flag);
482+
return get_op_for_self(allow_suspend, check_async_exception) != NULL;
483+
}
484+
485+
bool HandshakeState::has_async_exception_operation(bool ThreadDeath_only) {
486+
if (!has_operation()) return false;
487+
MutexLocker ml(_lock.owned_by_self() ? NULL : &_lock, Mutex::_no_safepoint_check_flag);
488+
if (!ThreadDeath_only) {
489+
return _queue.peek(async_exception_filter) != NULL;
490+
} else {
491+
return _queue.peek(is_ThreadDeath_filter) != NULL;
492+
}
462493
}
463494

464495
bool HandshakeState::have_non_self_executable_operation() {
465496
assert(_handshakee != Thread::current(), "Must not be called by self");
466497
assert(_lock.owned_by_self(), "Lock must be held");
467-
return _queue.contains(non_self_queue_filter);
468-
}
469-
470-
bool HandshakeState::has_a_non_suspend_operation() {
471-
MutexLocker ml(&_lock, Mutex::_no_safepoint_check_flag);
472-
return _queue.contains(no_suspend_filter);
498+
return _queue.contains(non_self_executable_filter);
473499
}
474500

475501
HandshakeOperation* HandshakeState::get_op() {
476502
assert(_handshakee != Thread::current(), "Must not be called by self");
477503
assert(_lock.owned_by_self(), "Lock must be held");
478-
return _queue.peek(non_self_queue_filter);
504+
return _queue.peek(non_self_executable_filter);
479505
};
480506

481507
void HandshakeState::remove_op(HandshakeOperation* op) {
@@ -485,23 +511,20 @@ void HandshakeState::remove_op(HandshakeOperation* op) {
485511
assert(ret == op, "Popped op must match requested op");
486512
};
487513

488-
bool HandshakeState::process_by_self(bool allow_suspend) {
514+
bool HandshakeState::process_by_self(bool allow_suspend, bool check_async_exception) {
489515
assert(Thread::current() == _handshakee, "should call from _handshakee");
490516
assert(!_handshakee->is_terminated(), "should not be a terminated thread");
491517

492518
_handshakee->frame_anchor()->make_walkable(_handshakee);
493519
// Threads shouldn't block if they are in the middle of printing, but...
494520
ttyLocker::break_tty_lock_for_safepoint(os::current_thread_id());
495521

496-
// Handshakes cannot safely safepoint.
497-
// The exception to this rule is the asynchronous suspension handshake.
498-
// It by-passes the NSV by manually doing the transition.
499-
NoSafepointVerifier nsv;
500-
501522
while (has_operation()) {
523+
// Handshakes cannot safely safepoint. The exceptions to this rule are
524+
// the asynchronous suspension and unsafe access error handshakes.
502525
MutexLocker ml(&_lock, Mutex::_no_safepoint_check_flag);
503526

504-
HandshakeOperation* op = get_op_for_self(allow_suspend);
527+
HandshakeOperation* op = get_op_for_self(allow_suspend, check_async_exception);
505528
if (op != NULL) {
506529
assert(op->_target == NULL || op->_target == Thread::current(), "Wrong thread");
507530
bool async = op->is_async();
@@ -517,8 +540,8 @@ bool HandshakeState::process_by_self(bool allow_suspend) {
517540
// An asynchronous handshake may put the JavaThread in blocked state (safepoint safe).
518541
// The destructor ~PreserveExceptionMark touches the exception oop so it must not be executed,
519542
// since a safepoint may be in-progress when returning from the async handshake.
520-
op->do_handshake(_handshakee); // acquire, op removed after
521543
remove_op(op);
544+
op->do_handshake(_handshakee);
522545
log_handshake_info(((AsyncHandshakeOperation*)op)->start_time(), op->name(), 1, 0, "asynchronous");
523546
delete op;
524547
return true; // Must check for safepoints
@@ -730,3 +753,29 @@ bool HandshakeState::resume() {
730753
_lock.notify();
731754
return true;
732755
}
756+
757+
void HandshakeState::handle_unsafe_access_error() {
758+
if (is_suspended()) {
759+
// A suspend handshake was added to the queue after the
760+
// unsafe access error. Since the suspender has already
761+
// considered this JT as suspended and assumes it won't go
762+
// back to Java until resumed we cannot create the exception
763+
// object yet. Add a new unsafe access error operation to
764+
// the end of the queue and try again in the next attempt.
765+
Handshake::execute(new UnsafeAccessErrorHandshake(), _handshakee);
766+
log_info(handshake)("JavaThread " INTPTR_FORMAT " skipping unsafe access processing due to suspend.", p2i(_handshakee));
767+
return;
768+
}
769+
// Release the handshake lock before constructing the oop to
770+
// avoid deadlocks since that can block. This will allow the
771+
// JavaThread to execute normally as if it was outside a handshake.
772+
// We will reacquire the handshake lock at return from ~MutexUnlocker.
773+
MutexUnlocker ml(&_lock, Mutex::_no_safepoint_check_flag);
774+
// We may be at method entry which requires we save the do-not-unlock flag.
775+
UnlockFlagSaver fs(_handshakee);
776+
Handle h_exception = Exceptions::new_exception(_handshakee, vmSymbols::java_lang_InternalError(), "a fault occurred in an unsafe memory access operation");
777+
if (h_exception()->is_a(vmClasses::InternalError_klass())) {
778+
java_lang_InternalError::set_during_unsafe_access(h_exception());
779+
}
780+
_handshakee->handle_async_exception(h_exception());
781+
}

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

+18-6
Original file line numberDiff line numberDiff line change
@@ -33,9 +33,11 @@
3333
#include "utilities/filterQueue.hpp"
3434

3535
class HandshakeOperation;
36+
class AsyncHandshakeOperation;
3637
class JavaThread;
3738
class SuspendThreadHandshake;
3839
class ThreadSelfSuspensionHandshake;
40+
class UnsafeAccessErrorHandshake;
3941
class ThreadsListHandle;
4042

4143
// A handshake closure is a callback that is executed for a JavaThread
@@ -51,6 +53,8 @@ class HandshakeClosure : public ThreadClosure, public CHeapObj<mtThread> {
5153
const char* name() const { return _name; }
5254
virtual bool is_async() { return false; }
5355
virtual bool is_suspend() { return false; }
56+
virtual bool is_async_exception() { return false; }
57+
virtual bool is_ThreadDeath() { return false; }
5458
virtual void do_thread(Thread* thread) = 0;
5559
};
5660

@@ -87,6 +91,7 @@ class JvmtiRawMonitor;
8791
class HandshakeState {
8892
friend ThreadSelfSuspensionHandshake;
8993
friend SuspendThreadHandshake;
94+
friend UnsafeAccessErrorHandshake;
9095
friend JavaThread;
9196
// This a back reference to the JavaThread,
9297
// the target for all operation in the queue.
@@ -104,7 +109,7 @@ class HandshakeState {
104109
bool can_process_handshake();
105110

106111
bool have_non_self_executable_operation();
107-
HandshakeOperation* get_op_for_self(bool allow_suspend);
112+
HandshakeOperation* get_op_for_self(bool allow_suspend, bool check_async_exception);
108113
HandshakeOperation* get_op();
109114
void remove_op(HandshakeOperation* op);
110115

@@ -124,17 +129,16 @@ class HandshakeState {
124129

125130
void add_operation(HandshakeOperation* op);
126131

127-
bool has_operation() {
128-
return !_queue.is_empty();
129-
}
130-
bool has_a_non_suspend_operation();
132+
bool has_operation() { return !_queue.is_empty(); }
133+
bool has_operation(bool allow_suspend, bool check_async_exception);
134+
bool has_async_exception_operation(bool ThreadDeath_only);
131135

132136
bool operation_pending(HandshakeOperation* op);
133137

134138
// If the method returns true we need to check for a possible safepoint.
135139
// This is due to a suspension handshake which put the JavaThread in blocked
136140
// state so a safepoint may be in-progress.
137-
bool process_by_self(bool allow_suspend);
141+
bool process_by_self(bool allow_suspend, bool check_async_exception);
138142

139143
enum ProcessResult {
140144
_no_operation = 0,
@@ -148,6 +152,14 @@ class HandshakeState {
148152

149153
Thread* active_handshaker() const { return Atomic::load(&_active_handshaker); }
150154

155+
// Support for asynchronous exceptions
156+
private:
157+
bool _async_exceptions_blocked;
158+
159+
bool async_exceptions_blocked() { return _async_exceptions_blocked; }
160+
void set_async_exceptions_blocked(bool b) { _async_exceptions_blocked = b; }
161+
void handle_unsafe_access_error();
162+
151163
// Suspend/resume support
152164
private:
153165
// This flag is true when the thread owning this

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -210,7 +210,7 @@ class ThreadBlockInVMPreprocess : public ThreadStateTransition {
210210

211211
if (SafepointMechanism::should_process(_thread, _allow_suspend)) {
212212
_pr(_thread);
213-
SafepointMechanism::process_if_requested(_thread, _allow_suspend);
213+
SafepointMechanism::process_if_requested(_thread, _allow_suspend, false /* check_async_exception */);
214214
}
215215
}
216216
};

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

+4-1
Original file line numberDiff line numberDiff line change
@@ -1897,7 +1897,10 @@ int ObjectMonitor::TrySpin(JavaThread* current) {
18971897
// This is in keeping with the "no loitering in runtime" rule.
18981898
// We periodically check to see if there's a safepoint pending.
18991899
if ((ctr & 0xFF) == 0) {
1900-
if (SafepointMechanism::should_process(current)) {
1900+
// Can't call SafepointMechanism::should_process() since that
1901+
// might update the poll values and we could be in a thread_blocked
1902+
// state here which is not allowed so just check the poll.
1903+
if (SafepointMechanism::local_poll_armed(current)) {
19011904
goto Abort; // abrupt spin egress
19021905
}
19031906
SpinPause();

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

+17-16
Original file line numberDiff line numberDiff line change
@@ -963,33 +963,34 @@ void ThreadSafepointState::handle_polling_page_exception() {
963963
set_at_poll_safepoint(true);
964964
// Process pending operation
965965
// We never deliver an async exception at a polling point as the
966-
// compiler may not have an exception handler for it. The polling
967-
// code will notice the pending async exception, deoptimize and
968-
// the exception will be delivered. (Polling at a return point
969-
// is ok though). Sure is a lot of bother for a deprecated feature...
966+
// compiler may not have an exception handler for it (polling at
967+
// a return point is ok though). We will check for a pending async
968+
// exception below and deoptimize if needed. We also cannot deoptimize
969+
// and still install the exception here because live registers needed
970+
// during deoptimization are clobbered by the exception path. The
971+
// exception will just be delivered once we get into the interpreter.
970972
SafepointMechanism::process_if_requested_with_exit_check(self, false /* check asyncs */);
971973
set_at_poll_safepoint(false);
972974

973-
// If we have a pending async exception deoptimize the frame
974-
// as otherwise we may never deliver it.
975975
if (self->has_async_exception_condition()) {
976976
Deoptimization::deoptimize_frame(self, caller_fr.id());
977+
log_info(exceptions)("deferred async exception at compiled safepoint");
977978
}
978979

979-
// If an exception has been installed we must check for a pending deoptimization
980-
// Deoptimize frame if exception has been thrown.
981-
980+
// If an exception has been installed we must verify that the top frame wasn't deoptimized.
982981
if (self->has_pending_exception() ) {
983982
RegisterMap map(self, true, false);
984983
frame caller_fr = stub_fr.sender(&map);
985984
if (caller_fr.is_deoptimized_frame()) {
986-
// The exception patch will destroy registers that are still
987-
// live and will be needed during deoptimization. Defer the
988-
// Async exception should have deferred the exception until the
989-
// next safepoint which will be detected when we get into
990-
// the interpreter so if we have an exception now things
991-
// are messed up.
992-
985+
// The exception path will destroy registers that are still
986+
// live and will be needed during deoptimization, so if we
987+
// have an exception now things are messed up. We only check
988+
// at this scope because for a poll return it is ok to deoptimize
989+
// while having a pending exception since the call we are returning
990+
// from already collides with exception handling registers and
991+
// so there is no issue (the exception handling path kills call
992+
// result registers but this is ok since the exception kills
993+
// the result anyway).
993994
fatal("Exception installed and deoptimization is pending");
994995
}
995996
}

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

+2-2
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,7 @@ void SafepointMechanism::update_poll_values(JavaThread* thread) {
111111
}
112112
}
113113

114-
void SafepointMechanism::process(JavaThread *thread, bool allow_suspend) {
114+
void SafepointMechanism::process(JavaThread *thread, bool allow_suspend, bool check_async_exception) {
115115
// Read global poll and has_handshake after local poll
116116
OrderAccess::loadload();
117117

@@ -135,7 +135,7 @@ void SafepointMechanism::process(JavaThread *thread, bool allow_suspend) {
135135
// 3) Before the handshake code is run
136136
StackWatermarkSet::on_safepoint(thread);
137137

138-
need_rechecking = thread->handshake_state()->has_operation() && thread->handshake_state()->process_by_self(allow_suspend);
138+
need_rechecking = thread->handshake_state()->has_operation() && thread->handshake_state()->process_by_self(allow_suspend, check_async_exception);
139139
} while (need_rechecking);
140140

141141
update_poll_values(thread);

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

+3-3
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ class SafepointMechanism : public AllStatic {
4949

5050
static inline bool global_poll();
5151

52-
static void process(JavaThread *thread, bool allow_suspend);
52+
static void process(JavaThread *thread, bool allow_suspend, bool check_async_exception);
5353

5454
static void default_initialize();
5555

@@ -80,8 +80,8 @@ class SafepointMechanism : public AllStatic {
8080
static inline bool should_process(JavaThread* thread, bool allow_suspend = true);
8181

8282
// Processes a pending requested operation.
83-
static inline void process_if_requested(JavaThread* thread, bool allow_suspend = true);
84-
static inline void process_if_requested_with_exit_check(JavaThread* thread, bool check_asyncs);
83+
static inline void process_if_requested(JavaThread* thread, bool allow_suspend, bool check_async_exception);
84+
static inline void process_if_requested_with_exit_check(JavaThread* thread, bool check_async_exception);
8585
// Compute what the poll values should be and install them.
8686
static void update_poll_values(JavaThread* thread);
8787

0 commit comments

Comments
 (0)
Please sign in to comment.