diff --git a/src/share/vm/opto/callnode.cpp b/src/share/vm/opto/callnode.cpp index 1059dd5a56e27a0fc4879e92a24013539df18ab4..2811264846e11fe3bfc8ff7b8adc337608f4b7f7 100644 --- a/src/share/vm/opto/callnode.cpp +++ b/src/share/vm/opto/callnode.cpp @@ -1625,21 +1625,20 @@ Node *LockNode::Ideal(PhaseGVN *phase, bool can_reshape) { //============================================================================= bool LockNode::is_nested_lock_region() { - Node* box = box_node(); - if (!box->is_BoxLock() || box->as_BoxLock()->stack_slot() <= 0) + BoxLockNode* box = box_node()->as_BoxLock(); + int stk_slot = box->stack_slot(); + if (stk_slot <= 0) return false; // External lock or it is not Box (Phi node). // Ignore complex cases: merged locks or multiple locks. - BoxLockNode* box_lock = box->as_BoxLock(); Node* obj = obj_node(); LockNode* unique_lock = NULL; - if (!box_lock->is_simple_lock_region(&unique_lock, obj) || + if (!box->is_simple_lock_region(&unique_lock, obj) || (unique_lock != this)) { return false; } // Look for external lock for the same object. - int stk_slot = box_lock->stack_slot(); SafePointNode* sfn = this->as_SafePoint(); JVMState* youngest_jvms = sfn->jvms(); int max_depth = youngest_jvms->depth(); @@ -1649,7 +1648,7 @@ bool LockNode::is_nested_lock_region() { // Loop over monitors for (int idx = 0; idx < num_mon; idx++) { Node* obj_node = sfn->monitor_obj(jvms, idx); - BoxLockNode* box_node = BoxLockNode::box_node(sfn->monitor_box(jvms, idx)); + BoxLockNode* box_node = sfn->monitor_box(jvms, idx)->as_BoxLock(); if ((box_node->stack_slot() < stk_slot) && obj_node->eqv_uncast(obj)) { return true; } diff --git a/src/share/vm/opto/locknode.cpp b/src/share/vm/opto/locknode.cpp index bae284421fb332aa5a2aff07a2996bfa2654447e..4147ed33dc7693fe32853af9f6b53283640b46e3 100644 --- a/src/share/vm/opto/locknode.cpp +++ b/src/share/vm/opto/locknode.cpp @@ -63,7 +63,7 @@ uint BoxLockNode::cmp( const Node &n ) const { } BoxLockNode* BoxLockNode::box_node(Node* box) { - // Chase down the BoxNode + // Chase down the BoxNode after RA which may spill box nodes. while (!box->is_BoxLock()) { // if (box_node->is_SpillCopy()) { // Node *m = box_node->in(1); @@ -84,18 +84,13 @@ OptoReg::Name BoxLockNode::reg(Node* box) { return box_node(box)->in_RegMask(0).find_first_elem(); } -bool BoxLockNode::same_slot(Node* box1, Node* box2) { - return box_node(box1)->_slot == box_node(box2)->_slot; -} - // Is BoxLock node used for one simple lock region (same box and obj)? bool BoxLockNode::is_simple_lock_region(LockNode** unique_lock, Node* obj) { LockNode* lock = NULL; bool has_one_lock = false; for (uint i = 0; i < this->outcnt(); i++) { Node* n = this->raw_out(i); - if (n->is_Phi()) - return false; // Merged regions + assert(!n->is_Phi(), "should not merge BoxLock nodes"); if (n->is_AbstractLock()) { AbstractLockNode* alock = n->as_AbstractLock(); // Check lock's box since box could be referenced by Lock's debug info. @@ -123,23 +118,12 @@ bool BoxLockNode::is_simple_lock_region(LockNode** unique_lock, Node* obj) { FastLockNode* flock = n->as_FastLock(); assert((flock->box_node() == this) && flock->obj_node()->eqv_uncast(obj),""); } - if (n->is_SafePoint() && n->as_SafePoint()->jvms()) { - SafePointNode* sfn = n->as_SafePoint(); - JVMState* youngest_jvms = sfn->jvms(); - int max_depth = youngest_jvms->depth(); - for (int depth = 1; depth <= max_depth; depth++) { - JVMState* jvms = youngest_jvms->of_depth(depth); - int num_mon = jvms->nof_monitors(); - // Loop over monitors - for (int idx = 0; idx < num_mon; idx++) { - Node* obj_node = sfn->monitor_obj(jvms, idx); - Node* box_node = sfn->monitor_box(jvms, idx); - if (box_node == this) { - assert(obj_node->eqv_uncast(obj),""); - } - } - } - } + // Don't check monitor info in safepoints since the referenced object could + // be different from the locked object. It could be Phi node of different + // cast nodes which point to this locked object. + // We assume that no other objects could be referenced in monitor info + // associated with this BoxLock node because all associated locks and + // unlocks are reference only this one object. } #endif if (unique_lock != NULL && has_one_lock) { diff --git a/src/share/vm/opto/locknode.hpp b/src/share/vm/opto/locknode.hpp index b362537ea7c76de29bd350ad5e3fc9376a634149..91b99bc0052aee019303567d36967d35b9942fe5 100644 --- a/src/share/vm/opto/locknode.hpp +++ b/src/share/vm/opto/locknode.hpp @@ -49,9 +49,9 @@ //------------------------------BoxLockNode------------------------------------ class BoxLockNode : public Node { - const int _slot; - RegMask _inmask; - bool _is_eliminated; // indicates this lock was safely eliminated + const int _slot; // stack slot + RegMask _inmask; // OptoReg corresponding to stack slot + bool _is_eliminated; // Associated locks were safely eliminated public: BoxLockNode( int lock ); @@ -68,7 +68,9 @@ public: static OptoReg::Name reg(Node* box_node); static BoxLockNode* box_node(Node* box_node); - static bool same_slot(Node* box1, Node* box2); + static bool same_slot(Node* box1, Node* box2) { + return box1->as_BoxLock()->_slot == box2->as_BoxLock()->_slot; + } int stack_slot() const { return _slot; } bool is_eliminated() const { return _is_eliminated; } diff --git a/src/share/vm/opto/macro.cpp b/src/share/vm/opto/macro.cpp index 116d83ec81ddb0e745a13d9a5e4248846571373f..adcf8e48f553fed32e4160b46a18c9130006a90b 100644 --- a/src/share/vm/opto/macro.cpp +++ b/src/share/vm/opto/macro.cpp @@ -1802,10 +1802,14 @@ void PhaseMacroExpand::expand_allocate_array(AllocateArrayNode *alloc) { // Mark all associated (same box and obj) lock and unlock nodes for // elimination if some of them marked already. void PhaseMacroExpand::mark_eliminated_box(Node* oldbox, Node* obj) { - if (oldbox->is_BoxLock() && oldbox->as_BoxLock()->is_eliminated()) - return; - - if (oldbox->is_BoxLock() && + if (oldbox->as_BoxLock()->is_eliminated()) + return; // This BoxLock node was processed already. + + // New implementation (EliminateNestedLocks) has separate BoxLock + // node for each locked region so mark all associated locks/unlocks as + // eliminated even if different objects are referenced in one locked region + // (for example, OSR compilation of nested loop inside locked scope). + if (EliminateNestedLocks || oldbox->as_BoxLock()->is_simple_lock_region(NULL, obj)) { // Box is used only in one lock region. Mark this box as eliminated. _igvn.hash_delete(oldbox); @@ -1818,7 +1822,6 @@ void PhaseMacroExpand::mark_eliminated_box(Node* oldbox, Node* obj) { AbstractLockNode* alock = u->as_AbstractLock(); // Check lock's box since box could be referenced by Lock's debug info. if (alock->box_node() == oldbox) { - assert(alock->obj_node()->eqv_uncast(obj), ""); // Mark eliminated all related locks and unlocks. alock->set_non_esc_obj(); } @@ -1829,8 +1832,7 @@ void PhaseMacroExpand::mark_eliminated_box(Node* oldbox, Node* obj) { // Create new "eliminated" BoxLock node and use it in monitor debug info // instead of oldbox for the same object. - BoxLockNode* box = BoxLockNode::box_node(oldbox); - BoxLockNode* newbox = box->clone()->as_BoxLock(); + BoxLockNode* newbox = oldbox->clone()->as_BoxLock(); // Note: BoxLock node is marked eliminated only here and it is used // to indicate that all associated lock and unlock nodes are marked @@ -2047,7 +2049,7 @@ void PhaseMacroExpand::expand_lock_node(LockNode *lock) { Node* box = lock->box_node(); Node* flock = lock->fastlock_node(); - assert(!BoxLockNode::box_node(box)->is_eliminated(), "sanity"); + assert(!box->as_BoxLock()->is_eliminated(), "sanity"); // Make the merge point Node *region; @@ -2283,7 +2285,7 @@ void PhaseMacroExpand::expand_unlock_node(UnlockNode *unlock) { Node* obj = unlock->obj_node(); Node* box = unlock->box_node(); - assert(!BoxLockNode::box_node(box)->is_eliminated(), "sanity"); + assert(!box->as_BoxLock()->is_eliminated(), "sanity"); // No need for a null check on unlock diff --git a/src/share/vm/opto/parse1.cpp b/src/share/vm/opto/parse1.cpp index 1505123c32d3187b43412686d83b792fb553c472..99bdf93b933d7fb890075a4fcf21ab7425aae1c1 100644 --- a/src/share/vm/opto/parse1.cpp +++ b/src/share/vm/opto/parse1.cpp @@ -1604,7 +1604,16 @@ void Parse::merge_common(Parse::Block* target, int pnum) { continue; default: // All normal stuff if (phi == NULL) { - if (!check_elide_phi || !target->can_elide_SEL_phi(j)) { + const JVMState* jvms = map()->jvms(); + if (EliminateNestedLocks && + jvms->is_mon(j) && jvms->is_monitor_box(j)) { + // BoxLock nodes are not commoning. + // Use old BoxLock node as merged box. + assert(newin->jvms()->is_monitor_box(j), "sanity"); + // This assert also tests that nodes are BoxLock. + assert(BoxLockNode::same_slot(n, m), "sanity"); + C->gvn_replace_by(n, m); + } else if (!check_elide_phi || !target->can_elide_SEL_phi(j)) { phi = ensure_phi(j, nophi); } } @@ -1819,12 +1828,8 @@ PhiNode *Parse::ensure_phi(int idx, bool nocreate) { } else if (jvms->is_stk(idx)) { t = block()->stack_type_at(idx - jvms->stkoff()); } else if (jvms->is_mon(idx)) { - if (EliminateNestedLocks && jvms->is_monitor_box(idx)) { - // BoxLock nodes are not commoning. Create Phi. - t = o->bottom_type(); // TypeRawPtr::BOTTOM - } else { - t = TypeInstPtr::BOTTOM; // this is sufficient for a lock object - } + assert(!jvms->is_monitor_box(idx), "no phis for boxes"); + t = TypeInstPtr::BOTTOM; // this is sufficient for a lock object } else if ((uint)idx < TypeFunc::Parms) { t = o->bottom_type(); // Type::RETURN_ADDRESS or such-like. } else {