@@ -59,7 +59,7 @@ class BlockListBuilder {
59
59
// fields used by mark_loops
60
60
ResourceBitMap _active; // for iteration of control flow graph
61
61
ResourceBitMap _visited; // for iteration of control flow graph
62
- intArray _loop_map; // caches the information if a block is contained in a loop
62
+ GrowableArray<ResourceBitMap> _loop_map; // caches the information if a block is contained in a loop
63
63
int _next_loop_index; // next free loop number
64
64
int _next_block_number; // for reverse postorder numbering of blocks
65
65
@@ -84,7 +84,7 @@ class BlockListBuilder {
84
84
85
85
void make_loop_header (BlockBegin* block);
86
86
void mark_loops ();
87
- int mark_loops (BlockBegin* b, bool in_subroutine);
87
+ BitMap& mark_loops (BlockBegin* b, bool in_subroutine);
88
88
89
89
// debugging
90
90
#ifndef PRODUCT
@@ -376,17 +376,36 @@ void BlockListBuilder::mark_loops() {
376
376
377
377
_active.initialize (BlockBegin::number_of_blocks ());
378
378
_visited.initialize (BlockBegin::number_of_blocks ());
379
- _loop_map = intArray (BlockBegin::number_of_blocks (), BlockBegin::number_of_blocks (), 0 );
379
+ _loop_map = GrowableArray<ResourceBitMap>(BlockBegin::number_of_blocks (), BlockBegin::number_of_blocks (), ResourceBitMap ());
380
+ for (int i = 0 ; i < BlockBegin::number_of_blocks (); i++) {
381
+ _loop_map.at (i).initialize (BlockBegin::number_of_blocks ());
382
+ }
380
383
_next_loop_index = 0 ;
381
384
_next_block_number = _blocks.length ();
382
385
383
- // recursively iterate the control flow graph
384
- mark_loops (_bci2block->at (0 ), false );
386
+ // The loop detection algorithm works as follows:
387
+ // - We maintain the _loop_map, where for each block we have a bitmap indicating which loops contain this block.
388
+ // - The CFG is recursively traversed (depth-first) and if we detect a loop, we assign the loop a unique number that is stored
389
+ // in the bitmap associated with the loop header block. Until we return back through that loop header the bitmap contains
390
+ // only a single bit corresponding to the loop number.
391
+ // - The bit is then propagated for all the blocks in the loop after we exit them (post-order). There could be multiple bits
392
+ // of course in case of nested loops.
393
+ // - When we exit the loop header we remove that single bit and assign the real loop state for it.
394
+ // - Now, the tricky part here is how we detect irriducible loops. In the algorithm above the loop state bits
395
+ // are propagated to the predecessors. If we encounter an irreducible loop (a loop with multiple heads) we would see
396
+ // a node with some loop bit set that would then propagate back and be never cleared because we would
397
+ // never go back through the original loop header. Therefore if there are any irreducible loops the bits in the states
398
+ // for these loops are going to propagate back to the root.
399
+ BitMap& loop_state = mark_loops (_bci2block->at (0 ), false );
400
+ if (!loop_state.is_empty ()) {
401
+ compilation ()->set_has_irreducible_loops (true );
402
+ }
385
403
assert (_next_block_number >= 0 , " invalid block numbers" );
386
404
387
405
// Remove dangling Resource pointers before the ResourceMark goes out-of-scope.
388
406
_active.resize (0 );
389
407
_visited.resize (0 );
408
+ _loop_map.clear ();
390
409
}
391
410
392
411
void BlockListBuilder::make_loop_header (BlockBegin* block) {
@@ -398,19 +417,17 @@ void BlockListBuilder::make_loop_header(BlockBegin* block) {
398
417
if (!block->is_set (BlockBegin::parser_loop_header_flag)) {
399
418
block->set (BlockBegin::parser_loop_header_flag);
400
419
401
- assert (_loop_map.at (block->block_id ()) == 0 , " must not be set yet" );
402
- assert (0 <= _next_loop_index && _next_loop_index < BitsPerInt, " _next_loop_index is used as a bit-index in integer" );
403
- _loop_map.at_put (block->block_id (), 1 << _next_loop_index);
404
- if (_next_loop_index < 31 ) _next_loop_index++;
420
+ assert (_loop_map.at (block->block_id ()).is_empty (), " must not be set yet" );
421
+ assert (0 <= _next_loop_index && _next_loop_index < BlockBegin::number_of_blocks (), " _next_loop_index is too large" );
422
+ _loop_map.at (block->block_id ()).set_bit (_next_loop_index++);
405
423
} else {
406
424
// block already marked as loop header
407
- assert (is_power_of_2 (( unsigned int ) _loop_map.at (block->block_id ())) , " exactly one bit must be set" );
425
+ assert (_loop_map.at (block->block_id ()). count_one_bits () == 1 , " exactly one bit must be set" );
408
426
}
409
427
}
410
428
411
- int BlockListBuilder::mark_loops (BlockBegin* block, bool in_subroutine) {
429
+ BitMap& BlockListBuilder::mark_loops (BlockBegin* block, bool in_subroutine) {
412
430
int block_id = block->block_id ();
413
-
414
431
if (_visited.at (block_id)) {
415
432
if (_active.at (block_id)) {
416
433
// reached block via backward branch
@@ -428,10 +445,11 @@ int BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) {
428
445
_visited.set_bit (block_id);
429
446
_active.set_bit (block_id);
430
447
431
- intptr_t loop_state = 0 ;
448
+ ResourceMark rm;
449
+ ResourceBitMap loop_state (BlockBegin::number_of_blocks ());
432
450
for (int i = number_of_successors (block) - 1 ; i >= 0 ; i--) {
433
451
// recursively process all successors
434
- loop_state |= mark_loops (successor_at (block, i), in_subroutine);
452
+ loop_state. set_union ( mark_loops (successor_at (block, i), in_subroutine) );
435
453
}
436
454
437
455
// clear active-bit after all successors are processed
@@ -441,26 +459,22 @@ int BlockListBuilder::mark_loops(BlockBegin* block, bool in_subroutine) {
441
459
block->set_depth_first_number (_next_block_number);
442
460
_next_block_number--;
443
461
444
- if (loop_state != 0 || in_subroutine ) {
462
+ if (!loop_state. is_empty () || in_subroutine ) {
445
463
// block is contained at least in one loop, so phi functions are necessary
446
464
// phi functions are also necessary for all locals stored in a subroutine
447
465
scope ()->requires_phi_function ().set_union (block->stores_to_locals ());
448
466
}
449
467
450
468
if (block->is_set (BlockBegin::parser_loop_header_flag)) {
451
- int header_loop_state = _loop_map.at (block_id);
452
- assert (is_power_of_2 ((unsigned )header_loop_state), " exactly one bit must be set" );
453
-
454
- // If the highest bit is set (i.e. when integer value is negative), the method
455
- // has 32 or more loops. This bit is never cleared because it is used for multiple loops
456
- if (header_loop_state >= 0 ) {
457
- clear_bits (loop_state, header_loop_state);
458
- }
469
+ BitMap& header_loop_state = _loop_map.at (block_id);
470
+ assert (header_loop_state.count_one_bits () == 1 , " exactly one bit must be set" );
471
+ // remove the bit with the loop number for the state (header is outside of the loop)
472
+ loop_state.set_difference (header_loop_state);
459
473
}
460
474
461
475
// cache and return loop information for this block
462
- _loop_map.at_put (block_id, loop_state);
463
- return loop_state ;
476
+ _loop_map.at (block_id). set_from ( loop_state);
477
+ return _loop_map. at (block_id) ;
464
478
}
465
479
466
480
inline int BlockListBuilder::number_of_successors (BlockBegin* block)
@@ -2496,7 +2510,7 @@ XHandlers* GraphBuilder::handle_exception(Instruction* instruction) {
2496
2510
// The only test case we've seen so far which exhibits this
2497
2511
// problem is caught by the infinite recursion test in
2498
2512
// GraphBuilder::jsr() if the join doesn't work.
2499
- if (!entry->try_merge (cur_state)) {
2513
+ if (!entry->try_merge (cur_state, compilation ()-> has_irreducible_loops () )) {
2500
2514
BAILOUT_ (" error while joining with exception handler, prob. due to complicated jsr/rets" , exception_handlers);
2501
2515
}
2502
2516
@@ -2982,7 +2996,7 @@ BlockEnd* GraphBuilder::iterate_bytecodes_for_block(int bci) {
2982
2996
BlockBegin* sux = end->sux_at (i);
2983
2997
assert (sux->is_predecessor (block ()), " predecessor missing" );
2984
2998
// be careful, bailout if bytecodes are strange
2985
- if (!sux->try_merge (end->state ())) BAILOUT_ (" block join failed" , NULL );
2999
+ if (!sux->try_merge (end->state (), compilation ()-> has_irreducible_loops () )) BAILOUT_ (" block join failed" , NULL );
2986
3000
scope_data ()->add_to_work_list (end->sux_at (i));
2987
3001
}
2988
3002
@@ -3136,7 +3150,7 @@ BlockBegin* GraphBuilder::setup_start_block(int osr_bci, BlockBegin* std_entry,
3136
3150
3137
3151
if (base->std_entry ()->state () == NULL ) {
3138
3152
// setup states for header blocks
3139
- base->std_entry ()->merge (state);
3153
+ base->std_entry ()->merge (state, compilation ()-> has_irreducible_loops () );
3140
3154
}
3141
3155
3142
3156
assert (base->std_entry ()->state () != NULL , " " );
@@ -3219,7 +3233,7 @@ void GraphBuilder::setup_osr_entry_block() {
3219
3233
Goto* g = new Goto (target, false );
3220
3234
append (g);
3221
3235
_osr_entry->set_end (g);
3222
- target->merge (_osr_entry->end ()->state ());
3236
+ target->merge (_osr_entry->end ()->state (), compilation ()-> has_irreducible_loops () );
3223
3237
3224
3238
scope_data ()->set_stream (NULL );
3225
3239
}
@@ -3278,7 +3292,7 @@ GraphBuilder::GraphBuilder(Compilation* compilation, IRScope* scope)
3278
3292
3279
3293
// setup state for std entry
3280
3294
_initial_state = state_at_entry ();
3281
- start_block->merge (_initial_state);
3295
+ start_block->merge (_initial_state, compilation-> has_irreducible_loops () );
3282
3296
3283
3297
// End nulls still exist here
3284
3298
@@ -4029,7 +4043,7 @@ bool GraphBuilder::try_inline_full(ciMethod* callee, bool holder_known, bool ign
4029
4043
// the entry bci for the callee instead of the call site bci.
4030
4044
append_with_bci (goto_callee, 0 );
4031
4045
_block->set_end (goto_callee);
4032
- callee_start_block->merge (callee_state);
4046
+ callee_start_block->merge (callee_state, compilation ()-> has_irreducible_loops () );
4033
4047
4034
4048
_last = _block = callee_start_block;
4035
4049
0 commit comments