diff --git a/.hgtags b/.hgtags index b256e510477..db42d6781c3 100644 --- a/.hgtags +++ b/.hgtags @@ -648,4 +648,5 @@ b58fc60580550a4a587cab729d8fd87223ad6932 jdk-15+29 e2622818f0bd30e736252eba101fe7d2c27f400b jdk-16+4 a32f58c6b8be81877411767de7ba9c4cf087c1b5 jdk-15+31 143e258f64af490010eb7e0bacc1cfaeceff0993 jdk-16+5 +2dad000726b8d5db9f3df647fb4949d88f269dd4 jdk-15+32 4a8fd81d64bafa523cddb45f82805536edace106 jdk-16+6 diff --git a/make/data/jdwp/jdwp.spec b/make/data/jdwp/jdwp.spec index 8a4f0bac95f..b55a286aaa1 100644 --- a/make/data/jdwp/jdwp.spec +++ b/make/data/jdwp/jdwp.spec @@ -530,7 +530,7 @@ JDWP "Java(tm) Debug Wire Protocol" "returned for each class. " "Generic signatures are described in the signature attribute " "section in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "Since JDWP version 1.5." (Out ) @@ -643,7 +643,7 @@ JDWP "Java(tm) Debug Wire Protocol" ) (Reply (int modBits "Modifier bits as defined in Chapter 4 of " - "The Java™ Virtual Machine Specification") + "The Java Virtual Machine Specification") ) (ErrorSet (Error INVALID_CLASS "refType is not the ID of a reference " @@ -671,7 +671,7 @@ JDWP "Java(tm) Debug Wire Protocol" "which provide additional information on the " "field declaration. Individual flag values are " "defined in Chapter 4 of " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "In addition, The 0xf0000000 bit identifies " "the field as synthetic, if the synthetic attribute " "capability is available.") @@ -706,7 +706,7 @@ JDWP "Java(tm) Debug Wire Protocol" "which provide additional information on the " "method declaration. Individual flag values are " "defined in Chapter 4 of " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "In addition, The 0xf0000000 bit identifies " "the method as synthetic, if the synthetic attribute " "capability is available.") @@ -793,7 +793,7 @@ JDWP "Java(tm) Debug Wire Protocol" "Returns the current status of the reference type. The status " "indicates the extent to which the reference type has been " "initialized, as described in section 2.1.6 of " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "If the class is linked the PREPARED and VERIFIED bits in the returned status bits " "will be set. If the class is initialized the INITIALIZED bit in the returned " "status bits will be set. If an error occured during initialization then the " @@ -872,7 +872,7 @@ JDWP "Java(tm) Debug Wire Protocol" "generic signature if there is one. " "Generic signatures are described in the signature attribute " "section in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "Since JDWP version 1.5." (Out (referenceType refType "The reference type ID.") @@ -900,7 +900,7 @@ JDWP "Java(tm) Debug Wire Protocol" "Fields are returned in the order they occur in the class file. " "Generic signatures are described in the signature attribute " "section in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "Since JDWP version 1.5." (Out (referenceType refType "The reference type ID.") @@ -917,7 +917,7 @@ JDWP "Java(tm) Debug Wire Protocol" "which provide additional information on the " "field declaration. Individual flag values are " "defined in Chapter 4 of " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "In addition, The 0xf0000000 bit identifies " "the field as synthetic, if the synthetic attribute " "capability is available.") @@ -942,7 +942,7 @@ JDWP "Java(tm) Debug Wire Protocol" "Methods are returned in the order they occur in the class file. " "Generic signatures are described in the signature attribute " "section in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "Since JDWP version 1.5." (Out (referenceType refType "The reference type ID.") @@ -959,7 +959,7 @@ JDWP "Java(tm) Debug Wire Protocol" "which provide additional information on the " "method declaration. Individual flag values are " "defined in Chapter 4 of " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "In addition, The 0xf0000000 bit identifies " "the method as synthetic, if the synthetic attribute " "capability is available.") @@ -1022,7 +1022,7 @@ JDWP "Java(tm) Debug Wire Protocol" (Command ConstantPool=18 "Return the raw bytes of the constant pool in the format of the " "constant_pool item of the Class File Format in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "

Since JDWP version 1.6. Requires canGetConstantPool capability - see " "CapabilitiesNew."" (Out @@ -1032,7 +1032,7 @@ JDWP "Java(tm) Debug Wire Protocol" (int count "Total number of constant pool entries plus one. This " "corresponds to the constant_pool_count item of the " "Class File Format in " - "The Java™ Virtual Machine Specification. ") + "The Java Virtual Machine Specification. ") (Repeat bytes (byte cpbytes "Raw bytes of constant pool") ) @@ -1435,7 +1435,7 @@ JDWP "Java(tm) Debug Wire Protocol" ) (Command Bytecodes=3 "Retrieve the method's bytecodes as defined in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "Requires canGetBytecodes capability - see " "CapabilitiesNew." (Out @@ -1491,7 +1491,7 @@ JDWP "Java(tm) Debug Wire Protocol" "table. Also, synthetic variables may be present. " "Generic signatures are described in the signature attribute " "section in " - "The Java™ Virtual Machine Specification. " + "The Java Virtual Machine Specification. " "Since JDWP version 1.5." (Out (referenceType refType "The class.") @@ -2082,7 +2082,7 @@ JDWP "Java(tm) Debug Wire Protocol" "The method which will return early is referred to as the " "called method. The called method is the current method (as " "defined by the Frames section in " - "The Java™ Virtual Machine Specification) " + "The Java Virtual Machine Specification) " "for the specified thread at the time this command " "is received. " "

" diff --git a/src/hotspot/share/jvmci/jvmciEnv.cpp b/src/hotspot/share/jvmci/jvmciEnv.cpp index b8deb73913d..bbcbaeefdde 100644 --- a/src/hotspot/share/jvmci/jvmciEnv.cpp +++ b/src/hotspot/share/jvmci/jvmciEnv.cpp @@ -568,26 +568,13 @@ const char* JVMCIEnv::as_utf8_string(JVMCIObject str) { } else { JNIAccessMark jni(this); int length = jni()->GetStringLength(str.as_jstring()); - char* result = NEW_RESOURCE_ARRAY(char, length + 1); + int utf8_length = jni()->GetStringUTFLength(str.as_jstring()); + char* result = NEW_RESOURCE_ARRAY(char, utf8_length + 1); jni()->GetStringUTFRegion(str.as_jstring(), 0, length, result); return result; } } -char* JVMCIEnv::as_utf8_string(JVMCIObject str, char* buf, int buflen) { - if (is_hotspot()) { - return java_lang_String::as_utf8_string(HotSpotJVMCI::resolve(str), buf, buflen); - } else { - JNIAccessMark jni(this); - int length = jni()->GetStringLength(str.as_jstring()); - if (length >= buflen) { - length = buflen; - } - jni()->GetStringUTFRegion(str.as_jstring(), 0, length, buf); - return buf; - } -} - #define DO_THROW(name) \ void JVMCIEnv::throw_##name(const char* msg) { \ if (is_hotspot()) { \ diff --git a/src/hotspot/share/jvmci/jvmciEnv.hpp b/src/hotspot/share/jvmci/jvmciEnv.hpp index a29e122f6bc..333525d5d69 100644 --- a/src/hotspot/share/jvmci/jvmciEnv.hpp +++ b/src/hotspot/share/jvmci/jvmciEnv.hpp @@ -255,7 +255,6 @@ public: JVMCIObject create_box(BasicType type, jvalue* value, JVMCI_TRAPS); const char* as_utf8_string(JVMCIObject str); - char* as_utf8_string(JVMCIObject str, char* buf, int buflen); JVMCIObject create_string(Symbol* str, JVMCI_TRAPS) { JVMCIObject s = create_string(str->as_C_string(), JVMCI_CHECK_(JVMCIObject())); diff --git a/src/hotspot/share/opto/classes.hpp b/src/hotspot/share/opto/classes.hpp index f0a12ef1dfd..e609851772f 100644 --- a/src/hotspot/share/opto/classes.hpp +++ b/src/hotspot/share/opto/classes.hpp @@ -234,6 +234,8 @@ macro(NegF) macro(NeverBranch) macro(OnSpinWait) macro(Opaque1) +macro(OpaqueLoopInit) +macro(OpaqueLoopStride) macro(Opaque2) macro(Opaque3) macro(Opaque4) diff --git a/src/hotspot/share/opto/compile.cpp b/src/hotspot/share/opto/compile.cpp index 9cd4b980fa8..7961cfbc3e6 100644 --- a/src/hotspot/share/opto/compile.cpp +++ b/src/hotspot/share/opto/compile.cpp @@ -1800,7 +1800,17 @@ void Compile::remove_opaque4_nodes(PhaseIterGVN &igvn) { for (int i = opaque4_count(); i > 0; i--) { Node* opaq = opaque4_node(i-1); assert(opaq->Opcode() == Op_Opaque4, "Opaque4 only"); + // With Opaque4 nodes, the expectation is that the test of input 1 + // is always equal to the constant value of input 2. So we can + // remove the Opaque4 and replace it by input 2. In debug builds, + // leave the non constant test in instead to sanity check that it + // never fails (if it does, that subgraph was constructed so, at + // runtime, a Halt node is executed). +#ifdef ASSERT + igvn.replace_node(opaq, opaq->in(1)); +#else igvn.replace_node(opaq, opaq->in(2)); +#endif } assert(opaque4_count() == 0, "should be empty"); } diff --git a/src/hotspot/share/opto/loopPredicate.cpp b/src/hotspot/share/opto/loopPredicate.cpp index c15c278231c..91aad8cad84 100644 --- a/src/hotspot/share/opto/loopPredicate.cpp +++ b/src/hotspot/share/opto/loopPredicate.cpp @@ -1255,8 +1255,9 @@ ProjNode* PhaseIdealLoop::insert_initial_skeleton_predicate(IfNode* iff, IdealLo Node* init, Node* limit, jint stride, Node* rng, bool &overflow, Deoptimization::DeoptReason reason) { + // First predicate for the initial value on first loop iteration assert(proj->_con && predicate_proj->_con, "not a range check?"); - Node* opaque_init = new Opaque1Node(C, init); + Node* opaque_init = new OpaqueLoopInitNode(C, init); register_new_node(opaque_init, upper_bound_proj); BoolNode* bol = rc_predicate(loop, upper_bound_proj, scale, offset, opaque_init, limit, stride, rng, (stride > 0) != (scale > 0), overflow); Node* opaque_bol = new Opaque4Node(C, bol, _igvn.intcon(1)); // This will go away once loop opts are over @@ -1264,6 +1265,24 @@ ProjNode* PhaseIdealLoop::insert_initial_skeleton_predicate(IfNode* iff, IdealLo ProjNode* new_proj = create_new_if_for_predicate(predicate_proj, NULL, reason, overflow ? Op_If : iff->Opcode()); _igvn.replace_input_of(new_proj->in(0), 1, opaque_bol); assert(opaque_init->outcnt() > 0, "should be used"); + + // Second predicate for init + (current stride - initial stride) + // This is identical to the previous predicate initially but as + // unrolling proceeds current stride is updated. + Node* init_stride = loop->_head->as_CountedLoop()->stride(); + Node* opaque_stride = new OpaqueLoopStrideNode(C, init_stride); + register_new_node(opaque_stride, new_proj); + Node* max_value = new SubINode(opaque_stride, init_stride); + register_new_node(max_value, new_proj); + max_value = new AddINode(opaque_init, max_value); + register_new_node(max_value, new_proj); + bol = rc_predicate(loop, new_proj, scale, offset, max_value, limit, stride, rng, (stride > 0) != (scale > 0), overflow); + opaque_bol = new Opaque4Node(C, bol, _igvn.intcon(1)); + register_new_node(opaque_bol, new_proj); + new_proj = create_new_if_for_predicate(predicate_proj, NULL, reason, overflow ? Op_If : iff->Opcode()); + _igvn.replace_input_of(new_proj->in(0), 1, opaque_bol); + assert(max_value->outcnt() > 0, "should be used"); + return new_proj; } diff --git a/src/hotspot/share/opto/loopTransform.cpp b/src/hotspot/share/opto/loopTransform.cpp index 19af0c9e8b5..133d47a06e6 100644 --- a/src/hotspot/share/opto/loopTransform.cpp +++ b/src/hotspot/share/opto/loopTransform.cpp @@ -1141,7 +1141,7 @@ void PhaseIdealLoop::ensure_zero_trip_guard_proj(Node* node, bool is_main_loop) // CastII/ConvI2L nodes cause some data paths to die. For consistency, // the control paths must die too but the range checks were removed by // predication. The range checks that we add here guarantee that they do. -void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop_helper(Node* predicate, Node* start, Node* end, +void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop_helper(Node* predicate, Node* init, Node* stride, IdealLoopTree* outer_loop, LoopNode* outer_main_head, uint dd_main_head, const uint idx_before_pre_post, const uint idx_after_post_before_pre, Node* zero_trip_guard_proj_main, @@ -1159,6 +1159,11 @@ void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop_helper(Node* predicat predicate = iff->in(0); Node* current_proj = outer_main_head->in(LoopNode::EntryControl); Node* prev_proj = current_proj; + Node* opaque_init = new OpaqueLoopInitNode(C, init); + register_new_node(opaque_init, outer_main_head->in(LoopNode::EntryControl)); + Node* opaque_stride = new OpaqueLoopStrideNode(C, stride); + register_new_node(opaque_stride, outer_main_head->in(LoopNode::EntryControl)); + while (predicate != NULL && predicate->is_Proj() && predicate->in(0)->is_If()) { iff = predicate->in(0)->as_If(); uncommon_proj = iff->proj_out(1 - predicate->as_Proj()->_con); @@ -1169,10 +1174,11 @@ void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop_helper(Node* predicat // Clone the skeleton predicate twice and initialize one with the initial // value of the loop induction variable. Leave the other predicate // to be initialized when increasing the stride during loop unrolling. - prev_proj = clone_skeleton_predicate(iff, start, predicate, uncommon_proj, current_proj, outer_loop, prev_proj); - assert(skeleton_predicate_has_opaque(prev_proj->in(0)->as_If()) == (start->Opcode() == Op_Opaque1), ""); - prev_proj = clone_skeleton_predicate(iff, end, predicate, uncommon_proj, current_proj, outer_loop, prev_proj); - assert(skeleton_predicate_has_opaque(prev_proj->in(0)->as_If()) == (end->Opcode() == Op_Opaque1), ""); + prev_proj = clone_skeleton_predicate(iff, opaque_init, NULL, predicate, uncommon_proj, current_proj, outer_loop, prev_proj); + assert(skeleton_predicate_has_opaque(prev_proj->in(0)->as_If()), ""); + + prev_proj = clone_skeleton_predicate(iff, init, stride, predicate, uncommon_proj, current_proj, outer_loop, prev_proj); + assert(!skeleton_predicate_has_opaque(prev_proj->in(0)->as_If()), ""); // Rewire any control inputs from the cloned skeleton predicates down to the main and post loop for data nodes that are part of the // main loop (and were cloned to the pre and post loop). @@ -1238,14 +1244,14 @@ bool PhaseIdealLoop::skeleton_predicate_has_opaque(IfNode* iff) { } continue; } - if (op == Op_Opaque1) { + if (n->is_Opaque1()) { return true; } } return false; } -Node* PhaseIdealLoop::clone_skeleton_predicate(Node* iff, Node* value, Node* predicate, Node* uncommon_proj, +Node* PhaseIdealLoop::clone_skeleton_predicate(Node* iff, Node* new_init, Node* new_stride, Node* predicate, Node* uncommon_proj, Node* current_proj, IdealLoopTree* outer_loop, Node* prev_proj) { Node_Stack to_clone(2); to_clone.push(iff->in(1), 1); @@ -1265,12 +1271,19 @@ Node* PhaseIdealLoop::clone_skeleton_predicate(Node* iff, Node* value, Node* pre to_clone.push(m, 1); continue; } - if (op == Op_Opaque1) { + if (m->is_Opaque1()) { if (n->_idx < current) { n = n->clone(); + register_new_node(n, current_proj); + } + if (op == Op_OpaqueLoopInit) { + n->set_req(i, new_init); + } else { + assert(op == Op_OpaqueLoopStride, "unexpected opaque node"); + if (new_stride != NULL) { + n->set_req(i, new_stride); + } } - n->set_req(i, value); - register_new_node(n, current_proj); to_clone.set_node(n); } for (;;) { @@ -1320,7 +1333,7 @@ Node* PhaseIdealLoop::clone_skeleton_predicate(Node* iff, Node* value, Node* pre return proj; } -void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop(CountedLoopNode* pre_head, Node* start, Node* end, +void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop(CountedLoopNode* pre_head, Node* init, Node* stride, IdealLoopTree* outer_loop, LoopNode* outer_main_head, uint dd_main_head, const uint idx_before_pre_post, const uint idx_after_post_before_pre, Node* zero_trip_guard_proj_main, @@ -1340,10 +1353,10 @@ void PhaseIdealLoop::copy_skeleton_predicates_to_main_loop(CountedLoopNode* pre_ } } predicate = find_predicate_insertion_point(entry, Deoptimization::Reason_predicate); - copy_skeleton_predicates_to_main_loop_helper(predicate, start, end, outer_loop, outer_main_head, dd_main_head, + copy_skeleton_predicates_to_main_loop_helper(predicate, init, stride, outer_loop, outer_main_head, dd_main_head, idx_before_pre_post, idx_after_post_before_pre, zero_trip_guard_proj_main, zero_trip_guard_proj_post, old_new); - copy_skeleton_predicates_to_main_loop_helper(profile_predicate, start, end, outer_loop, outer_main_head, dd_main_head, + copy_skeleton_predicates_to_main_loop_helper(profile_predicate, init, stride, outer_loop, outer_main_head, dd_main_head, idx_before_pre_post, idx_after_post_before_pre, zero_trip_guard_proj_main, zero_trip_guard_proj_post, old_new); } @@ -1493,10 +1506,8 @@ void PhaseIdealLoop::insert_pre_post_loops(IdealLoopTree *loop, Node_List &old_n // CastII for the main loop: Node* castii = cast_incr_before_loop(pre_incr, min_taken, main_head); assert(castii != NULL, "no castII inserted"); - Node* opaque_castii = new Opaque1Node(C, castii); - register_new_node(opaque_castii, outer_main_head->in(LoopNode::EntryControl)); assert(post_head->in(1)->is_IfProj(), "must be zero-trip guard If node projection of the post loop"); - copy_skeleton_predicates_to_main_loop(pre_head, castii, opaque_castii, outer_loop, outer_main_head, dd_main_head, + copy_skeleton_predicates_to_main_loop(pre_head, castii, stride, outer_loop, outer_main_head, dd_main_head, idx_before_pre_post, idx_after_post_before_pre, min_taken, post_head->in(1), old_new); // Step B4: Shorten the pre-loop to run only 1 iteration (for now). @@ -1788,6 +1799,13 @@ void PhaseIdealLoop::update_main_loop_skeleton_predicates(Node* ctrl, CountedLoo Node* prev_proj = ctrl; LoopNode* outer_loop_head = loop_head->skip_strip_mined(); IdealLoopTree* outer_loop = get_loop(outer_loop_head); + + // Compute the value of the loop induction variable at the end of the + // first iteration of the unrolled loop: init + new_stride_con - init_inc + int new_stride_con = stride_con * 2; + Node* max_value = _igvn.intcon(new_stride_con); + set_ctrl(max_value, C->root()); + while (entry != NULL && entry->is_Proj() && entry->in(0)->is_If()) { IfNode* iff = entry->in(0)->as_If(); ProjNode* proj = iff->proj_out(1 - entry->as_Proj()->_con); @@ -1803,18 +1821,8 @@ void PhaseIdealLoop::update_main_loop_skeleton_predicates(Node* ctrl, CountedLoo // tell. Kill it in any case. _igvn.replace_input_of(iff, 1, iff->in(1)->in(2)); } else { - // Add back the predicate for the value at the beginning of the first entry - prev_proj = clone_skeleton_predicate(iff, init, entry, proj, ctrl, outer_loop, prev_proj); - assert(!skeleton_predicate_has_opaque(prev_proj->in(0)->as_If()), "unexpected"); - // Compute the value of the loop induction variable at the end of the - // first iteration of the unrolled loop: init + new_stride_con - init_inc - int init_inc = stride_con/loop_head->unrolled_count(); - assert(init_inc != 0, "invalid loop increment"); - int new_stride_con = stride_con * 2; - Node* max_value = _igvn.intcon(new_stride_con - init_inc); - max_value = new AddINode(init, max_value); - register_new_node(max_value, get_ctrl(iff->in(1))); - prev_proj = clone_skeleton_predicate(iff, max_value, entry, proj, ctrl, outer_loop, prev_proj); + // Add back predicates updated for the new stride. + prev_proj = clone_skeleton_predicate(iff, init, max_value, entry, proj, ctrl, outer_loop, prev_proj); assert(!skeleton_predicate_has_opaque(prev_proj->in(0)->as_If()), "unexpected"); } } @@ -2667,22 +2675,26 @@ int PhaseIdealLoop::do_range_check(IdealLoopTree *loop, Node_List &old_new) { // (0-offset)/scale could be outside of loop iterations range. conditional_rc = true; Node* init = cl->init_trip(); - Node* opaque_init = new Opaque1Node(C, init); + Node* opaque_init = new OpaqueLoopInitNode(C, init); register_new_node(opaque_init, predicate_proj); - // template predicate so it can be updated on next unrolling - predicate_proj = add_range_check_predicate(loop, cl, predicate_proj, scale_con, offset, limit, stride_con, opaque_init); - assert(skeleton_predicate_has_opaque(predicate_proj->in(0)->as_If()), "unexpected"); + // predicate on first value of first iteration predicate_proj = add_range_check_predicate(loop, cl, predicate_proj, scale_con, offset, limit, stride_con, init); assert(!skeleton_predicate_has_opaque(predicate_proj->in(0)->as_If()), "unexpected"); - int init_inc = stride_con/cl->unrolled_count(); - assert(init_inc != 0, "invalid loop increment"); - Node* max_value = _igvn.intcon(stride_con - init_inc); - max_value = new AddINode(init, max_value); + + // template predicate so it can be updated on next unrolling + predicate_proj = add_range_check_predicate(loop, cl, predicate_proj, scale_con, offset, limit, stride_con, opaque_init); + assert(skeleton_predicate_has_opaque(predicate_proj->in(0)->as_If()), "unexpected"); + + Node* opaque_stride = new OpaqueLoopStrideNode(C, cl->stride()); + register_new_node(opaque_stride, predicate_proj); + Node* max_value = new SubINode(opaque_stride, cl->stride()); + register_new_node(max_value, predicate_proj); + max_value = new AddINode(opaque_init, max_value); register_new_node(max_value, predicate_proj); - // predicate on last value of first iteration (in case unrolling has already happened) predicate_proj = add_range_check_predicate(loop, cl, predicate_proj, scale_con, offset, limit, stride_con, max_value); - assert(!skeleton_predicate_has_opaque(predicate_proj->in(0)->as_If()), "unexpected"); + assert(skeleton_predicate_has_opaque(predicate_proj->in(0)->as_If()), "unexpected"); + } else { if (PrintOpto) { tty->print_cr("missed RCE opportunity"); diff --git a/src/hotspot/share/opto/loopnode.cpp b/src/hotspot/share/opto/loopnode.cpp index 2ae128d2c1b..920d1bb1919 100644 --- a/src/hotspot/share/opto/loopnode.cpp +++ b/src/hotspot/share/opto/loopnode.cpp @@ -2580,7 +2580,7 @@ uint IdealLoopTree::est_loop_clone_sz(uint factor) const { uint const bc = 13; uint const cc = 17; - uint const sz = _body.size() + (_body.size() + 7) / 8; + uint const sz = _body.size() + (_body.size() + 7) / 2; uint estimate = factor * (sz + bc) + cc; assert((estimate - cc) / factor == sz + bc, "overflow"); diff --git a/src/hotspot/share/opto/loopnode.hpp b/src/hotspot/share/opto/loopnode.hpp index 97a97ff8290..2fbeb04d5f9 100644 --- a/src/hotspot/share/opto/loopnode.hpp +++ b/src/hotspot/share/opto/loopnode.hpp @@ -788,13 +788,13 @@ private: #ifdef ASSERT void ensure_zero_trip_guard_proj(Node* node, bool is_main_loop); #endif - void copy_skeleton_predicates_to_main_loop_helper(Node* predicate, Node* start, Node* end, IdealLoopTree* outer_loop, LoopNode* outer_main_head, + void copy_skeleton_predicates_to_main_loop_helper(Node* predicate, Node* init, Node* stride, IdealLoopTree* outer_loop, LoopNode* outer_main_head, uint dd_main_head, const uint idx_before_pre_post, const uint idx_after_post_before_pre, Node* zero_trip_guard_proj_main, Node* zero_trip_guard_proj_post, const Node_List &old_new); - void copy_skeleton_predicates_to_main_loop(CountedLoopNode* pre_head, Node* start, Node* end, IdealLoopTree* outer_loop, LoopNode* outer_main_head, + void copy_skeleton_predicates_to_main_loop(CountedLoopNode* pre_head, Node* init, Node* stride, IdealLoopTree* outer_loop, LoopNode* outer_main_head, uint dd_main_head, const uint idx_before_pre_post, const uint idx_after_post_before_pre, Node* zero_trip_guard_proj_main, Node* zero_trip_guard_proj_post, const Node_List &old_new); - Node* clone_skeleton_predicate(Node* iff, Node* value, Node* predicate, Node* uncommon_proj, + Node* clone_skeleton_predicate(Node* iff, Node* new_init, Node* new_stride, Node* predicate, Node* uncommon_proj, Node* current_proj, IdealLoopTree* outer_loop, Node* prev_proj); bool skeleton_predicate_has_opaque(IfNode* iff); void update_main_loop_skeleton_predicates(Node* ctrl, CountedLoopNode* loop_head, Node* init, int stride_con); diff --git a/src/hotspot/share/opto/loopopts.cpp b/src/hotspot/share/opto/loopopts.cpp index 35f97826e42..e519305eeea 100644 --- a/src/hotspot/share/opto/loopopts.cpp +++ b/src/hotspot/share/opto/loopopts.cpp @@ -952,29 +952,43 @@ void PhaseIdealLoop::try_move_store_after_loop(Node* n) { Node *PhaseIdealLoop::split_if_with_blocks_pre( Node *n ) { // Cloning these guys is unlikely to win int n_op = n->Opcode(); - if( n_op == Op_MergeMem ) return n; - if( n->is_Proj() ) return n; + if (n_op == Op_MergeMem) { + return n; + } + if (n->is_Proj()) { + return n; + } // Do not clone-up CmpFXXX variations, as these are always // followed by a CmpI - if( n->is_Cmp() ) return n; - // Attempt to use a conditional move instead of a phi/branch - if( ConditionalMoveLimit > 0 && n_op == Op_Region ) { - Node *cmov = conditional_move( n ); - if( cmov ) return cmov; - } - if( n->is_CFG() || n->is_LoadStore() ) + if (n->is_Cmp()) { return n; - if( n_op == Op_Opaque1 || // Opaque nodes cannot be mod'd - n_op == Op_Opaque2 ) { - if( !C->major_progress() ) // If chance of no more loop opts... + } + // Attempt to use a conditional move instead of a phi/branch + if (ConditionalMoveLimit > 0 && n_op == Op_Region) { + Node *cmov = conditional_move( n ); + if (cmov) { + return cmov; + } + } + if (n->is_CFG() || n->is_LoadStore()) { + return n; + } + if (n->is_Opaque1() || // Opaque nodes cannot be mod'd + n_op == Op_Opaque2) { + if (!C->major_progress()) { // If chance of no more loop opts... _igvn._worklist.push(n); // maybe we'll remove them + } return n; } - if( n->is_Con() ) return n; // No cloning for Con nodes + if (n->is_Con()) { + return n; // No cloning for Con nodes + } Node *n_ctrl = get_ctrl(n); - if( !n_ctrl ) return n; // Dead node + if (!n_ctrl) { + return n; // Dead node + } Node* res = try_move_store_before_loop(n, n_ctrl); if (res != NULL) { diff --git a/src/hotspot/share/opto/macro.cpp b/src/hotspot/share/opto/macro.cpp index 19b75b81d62..b82a36ea891 100644 --- a/src/hotspot/share/opto/macro.cpp +++ b/src/hotspot/share/opto/macro.cpp @@ -2636,9 +2636,10 @@ void PhaseMacroExpand::eliminate_macro_nodes() { break; case Node::Class_SubTypeCheck: break; + case Node::Class_Opaque1: + break; default: assert(n->Opcode() == Op_LoopLimit || - n->Opcode() == Op_Opaque1 || n->Opcode() == Op_Opaque2 || n->Opcode() == Op_Opaque3 || BarrierSet::barrier_set()->barrier_set_c2()->is_gc_barrier_node(n), @@ -2674,7 +2675,7 @@ bool PhaseMacroExpand::expand_macro_nodes() { C->remove_macro_node(n); _igvn._worklist.push(n); success = true; - } else if (n->Opcode() == Op_Opaque1 || n->Opcode() == Op_Opaque2) { + } else if (n->is_Opaque1() || n->Opcode() == Op_Opaque2) { _igvn.replace_node(n, n->in(1)); success = true; #if INCLUDE_RTM_OPT diff --git a/src/hotspot/share/opto/node.hpp b/src/hotspot/share/opto/node.hpp index f7a2e0d4d7f..6c40abc1576 100644 --- a/src/hotspot/share/opto/node.hpp +++ b/src/hotspot/share/opto/node.hpp @@ -116,6 +116,7 @@ class MulNode; class MultiNode; class MultiBranchNode; class NeverBranchNode; +class Opaque1Node; class OuterStripMinedLoopNode; class OuterStripMinedLoopEndNode; class Node; @@ -609,9 +610,9 @@ public: // This enum is used only for C2 ideal and mach nodes with is_() methods // so that it's values fits into 16 bits. enum NodeClasses { - Bit_Node = 0x0000, - Class_Node = 0x0000, - ClassMask_Node = 0xFFFF, + Bit_Node = 0x00000000, + Class_Node = 0x00000000, + ClassMask_Node = 0xFFFFFFFF, DEFINE_CLASS_ID(Multi, Node, 0) DEFINE_CLASS_ID(SafePoint, Multi, 0) @@ -716,6 +717,7 @@ public: DEFINE_CLASS_ID(Vector, Node, 13) DEFINE_CLASS_ID(ClearArray, Node, 14) DEFINE_CLASS_ID(Halt, Node, 15) + DEFINE_CLASS_ID(Opaque1, Node, 16) _max_classes = ClassMask_Halt }; @@ -744,14 +746,14 @@ public: class PD; private: - jushort _class_id; + juint _class_id; jushort _flags; static juint max_flags(); protected: // These methods should be called from constructors only. - void init_class_id(jushort c) { + void init_class_id(juint c) { _class_id = c; // cast out const } void init_flags(uint fl) { @@ -764,7 +766,7 @@ protected: } public: - const jushort class_id() const { return _class_id; } + const juint class_id() const { return _class_id; } const jushort flags() const { return _flags; } @@ -865,6 +867,7 @@ public: DEFINE_CLASS_QUERY(Mul) DEFINE_CLASS_QUERY(Multi) DEFINE_CLASS_QUERY(MultiBranch) + DEFINE_CLASS_QUERY(Opaque1) DEFINE_CLASS_QUERY(OuterStripMinedLoop) DEFINE_CLASS_QUERY(OuterStripMinedLoopEnd) DEFINE_CLASS_QUERY(Parm) diff --git a/src/hotspot/share/opto/opaquenode.hpp b/src/hotspot/share/opto/opaquenode.hpp index 18cbbb08c64..c6d27948748 100644 --- a/src/hotspot/share/opto/opaquenode.hpp +++ b/src/hotspot/share/opto/opaquenode.hpp @@ -38,6 +38,7 @@ class Opaque1Node : public Node { Opaque1Node(Compile* C, Node *n) : Node(NULL, n) { // Put it on the Macro nodes list to removed during macro nodes expansion. init_flags(Flag_is_macro); + init_class_id(Class_Opaque1); C->add_macro_node(this); } // Special version for the pre-loop to hold the original loop limit @@ -45,6 +46,7 @@ class Opaque1Node : public Node { Opaque1Node(Compile* C, Node *n, Node* orig_limit) : Node(NULL, n, orig_limit) { // Put it on the Macro nodes list to removed during macro nodes expansion. init_flags(Flag_is_macro); + init_class_id(Class_Opaque1); C->add_macro_node(this); } Node* original_loop_limit() { return req()==3 ? in(2) : NULL; } @@ -53,6 +55,21 @@ class Opaque1Node : public Node { virtual Node* Identity(PhaseGVN* phase); }; +// Opaque nodes specific to range check elimination handling +class OpaqueLoopInitNode : public Opaque1Node { + public: + OpaqueLoopInitNode(Compile* C, Node *n) : Opaque1Node(C, n) { + } + virtual int Opcode() const; +}; + +class OpaqueLoopStrideNode : public Opaque1Node { + public: + OpaqueLoopStrideNode(Compile* C, Node *n) : Opaque1Node(C, n) { + } + virtual int Opcode() const; +}; + //------------------------------Opaque2Node------------------------------------ // A node to prevent unwanted optimizations. Allows constant folding. Stops // value-numbering, most Ideal calls or Identity functions. This Node is diff --git a/src/hotspot/share/runtime/vmStructs.cpp b/src/hotspot/share/runtime/vmStructs.cpp index dc3c6936d69..1fb9029606a 100644 --- a/src/hotspot/share/runtime/vmStructs.cpp +++ b/src/hotspot/share/runtime/vmStructs.cpp @@ -918,7 +918,7 @@ typedef HashtableEntry KlassHashtableEntry; c2_nonstatic_field(Node, _outcnt, node_idx_t) \ c2_nonstatic_field(Node, _outmax, node_idx_t) \ c2_nonstatic_field(Node, _idx, const node_idx_t) \ - c2_nonstatic_field(Node, _class_id, jushort) \ + c2_nonstatic_field(Node, _class_id, juint) \ c2_nonstatic_field(Node, _flags, jushort) \ \ c2_nonstatic_field(Compile, _root, RootNode*) \ diff --git a/src/java.base/share/classes/java/io/Console.java b/src/java.base/share/classes/java/io/Console.java index 9f03f5358b9..b82673eeb0a 100644 --- a/src/java.base/share/classes/java/io/Console.java +++ b/src/java.base/share/classes/java/io/Console.java @@ -152,7 +152,7 @@ public final class Console implements Flushable * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -192,7 +192,7 @@ public final class Console implements Flushable * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -225,7 +225,7 @@ public final class Console implements Flushable * string. If there are more arguments than format specifiers, the * extra arguments are ignored. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * * @throws IllegalFormatException * If a format string contains an illegal syntax, a format @@ -289,7 +289,7 @@ public final class Console implements Flushable * string. If there are more arguments than format specifiers, the * extra arguments are ignored. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * * @throws IllegalFormatException * If a format string contains an illegal syntax, a format diff --git a/src/java.base/share/classes/java/io/PrintStream.java b/src/java.base/share/classes/java/io/PrintStream.java index a8566a0b873..b34f6c54e12 100644 --- a/src/java.base/share/classes/java/io/PrintStream.java +++ b/src/java.base/share/classes/java/io/PrintStream.java @@ -1080,7 +1080,7 @@ public class PrintStream extends FilterOutputStream * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -1132,7 +1132,7 @@ public class PrintStream extends FilterOutputStream * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -1177,7 +1177,7 @@ public class PrintStream extends FilterOutputStream * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -1236,7 +1236,7 @@ public class PrintStream extends FilterOutputStream * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. diff --git a/src/java.base/share/classes/java/io/PrintWriter.java b/src/java.base/share/classes/java/io/PrintWriter.java index 07dbd34a967..a338af28e8e 100644 --- a/src/java.base/share/classes/java/io/PrintWriter.java +++ b/src/java.base/share/classes/java/io/PrintWriter.java @@ -863,7 +863,7 @@ public class PrintWriter extends Writer { * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -916,7 +916,7 @@ public class PrintWriter extends Writer { * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -960,7 +960,7 @@ public class PrintWriter extends Writer { * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. @@ -1020,7 +1020,7 @@ public class PrintWriter extends Writer { * extra arguments are ignored. The number of arguments is * variable and may be zero. The maximum number of arguments is * limited by the maximum dimension of a Java array as defined by - * The Java™ Virtual Machine Specification. + * The Java Virtual Machine Specification. * The behaviour on a * {@code null} argument depends on the conversion. diff --git a/src/java.base/share/classes/java/lang/AssertionError.java b/src/java.base/share/classes/java/lang/AssertionError.java index 3643e42e81f..704e774468a 100644 --- a/src/java.base/share/classes/java/lang/AssertionError.java +++ b/src/java.base/share/classes/java/lang/AssertionError.java @@ -35,7 +35,7 @@ package java.lang; * * has as its detail message the string conversion of * expression (as defined in section 15.18.1.1 of - * The Java™ Language Specification), + * The Java Language Specification), * regardless of the type of expression. * * @since 1.4 @@ -63,7 +63,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified object, which is converted to a string as * defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. *

* If the specified object is an instance of {@code Throwable}, it * becomes the cause of the newly constructed assertion error. @@ -81,7 +81,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified {@code boolean}, which is converted to * a string as defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. * * @param detailMessage value to be used in constructing detail message */ @@ -93,7 +93,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified {@code char}, which is converted to a * string as defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. * * @param detailMessage value to be used in constructing detail message */ @@ -105,7 +105,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified {@code int}, which is converted to a * string as defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. * * @param detailMessage value to be used in constructing detail message */ @@ -117,7 +117,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified {@code long}, which is converted to a * string as defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. * * @param detailMessage value to be used in constructing detail message */ @@ -129,7 +129,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified {@code float}, which is converted to a * string as defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. * * @param detailMessage value to be used in constructing detail message */ @@ -141,7 +141,7 @@ public class AssertionError extends Error { * Constructs an AssertionError with its detail message derived * from the specified {@code double}, which is converted to a * string as defined in section 15.18.1.1 of - * The Java™ Language Specification. + * The Java Language Specification. * * @param detailMessage value to be used in constructing detail message */ diff --git a/src/java.base/share/classes/java/lang/Byte.java b/src/java.base/share/classes/java/lang/Byte.java index c43b319c9e7..98c2ef422d4 100644 --- a/src/java.base/share/classes/java/lang/Byte.java +++ b/src/java.base/share/classes/java/lang/Byte.java @@ -290,7 +290,7 @@ public final class Byte extends Number implements Comparable, Constable { * * DecimalNumeral, HexDigits, and OctalDigits * are as defined in section 3.10.1 of - * The Java™ Language Specification, + * The Java Language Specification, * except that underscores are not accepted between digits. * *

The sequence of characters following an optional diff --git a/src/java.base/share/classes/java/lang/Class.java b/src/java.base/share/classes/java/lang/Class.java index 05d2415ed8a..1fabad43738 100644 --- a/src/java.base/share/classes/java/lang/Class.java +++ b/src/java.base/share/classes/java/lang/Class.java @@ -413,7 +413,7 @@ public final class Class implements java.io.Serializable, * * Note that this method throws errors related to loading, linking * or initializing as specified in Sections {@jls 12.2}, {@jls - * 12.3}, and {@jls 12.4} of The Java™ Language + * 12.3}, and {@jls 12.4} of The Java Language * Specification. * Note that this method does not check whether the requested class * is accessible to its caller. @@ -422,7 +422,7 @@ public final class Class implements java.io.Serializable, * @param initialize if {@code true} the class will be initialized * (which implies linking). See Section {@jls - * 12.4} of The Java™ Language + * 12.4} of The Java Language * Specification. * @param loader class loader from which the class must be loaded * @return class object representing the desired class @@ -700,7 +700,7 @@ public final class Class implements java.io.Serializable, *

Specifically, this method tests whether the type represented by the * specified {@code Class} parameter can be converted to the type * represented by this {@code Class} object via an identity conversion - * or via a widening reference conversion. See The Java™ Language + * or via a widening reference conversion. See The Java Language * Specification, sections {@jls 5.1.1} and {@jls 5.1.4}, * for details. * @@ -952,7 +952,7 @@ public final class Class implements java.io.Serializable, * @throws java.lang.reflect.GenericSignatureFormatError if the generic * signature of this generic declaration does not conform to * the format specified in section {@jvms 4.7.9} of - * The Java™ Virtual Machine Specification + * The Java Virtual Machine Specification * @since 1.5 */ @SuppressWarnings("unchecked") @@ -1000,7 +1000,7 @@ public final class Class implements java.io.Serializable, * * @throws java.lang.reflect.GenericSignatureFormatError if the generic * class signature does not conform to the format specified in - * section {@jvms 4.7.9} of The Java™ Virtual + * section {@jvms 4.7.9} of The Java Virtual * Machine Specification * @throws TypeNotPresentException if the generic superclass * refers to a non-existent type declaration @@ -1197,7 +1197,7 @@ public final class Class implements java.io.Serializable, * @throws java.lang.reflect.GenericSignatureFormatError * if the generic class signature does not conform to the * format specified in section {@jvms 4.7.9} of The - * Java™ Virtual Machine Specification + * Java Virtual Machine Specification * @throws TypeNotPresentException if any of the generic * superinterfaces refers to a non-existent type declaration * @throws java.lang.reflect.MalformedParameterizedTypeException @@ -1269,7 +1269,7 @@ public final class Class implements java.io.Serializable, * by this specification. * *

The modifier encodings are defined in section {@jvms 4.1} - * of The Java™ Virtual Machine Specification. + * of The Java Virtual Machine Specification. * * @return the {@code int} representing the modifiers for this class * @see java.lang.reflect.Modifier @@ -1678,7 +1678,7 @@ public final class Class implements java.io.Serializable, /** * Returns the canonical name of the underlying class as - * defined by The Java™ Language Specification. + * defined by The Java Language Specification. * Returns {@code null} if the underlying class does not have a canonical * name. Classes without canonical names include: *