diff --git a/src/hotspot/share/c1/c1_CFGPrinter.cpp b/src/hotspot/share/c1/c1_CFGPrinter.cpp index 9cd617d5428..73a1d2d649e 100644 --- a/src/hotspot/share/c1/c1_CFGPrinter.cpp +++ b/src/hotspot/share/c1/c1_CFGPrinter.cpp @@ -231,9 +231,7 @@ void CFGPrinterOutput::print_LIR(BlockBegin* block) { void CFGPrinterOutput::print_block(BlockBegin* block) { print_begin("block"); - print("name \"B%d\"", block->block_id()); - print("from_bci %d", block->bci()); print("to_bci %d", (block->end() == NULL ? -1 : block->end()->printable_bci())); @@ -246,9 +244,13 @@ void CFGPrinterOutput::print_block(BlockBegin* block) { output()->cr(); output()->indent(); - output()->print("successors "); - for (i = 0; i < block->number_of_sux(); i++) { - output()->print("\"B%d\" ", block->sux_at(i)->block_id()); + if (block->end() != NULL) { + output()->print("successors "); + for (i = 0; i < block->number_of_sux(); i++) { + output()->print("\"B%d\" ", block->sux_at(i)->block_id()); + } + } else { + output()->print("(block has no end, cannot print successors)"); } output()->cr(); diff --git a/src/hotspot/share/c1/c1_GraphBuilder.cpp b/src/hotspot/share/c1/c1_GraphBuilder.cpp index 2c10642f8ad..2f9355f1099 100644 --- a/src/hotspot/share/c1/c1_GraphBuilder.cpp +++ b/src/hotspot/share/c1/c1_GraphBuilder.cpp @@ -53,6 +53,7 @@ class BlockListBuilder { BlockList _blocks; // internal list of all blocks BlockList* _bci2block; // mapping from bci to blocks for GraphBuilder + GrowableArray _bci2block_successors; // Mapping bcis to their blocks successors while we dont have a blockend // fields used by mark_loops ResourceBitMap _active; // for iteration of control flow graph @@ -89,6 +90,11 @@ class BlockListBuilder { void print(); #endif + int number_of_successors(BlockBegin* block); + BlockBegin* successor_at(BlockBegin* block, int i); + void add_successor(BlockBegin* block, BlockBegin* sux); + bool is_successor(BlockBegin* block, BlockBegin* sux); + public: // creation BlockListBuilder(Compilation* compilation, IRScope* scope, int osr_bci); @@ -105,6 +111,7 @@ BlockListBuilder::BlockListBuilder(Compilation* compilation, IRScope* scope, int , _scope(scope) , _blocks(16) , _bci2block(new BlockList(scope->method()->code_size(), NULL)) + , _bci2block_successors(scope->method()->code_size()) , _active() // size not known yet , _visited() // size not known yet , _loop_map() // size not known yet @@ -118,6 +125,8 @@ BlockListBuilder::BlockListBuilder(Compilation* compilation, IRScope* scope, int mark_loops(); NOT_PRODUCT(if (PrintInitialBlockList) print()); + // _bci2block still contains blocks with _end == null and > 0 sux in _bci2block_successors. + #ifndef PRODUCT if (PrintCFGToFile) { stringStream title; @@ -160,6 +169,7 @@ BlockBegin* BlockListBuilder::make_block_at(int cur_bci, BlockBegin* predecessor block = new BlockBegin(cur_bci); block->init_stores_to_locals(method()->max_locals()); _bci2block->at_put(cur_bci, block); + _bci2block_successors.at_put_grow(cur_bci, BlockList()); _blocks.append(block); assert(predecessor == NULL || predecessor->bci() < cur_bci, "targets for backward branches must already exist"); @@ -170,7 +180,7 @@ BlockBegin* BlockListBuilder::make_block_at(int cur_bci, BlockBegin* predecessor BAILOUT_("Exception handler can be reached by both normal and exceptional control flow", block); } - predecessor->add_successor(block); + add_successor(predecessor, block); block->increment_total_preds(); } @@ -201,8 +211,8 @@ void BlockListBuilder::handle_exceptions(BlockBegin* current, int cur_bci) { assert(entry->is_set(BlockBegin::exception_entry_flag), "flag must be set"); // add each exception handler only once - if (!current->is_successor(entry)) { - current->add_successor(entry); + if(!is_successor(current, entry)) { + add_successor(current, entry); entry->increment_total_preds(); } @@ -418,9 +428,9 @@ int BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) { _active.set_bit(block_id); intptr_t loop_state = 0; - for (int i = block->number_of_sux() - 1; i >= 0; i--) { + for (int i = number_of_successors(block) - 1; i >= 0; i--) { // recursively process all successors - loop_state |= mark_loops(block->sux_at(i), in_subroutine); + loop_state |= mark_loops(successor_at(block, i), in_subroutine); } // clear active-bit after all successors are processed @@ -452,6 +462,28 @@ int BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) { return loop_state; } +inline int BlockListBuilder::number_of_successors(BlockBegin* block) +{ + assert(_bci2block_successors.length() > block->bci(), "sux must exist"); + return _bci2block_successors.at(block->bci()).length(); +} + +inline BlockBegin* BlockListBuilder::successor_at(BlockBegin* block, int i) +{ + assert(_bci2block_successors.length() > block->bci(), "sux must exist"); + return _bci2block_successors.at(block->bci()).at(i); +} + +inline void BlockListBuilder::add_successor(BlockBegin* block, BlockBegin* sux) +{ + assert(_bci2block_successors.length() > block->bci(), "sux must exist"); + _bci2block_successors.at(block->bci()).append(sux); +} + +inline bool BlockListBuilder::is_successor(BlockBegin* block, BlockBegin* sux) { + assert(_bci2block_successors.length() > block->bci(), "sux must exist"); + return _bci2block_successors.at(block->bci()).contains(sux); +} #ifndef PRODUCT @@ -477,10 +509,10 @@ void BlockListBuilder::print() { tty->print(cur->is_set(BlockBegin::subroutine_entry_flag) ? " sr" : " "); tty->print(cur->is_set(BlockBegin::parser_loop_header_flag) ? " lh" : " "); - if (cur->number_of_sux() > 0) { + if (number_of_successors(cur) > 0) { tty->print(" sux: "); - for (int j = 0; j < cur->number_of_sux(); j++) { - BlockBegin* sux = cur->sux_at(j); + for (int j = 0; j < number_of_successors(cur); j++) { + BlockBegin* sux = successor_at(cur, j); tty->print("B%d ", sux->block_id()); } } @@ -3230,6 +3262,8 @@ GraphBuilder::GraphBuilder(Compilation* compilation, IRScope* scope) _initial_state = state_at_entry(); start_block->merge(_initial_state); + // End nulls still exist here + // complete graph _vmap = new ValueMap(); switch (scope->method()->intrinsic_id()) { @@ -3333,6 +3367,27 @@ GraphBuilder::GraphBuilder(Compilation* compilation, IRScope* scope) } CHECK_BAILOUT(); +# ifdef ASSERT + //All blocks reachable from start_block have _end != NULL + { + BlockList processed; + BlockList to_go; + to_go.append(start_block); + while(to_go.length() > 0) { + BlockBegin* current = to_go.pop(); + assert(current != NULL, "Should not happen."); + assert(current->end() != NULL, "All blocks reachable from start_block should have end() != NULL."); + processed.append(current); + for(int i = 0; i < current->number_of_sux(); i++) { + BlockBegin* s = current->sux_at(i); + if (!processed.contains(s)) { + to_go.append(s); + } + } + } + } +#endif // ASSERT + _start = setup_start_block(osr_bci, start_block, _osr_entry, _initial_state); eliminate_redundant_phis(_start); diff --git a/src/hotspot/share/c1/c1_IR.cpp b/src/hotspot/share/c1/c1_IR.cpp index 0b7f7a2b2c1..53c19c78683 100644 --- a/src/hotspot/share/c1/c1_IR.cpp +++ b/src/hotspot/share/c1/c1_IR.cpp @@ -1261,6 +1261,16 @@ void IR::print(bool cfg_only, bool live_only) { } } +class EndNotNullValidator : public BlockClosure { + public: + EndNotNullValidator(IR* hir) { + hir->start()->iterate_postorder(this); + } + + void block_do(BlockBegin* block) { + assert(block->end() != NULL, "Expect block end to exist."); + } +}; typedef GrowableArray BlockListList; @@ -1363,6 +1373,7 @@ public: void IR::verify() { #ifdef ASSERT PredecessorValidator pv(this); + EndNotNullValidator(this); VerifyBlockBeginField verifier; this->iterate_postorder(&verifier); #endif diff --git a/src/hotspot/share/c1/c1_Instruction.cpp b/src/hotspot/share/c1/c1_Instruction.cpp index 687d2dae707..801aeb054aa 100644 --- a/src/hotspot/share/c1/c1_Instruction.cpp +++ b/src/hotspot/share/c1/c1_Instruction.cpp @@ -524,39 +524,21 @@ Constant::CompareResult Constant::compare(Instruction::Condition cond, Value rig // Implementation of BlockBegin -void BlockBegin::set_end(BlockEnd* end) { - assert(end != NULL, "should not reset block end to NULL"); - if (end == _end) { - return; +void BlockBegin::set_end(BlockEnd* new_end) { // Assumes that no predecessor of new_end still has it as its successor + assert(new_end != NULL, "Should not reset block new_end to NULL"); + if (new_end == _end) return; + + // Remove this block as predecessor of its current successors + if (_end != NULL) + for (int i = 0; i < number_of_sux(); i++) { + sux_at(i)->remove_predecessor(this); } - clear_end(); - // Set the new end - _end = end; + _end = new_end; - _successors.clear(); - // Now reset successors list based on BlockEnd - for (int i = 0; i < end->number_of_sux(); i++) { - BlockBegin* sux = end->sux_at(i); - _successors.append(sux); - sux->_predecessors.append(this); - } - _end->set_begin(this); -} - - -void BlockBegin::clear_end() { - // Must make the predecessors/successors match up with the - // BlockEnd's notion. - if (_end != NULL) { - // disconnect from the old end - _end->set_begin(NULL); - - // disconnect this block from it's current successors - for (int i = 0; i < _successors.length(); i++) { - _successors.at(i)->remove_predecessor(this); - } - _end = NULL; + // Add this block as predecessor of its new successors + for (int i = 0; i < number_of_sux(); i++) { + sux_at(i)->add_predecessor(this); } } @@ -575,7 +557,7 @@ void BlockBegin::disconnect_edge(BlockBegin* from, BlockBegin* to) { if (index >= 0) { sux->_predecessors.remove_at(index); } - from->_successors.remove_at(s); + from->end()->remove_sux_at(s); } else { s++; } @@ -583,16 +565,6 @@ void BlockBegin::disconnect_edge(BlockBegin* from, BlockBegin* to) { } -void BlockBegin::disconnect_from_graph() { - // disconnect this block from all other blocks - for (int p = 0; p < number_of_preds(); p++) { - pred_at(p)->remove_successor(this); - } - for (int s = 0; s < number_of_sux(); s++) { - sux_at(s)->remove_predecessor(this); - } -} - void BlockBegin::substitute_sux(BlockBegin* old_sux, BlockBegin* new_sux) { // modify predecessors before substituting successors for (int i = 0; i < number_of_sux(); i++) { @@ -669,14 +641,6 @@ BlockBegin* BlockBegin::insert_block_between(BlockBegin* sux) { } -void BlockBegin::remove_successor(BlockBegin* pred) { - int idx; - while ((idx = _successors.find(pred)) >= 0) { - _successors.remove_at(idx); - } -} - - void BlockBegin::add_predecessor(BlockBegin* pred) { _predecessors.append(pred); } @@ -953,26 +917,10 @@ void BlockList::print(bool cfg_only, bool live_only) { // Implementation of BlockEnd -void BlockEnd::set_begin(BlockBegin* begin) { - BlockList* sux = NULL; - if (begin != NULL) { - sux = begin->successors(); - } else if (this->begin() != NULL) { - // copy our sux list - BlockList* sux = new BlockList(this->begin()->number_of_sux()); - for (int i = 0; i < this->begin()->number_of_sux(); i++) { - sux->append(this->begin()->sux_at(i)); - } - } - _sux = sux; -} - - void BlockEnd::substitute_sux(BlockBegin* old_sux, BlockBegin* new_sux) { substitute(*_sux, old_sux, new_sux); } - // Implementation of Phi // Normal phi functions take their operands from the last instruction of the diff --git a/src/hotspot/share/c1/c1_Instruction.hpp b/src/hotspot/share/c1/c1_Instruction.hpp index 8d360b094b8..5b5ea8a863c 100644 --- a/src/hotspot/share/c1/c1_Instruction.hpp +++ b/src/hotspot/share/c1/c1_Instruction.hpp @@ -1595,7 +1595,6 @@ LEAF(BlockBegin, StateSplit) ResourceBitMap _stores_to_locals; // bit is set when a local variable is stored in the block // SSA specific fields: (factor out later) - BlockList _successors; // the successors of this block BlockList _predecessors; // the predecessors of this block BlockList _dominates; // list of blocks that are dominated by this block BlockBegin* _dominator; // the dominator of this block @@ -1649,7 +1648,6 @@ LEAF(BlockBegin, StateSplit) , _flags(0) , _total_preds(0) , _stores_to_locals() - , _successors(2) , _predecessors(2) , _dominates(2) , _dominator(NULL) @@ -1676,7 +1674,6 @@ LEAF(BlockBegin, StateSplit) // accessors int block_id() const { return _block_id; } int bci() const { return _bci; } - BlockList* successors() { return &_successors; } BlockList* dominates() { return &_dominates; } BlockBegin* dominator() const { return _dominator; } int loop_depth() const { return _loop_depth; } @@ -1704,9 +1701,7 @@ LEAF(BlockBegin, StateSplit) void set_dominator_depth(int d) { _dominator_depth = d; } void set_depth_first_number(int dfn) { _depth_first_number = dfn; } void set_linear_scan_number(int lsn) { _linear_scan_number = lsn; } - void set_end(BlockEnd* end); - void clear_end(); - void disconnect_from_graph(); + void set_end(BlockEnd* new_end); static void disconnect_edge(BlockBegin* from, BlockBegin* to); BlockBegin* insert_block_between(BlockBegin* sux); void substitute_sux(BlockBegin* old_sux, BlockBegin* new_sux); @@ -1729,10 +1724,6 @@ LEAF(BlockBegin, StateSplit) // successors and predecessors int number_of_sux() const; BlockBegin* sux_at(int i) const; - void add_successor(BlockBegin* sux); - void remove_successor(BlockBegin* pred); - bool is_successor(BlockBegin* sux) const { return _successors.contains(sux); } - void add_predecessor(BlockBegin* pred); void remove_predecessor(BlockBegin* pred); bool is_predecessor(BlockBegin* pred) const { return _predecessors.contains(pred); } @@ -1792,6 +1783,7 @@ LEAF(BlockBegin, StateSplit) // debugging void print_block() PRODUCT_RETURN; void print_block(InstructionPrinter& ip, bool live_only = false) PRODUCT_RETURN; + }; @@ -1825,14 +1817,13 @@ BASE(BlockEnd, StateSplit) BlockBegin* begin() const { return _block; } // manipulation - void set_begin(BlockBegin* begin); + inline void remove_sux_at(int i) { _sux->remove_at(i);} + inline int find_sux(BlockBegin* sux) {return _sux->find(sux);} // successors int number_of_sux() const { return _sux != NULL ? _sux->length() : 0; } BlockBegin* sux_at(int i) const { return _sux->at(i); } BlockBegin* default_sux() const { return sux_at(number_of_sux() - 1); } - BlockBegin** addr_sux_at(int i) const { return _sux->adr_at(i); } - int sux_index(BlockBegin* sux) const { return _sux->find(sux); } void substitute_sux(BlockBegin* old_sux, BlockBegin* new_sux); }; @@ -2439,9 +2430,8 @@ class BlockPair: public CompilationResourceObj { typedef GrowableArray BlockPairList; -inline int BlockBegin::number_of_sux() const { assert(_end == NULL || _end->number_of_sux() == _successors.length(), "mismatch"); return _successors.length(); } -inline BlockBegin* BlockBegin::sux_at(int i) const { assert(_end == NULL || _end->sux_at(i) == _successors.at(i), "mismatch"); return _successors.at(i); } -inline void BlockBegin::add_successor(BlockBegin* sux) { assert(_end == NULL, "Would create mismatch with successors of BlockEnd"); _successors.append(sux); } +inline int BlockBegin::number_of_sux() const { assert(_end != NULL, "need end"); return _end->number_of_sux(); } +inline BlockBegin* BlockBegin::sux_at(int i) const { assert(_end != NULL , "need end"); return _end->sux_at(i); } #undef ASSERT_VALUES diff --git a/src/hotspot/share/c1/c1_InstructionPrinter.cpp b/src/hotspot/share/c1/c1_InstructionPrinter.cpp index 3e0708351a6..1a074a98ff8 100644 --- a/src/hotspot/share/c1/c1_InstructionPrinter.cpp +++ b/src/hotspot/share/c1/c1_InstructionPrinter.cpp @@ -612,14 +612,7 @@ void InstructionPrinter::do_BlockBegin(BlockBegin* x) { output()->print(" dom B%d", x->dominator()->block_id()); } - // print predecessors and successors - if (x->successors()->length() > 0) { - output()->print(" sux:"); - for (int i = 0; i < x->successors()->length(); i ++) { - output()->print(" B%d", x->successors()->at(i)->block_id()); - } - } - + // print predecessors if (x->number_of_preds() > 0) { output()->print(" pred:"); for (int i = 0; i < x->number_of_preds(); i ++) { diff --git a/src/hotspot/share/c1/c1_LIR.cpp b/src/hotspot/share/c1/c1_LIR.cpp index c726c94daef..d27b9175c9d 100644 --- a/src/hotspot/share/c1/c1_LIR.cpp +++ b/src/hotspot/share/c1/c1_LIR.cpp @@ -1580,7 +1580,7 @@ static void print_block(BlockBegin* x) { } } - if (x->number_of_sux() > 0) { + if (end != NULL && x->number_of_sux() > 0) { tty->print("sux: "); for (int i = 0; i < x->number_of_sux(); i ++) { tty->print("B%d ", x->sux_at(i)->block_id()); diff --git a/src/hotspot/share/c1/c1_Optimizer.cpp b/src/hotspot/share/c1/c1_Optimizer.cpp index 8e32b813ef7..e406a530b73 100644 --- a/src/hotspot/share/c1/c1_Optimizer.cpp +++ b/src/hotspot/share/c1/c1_Optimizer.cpp @@ -312,6 +312,19 @@ void Optimizer::eliminate_conditional_expressions() { CE_Eliminator ce(ir()); } +void disconnect_from_graph(BlockBegin* block) { + for (int p = 0; p < block->number_of_preds(); p++) { + BlockBegin* pred = block->pred_at(p); + int idx; + while ((idx = pred->end()->find_sux(block)) >= 0) { + pred->end()->remove_sux_at(idx); + } + } + for (int s = 0; s < block->number_of_sux(); s++) { + block->sux_at(s)->remove_predecessor(block); + } +} + class BlockMerger: public BlockClosure { private: IR* _hir; @@ -369,8 +382,8 @@ class BlockMerger: public BlockClosure { for_each_local_value(sux_state, index, sux_value) { Phi* sux_phi = sux_value->as_Phi(); if (sux_phi != NULL && sux_phi->is_illegal()) continue; - assert(sux_value == end_state->local_at(index), "locals not equal"); - } + assert(sux_value == end_state->local_at(index), "locals not equal"); + } assert(sux_state->caller_state() == end_state->caller_state(), "caller not equal"); #endif @@ -380,8 +393,12 @@ class BlockMerger: public BlockClosure { assert(prev->as_BlockEnd() == NULL, "must not be a BlockEnd"); prev->set_next(next); prev->fixup_block_pointers(); - sux->disconnect_from_graph(); + + // disconnect this block from all other blocks + disconnect_from_graph(sux); block->set_end(sux->end()); + + // TODO Should this be done in set_end universally? // add exception handlers of deleted block, if any for (int k = 0; k < sux->number_of_exception_handlers(); k++) { BlockBegin* xhandler = sux->exception_handler_at(k);