@@ -878,6 +878,19 @@ void Predicates::dump_for_loop(LoopNode* loop_node) {
878
878
}
879
879
#endif // NOT PRODUCT
880
880
881
+ CreateAssertionPredicatesVisitor::CreateAssertionPredicatesVisitor (CountedLoopNode* target_loop_head,
882
+ PhaseIdealLoop* phase,
883
+ const NodeInLoopBody& node_in_loop_body,
884
+ const bool clone_template)
885
+ : _init(target_loop_head->init_trip ()),
886
+ _stride(target_loop_head->stride ()),
887
+ _old_target_loop_entry(target_loop_head->skip_strip_mined ()->in(LoopNode::EntryControl)),
888
+ _current_predicate_chain_head(target_loop_head->skip_strip_mined ()), // Initially no predicates, yet.
889
+ _phase(phase),
890
+ _has_hoisted_check_parse_predicates(false ),
891
+ _node_in_loop_body(node_in_loop_body),
892
+ _clone_template(clone_template) {}
893
+
881
894
// Keep track of whether we are in the correct Predicate Block where Template Assertion Predicates can be found.
882
895
// The PredicateIterator will always start at the loop entry and first visits the Loop Limit Check Predicate Block.
883
896
void CreateAssertionPredicatesVisitor::visit (const ParsePredicate& parse_predicate) {
@@ -894,31 +907,74 @@ void CreateAssertionPredicatesVisitor::visit(const TemplateAssertionPredicate& t
894
907
return ;
895
908
}
896
909
if (_clone_template) {
897
- _new_control = clone_template_and_replace_init_input (template_assertion_predicate);
910
+ IfTrueNode* cloned_template_success_proj = clone_template_and_replace_init_input (template_assertion_predicate);
911
+ initialize_from_template (template_assertion_predicate, cloned_template_success_proj);
912
+ _current_predicate_chain_head = cloned_template_success_proj->in (0 );
913
+ } else {
914
+ IfTrueNode* initialized_success_proj = initialize_from_template (template_assertion_predicate, _old_target_loop_entry);
915
+ _current_predicate_chain_head = initialized_success_proj->in (0 );
898
916
}
899
- _new_control = initialize_from_template (template_assertion_predicate);
900
917
}
901
918
902
919
// Create an Initialized Assertion Predicate from the provided Template Assertion Predicate.
903
920
IfTrueNode* CreateAssertionPredicatesVisitor::initialize_from_template (
904
- const TemplateAssertionPredicate& template_assertion_predicate) const {
921
+ const TemplateAssertionPredicate& template_assertion_predicate, Node* new_control ) const {
905
922
DEBUG_ONLY (template_assertion_predicate.verify ();)
906
923
IfNode* template_head = template_assertion_predicate.head ();
907
924
InitializedAssertionPredicateCreator initialized_assertion_predicate_creator (_phase);
908
925
IfTrueNode* initialized_predicate = initialized_assertion_predicate_creator.create_from_template (template_head,
909
- _new_control ,
926
+ new_control ,
910
927
_init, _stride);
911
928
DEBUG_ONLY (InitializedAssertionPredicate::verify (initialized_predicate);)
912
929
template_assertion_predicate.rewire_loop_data_dependencies (initialized_predicate, _node_in_loop_body, _phase);
930
+ rewire_to_old_predicate_chain_head (initialized_predicate);
913
931
return initialized_predicate;
914
932
}
915
933
916
934
// Clone the provided 'template_assertion_predicate' and set '_init' as new input for the OpaqueLoopInitNode.
917
935
IfTrueNode* CreateAssertionPredicatesVisitor::clone_template_and_replace_init_input (
918
936
const TemplateAssertionPredicate& template_assertion_predicate) {
919
937
OpaqueLoopInitNode* opaque_init = new OpaqueLoopInitNode (_phase->C , _init);
920
- _phase->register_new_node (opaque_init, _new_control);
921
- return template_assertion_predicate.clone_and_replace_init (_new_control, opaque_init, _phase);
938
+ _phase->register_new_node (opaque_init, _old_target_loop_entry);
939
+ return template_assertion_predicate.clone_and_replace_init (_old_target_loop_entry, opaque_init, _phase);
940
+ }
941
+
942
+ // Rewire the newly created predicates to the old predicate chain head (i.e. '_current_predicate_chain_head') by
943
+ // rewiring the current control input of '_current_predicate_chain_head' from '_old_target_loop_entry' to
944
+ // 'initialized_assertion_predicate_success_proj'. This is required because we walk the predicate chain from the loop
945
+ // up and clone Template Assertion Predicates on the fly:
946
+ //
947
+ // x
948
+ // | old target
949
+ // Template Assertion loop entry
950
+ // Predicate 1 old target clone | \
951
+ // | loop entry TAP 2 | cloned Template Assertion
952
+ // Template Assertion | ======> | Predicate 2
953
+ // Predicate 2 target loop |
954
+ // | target loop #_current_predicate_chain_head
955
+ // source loop
956
+ //
957
+ //
958
+ // old target old target
959
+ // loop entry loop entry
960
+ // | \ rewire |
961
+ // | cloned Template Assertion to old cloned Template Assertion #current_predicate
962
+ // initialize | Predicate 2 predicate Predicate 2 _chain_head (new)
963
+ // TAP 2 | | chain head |
964
+ // ======> | Initialized Assertion ======> Initialized Assertion
965
+ // | Predicate 2 Predicate 2
966
+ // | |
967
+ // target loop #_current_predicate_chain_head target loop
968
+ //
969
+ void CreateAssertionPredicatesVisitor::rewire_to_old_predicate_chain_head (
970
+ Node* initialized_assertion_predicate_success_proj) const {
971
+ if (_current_predicate_chain_head->is_Loop ()) {
972
+ assert (_current_predicate_chain_head->in (LoopNode::EntryControl) == _old_target_loop_entry, " must be old loop entry" );
973
+ _phase->replace_loop_entry (_current_predicate_chain_head->as_Loop (), initialized_assertion_predicate_success_proj);
974
+ } else {
975
+ assert (_current_predicate_chain_head->in (0 ) == _old_target_loop_entry, " must be old loop entry" );
976
+ _phase->replace_control (_current_predicate_chain_head, initialized_assertion_predicate_success_proj);
977
+ }
922
978
}
923
979
924
980
// Clone the Template Assertion Predicate and set a new input for the OpaqueLoopStrideNode.
@@ -951,9 +1007,8 @@ IfTrueNode* UpdateStrideForAssertionPredicates::initialize_from_updated_template
951
1007
void UpdateStrideForAssertionPredicates::connect_initialized_assertion_predicate (
952
1008
Node* new_control_out, IfTrueNode* initialized_success_proj) const {
953
1009
if (new_control_out->is_Loop ()) {
954
- _phase->igvn (). replace_input_of ( new_control_out, LoopNode::EntryControl , initialized_success_proj);
1010
+ _phase->replace_loop_entry ( new_control_out-> as_Loop () , initialized_success_proj);
955
1011
} else {
956
- _phase->igvn (). replace_input_of ( new_control_out, 0 , initialized_success_proj);
1012
+ _phase->replace_control ( new_control_out, initialized_success_proj);
957
1013
}
958
- _phase->set_idom (new_control_out, initialized_success_proj, _phase->dom_depth (new_control_out));
959
1014
}
0 commit comments