提交 44617907 编写于 作者: N never

7032963: StoreCM shouldn't participate in store elimination

Reviewed-by: kvn
上级 50619040
...@@ -2050,6 +2050,52 @@ static bool oop_offset_is_sane(const TypeInstPtr* tp) { ...@@ -2050,6 +2050,52 @@ static bool oop_offset_is_sane(const TypeInstPtr* tp) {
// Note that OffsetBot and OffsetTop are very negative. // Note that OffsetBot and OffsetTop are very negative.
} }
// Eliminate trivially redundant StoreCMs and accumulate their
// precedence edges.
static void eliminate_redundant_card_marks(Node* n) {
assert(n->Opcode() == Op_StoreCM, "expected StoreCM");
if (n->in(MemNode::Address)->outcnt() > 1) {
// There are multiple users of the same address so it might be
// possible to eliminate some of the StoreCMs
Node* mem = n->in(MemNode::Memory);
Node* adr = n->in(MemNode::Address);
Node* val = n->in(MemNode::ValueIn);
Node* prev = n;
bool done = false;
// Walk the chain of StoreCMs eliminating ones that match. As
// long as it's a chain of single users then the optimization is
// safe. Eliminating partially redundant StoreCMs would require
// cloning copies down the other paths.
while (mem->Opcode() == Op_StoreCM && mem->outcnt() == 1 && !done) {
if (adr == mem->in(MemNode::Address) &&
val == mem->in(MemNode::ValueIn)) {
// redundant StoreCM
if (mem->req() > MemNode::OopStore) {
// Hasn't been processed by this code yet.
n->add_prec(mem->in(MemNode::OopStore));
} else {
// Already converted to precedence edge
for (uint i = mem->req(); i < mem->len(); i++) {
// Accumulate any precedence edges
if (mem->in(i) != NULL) {
n->add_prec(mem->in(i));
}
}
// Everything above this point has been processed.
done = true;
}
// Eliminate the previous StoreCM
prev->set_req(MemNode::Memory, mem->in(MemNode::Memory));
assert(mem->outcnt() == 0, "should be dead");
mem->disconnect_inputs(NULL);
} else {
prev = mem;
}
mem = prev->in(MemNode::Memory);
}
}
}
//------------------------------final_graph_reshaping_impl---------------------- //------------------------------final_graph_reshaping_impl----------------------
// Implement items 1-5 from final_graph_reshaping below. // Implement items 1-5 from final_graph_reshaping below.
static void final_graph_reshaping_impl( Node *n, Final_Reshape_Counts &frc ) { static void final_graph_reshaping_impl( Node *n, Final_Reshape_Counts &frc ) {
...@@ -2176,9 +2222,19 @@ static void final_graph_reshaping_impl( Node *n, Final_Reshape_Counts &frc ) { ...@@ -2176,9 +2222,19 @@ static void final_graph_reshaping_impl( Node *n, Final_Reshape_Counts &frc ) {
frc.inc_float_count(); frc.inc_float_count();
goto handle_mem; goto handle_mem;
case Op_StoreCM:
{
// Convert OopStore dependence into precedence edge
Node* prec = n->in(MemNode::OopStore);
n->del_req(MemNode::OopStore);
n->add_prec(prec);
eliminate_redundant_card_marks(n);
}
// fall through
case Op_StoreB: case Op_StoreB:
case Op_StoreC: case Op_StoreC:
case Op_StoreCM:
case Op_StorePConditional: case Op_StorePConditional:
case Op_StoreI: case Op_StoreI:
case Op_StoreL: case Op_StoreL:
......
...@@ -685,20 +685,22 @@ bool Block::schedule_local(PhaseCFG *cfg, Matcher &matcher, int *ready_cnt, Vect ...@@ -685,20 +685,22 @@ bool Block::schedule_local(PhaseCFG *cfg, Matcher &matcher, int *ready_cnt, Vect
} }
ready_cnt[n->_idx] = local; // Count em up ready_cnt[n->_idx] = local; // Count em up
// A few node types require changing a required edge to a precedence edge #ifdef ASSERT
// before allocation.
if( UseConcMarkSweepGC || UseG1GC ) { if( UseConcMarkSweepGC || UseG1GC ) {
if( n->is_Mach() && n->as_Mach()->ideal_Opcode() == Op_StoreCM ) { if( n->is_Mach() && n->as_Mach()->ideal_Opcode() == Op_StoreCM ) {
// Note: Required edges with an index greater than oper_input_base // Check the precedence edges
// are not supported by the allocator. for (uint prec = n->req(); prec < n->len(); prec++) {
// Note2: Can only depend on unmatched edge being last, Node* oop_store = n->in(prec);
// can not depend on its absolute position. if (oop_store != NULL) {
Node *oop_store = n->in(n->req() - 1);
n->del_req(n->req() - 1);
n->add_prec(oop_store);
assert(cfg->_bbs[oop_store->_idx]->_dom_depth <= this->_dom_depth, "oop_store must dominate card-mark"); assert(cfg->_bbs[oop_store->_idx]->_dom_depth <= this->_dom_depth, "oop_store must dominate card-mark");
} }
} }
}
}
#endif
// A few node types require changing a required edge to a precedence edge
// before allocation.
if( n->is_Mach() && n->req() > TypeFunc::Parms && if( n->is_Mach() && n->req() > TypeFunc::Parms &&
(n->as_Mach()->ideal_Opcode() == Op_MemBarAcquire || (n->as_Mach()->ideal_Opcode() == Op_MemBarAcquire ||
n->as_Mach()->ideal_Opcode() == Op_MemBarVolatile) ) { n->as_Mach()->ideal_Opcode() == Op_MemBarVolatile) ) {
......
...@@ -2159,9 +2159,12 @@ Node *StoreNode::Ideal(PhaseGVN *phase, bool can_reshape) { ...@@ -2159,9 +2159,12 @@ Node *StoreNode::Ideal(PhaseGVN *phase, bool can_reshape) {
Node* mem = in(MemNode::Memory); Node* mem = in(MemNode::Memory);
Node* address = in(MemNode::Address); Node* address = in(MemNode::Address);
// Back-to-back stores to same address? Fold em up. // Back-to-back stores to same address? Fold em up. Generally
// Generally unsafe if I have intervening uses... // unsafe if I have intervening uses... Also disallowed for StoreCM
if (mem->is_Store() && phase->eqv_uncast(mem->in(MemNode::Address), address)) { // since they must follow each StoreP operation. Redundant StoreCMs
// are eliminated just before matching in final_graph_reshape.
if (mem->is_Store() && phase->eqv_uncast(mem->in(MemNode::Address), address) &&
mem->Opcode() != Op_StoreCM) {
// Looking at a dead closed cycle of memory? // Looking at a dead closed cycle of memory?
assert(mem != mem->in(MemNode::Memory), "dead loop in StoreNode::Ideal"); assert(mem != mem->in(MemNode::Memory), "dead loop in StoreNode::Ideal");
......
...@@ -1354,8 +1354,11 @@ void Compile::Fill_buffer() { ...@@ -1354,8 +1354,11 @@ void Compile::Fill_buffer() {
// Check that oop-store precedes the card-mark // Check that oop-store precedes the card-mark
else if( mach->ideal_Opcode() == Op_StoreCM ) { else if( mach->ideal_Opcode() == Op_StoreCM ) {
uint storeCM_idx = j; uint storeCM_idx = j;
Node *oop_store = mach->in(mach->_cnt); // First precedence edge int count = 0;
assert( oop_store != NULL, "storeCM expects a precedence edge"); for (uint prec = mach->req(); prec < mach->len(); prec++) {
Node *oop_store = mach->in(prec); // Precedence edge
if (oop_store == NULL) continue;
count++;
uint i4; uint i4;
for( i4 = 0; i4 < last_inst; ++i4 ) { for( i4 = 0; i4 < last_inst; ++i4 ) {
if( b->_nodes[i4] == oop_store ) break; if( b->_nodes[i4] == oop_store ) break;
...@@ -1364,6 +1367,8 @@ void Compile::Fill_buffer() { ...@@ -1364,6 +1367,8 @@ void Compile::Fill_buffer() {
// edges have been added to the storeCMNode. // edges have been added to the storeCMNode.
assert( i4 == last_inst || i4 < storeCM_idx, "CM card-mark executes before oop-store"); assert( i4 == last_inst || i4 < storeCM_idx, "CM card-mark executes before oop-store");
} }
assert(count > 0, "storeCM expects at least one precedence edge");
}
#endif #endif
else if( !n->is_Proj() ) { else if( !n->is_Proj() ) {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册