8289069: Very slow C1 arraycopy jcstress tests after JDK-8279886
Reviewed-by: thartmann, kvn
This commit is contained in:
parent
c42b796f91
commit
9b7805e3b4
@ -66,7 +66,9 @@ class BlockListBuilder {
|
|||||||
GrowableArray<ResourceBitMap> _loop_map; // caches the information if a block is contained in a loop
|
GrowableArray<ResourceBitMap> _loop_map; // caches the information if a block is contained in a loop
|
||||||
int _next_loop_index; // next free loop number
|
int _next_loop_index; // next free loop number
|
||||||
int _next_block_number; // for reverse postorder numbering of blocks
|
int _next_block_number; // for reverse postorder numbering of blocks
|
||||||
|
int _block_id_start;
|
||||||
|
|
||||||
|
int bit_number(int block_id) const { return block_id - _block_id_start; }
|
||||||
// accessors
|
// accessors
|
||||||
Compilation* compilation() const { return _compilation; }
|
Compilation* compilation() const { return _compilation; }
|
||||||
IRScope* scope() const { return _scope; }
|
IRScope* scope() const { return _scope; }
|
||||||
@ -122,6 +124,7 @@ BlockListBuilder::BlockListBuilder(Compilation* compilation, IRScope* scope, int
|
|||||||
, _loop_map() // size not known yet
|
, _loop_map() // size not known yet
|
||||||
, _next_loop_index(0)
|
, _next_loop_index(0)
|
||||||
, _next_block_number(0)
|
, _next_block_number(0)
|
||||||
|
, _block_id_start(0)
|
||||||
{
|
{
|
||||||
set_entries(osr_bci);
|
set_entries(osr_bci);
|
||||||
set_leaders();
|
set_leaders();
|
||||||
@ -384,11 +387,12 @@ void BlockListBuilder::set_leaders() {
|
|||||||
void BlockListBuilder::mark_loops() {
|
void BlockListBuilder::mark_loops() {
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
|
|
||||||
_active.initialize(BlockBegin::number_of_blocks());
|
const int number_of_blocks = _blocks.length();
|
||||||
_visited.initialize(BlockBegin::number_of_blocks());
|
_active.initialize(number_of_blocks);
|
||||||
_loop_map = GrowableArray<ResourceBitMap>(BlockBegin::number_of_blocks(), BlockBegin::number_of_blocks(), ResourceBitMap());
|
_visited.initialize(number_of_blocks);
|
||||||
for (int i = 0; i < BlockBegin::number_of_blocks(); i++) {
|
_loop_map = GrowableArray<ResourceBitMap>(number_of_blocks, number_of_blocks, ResourceBitMap());
|
||||||
_loop_map.at(i).initialize(BlockBegin::number_of_blocks());
|
for (int i = 0; i < number_of_blocks; i++) {
|
||||||
|
_loop_map.at(i).initialize(number_of_blocks);
|
||||||
}
|
}
|
||||||
_next_loop_index = 0;
|
_next_loop_index = 0;
|
||||||
_next_block_number = _blocks.length();
|
_next_block_number = _blocks.length();
|
||||||
@ -401,12 +405,14 @@ void BlockListBuilder::mark_loops() {
|
|||||||
// - The bit is then propagated for all the blocks in the loop after we exit them (post-order). There could be multiple bits
|
// - The bit is then propagated for all the blocks in the loop after we exit them (post-order). There could be multiple bits
|
||||||
// of course in case of nested loops.
|
// of course in case of nested loops.
|
||||||
// - When we exit the loop header we remove that single bit and assign the real loop state for it.
|
// - When we exit the loop header we remove that single bit and assign the real loop state for it.
|
||||||
// - Now, the tricky part here is how we detect irriducible loops. In the algorithm above the loop state bits
|
// - Now, the tricky part here is how we detect irreducible loops. In the algorithm above the loop state bits
|
||||||
// are propagated to the predecessors. If we encounter an irreducible loop (a loop with multiple heads) we would see
|
// are propagated to the predecessors. If we encounter an irreducible loop (a loop with multiple heads) we would see
|
||||||
// a node with some loop bit set that would then propagate back and be never cleared because we would
|
// a node with some loop bit set that would then propagate back and be never cleared because we would
|
||||||
// never go back through the original loop header. Therefore if there are any irreducible loops the bits in the states
|
// never go back through the original loop header. Therefore if there are any irreducible loops the bits in the states
|
||||||
// for these loops are going to propagate back to the root.
|
// for these loops are going to propagate back to the root.
|
||||||
BitMap& loop_state = mark_loops(_bci2block->at(0), false);
|
BlockBegin* start = _bci2block->at(0);
|
||||||
|
_block_id_start = start->block_id();
|
||||||
|
BitMap& loop_state = mark_loops(start, false);
|
||||||
if (!loop_state.is_empty()) {
|
if (!loop_state.is_empty()) {
|
||||||
compilation()->set_has_irreducible_loops(true);
|
compilation()->set_has_irreducible_loops(true);
|
||||||
}
|
}
|
||||||
@ -419,6 +425,8 @@ void BlockListBuilder::mark_loops() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BlockListBuilder::make_loop_header(BlockBegin* block) {
|
void BlockListBuilder::make_loop_header(BlockBegin* block) {
|
||||||
|
int block_id = block->block_id();
|
||||||
|
int block_bit = bit_number(block_id);
|
||||||
if (block->is_set(BlockBegin::exception_entry_flag)) {
|
if (block->is_set(BlockBegin::exception_entry_flag)) {
|
||||||
// exception edges may look like loops but don't mark them as such
|
// exception edges may look like loops but don't mark them as such
|
||||||
// since it screws up block ordering.
|
// since it screws up block ordering.
|
||||||
@ -427,24 +435,25 @@ void BlockListBuilder::make_loop_header(BlockBegin* block) {
|
|||||||
if (!block->is_set(BlockBegin::parser_loop_header_flag)) {
|
if (!block->is_set(BlockBegin::parser_loop_header_flag)) {
|
||||||
block->set(BlockBegin::parser_loop_header_flag);
|
block->set(BlockBegin::parser_loop_header_flag);
|
||||||
|
|
||||||
assert(_loop_map.at(block->block_id()).is_empty(), "must not be set yet");
|
assert(_loop_map.at(block_bit).is_empty(), "must not be set yet");
|
||||||
assert(0 <= _next_loop_index && _next_loop_index < BlockBegin::number_of_blocks(), "_next_loop_index is too large");
|
assert(0 <= _next_loop_index && _next_loop_index < _loop_map.length(), "_next_loop_index is too large");
|
||||||
_loop_map.at(block->block_id()).set_bit(_next_loop_index++);
|
_loop_map.at(block_bit).set_bit(_next_loop_index++);
|
||||||
} else {
|
} else {
|
||||||
// block already marked as loop header
|
// block already marked as loop header
|
||||||
assert(_loop_map.at(block->block_id()).count_one_bits() == 1, "exactly one bit must be set");
|
assert(_loop_map.at(block_bit).count_one_bits() == 1, "exactly one bit must be set");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
BitMap& BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) {
|
BitMap& BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) {
|
||||||
int block_id = block->block_id();
|
int block_id = block->block_id();
|
||||||
if (_visited.at(block_id)) {
|
int block_bit = bit_number(block_id);
|
||||||
if (_active.at(block_id)) {
|
if (_visited.at(block_bit)) {
|
||||||
|
if (_active.at(block_bit)) {
|
||||||
// reached block via backward branch
|
// reached block via backward branch
|
||||||
make_loop_header(block);
|
make_loop_header(block);
|
||||||
}
|
}
|
||||||
// return cached loop information for this block
|
// return cached loop information for this block
|
||||||
return _loop_map.at(block_id);
|
return _loop_map.at(block_bit);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (block->is_set(BlockBegin::subroutine_entry_flag)) {
|
if (block->is_set(BlockBegin::subroutine_entry_flag)) {
|
||||||
@ -452,18 +461,19 @@ BitMap& BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// set active and visited bits before successors are processed
|
// set active and visited bits before successors are processed
|
||||||
_visited.set_bit(block_id);
|
_visited.set_bit(block_bit);
|
||||||
_active.set_bit(block_id);
|
_active.set_bit(block_bit);
|
||||||
|
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
ResourceBitMap loop_state(BlockBegin::number_of_blocks());
|
ResourceBitMap loop_state(_loop_map.length());
|
||||||
for (int i = number_of_successors(block) - 1; i >= 0; i--) {
|
for (int i = number_of_successors(block) - 1; i >= 0; i--) {
|
||||||
|
BlockBegin* sux = successor_at(block, i);
|
||||||
// recursively process all successors
|
// recursively process all successors
|
||||||
loop_state.set_union(mark_loops(successor_at(block, i), in_subroutine));
|
loop_state.set_union(mark_loops(sux, in_subroutine));
|
||||||
}
|
}
|
||||||
|
|
||||||
// clear active-bit after all successors are processed
|
// clear active-bit after all successors are processed
|
||||||
_active.clear_bit(block_id);
|
_active.clear_bit(block_bit);
|
||||||
|
|
||||||
// reverse-post-order numbering of all blocks
|
// reverse-post-order numbering of all blocks
|
||||||
block->set_depth_first_number(_next_block_number);
|
block->set_depth_first_number(_next_block_number);
|
||||||
@ -476,15 +486,15 @@ BitMap& BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (block->is_set(BlockBegin::parser_loop_header_flag)) {
|
if (block->is_set(BlockBegin::parser_loop_header_flag)) {
|
||||||
BitMap& header_loop_state = _loop_map.at(block_id);
|
BitMap& header_loop_state = _loop_map.at(block_bit);
|
||||||
assert(header_loop_state.count_one_bits() == 1, "exactly one bit must be set");
|
assert(header_loop_state.count_one_bits() == 1, "exactly one bit must be set");
|
||||||
// remove the bit with the loop number for the state (header is outside of the loop)
|
// remove the bit with the loop number for the state (header is outside of the loop)
|
||||||
loop_state.set_difference(header_loop_state);
|
loop_state.set_difference(header_loop_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
// cache and return loop information for this block
|
// cache and return loop information for this block
|
||||||
_loop_map.at(block_id).set_from(loop_state);
|
_loop_map.at(block_bit).set_from(loop_state);
|
||||||
return _loop_map.at(block_id);
|
return _loop_map.at(block_bit);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline int BlockListBuilder::number_of_successors(BlockBegin* block)
|
inline int BlockListBuilder::number_of_successors(BlockBegin* block)
|
||||||
|
Loading…
Reference in New Issue
Block a user