24
24
25
25
#include " precompiled.hpp"
26
26
#include " jvm_io.h"
27
+ #include " classfile/javaClasses.hpp"
28
+ #include " classfile/vmSymbols.hpp"
27
29
#include " logging/log.hpp"
28
30
#include " logging/logStream.hpp"
29
31
#include " memory/resourceArea.hpp"
@@ -78,6 +80,8 @@ class HandshakeOperation : public CHeapObj<mtThread> {
78
80
const char * name () { return _handshake_cl->name (); }
79
81
bool is_async () { return _handshake_cl->is_async (); }
80
82
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 (); }
81
85
};
82
86
83
87
class AsyncHandshakeOperation : public HandshakeOperation {
@@ -313,7 +317,6 @@ void HandshakeOperation::do_handshake(JavaThread* thread) {
313
317
314
318
// Only actually execute the operation for non terminated threads.
315
319
if (!thread->is_terminated ()) {
316
- NoSafepointVerifier nsv;
317
320
_handshake_cl->do_thread (thread);
318
321
}
319
322
@@ -426,6 +429,7 @@ HandshakeState::HandshakeState(JavaThread* target) :
426
429
_queue(),
427
430
_lock(Monitor::nosafepoint, " HandshakeState_lock" ),
428
431
_active_handshaker(),
432
+ _async_exceptions_blocked(false ),
429
433
_suspended(false ),
430
434
_async_suspend_handshake(false )
431
435
{
@@ -443,39 +447,61 @@ bool HandshakeState::operation_pending(HandshakeOperation* op) {
443
447
return _queue.contains (mo);
444
448
}
445
449
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 ();
448
465
}
449
466
450
- HandshakeOperation* HandshakeState::get_op_for_self (bool allow_suspend) {
467
+ HandshakeOperation* HandshakeState::get_op_for_self (bool allow_suspend, bool check_async_exception ) {
451
468
assert (_handshakee == Thread::current (), " Must be called by self" );
452
469
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) {
454
474
return _queue.peek ();
455
475
} else {
456
- return _queue.peek (no_suspend_filter );
476
+ return _queue.peek (no_async_exception_filter );
457
477
}
458
478
}
459
479
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
+ }
462
493
}
463
494
464
495
bool HandshakeState::have_non_self_executable_operation () {
465
496
assert (_handshakee != Thread::current (), " Must not be called by self" );
466
497
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);
473
499
}
474
500
475
501
HandshakeOperation* HandshakeState::get_op () {
476
502
assert (_handshakee != Thread::current (), " Must not be called by self" );
477
503
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 );
479
505
};
480
506
481
507
void HandshakeState::remove_op (HandshakeOperation* op) {
@@ -485,23 +511,20 @@ void HandshakeState::remove_op(HandshakeOperation* op) {
485
511
assert (ret == op, " Popped op must match requested op" );
486
512
};
487
513
488
- bool HandshakeState::process_by_self (bool allow_suspend) {
514
+ bool HandshakeState::process_by_self (bool allow_suspend, bool check_async_exception ) {
489
515
assert (Thread::current () == _handshakee, " should call from _handshakee" );
490
516
assert (!_handshakee->is_terminated (), " should not be a terminated thread" );
491
517
492
518
_handshakee->frame_anchor ()->make_walkable (_handshakee);
493
519
// Threads shouldn't block if they are in the middle of printing, but...
494
520
ttyLocker::break_tty_lock_for_safepoint (os::current_thread_id ());
495
521
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
-
501
522
while (has_operation ()) {
523
+ // Handshakes cannot safely safepoint. The exceptions to this rule are
524
+ // the asynchronous suspension and unsafe access error handshakes.
502
525
MutexLocker ml (&_lock, Mutex::_no_safepoint_check_flag);
503
526
504
- HandshakeOperation* op = get_op_for_self (allow_suspend);
527
+ HandshakeOperation* op = get_op_for_self (allow_suspend, check_async_exception );
505
528
if (op != NULL ) {
506
529
assert (op->_target == NULL || op->_target == Thread::current (), " Wrong thread" );
507
530
bool async = op->is_async ();
@@ -517,8 +540,8 @@ bool HandshakeState::process_by_self(bool allow_suspend) {
517
540
// An asynchronous handshake may put the JavaThread in blocked state (safepoint safe).
518
541
// The destructor ~PreserveExceptionMark touches the exception oop so it must not be executed,
519
542
// since a safepoint may be in-progress when returning from the async handshake.
520
- op->do_handshake (_handshakee); // acquire, op removed after
521
543
remove_op (op);
544
+ op->do_handshake (_handshakee);
522
545
log_handshake_info (((AsyncHandshakeOperation*)op)->start_time (), op->name (), 1 , 0 , " asynchronous" );
523
546
delete op;
524
547
return true ; // Must check for safepoints
@@ -730,3 +753,29 @@ bool HandshakeState::resume() {
730
753
_lock.notify ();
731
754
return true ;
732
755
}
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
+ }
0 commit comments