8343745: Only update Last Value Assertion Predicates in Loop Unrolling

Reviewed-by: thartmann, kvn
This commit is contained in:
Christian Hagedorn 2024-11-12 10:11:55 +00:00
parent b53ee053f7
commit 3727f40461
7 changed files with 88 additions and 85 deletions

View File

@ -324,7 +324,7 @@ class IfNode : public MultiBranchNode {
float _fcnt; // Frequency counter float _fcnt; // Frequency counter
private: private:
NOT_PRODUCT(AssertionPredicateType _assertion_predicate_type;) AssertionPredicateType _assertion_predicate_type;
void init_node(Node* control, Node* bol) { void init_node(Node* control, Node* bol) {
init_class_id(Class_If); init_class_id(Class_If);
@ -426,7 +426,7 @@ public:
// gen_subtype_check() and catch_inline_exceptions(). // gen_subtype_check() and catch_inline_exceptions().
IfNode(Node* control, Node* bol, float p, float fcnt); IfNode(Node* control, Node* bol, float p, float fcnt);
NOT_PRODUCT(IfNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicateType assertion_predicate_type);) IfNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicateType assertion_predicate_type);
static IfNode* make_with_same_profile(IfNode* if_node_profile, Node* ctrl, BoolNode* bol); static IfNode* make_with_same_profile(IfNode* if_node_profile, Node* ctrl, BoolNode* bol);
@ -448,11 +448,11 @@ public:
// Returns null is it couldn't improve the type. // Returns null is it couldn't improve the type.
static const TypeInt* filtered_int_type(PhaseGVN* phase, Node* val, Node* if_proj); static const TypeInt* filtered_int_type(PhaseGVN* phase, Node* val, Node* if_proj);
#ifndef PRODUCT
AssertionPredicateType assertion_predicate_type() const { AssertionPredicateType assertion_predicate_type() const {
return _assertion_predicate_type; return _assertion_predicate_type;
} }
#ifndef PRODUCT
virtual void dump_spec(outputStream *st) const; virtual void dump_spec(outputStream *st) const;
#endif #endif
@ -468,12 +468,10 @@ public:
init_class_id(Class_RangeCheck); init_class_id(Class_RangeCheck);
} }
#ifndef PRODUCT
RangeCheckNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicateType assertion_predicate_type) RangeCheckNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicateType assertion_predicate_type)
: IfNode(control, bol, p, fcnt, assertion_predicate_type) { : IfNode(control, bol, p, fcnt, assertion_predicate_type) {
init_class_id(Class_RangeCheck); init_class_id(Class_RangeCheck);
} }
#endif // NOT PRODUCT
virtual int Opcode() const; virtual int Opcode() const;
virtual Node* Ideal(PhaseGVN *phase, bool can_reshape); virtual Node* Ideal(PhaseGVN *phase, bool can_reshape);

View File

@ -50,12 +50,11 @@ extern uint explicit_null_checks_elided;
IfNode::IfNode(Node* control, Node* bol, float p, float fcnt) IfNode::IfNode(Node* control, Node* bol, float p, float fcnt)
: MultiBranchNode(2), : MultiBranchNode(2),
_prob(p), _prob(p),
_fcnt(fcnt) _fcnt(fcnt),
NOT_PRODUCT(COMMA _assertion_predicate_type(AssertionPredicateType::None)) { _assertion_predicate_type(AssertionPredicateType::None) {
init_node(control, bol); init_node(control, bol);
} }
#ifndef PRODUCT
IfNode::IfNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicateType assertion_predicate_type) IfNode::IfNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicateType assertion_predicate_type)
: MultiBranchNode(2), : MultiBranchNode(2),
_prob(p), _prob(p),
@ -63,7 +62,6 @@ IfNode::IfNode(Node* control, Node* bol, float p, float fcnt, AssertionPredicate
_assertion_predicate_type(assertion_predicate_type) { _assertion_predicate_type(assertion_predicate_type) {
init_node(control, bol); init_node(control, bol);
} }
#endif // NOT_PRODUCT
//============================================================================= //=============================================================================
//------------------------------Value------------------------------------------ //------------------------------Value------------------------------------------

View File

@ -100,8 +100,8 @@ void PhaseIdealLoop::register_control(Node* n, IdealLoopTree *loop, Node* pred,
// is an IfTrue projection. This code is also used to clone predicates to cloned loops. // is an IfTrue projection. This code is also used to clone predicates to cloned loops.
IfTrueNode* PhaseIdealLoop::create_new_if_for_predicate(ParsePredicateSuccessProj* parse_predicate_success_proj, IfTrueNode* PhaseIdealLoop::create_new_if_for_predicate(ParsePredicateSuccessProj* parse_predicate_success_proj,
Node* new_entry, const Deoptimization::DeoptReason reason, Node* new_entry, const Deoptimization::DeoptReason reason,
const int opcode, const bool rewire_uncommon_proj_phi_inputs const int opcode, const bool rewire_uncommon_proj_phi_inputs,
NOT_PRODUCT (COMMA AssertionPredicateType assertion_predicate_type)) { AssertionPredicateType assertion_predicate_type) {
assert(parse_predicate_success_proj->is_uncommon_trap_if_pattern(reason), "must be a uct if pattern!"); assert(parse_predicate_success_proj->is_uncommon_trap_if_pattern(reason), "must be a uct if pattern!");
ParsePredicateNode* parse_predicate = parse_predicate_success_proj->in(0)->as_ParsePredicate(); ParsePredicateNode* parse_predicate = parse_predicate_success_proj->in(0)->as_ParsePredicate();
ParsePredicateUncommonProj* uncommon_proj = parse_predicate->uncommon_proj(); ParsePredicateUncommonProj* uncommon_proj = parse_predicate->uncommon_proj();

View File

@ -2759,8 +2759,8 @@ void PhaseIdealLoop::do_range_check(IdealLoopTree *loop, Node_List &old_new) {
// cannot remove an empty loop with a constant limit when init is not a constant as well. We will use // cannot remove an empty loop with a constant limit when init is not a constant as well. We will use
// a LoopLimitCheck node that can only be folded if the zero grip guard is also foldable. // a LoopLimitCheck node that can only be folded if the zero grip guard is also foldable.
loop_entry = initialized_assertion_predicate_creator.create(final_iv_placeholder, loop_entry, stride_con, loop_entry = initialized_assertion_predicate_creator.create(final_iv_placeholder, loop_entry, stride_con,
scale_con, int_offset, int_limit NOT_PRODUCT( scale_con, int_offset, int_limit,
COMMA AssertionPredicateType::FinalIv)); AssertionPredicateType::FinalIv);
assert(!assertion_predicate_has_loop_opaque_node(loop_entry->in(0)->as_If()), "unexpected"); assert(!assertion_predicate_has_loop_opaque_node(loop_entry->in(0)->as_If()), "unexpected");
} }
@ -2773,8 +2773,8 @@ void PhaseIdealLoop::do_range_check(IdealLoopTree *loop, Node_List &old_new) {
// Initialized Assertion Predicate for the value of the initial main-loop. // Initialized Assertion Predicate for the value of the initial main-loop.
loop_entry = initialized_assertion_predicate_creator.create(init, loop_entry, stride_con, scale_con, loop_entry = initialized_assertion_predicate_creator.create(init, loop_entry, stride_con, scale_con,
int_offset, int_limit NOT_PRODUCT(COMMA int_offset, int_limit,
AssertionPredicateType::InitValue)); AssertionPredicateType::InitValue);
assert(!assertion_predicate_has_loop_opaque_node(loop_entry->in(0)->as_If()), "unexpected"); assert(!assertion_predicate_has_loop_opaque_node(loop_entry->in(0)->as_If()), "unexpected");
} else { } else {

View File

@ -1352,8 +1352,8 @@ public:
// Create a new if above the uncommon_trap_if_pattern for the predicate to be promoted // Create a new if above the uncommon_trap_if_pattern for the predicate to be promoted
IfTrueNode* create_new_if_for_predicate( IfTrueNode* create_new_if_for_predicate(
ParsePredicateSuccessProj* parse_predicate_proj, Node* new_entry, Deoptimization::DeoptReason reason, int opcode, ParsePredicateSuccessProj* parse_predicate_proj, Node* new_entry, Deoptimization::DeoptReason reason, int opcode,
bool rewire_uncommon_proj_phi_inputs = false bool rewire_uncommon_proj_phi_inputs = false,
NOT_PRODUCT (COMMA AssertionPredicateType assertion_predicate_type = AssertionPredicateType::None)); AssertionPredicateType assertion_predicate_type = AssertionPredicateType::None);
private: private:
// Helper functions for create_new_if_for_predicate() // Helper functions for create_new_if_for_predicate()

View File

@ -171,8 +171,8 @@ IfTrueNode* TemplateAssertionPredicate::clone_and_replace_init(Node* new_control
template_assertion_expression.clone_and_replace_init(new_opaque_init, new_control, phase); template_assertion_expression.clone_and_replace_init(new_opaque_init, new_control, phase);
AssertionPredicateIfCreator assertion_predicate_if_creator(phase); AssertionPredicateIfCreator assertion_predicate_if_creator(phase);
IfTrueNode* success_proj = assertion_predicate_if_creator.create_for_template(new_control, _if_node->Opcode(), IfTrueNode* success_proj = assertion_predicate_if_creator.create_for_template(new_control, _if_node->Opcode(),
new_opaque_node NOT_PRODUCT(COMMA new_opaque_node,
_if_node->assertion_predicate_type())); _if_node->assertion_predicate_type());
assert(PhaseIdealLoop::assertion_predicate_has_loop_opaque_node(success_proj->in(0)->as_If()), assert(PhaseIdealLoop::assertion_predicate_has_loop_opaque_node(success_proj->in(0)->as_If()),
"Template Assertion Predicates must have OpaqueLoop* nodes in the bool expression"); "Template Assertion Predicates must have OpaqueLoop* nodes in the bool expression");
return success_proj; return success_proj;
@ -546,19 +546,19 @@ class AssertionPredicateExpressionCreator : public StackObj {
// Creates an If with a success and a fail path with the given assertion_expression. The only difference to // Creates an If with a success and a fail path with the given assertion_expression. The only difference to
// create_for_initialized() is that we use a template specific Halt message on the fail path. // create_for_initialized() is that we use a template specific Halt message on the fail path.
IfTrueNode* AssertionPredicateIfCreator::create_for_template(Node* new_control, const int if_opcode, IfTrueNode* AssertionPredicateIfCreator::create_for_template(Node* new_control, const int if_opcode,
Node* assertion_expression NOT_PRODUCT(COMMA Node* assertion_expression,
const AssertionPredicateType assertion_predicate_type)) { const AssertionPredicateType assertion_predicate_type) {
const char* halt_message = "Template Assertion Predicates are always removed before code generation"; const char* halt_message = "Template Assertion Predicates are always removed before code generation";
return create(new_control, if_opcode, assertion_expression, halt_message NOT_PRODUCT(COMMA assertion_predicate_type)); return create(new_control, if_opcode, assertion_expression, halt_message, assertion_predicate_type);
} }
// Creates an If with a success and a fail path with the given assertion_expression. The only difference to // Creates an If with a success and a fail path with the given assertion_expression. The only difference to
// create_for_template() is that we use a initialized specific Halt message on the fail path. // create_for_template() is that we use a initialized specific Halt message on the fail path.
IfTrueNode* AssertionPredicateIfCreator::create_for_initialized(Node* new_control, const int if_opcode, IfTrueNode* AssertionPredicateIfCreator::create_for_initialized(Node* new_control, const int if_opcode,
Node* assertion_expression NOT_PRODUCT(COMMA Node* assertion_expression,
const AssertionPredicateType assertion_predicate_type)) { const AssertionPredicateType assertion_predicate_type) {
const char* halt_message = "Initialized Assertion Predicate cannot fail"; const char* halt_message = "Initialized Assertion Predicate cannot fail";
return create(new_control, if_opcode, assertion_expression, halt_message NOT_PRODUCT(COMMA assertion_predicate_type)); return create(new_control, if_opcode, assertion_expression, halt_message, assertion_predicate_type);
} }
// Creates the If node for an Assertion Predicate with a success path and a fail path having a Halt node: // Creates the If node for an Assertion Predicate with a success path and a fail path having a Halt node:
@ -571,28 +571,25 @@ IfTrueNode* AssertionPredicateIfCreator::create_for_initialized(Node* new_contro
// proj with Halt // proj with Halt
// //
IfTrueNode* AssertionPredicateIfCreator::create(Node* new_control, const int if_opcode, Node* assertion_expression, IfTrueNode* AssertionPredicateIfCreator::create(Node* new_control, const int if_opcode, Node* assertion_expression,
const char* halt_message NOT_PRODUCT(COMMA const char* halt_message,
const AssertionPredicateType assertion_predicate_type)) { const AssertionPredicateType assertion_predicate_type) {
assert(assertion_expression->is_OpaqueTemplateAssertionPredicate() || assert(assertion_expression->is_OpaqueTemplateAssertionPredicate() ||
assertion_expression->is_OpaqueInitializedAssertionPredicate(), "not a valid assertion expression"); assertion_expression->is_OpaqueInitializedAssertionPredicate(), "not a valid assertion expression");
IdealLoopTree* loop = _phase->get_loop(new_control); IdealLoopTree* loop = _phase->get_loop(new_control);
IfNode* if_node = create_if_node(new_control, if_opcode, assertion_expression, loop IfNode* if_node = create_if_node(new_control, if_opcode, assertion_expression, loop, assertion_predicate_type);
NOT_PRODUCT(COMMA assertion_predicate_type));
create_fail_path(if_node, loop, halt_message); create_fail_path(if_node, loop, halt_message);
return create_success_path(if_node, loop); return create_success_path(if_node, loop);
} }
IfNode* AssertionPredicateIfCreator::create_if_node(Node* new_control, const int if_opcode, Node* assertion_expression, IfNode* AssertionPredicateIfCreator::create_if_node(Node* new_control, const int if_opcode, Node* assertion_expression,
IdealLoopTree* loop NOT_PRODUCT(COMMA IdealLoopTree* loop,
const AssertionPredicateType assertion_predicate_type)) { const AssertionPredicateType assertion_predicate_type) {
IfNode* if_node; IfNode* if_node;
if (if_opcode == Op_If) { if (if_opcode == Op_If) {
if_node = new IfNode(new_control, assertion_expression, PROB_MAX, COUNT_UNKNOWN if_node = new IfNode(new_control, assertion_expression, PROB_MAX, COUNT_UNKNOWN, assertion_predicate_type);
NOT_PRODUCT(COMMA assertion_predicate_type));
} else { } else {
assert(if_opcode == Op_RangeCheck, "must be range check"); assert(if_opcode == Op_RangeCheck, "must be range check");
if_node = new RangeCheckNode(new_control, assertion_expression, PROB_MAX, COUNT_UNKNOWN if_node = new RangeCheckNode(new_control, assertion_expression, PROB_MAX, COUNT_UNKNOWN, assertion_predicate_type);
NOT_PRODUCT(COMMA assertion_predicate_type));
} }
_phase->register_control(if_node, loop, new_control); _phase->register_control(if_node, loop, new_control);
return if_node; return if_node;
@ -631,13 +628,13 @@ IfTrueNode* TemplateAssertionPredicateCreator::create_with_uncommon_trap(
create_for_init_value(new_control, opaque_init, does_overflow); create_for_init_value(new_control, opaque_init, does_overflow);
IfTrueNode* template_predicate_success_proj = IfTrueNode* template_predicate_success_proj =
create_if_node_with_uncommon_trap(template_assertion_predicate_expression, parse_predicate_success_proj, create_if_node_with_uncommon_trap(template_assertion_predicate_expression, parse_predicate_success_proj,
deopt_reason, if_opcode, does_overflow deopt_reason, if_opcode, does_overflow,
NOT_PRODUCT(COMMA AssertionPredicateType::InitValue)); AssertionPredicateType::InitValue);
template_assertion_predicate_expression = create_for_last_value(template_predicate_success_proj, opaque_init, template_assertion_predicate_expression = create_for_last_value(template_predicate_success_proj, opaque_init,
does_overflow); does_overflow);
return create_if_node_with_uncommon_trap(template_assertion_predicate_expression, parse_predicate_success_proj, return create_if_node_with_uncommon_trap(template_assertion_predicate_expression, parse_predicate_success_proj,
deopt_reason, if_opcode, does_overflow deopt_reason, if_opcode, does_overflow,
NOT_PRODUCT(COMMA AssertionPredicateType::LastValue)); AssertionPredicateType::LastValue);
} }
OpaqueLoopInitNode* TemplateAssertionPredicateCreator::create_opaque_init(Node* new_control) { OpaqueLoopInitNode* TemplateAssertionPredicateCreator::create_opaque_init(Node* new_control) {
@ -656,11 +653,10 @@ TemplateAssertionPredicateCreator::create_for_init_value(Node* new_control, Opaq
IfTrueNode* TemplateAssertionPredicateCreator::create_if_node_with_uncommon_trap( IfTrueNode* TemplateAssertionPredicateCreator::create_if_node_with_uncommon_trap(
OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression, OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression,
ParsePredicateSuccessProj* parse_predicate_success_proj, const Deoptimization::DeoptReason deopt_reason, ParsePredicateSuccessProj* parse_predicate_success_proj, const Deoptimization::DeoptReason deopt_reason,
const int if_opcode, const bool does_overflow const int if_opcode, const bool does_overflow, const AssertionPredicateType assertion_predicate_type) {
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type)) {
IfTrueNode* success_proj = _phase->create_new_if_for_predicate(parse_predicate_success_proj, nullptr, deopt_reason, IfTrueNode* success_proj = _phase->create_new_if_for_predicate(parse_predicate_success_proj, nullptr, deopt_reason,
does_overflow ? Op_If : if_opcode, false does_overflow ? Op_If : if_opcode, false,
NOT_PRODUCT(COMMA assertion_predicate_type)); assertion_predicate_type);
_phase->igvn().replace_input_of(success_proj->in(0), 1, template_assertion_predicate_expression); _phase->igvn().replace_input_of(success_proj->in(0), 1, template_assertion_predicate_expression);
return success_proj; return success_proj;
} }
@ -688,12 +684,12 @@ Node* TemplateAssertionPredicateCreator::create_last_value(Node* new_control, Op
} }
IfTrueNode* TemplateAssertionPredicateCreator::create_if_node_with_halt( IfTrueNode* TemplateAssertionPredicateCreator::create_if_node_with_halt(
Node* new_control, OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression, bool does_overflow Node* new_control, OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression,
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type)) { const bool does_overflow, const AssertionPredicateType assertion_predicate_type) {
AssertionPredicateIfCreator assertion_predicate_if_creator(_phase); AssertionPredicateIfCreator assertion_predicate_if_creator(_phase);
return assertion_predicate_if_creator.create_for_template(new_control, does_overflow ? Op_If : Op_RangeCheck, return assertion_predicate_if_creator.create_for_template(new_control, does_overflow ? Op_If : Op_RangeCheck,
template_assertion_predicate_expression template_assertion_predicate_expression,
NOT_PRODUCT(COMMA assertion_predicate_type)); assertion_predicate_type);
} }
// Creates an init and last value Template Assertion Predicate connected together with a Halt node on the failing path. // Creates an init and last value Template Assertion Predicate connected together with a Halt node on the failing path.
@ -704,12 +700,12 @@ IfTrueNode* TemplateAssertionPredicateCreator::create_with_halt(Node* new_contro
OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression = OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression =
create_for_init_value(new_control, opaque_init, does_overflow); create_for_init_value(new_control, opaque_init, does_overflow);
IfTrueNode* template_predicate_success_proj = IfTrueNode* template_predicate_success_proj =
create_if_node_with_halt(new_control, template_assertion_predicate_expression, does_overflow create_if_node_with_halt(new_control, template_assertion_predicate_expression, does_overflow,
NOT_PRODUCT(COMMA AssertionPredicateType::InitValue)); AssertionPredicateType::InitValue);
template_assertion_predicate_expression = create_for_last_value(template_predicate_success_proj, opaque_init, template_assertion_predicate_expression = create_for_last_value(template_predicate_success_proj, opaque_init,
does_overflow); does_overflow);
return create_if_node_with_halt(template_predicate_success_proj, template_assertion_predicate_expression, return create_if_node_with_halt(template_predicate_success_proj, template_assertion_predicate_expression,
does_overflow NOT_PRODUCT(COMMA AssertionPredicateType::LastValue)); does_overflow, AssertionPredicateType::LastValue);
} }
InitializedAssertionPredicateCreator::InitializedAssertionPredicateCreator(PhaseIdealLoop* phase) InitializedAssertionPredicateCreator::InitializedAssertionPredicateCreator(PhaseIdealLoop* phase)
@ -740,8 +736,8 @@ IfTrueNode* InitializedAssertionPredicateCreator::create_from_template(IfNode* t
Node* new_stride) { Node* new_stride) {
OpaqueInitializedAssertionPredicateNode* assertion_expression = OpaqueInitializedAssertionPredicateNode* assertion_expression =
create_assertion_expression_from_template(template_assertion_predicate, new_control, new_init, new_stride); create_assertion_expression_from_template(template_assertion_predicate, new_control, new_init, new_stride);
return create_control_nodes(new_control, template_assertion_predicate->Opcode(), assertion_expression return create_control_nodes(new_control, template_assertion_predicate->Opcode(), assertion_expression,
NOT_PRODUCT(COMMA template_assertion_predicate->assertion_predicate_type())); template_assertion_predicate->assertion_predicate_type());
} }
// Create a new Initialized Assertion Predicate from 'template_assertion_predicate' by cloning it but omitting the // Create a new Initialized Assertion Predicate from 'template_assertion_predicate' by cloning it but omitting the
@ -753,29 +749,29 @@ IfTrueNode* InitializedAssertionPredicateCreator::create_from_template(IfNode* t
TemplateAssertionExpression template_assertion_expression(template_opaque); TemplateAssertionExpression template_assertion_expression(template_opaque);
OpaqueInitializedAssertionPredicateNode* assertion_expression = OpaqueInitializedAssertionPredicateNode* assertion_expression =
template_assertion_expression.clone_and_fold_opaque_loop_nodes(new_control, _phase); template_assertion_expression.clone_and_fold_opaque_loop_nodes(new_control, _phase);
return create_control_nodes(new_control, template_assertion_predicate->Opcode(), assertion_expression return create_control_nodes(new_control, template_assertion_predicate->Opcode(), assertion_expression,
NOT_PRODUCT(COMMA template_assertion_predicate->assertion_predicate_type())); template_assertion_predicate->assertion_predicate_type());
} }
// Create a new Initialized Assertion Predicate directly without a template. // Create a new Initialized Assertion Predicate directly without a template.
IfTrueNode* InitializedAssertionPredicateCreator::create(Node* operand, Node* new_control, const jint stride, IfTrueNode* InitializedAssertionPredicateCreator::create(Node* operand, Node* new_control, const jint stride,
const int scale, Node* offset, Node* range NOT_PRODUCT(COMMA const int scale, Node* offset, Node* range,
AssertionPredicateType assertion_predicate_type)) { const AssertionPredicateType assertion_predicate_type) {
AssertionPredicateExpressionCreator expression_creator(stride, scale, offset, range, _phase); AssertionPredicateExpressionCreator expression_creator(stride, scale, offset, range, _phase);
bool does_overflow; bool does_overflow;
OpaqueInitializedAssertionPredicateNode* assertion_expression = OpaqueInitializedAssertionPredicateNode* assertion_expression =
expression_creator.create_for_initialized(new_control, operand, does_overflow); expression_creator.create_for_initialized(new_control, operand, does_overflow);
return create_control_nodes(new_control, does_overflow ? Op_If : Op_RangeCheck, assertion_expression return create_control_nodes(new_control, does_overflow ? Op_If : Op_RangeCheck, assertion_expression,
NOT_PRODUCT(COMMA assertion_predicate_type)); assertion_predicate_type);
} }
// Creates the CFG nodes for the Initialized Assertion Predicate. // Creates the CFG nodes for the Initialized Assertion Predicate.
IfTrueNode* InitializedAssertionPredicateCreator::create_control_nodes( IfTrueNode* InitializedAssertionPredicateCreator::create_control_nodes(
Node* new_control, const int if_opcode, OpaqueInitializedAssertionPredicateNode* assertion_expression Node* new_control, const int if_opcode, OpaqueInitializedAssertionPredicateNode* assertion_expression,
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type)) { const AssertionPredicateType assertion_predicate_type) {
AssertionPredicateIfCreator assertion_predicate_if_creator(_phase); AssertionPredicateIfCreator assertion_predicate_if_creator(_phase);
return assertion_predicate_if_creator.create_for_initialized(new_control, if_opcode, assertion_expression return assertion_predicate_if_creator.create_for_initialized(new_control, if_opcode, assertion_expression,
NOT_PRODUCT(COMMA assertion_predicate_type)); assertion_predicate_type);
} }
// Create a new Assertion Expression based from the given template to be used as bool input for the Initialized // Create a new Assertion Expression based from the given template to be used as bool input for the Initialized
@ -879,6 +875,10 @@ IfTrueNode* CreateAssertionPredicatesVisitor::clone_template_and_replace_init_in
// Clone the Template Assertion Predicate and set a new input for the OpaqueLoopStrideNode. // Clone the Template Assertion Predicate and set a new input for the OpaqueLoopStrideNode.
void UpdateStrideForAssertionPredicates::visit(const TemplateAssertionPredicate& template_assertion_predicate) { void UpdateStrideForAssertionPredicates::visit(const TemplateAssertionPredicate& template_assertion_predicate) {
if (!template_assertion_predicate.is_last_value()) {
// Only Last Value Assertion Predicates have an OpaqueLoopStrideNode.
return;
}
replace_opaque_stride_input(template_assertion_predicate); replace_opaque_stride_input(template_assertion_predicate);
Node* template_tail_control_out = template_assertion_predicate.tail()->unique_ctrl_out(); Node* template_tail_control_out = template_assertion_predicate.tail()->unique_ctrl_out();
IfTrueNode* initialized_success_proj = initialize_from_updated_template(template_assertion_predicate); IfTrueNode* initialized_success_proj = initialize_from_updated_template(template_assertion_predicate);

View File

@ -201,7 +201,6 @@ class TemplateAssertionPredicate;
* Main Loop Head * Main Loop Head
*/ */
#ifndef PRODUCT
// Assertion Predicates are either emitted to check the initial value of a range check in the first iteration or the last // Assertion Predicates are either emitted to check the initial value of a range check in the first iteration or the last
// value of a range check in the last iteration of a loop. // value of a range check in the last iteration of a loop.
enum class AssertionPredicateType { enum class AssertionPredicateType {
@ -211,7 +210,6 @@ enum class AssertionPredicateType {
// Used for the Initialized Assertion Predicate emitted during Range Check Elimination for the final IV value. // Used for the Initialized Assertion Predicate emitted during Range Check Elimination for the final IV value.
FinalIv FinalIv
}; };
#endif // NOT PRODUCT
// Interface to represent a C2 predicate. A predicate is always represented by two CFG nodes: // Interface to represent a C2 predicate. A predicate is always represented by two CFG nodes:
// - An If node (head) // - An If node (head)
@ -377,8 +375,8 @@ class RuntimePredicate : public Predicate {
// Class to represent a Template Assertion Predicate. // Class to represent a Template Assertion Predicate.
class TemplateAssertionPredicate : public Predicate { class TemplateAssertionPredicate : public Predicate {
IfTrueNode* _success_proj; IfTrueNode* const _success_proj;
IfNode* _if_node; IfNode* const _if_node;
public: public:
explicit TemplateAssertionPredicate(IfTrueNode* success_proj) explicit TemplateAssertionPredicate(IfTrueNode* success_proj)
@ -403,6 +401,10 @@ class TemplateAssertionPredicate : public Predicate {
return _success_proj; return _success_proj;
} }
bool is_last_value() const {
return _if_node->assertion_predicate_type() == AssertionPredicateType::LastValue;
}
IfTrueNode* clone_and_replace_init(Node* new_control, OpaqueLoopInitNode* new_opaque_init, PhaseIdealLoop* phase) const; IfTrueNode* clone_and_replace_init(Node* new_control, OpaqueLoopInitNode* new_opaque_init, PhaseIdealLoop* phase) const;
void replace_opaque_stride_input(Node* new_stride, PhaseIterGVN& igvn) const; void replace_opaque_stride_input(Node* new_stride, PhaseIterGVN& igvn) const;
IfTrueNode* initialize(PhaseIdealLoop* phase, Node* new_control) const; IfTrueNode* initialize(PhaseIdealLoop* phase, Node* new_control) const;
@ -414,8 +416,8 @@ class TemplateAssertionPredicate : public Predicate {
// Class to represent an Initialized Assertion Predicate which always has a halt node on the failing path. // Class to represent an Initialized Assertion Predicate which always has a halt node on the failing path.
// This predicate should never fail at runtime by design. // This predicate should never fail at runtime by design.
class InitializedAssertionPredicate : public Predicate { class InitializedAssertionPredicate : public Predicate {
IfTrueNode* _success_proj; IfTrueNode* const _success_proj;
IfNode* _if_node; IfNode* const _if_node;
public: public:
explicit InitializedAssertionPredicate(IfTrueNode* success_proj) explicit InitializedAssertionPredicate(IfTrueNode* success_proj)
@ -436,6 +438,10 @@ class InitializedAssertionPredicate : public Predicate {
return _success_proj; return _success_proj;
} }
bool is_last_value() const {
return _if_node->assertion_predicate_type() == AssertionPredicateType::LastValue;
}
void kill(PhaseIdealLoop* phase) const; void kill(PhaseIdealLoop* phase) const;
static bool is_predicate(Node* node); static bool is_predicate(Node* node);
}; };
@ -550,15 +556,15 @@ class AssertionPredicateIfCreator : public StackObj {
explicit AssertionPredicateIfCreator(PhaseIdealLoop* const phase) : _phase(phase) {} explicit AssertionPredicateIfCreator(PhaseIdealLoop* const phase) : _phase(phase) {}
NONCOPYABLE(AssertionPredicateIfCreator); NONCOPYABLE(AssertionPredicateIfCreator);
IfTrueNode* create_for_initialized(Node* new_control, int if_opcode, Node* assertion_expression IfTrueNode* create_for_initialized(Node* new_control, int if_opcode, Node* assertion_expression,
NOT_PRODUCT(COMMA const AssertionPredicateType assertion_predicate_type)); AssertionPredicateType assertion_predicate_type);
IfTrueNode* create_for_template(Node* new_control, int if_opcode, Node* assertion_expression IfTrueNode* create_for_template(Node* new_control, int if_opcode, Node* assertion_expression,
NOT_PRODUCT(COMMA const AssertionPredicateType assertion_predicate_type)); AssertionPredicateType assertion_predicate_type);
private: private:
IfTrueNode* create(Node* new_control, int if_opcode, Node* assertion_expression, const char* halt_message IfTrueNode* create(Node* new_control, int if_opcode, Node* assertion_expression, const char* halt_message,
NOT_PRODUCT(COMMA const AssertionPredicateType assertion_predicate_type)); AssertionPredicateType assertion_predicate_type);
IfNode* create_if_node(Node* new_control, int if_opcode, Node* assertion_expression, IdealLoopTree* loop IfNode* create_if_node(Node* new_control, int if_opcode, Node* assertion_expression, IdealLoopTree* loop,
NOT_PRODUCT(COMMA const AssertionPredicateType assertion_predicate_type)); AssertionPredicateType assertion_predicate_type);
IfTrueNode* create_success_path(IfNode* if_node, IdealLoopTree* loop); IfTrueNode* create_success_path(IfNode* if_node, IdealLoopTree* loop);
void create_fail_path(IfNode* if_node, IdealLoopTree* loop, const char* halt_message); void create_fail_path(IfNode* if_node, IdealLoopTree* loop, const char* halt_message);
void create_halt_node(IfFalseNode* fail_proj, IdealLoopTree* loop, const char* halt_message); void create_halt_node(IfFalseNode* fail_proj, IdealLoopTree* loop, const char* halt_message);
@ -581,12 +587,10 @@ class TemplateAssertionPredicateCreator : public StackObj {
IfTrueNode* create_if_node_with_uncommon_trap(OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression, IfTrueNode* create_if_node_with_uncommon_trap(OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression,
ParsePredicateSuccessProj* parse_predicate_success_proj, ParsePredicateSuccessProj* parse_predicate_success_proj,
Deoptimization::DeoptReason deopt_reason, int if_opcode, Deoptimization::DeoptReason deopt_reason, int if_opcode,
bool does_overflow bool does_overflow, AssertionPredicateType assertion_predicate_type);
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type));
IfTrueNode* create_if_node_with_halt(Node* new_control, IfTrueNode* create_if_node_with_halt(Node* new_control,
OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression, OpaqueTemplateAssertionPredicateNode* template_assertion_predicate_expression,
bool does_overflow bool does_overflow, AssertionPredicateType assertion_predicate_type);
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type));
public: public:
TemplateAssertionPredicateCreator(CountedLoopNode* loop_head, int scale, Node* offset, Node* range, TemplateAssertionPredicateCreator(CountedLoopNode* loop_head, int scale, Node* offset, Node* range,
@ -614,16 +618,16 @@ class InitializedAssertionPredicateCreator : public StackObj {
IfTrueNode* create_from_template(IfNode* template_assertion_predicate, Node* new_control, Node* new_init, IfTrueNode* create_from_template(IfNode* template_assertion_predicate, Node* new_control, Node* new_init,
Node* new_stride); Node* new_stride);
IfTrueNode* create_from_template(IfNode* template_assertion_predicate, Node* new_control); IfTrueNode* create_from_template(IfNode* template_assertion_predicate, Node* new_control);
IfTrueNode* create(Node* operand, Node* new_control, jint stride, int scale, Node* offset, Node* range IfTrueNode* create(Node* operand, Node* new_control, jint stride, int scale, Node* offset, Node* range,
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type)); AssertionPredicateType assertion_predicate_type);
private: private:
OpaqueInitializedAssertionPredicateNode* create_assertion_expression_from_template(IfNode* template_assertion_predicate, OpaqueInitializedAssertionPredicateNode* create_assertion_expression_from_template(IfNode* template_assertion_predicate,
Node* new_control, Node* new_init, Node* new_control, Node* new_init,
Node* new_stride); Node* new_stride);
IfTrueNode* create_control_nodes(Node* new_control, int if_opcode, IfTrueNode* create_control_nodes(Node* new_control, int if_opcode,
OpaqueInitializedAssertionPredicateNode* assertion_expression OpaqueInitializedAssertionPredicateNode* assertion_expression,
NOT_PRODUCT(COMMA AssertionPredicateType assertion_predicate_type)); AssertionPredicateType assertion_predicate_type);
}; };
// This class iterates through all predicates of a Regular Predicate Block and applies the given visitor to each. // This class iterates through all predicates of a Regular Predicate Block and applies the given visitor to each.
@ -1063,7 +1067,10 @@ class UpdateStrideForAssertionPredicates : public PredicateVisitor {
// Predicates are inserted after the Template Assertion Predicate which ensures that we are not accidentally visiting // Predicates are inserted after the Template Assertion Predicate which ensures that we are not accidentally visiting
// and killing a newly created Initialized Assertion Predicate here. // and killing a newly created Initialized Assertion Predicate here.
void visit(const InitializedAssertionPredicate& initialized_assertion_predicate) override { void visit(const InitializedAssertionPredicate& initialized_assertion_predicate) override {
initialized_assertion_predicate.kill(_phase); if (initialized_assertion_predicate.is_last_value()) {
// Only Last Value Initialized Assertion Predicates need to be killed and updated.
initialized_assertion_predicate.kill(_phase);
}
} }
}; };
#endif // SHARE_OPTO_PREDICATES_HPP #endif // SHARE_OPTO_PREDICATES_HPP