8277496: Remove duplication in c1 Block successor lists

Reviewed-by: neliasso, kvn
This commit is contained in:
Ludvig Janiuk 2021-12-06 08:58:41 +00:00 committed by Nils Eliasson
parent 194cdf4e28
commit 8d190dd003
8 changed files with 122 additions and 106 deletions

View File

@ -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();

View File

@ -53,6 +53,7 @@ class BlockListBuilder {
BlockList _blocks; // internal list of all blocks
BlockList* _bci2block; // mapping from bci to blocks for GraphBuilder
GrowableArray<BlockList> _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);

View File

@ -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<BlockList*> BlockListList;
@ -1363,6 +1373,7 @@ public:
void IR::verify() {
#ifdef ASSERT
PredecessorValidator pv(this);
EndNotNullValidator(this);
VerifyBlockBeginField verifier;
this->iterate_postorder(&verifier);
#endif

View File

@ -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

View File

@ -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<BlockPair*> 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

View File

@ -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 ++) {

View File

@ -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());

View File

@ -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);