8231737: Cleanup JvmtiRawMonitor code

Reviewed-by: sspitsyn, pliden, coleenp, dcubed
This commit is contained in:
David Holmes 2019-10-08 17:30:48 -04:00
parent d7fcd0ccac
commit d113ca0727
2 changed files with 217 additions and 203 deletions

View File

@ -32,7 +32,7 @@
JvmtiRawMonitor::QNode::QNode(Thread* thread) : _next(NULL), _prev(NULL),
_event(thread->_ParkEvent),
_notified(0), TState(TS_RUN) {
_notified(0), _t_state(TS_RUN) {
}
GrowableArray<JvmtiRawMonitor*>* JvmtiPendingMonitors::_monitors =
@ -40,8 +40,8 @@ GrowableArray<JvmtiRawMonitor*> *JvmtiPendingMonitors::_monitors =
void JvmtiPendingMonitors::transition_raw_monitors() {
assert((Threads::number_of_threads()==1),
"Java thread has not been created yet or more than one java thread \
is running. Raw monitor transition will not work");
"Java thread has not been created yet or more than one java thread "
"is running. Raw monitor transition will not work");
JavaThread* current_java_thread = JavaThread::current();
assert(current_java_thread->thread_state() == _thread_in_vm, "Must be in vm");
for (int i = 0; i < count(); i++) {
@ -58,8 +58,8 @@ is running. Raw monitor transition will not work");
JvmtiRawMonitor::JvmtiRawMonitor(const char* name) : _owner(NULL),
_recursions(0),
_EntryList(NULL),
_WaitSet(NULL),
_entry_list(NULL),
_wait_set(NULL),
_waiters(0),
_magic(JVMTI_RM_MAGIC),
_name(NULL) {
@ -119,125 +119,128 @@ JvmtiRawMonitor::is_valid() {
//
// -------------------------------------------------------------------------
void JvmtiRawMonitor::SimpleEnter (Thread * Self) {
void JvmtiRawMonitor::simple_enter(Thread* self) {
for (;;) {
if (Atomic::replace_if_null(Self, &_owner)) {
if (Atomic::replace_if_null(self, &_owner)) {
return;
}
QNode Node (Self) ;
Self->_ParkEvent->reset() ; // strictly optional
Node.TState = QNode::TS_ENTER ;
QNode node(self);
self->_ParkEvent->reset(); // strictly optional
node._t_state = QNode::TS_ENTER;
RawMonitor_lock->lock_without_safepoint_check();
Node._next = _EntryList ;
_EntryList = &Node ;
node._next = _entry_list;
_entry_list = &node;
OrderAccess::fence();
if (_owner == NULL && Atomic::replace_if_null(Self, &_owner)) {
_EntryList = Node._next ;
if (_owner == NULL && Atomic::replace_if_null(self, &_owner)) {
_entry_list = node._next;
RawMonitor_lock->unlock();
return;
}
RawMonitor_lock->unlock();
while (Node.TState == QNode::TS_ENTER) {
Self->_ParkEvent->park() ;
while (node._t_state == QNode::TS_ENTER) {
self->_ParkEvent->park();
}
}
}
void JvmtiRawMonitor::SimpleExit (Thread * Self) {
guarantee (_owner == Self, "invariant") ;
void JvmtiRawMonitor::simple_exit(Thread* self) {
guarantee(_owner == self, "invariant");
OrderAccess::release_store(&_owner, (Thread*)NULL);
OrderAccess::fence();
if (_EntryList == NULL) return ;
QNode * w ;
if (_entry_list == NULL) {
return;
}
RawMonitor_lock->lock_without_safepoint_check();
w = _EntryList ;
QNode* w = _entry_list;
if (w != NULL) {
_EntryList = w->_next ;
_entry_list = w->_next;
}
RawMonitor_lock->unlock();
if (w != NULL) {
guarantee (w ->TState == QNode::TS_ENTER, "invariant") ;
// Once we set TState to TS_RUN the waiting thread can complete
// SimpleEnter and 'w' is pointing into random stack space. So we have
guarantee(w ->_t_state == QNode::TS_ENTER, "invariant");
// Once we set _t_state to TS_RUN the waiting thread can complete
// simple_enter and 'w' is pointing into random stack space. So we have
// to ensure we extract the ParkEvent (which is in type-stable memory)
// before we set the state, and then don't access 'w'.
ParkEvent* ev = w->_event;
OrderAccess::loadstore();
w->TState = QNode::TS_RUN ;
w->_t_state = QNode::TS_RUN;
OrderAccess::fence();
ev->unpark();
}
return;
}
int JvmtiRawMonitor::SimpleWait (Thread * Self, jlong millis) {
guarantee (_owner == Self , "invariant") ;
int JvmtiRawMonitor::simple_wait(Thread* self, jlong millis) {
guarantee(_owner == self , "invariant");
guarantee(_recursions == 0, "invariant");
QNode Node (Self) ;
Node._notified = 0 ;
Node.TState = QNode::TS_WAIT ;
QNode node(self);
node._notified = 0;
node._t_state = QNode::TS_WAIT;
RawMonitor_lock->lock_without_safepoint_check();
Node._next = _WaitSet ;
_WaitSet = &Node ;
node._next = _wait_set;
_wait_set = &node;
RawMonitor_lock->unlock();
SimpleExit (Self) ;
guarantee (_owner != Self, "invariant") ;
simple_exit(self);
guarantee(_owner != self, "invariant");
int ret = OS_OK;
if (millis <= 0) {
Self->_ParkEvent->park();
self->_ParkEvent->park();
} else {
ret = Self->_ParkEvent->park(millis);
ret = self->_ParkEvent->park(millis);
}
// If thread still resides on the waitset then unlink it.
// Double-checked locking -- the usage is safe in this context
// as TState is volatile and the lock-unlock operators are
// as _t_state is volatile and the lock-unlock operators are
// serializing (barrier-equivalent).
if (Node.TState == QNode::TS_WAIT) {
if (node._t_state == QNode::TS_WAIT) {
RawMonitor_lock->lock_without_safepoint_check();
if (Node.TState == QNode::TS_WAIT) {
if (node._t_state == QNode::TS_WAIT) {
// Simple O(n) unlink, but performance isn't critical here.
QNode* p;
QNode* q = NULL;
for (p = _WaitSet ; p != &Node; p = p->_next) {
for (p = _wait_set; p != &node; p = p->_next) {
q = p;
}
guarantee (p == &Node, "invariant") ;
guarantee(p == &node, "invariant");
if (q == NULL) {
guarantee (p == _WaitSet, "invariant") ;
_WaitSet = p->_next ;
guarantee (p == _wait_set, "invariant");
_wait_set = p->_next;
} else {
guarantee(p == q->_next, "invariant");
q->_next = p->_next;
}
Node.TState = QNode::TS_RUN ;
node._t_state = QNode::TS_RUN;
}
RawMonitor_lock->unlock();
}
guarantee (Node.TState == QNode::TS_RUN, "invariant") ;
SimpleEnter (Self) ;
guarantee(node._t_state == QNode::TS_RUN, "invariant");
simple_enter(self);
guarantee (_owner == Self, "invariant") ;
guarantee(_owner == self, "invariant");
guarantee(_recursions == 0, "invariant");
return ret;
}
void JvmtiRawMonitor::SimpleNotify (Thread * Self, bool All) {
guarantee (_owner == Self, "invariant") ;
if (_WaitSet == NULL) return ;
void JvmtiRawMonitor::simple_notify(Thread* self, bool all) {
guarantee(_owner == self, "invariant");
if (_wait_set == NULL) {
return;
}
// We have two options:
// A. Transfer the threads from the WaitSet to the EntryList
// B. Remove the thread from the WaitSet and unpark() it.
// A. Transfer the threads from the _wait_set to the _entry_list
// B. Remove the thread from the _wait_set and unpark() it.
//
// We use (B), which is crude and results in lots of futile
// context switching. In particular (B) induces lots of contention.
@ -245,29 +248,36 @@ void JvmtiRawMonitor::SimpleNotify (Thread * Self, bool All) {
ParkEvent* ev = NULL; // consider using a small auto array ...
RawMonitor_lock->lock_without_safepoint_check();
for (;;) {
QNode * w = _WaitSet ;
QNode* w = _wait_set;
if (w == NULL) break;
_WaitSet = w->_next ;
if (ev != NULL) { ev->unpark(); ev = NULL; }
_wait_set = w->_next;
if (ev != NULL) {
ev->unpark();
ev = NULL;
}
ev = w->_event;
OrderAccess::loadstore();
w->TState = QNode::TS_RUN ;
w->_t_state = QNode::TS_RUN;
OrderAccess::storeload();
if (!All) break ;
if (!all) {
break;
}
}
RawMonitor_lock->unlock();
if (ev != NULL) ev->unpark();
if (ev != NULL) {
ev->unpark();
}
return;
}
// Any JavaThread will enter here with state _thread_blocked
void JvmtiRawMonitor::raw_enter(Thread * Self) {
void * Contended ;
void JvmtiRawMonitor::raw_enter(Thread* self) {
void* contended;
JavaThread* jt = NULL;
// don't enter raw monitor if thread is being externally suspended, it will
// surprise the suspender if a "suspended" thread can still enter monitor
if (Self->is_Java_thread()) {
jt = (JavaThread*) Self;
if (self->is_Java_thread()) {
jt = (JavaThread*)self;
jt->SR_lock()->lock_without_safepoint_check();
while (jt->is_external_suspend()) {
jt->SR_lock()->unlock();
@ -275,37 +285,39 @@ void JvmtiRawMonitor::raw_enter(Thread * Self) {
jt->SR_lock()->lock_without_safepoint_check();
}
// guarded by SR_lock to avoid racing with new external suspend requests.
Contended = Atomic::cmpxchg(jt, &_owner, (Thread*)NULL);
contended = Atomic::cmpxchg(jt, &_owner, (Thread*)NULL);
jt->SR_lock()->unlock();
} else {
Contended = Atomic::cmpxchg(Self, &_owner, (Thread*)NULL);
contended = Atomic::cmpxchg(self, &_owner, (Thread*)NULL);
}
if (Contended == Self) {
if (contended == self) {
_recursions++;
return;
}
if (Contended == NULL) {
guarantee (_owner == Self, "invariant") ;
if (contended == NULL) {
guarantee(_owner == self, "invariant");
guarantee(_recursions == 0, "invariant");
return;
}
Self->set_current_pending_raw_monitor(this);
self->set_current_pending_raw_monitor(this);
if (!Self->is_Java_thread()) {
SimpleEnter (Self) ;
if (!self->is_Java_thread()) {
simple_enter(self);
} else {
guarantee(jt->thread_state() == _thread_blocked, "invariant");
for (;;) {
jt->set_suspend_equivalent();
// cleared by handle_special_suspend_equivalent_condition() or
// java_suspend_self()
SimpleEnter (jt) ;
simple_enter(jt);
// were we externally suspended while we were waiting?
if (!jt->handle_special_suspend_equivalent_condition()) break ;
if (!jt->handle_special_suspend_equivalent_condition()) {
break;
}
// This thread was externally suspended
// We have reentered the contended monitor, but while we were
@ -314,26 +326,26 @@ void JvmtiRawMonitor::raw_enter(Thread * Self) {
// thread that suspended us.
//
// Drop the lock
SimpleExit (jt) ;
simple_exit(jt);
jt->java_suspend_self();
}
}
Self->set_current_pending_raw_monitor(NULL);
self->set_current_pending_raw_monitor(NULL);
guarantee (_owner == Self, "invariant") ;
guarantee(_owner == self, "invariant");
guarantee(_recursions == 0, "invariant");
}
int JvmtiRawMonitor::raw_exit(Thread * Self) {
if (Self != _owner) {
int JvmtiRawMonitor::raw_exit(Thread* self) {
if (self != _owner) {
return M_ILLEGAL_MONITOR_STATE;
}
if (_recursions > 0) {
--_recursions ;
_recursions--;
} else {
SimpleExit (Self) ;
simple_exit(self);
}
return M_OK;
@ -341,46 +353,48 @@ int JvmtiRawMonitor::raw_exit(Thread * Self) {
// All JavaThreads will enter here with state _thread_blocked
int JvmtiRawMonitor::raw_wait(jlong millis, bool interruptible, Thread * Self) {
if (Self != _owner) {
int JvmtiRawMonitor::raw_wait(jlong millis, bool interruptible, Thread* self) {
if (self != _owner) {
return M_ILLEGAL_MONITOR_STATE;
}
// To avoid spurious wakeups we reset the parkevent -- This is strictly optional.
// To avoid spurious wakeups we reset the parkevent. This is strictly optional.
// The caller must be able to tolerate spurious returns from raw_wait().
Self->_ParkEvent->reset() ;
self->_ParkEvent->reset();
OrderAccess::fence();
JavaThread* jt = NULL;
// check interrupt event
if (interruptible) {
assert(Self->is_Java_thread(), "Only JavaThreads can be interruptible");
jt = (JavaThread*) Self;
assert(self->is_Java_thread(), "Only JavaThreads can be interruptible");
jt = (JavaThread*)self;
if (jt->is_interrupted(true)) {
return M_INTERRUPTED;
}
} else {
assert(!Self->is_Java_thread(), "JavaThreads must be interuptible");
assert(!self->is_Java_thread(), "JavaThreads must be interuptible");
}
intptr_t save = _recursions;
_recursions = 0;
_waiters++;
if (Self->is_Java_thread()) {
if (self->is_Java_thread()) {
guarantee(jt->thread_state() == _thread_blocked, "invariant");
jt->set_suspend_equivalent();
}
int rv = SimpleWait (Self, millis) ;
int rv = simple_wait(self, millis);
_recursions = save;
_waiters--;
guarantee (Self == _owner, "invariant") ;
if (Self->is_Java_thread()) {
guarantee(self == _owner, "invariant");
if (self->is_Java_thread()) {
for (;;) {
if (!jt->handle_special_suspend_equivalent_condition()) break ;
SimpleExit (jt) ;
if (!jt->handle_special_suspend_equivalent_condition()) {
break;
}
simple_exit(jt);
jt->java_suspend_self();
SimpleEnter (jt) ;
simple_enter(jt);
jt->set_suspend_equivalent();
}
guarantee(jt == _owner, "invariant");
@ -393,18 +407,18 @@ int JvmtiRawMonitor::raw_wait(jlong millis, bool interruptible, Thread * Self) {
return M_OK;
}
int JvmtiRawMonitor::raw_notify(Thread * Self) {
if (Self != _owner) {
int JvmtiRawMonitor::raw_notify(Thread* self) {
if (self != _owner) {
return M_ILLEGAL_MONITOR_STATE;
}
SimpleNotify (Self, false) ;
simple_notify(self, false);
return M_OK;
}
int JvmtiRawMonitor::raw_notifyAll(Thread * Self) {
if (Self != _owner) {
int JvmtiRawMonitor::raw_notifyAll(Thread* self) {
if (self != _owner) {
return M_ILLEGAL_MONITOR_STATE;
}
SimpleNotify (Self, true) ;
simple_notify(self, true);
return M_OK;
}

View File

@ -48,27 +48,27 @@ class JvmtiRawMonitor : public CHeapObj<mtSynchronizer> {
QNode* volatile _prev;
ParkEvent* _event;
volatile int _notified;
volatile TStates TState;
volatile TStates _t_state;
QNode(Thread* thread);
};
Thread* volatile _owner; // pointer to owning thread
volatile int _recursions; // recursion count, 0 for first entry
QNode* volatile _EntryList; // Threads blocked on entry or reentry.
QNode* volatile _entry_list; // Threads blocked on entry or reentry.
// The list is actually composed of nodes,
// acting as proxies for Threads.
QNode* volatile _WaitSet; // Threads wait()ing on the monitor
QNode* volatile _wait_set; // Threads wait()ing on the monitor
volatile jint _waiters; // number of waiting threads
int _magic;
char* _name;
// JVMTI_RM_MAGIC is set in contructor and unset in destructor.
enum { JVMTI_RM_MAGIC = (int)(('T' << 24) | ('I' << 16) | ('R' << 8) | 'M') };
void SimpleEnter (Thread * Self) ;
void SimpleExit (Thread * Self) ;
int SimpleWait (Thread * Self, jlong millis) ;
void SimpleNotify(Thread * Self, bool All) ;
void simple_enter(Thread* self);
void simple_exit(Thread* self);
int simple_wait(Thread* self, jlong millis);
void simple_notify(Thread* self, bool all);
public:
@ -90,11 +90,11 @@ public:
Thread* owner() const { return _owner; }
void set_owner(Thread* owner) { _owner = owner; }
int recursions() const { return _recursions; }
void raw_enter(Thread * Self);
int raw_exit(Thread * Self);
int raw_wait(jlong millis, bool interruptable, Thread * Self);
int raw_notify(Thread * Self);
int raw_notifyAll(Thread * Self);
void raw_enter(Thread* self);
int raw_exit(Thread* self);
int raw_wait(jlong millis, bool interruptible, Thread* self);
int raw_notify(Thread* self);
int raw_notifyAll(Thread* self);
int magic() const { return _magic; }
const char* get_name() const { return _name; }
bool is_valid();