Skip to content

Commit eab8455

Browse files
Wang HuangWu YanAi Jiaming
authored and
Vladimir Ivanov
committedApr 7, 2021
8261137: Optimization of Box nodes in uncommon_trap
Co-authored-by: Wu Yan <wuyan34@huawei.com> Co-authored-by: Ai Jiaming <aijiaming1@huawei.com> Reviewed-by: kvn, vlivanov, thartmann
1 parent 92fad1b commit eab8455

10 files changed

+306
-24
lines changed
 

‎src/hotspot/share/opto/callGenerator.cpp

+69-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2000, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2000, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -554,6 +554,57 @@ void LateInlineVirtualCallGenerator::do_late_inline() {
554554
CallGenerator::do_late_inline_helper();
555555
}
556556

557+
static bool has_non_debug_usages(Node* n) {
558+
for (DUIterator_Fast imax, i = n->fast_outs(imax); i < imax; i++) {
559+
Node* m = n->fast_out(i);
560+
if (!m->is_SafePoint()
561+
|| (m->is_Call() && m->as_Call()->has_non_debug_use(n))) {
562+
return true;
563+
}
564+
}
565+
return false;
566+
}
567+
568+
// delay box in runtime, treat box as a scalarized object
569+
static void scalarize_debug_usages(CallNode* call, Node* resproj) {
570+
GraphKit kit(call->jvms());
571+
PhaseGVN& gvn = kit.gvn();
572+
573+
ProjNode* res = resproj->as_Proj();
574+
ciInstanceKlass* klass = call->as_CallStaticJava()->method()->holder();
575+
int n_fields = klass->nof_nonstatic_fields();
576+
assert(n_fields == 1, "the klass must be an auto-boxing klass");
577+
578+
for (DUIterator_Last imin, i = res->last_outs(imin); i >= imin;) {
579+
SafePointNode* sfpt = res->last_out(i)->as_SafePoint();
580+
uint first_ind = sfpt->req() - sfpt->jvms()->scloff();
581+
Node* sobj = new SafePointScalarObjectNode(gvn.type(res)->isa_oopptr(),
582+
#ifdef ASSERT
583+
call,
584+
#endif // ASSERT
585+
first_ind, n_fields, true);
586+
sobj->init_req(0, kit.root());
587+
sfpt->add_req(call->in(TypeFunc::Parms));
588+
sobj = gvn.transform(sobj);
589+
JVMState* jvms = sfpt->jvms();
590+
jvms->set_endoff(sfpt->req());
591+
int start = jvms->debug_start();
592+
int end = jvms->debug_end();
593+
int num_edges = sfpt->replace_edges_in_range(res, sobj, start, end, &gvn);
594+
i -= num_edges;
595+
}
596+
597+
assert(res->outcnt() == 0, "the box must have no use after replace");
598+
599+
#ifndef PRODUCT
600+
if (PrintEliminateAllocations) {
601+
tty->print("++++ Eliminated: %d ", call->_idx);
602+
call->as_CallStaticJava()->method()->print_short_name(tty);
603+
tty->cr();
604+
}
605+
#endif
606+
}
607+
557608
void CallGenerator::do_late_inline_helper() {
558609
assert(is_late_inline(), "only late inline allowed");
559610

@@ -603,10 +654,23 @@ void CallGenerator::do_late_inline_helper() {
603654
C->remove_macro_node(call);
604655
}
605656

606-
bool result_not_used = (callprojs.resproj == NULL || callprojs.resproj->outcnt() == 0);
607-
if (is_pure_call() && result_not_used) {
657+
bool result_not_used = false;
658+
659+
if (is_pure_call()) {
660+
if (is_boxing_late_inline() && callprojs.resproj != nullptr) {
661+
// replace box node to scalar node only in case it is directly referenced by debug info
662+
assert(call->as_CallStaticJava()->is_boxing_method(), "sanity");
663+
if (!has_non_debug_usages(callprojs.resproj)) {
664+
scalarize_debug_usages(call, callprojs.resproj);
665+
}
666+
}
667+
608668
// The call is marked as pure (no important side effects), but result isn't used.
609669
// It's safe to remove the call.
670+
result_not_used = (callprojs.resproj == NULL || callprojs.resproj->outcnt() == 0);
671+
}
672+
673+
if (result_not_used) {
610674
GraphKit kit(call->jvms());
611675
kit.replace_call(call, C->top(), true);
612676
} else {
@@ -741,6 +805,8 @@ class LateInlineBoxingCallGenerator : public LateInlineCallGenerator {
741805
return new_jvms;
742806
}
743807

808+
virtual bool is_boxing_late_inline() const { return true; }
809+
744810
virtual CallGenerator* with_call_node(CallNode* call) {
745811
LateInlineBoxingCallGenerator* cg = new LateInlineBoxingCallGenerator(method(), _inline_cg);
746812
cg->set_call_node(call->as_CallStaticJava());

‎src/hotspot/share/opto/callGenerator.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,7 @@ class CallGenerator : public ResourceObj {
7474
virtual bool is_late_inline() const { return false; }
7575
// same but for method handle calls
7676
virtual bool is_mh_late_inline() const { return false; }
77+
virtual bool is_boxing_late_inline() const { return false; }
7778
virtual bool is_string_late_inline() const { return false; }
7879
virtual bool is_virtual_late_inline() const { return false; }
7980

‎src/hotspot/share/opto/callnode.cpp

+13-3
Original file line numberDiff line numberDiff line change
@@ -1515,17 +1515,27 @@ void SafePointNode::disconnect_from_root(PhaseIterGVN *igvn) {
15151515

15161516
SafePointScalarObjectNode::SafePointScalarObjectNode(const TypeOopPtr* tp,
15171517
#ifdef ASSERT
1518-
AllocateNode* alloc,
1518+
Node* alloc,
15191519
#endif
15201520
uint first_index,
1521-
uint n_fields) :
1521+
uint n_fields,
1522+
bool is_auto_box) :
15221523
TypeNode(tp, 1), // 1 control input -- seems required. Get from root.
15231524
_first_index(first_index),
1524-
_n_fields(n_fields)
1525+
_n_fields(n_fields),
1526+
_is_auto_box(is_auto_box)
15251527
#ifdef ASSERT
15261528
, _alloc(alloc)
15271529
#endif
15281530
{
1531+
#ifdef ASSERT
1532+
if (!alloc->is_Allocate()
1533+
&& !(alloc->Opcode() == Op_VectorBox)
1534+
&& (!alloc->is_CallStaticJava() || !alloc->as_CallStaticJava()->is_boxing_method())) {
1535+
alloc->dump();
1536+
assert(false, "unexpected call node");
1537+
}
1538+
#endif
15291539
init_class_id(Class_SafePointScalarObject);
15301540
}
15311541

‎src/hotspot/share/opto/callnode.hpp

+6-4
Original file line numberDiff line numberDiff line change
@@ -501,7 +501,8 @@ class SafePointScalarObjectNode: public TypeNode {
501501
// states of the scalarized object fields are collected.
502502
// It is relative to the last (youngest) jvms->_scloff.
503503
uint _n_fields; // Number of non-static fields of the scalarized object.
504-
DEBUG_ONLY(AllocateNode* _alloc;)
504+
bool _is_auto_box; // True if the scalarized object is an auto box.
505+
DEBUG_ONLY(Node* _alloc;)
505506

506507
virtual uint hash() const ; // { return NO_HASH; }
507508
virtual bool cmp( const Node &n ) const;
@@ -511,9 +512,9 @@ class SafePointScalarObjectNode: public TypeNode {
511512
public:
512513
SafePointScalarObjectNode(const TypeOopPtr* tp,
513514
#ifdef ASSERT
514-
AllocateNode* alloc,
515+
Node* alloc,
515516
#endif
516-
uint first_index, uint n_fields);
517+
uint first_index, uint n_fields, bool is_auto_box = false);
517518
virtual int Opcode() const;
518519
virtual uint ideal_reg() const;
519520
virtual const RegMask &in_RegMask(uint) const;
@@ -526,8 +527,9 @@ class SafePointScalarObjectNode: public TypeNode {
526527
}
527528
uint n_fields() const { return _n_fields; }
528529

530+
bool is_auto_box() const { return _is_auto_box; }
529531
#ifdef ASSERT
530-
AllocateNode* alloc() const { return _alloc; }
532+
Node* alloc() const { return _alloc; }
531533
#endif
532534

533535
virtual uint size_of() const { return sizeof(*this); }

‎src/hotspot/share/opto/output.cpp

+6-4
Original file line numberDiff line numberDiff line change
@@ -826,8 +826,9 @@ void PhaseOutput::FillLocArray( int idx, MachSafePointNode* sfpt, Node *local,
826826
ciKlass* cik = t->is_oopptr()->klass();
827827
assert(cik->is_instance_klass() ||
828828
cik->is_array_klass(), "Not supported allocation.");
829-
sv = new ObjectValue(spobj->_idx,
830-
new ConstantOopWriteValue(cik->java_mirror()->constant_encoding()));
829+
ScopeValue* klass_sv = new ConstantOopWriteValue(cik->java_mirror()->constant_encoding());
830+
sv = spobj->is_auto_box() ? new AutoBoxObjectValue(spobj->_idx, klass_sv)
831+
: new ObjectValue(spobj->_idx, klass_sv);
831832
set_sv_for_object_node(objs, sv);
832833

833834
uint first_ind = spobj->first_index(sfpt->jvms());
@@ -1099,8 +1100,9 @@ void PhaseOutput::Process_OopMap_Node(MachNode *mach, int current_offset) {
10991100
ciKlass* cik = t->is_oopptr()->klass();
11001101
assert(cik->is_instance_klass() ||
11011102
cik->is_array_klass(), "Not supported allocation.");
1102-
ObjectValue* sv = new ObjectValue(spobj->_idx,
1103-
new ConstantOopWriteValue(cik->java_mirror()->constant_encoding()));
1103+
ScopeValue* klass_sv = new ConstantOopWriteValue(cik->java_mirror()->constant_encoding());
1104+
ObjectValue* sv = spobj->is_auto_box() ? new AutoBoxObjectValue(spobj->_idx, klass_sv)
1105+
: new ObjectValue(spobj->_idx, klass_sv);
11041106
PhaseOutput::set_sv_for_object_node(objs, sv);
11051107

11061108
uint first_ind = spobj->first_index(youngest_jvms);

‎src/hotspot/share/opto/vector.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -248,7 +248,7 @@ void PhaseVector::scalarize_vbox_node(VectorBoxNode* vec_box) {
248248
uint first_ind = (sfpt->req() - sfpt->jvms()->scloff());
249249
Node* sobj = new SafePointScalarObjectNode(vec_box->box_type(),
250250
#ifdef ASSERT
251-
NULL,
251+
vec_box,
252252
#endif // ASSERT
253253
first_ind, n_fields);
254254
sobj->init_req(0, C->root());

‎src/hotspot/share/runtime/deoptimization.cpp

+7-8
Original file line numberDiff line numberDiff line change
@@ -898,7 +898,7 @@ Deoptimization::DeoptAction Deoptimization::_unloaded_action
898898

899899

900900

901-
#if INCLUDE_JVMCI || INCLUDE_AOT
901+
#if COMPILER2_OR_JVMCI || INCLUDE_AOT
902902
template<typename CacheType>
903903
class BoxCacheBase : public CHeapObj<mtCompiler> {
904904
protected:
@@ -1026,7 +1026,7 @@ oop Deoptimization::get_cached_box(AutoBoxObjectValue* bv, frame* fr, RegisterMa
10261026
}
10271027
return NULL;
10281028
}
1029-
#endif // INCLUDE_JVMCI || INCLUDE_AOT
1029+
#endif // COMPILER2_OR_JVMCI || INCLUDE_AOT
10301030

10311031
#if COMPILER2_OR_JVMCI
10321032
bool Deoptimization::realloc_objects(JavaThread* thread, frame* fr, RegisterMap* reg_map, GrowableArray<ScopeValue*>* objects, TRAPS) {
@@ -1045,17 +1045,16 @@ bool Deoptimization::realloc_objects(JavaThread* thread, frame* fr, RegisterMap*
10451045
oop obj = NULL;
10461046

10471047
if (k->is_instance_klass()) {
1048-
#if INCLUDE_JVMCI || INCLUDE_AOT
1049-
CompiledMethod* cm = fr->cb()->as_compiled_method_or_null();
1050-
if (cm->is_compiled_by_jvmci() && sv->is_auto_box()) {
1048+
#if COMPILER2_OR_JVMCI || INCLUDE_AOT
1049+
if (sv->is_auto_box()) {
10511050
AutoBoxObjectValue* abv = (AutoBoxObjectValue*) sv;
10521051
obj = get_cached_box(abv, fr, reg_map, THREAD);
10531052
if (obj != NULL) {
10541053
// Set the flag to indicate the box came from a cache, so that we can skip the field reassignment for it.
10551054
abv->set_cached(true);
10561055
}
10571056
}
1058-
#endif // INCLUDE_JVMCI || INCLUDE_AOT
1057+
#endif // COMPILER2_OR_JVMCI || INCLUDE_AOT
10591058
InstanceKlass* ik = InstanceKlass::cast(k);
10601059
if (obj == NULL) {
10611060
#ifdef COMPILER2
@@ -1397,12 +1396,12 @@ void Deoptimization::reassign_fields(frame* fr, RegisterMap* reg_map, GrowableAr
13971396
if (obj.is_null()) {
13981397
continue;
13991398
}
1400-
#if INCLUDE_JVMCI || INCLUDE_AOT
1399+
#if COMPILER2_OR_JVMCI || INCLUDE_AOT
14011400
// Don't reassign fields of boxes that came from a cache. Caches may be in CDS.
14021401
if (sv->is_auto_box() && ((AutoBoxObjectValue*) sv)->is_cached()) {
14031402
continue;
14041403
}
1405-
#endif // INCLUDE_JVMCI || INCLUDE_AOT
1404+
#endif // COMPILER2_OR_JVMCI || INCLUDE_AOT
14061405
#ifdef COMPILER2
14071406
if (EnableVectorSupport && VectorSupport::is_vector(k)) {
14081407
assert(sv->field_size() == 1, "%s not a vector", k->name()->as_C_string());

‎src/hotspot/share/runtime/deoptimization.hpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -164,9 +164,10 @@ class Deoptimization : AllStatic {
164164

165165
#if INCLUDE_JVMCI
166166
static address deoptimize_for_missing_exception_handler(CompiledMethod* cm);
167-
static oop get_cached_box(AutoBoxObjectValue* bv, frame* fr, RegisterMap* reg_map, TRAPS);
168167
#endif
169168

169+
static oop get_cached_box(AutoBoxObjectValue* bv, frame* fr, RegisterMap* reg_map, TRAPS);
170+
170171
private:
171172
// Does the actual work for deoptimizing a single frame
172173
static void deoptimize_single_frame(JavaThread* thread, frame fr, DeoptReason reason);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
/*
2+
* Copyright (c) 2021, Huawei Technologies Co., Ltd. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8261137
27+
* @requires vm.debug == true & vm.compiler2.enabled
28+
* @summary Verify that box object is scalarized in case it is directly referenced by debug info.
29+
* @library /test/lib
30+
*
31+
* @run main/othervm compiler.c2.TestEliminateBoxInDebugInfo
32+
*/
33+
package compiler.c2;
34+
35+
import java.util.regex.Matcher;
36+
import java.util.regex.Pattern;
37+
38+
import jdk.test.lib.Platform;
39+
import jdk.test.lib.process.OutputAnalyzer;
40+
import jdk.test.lib.process.ProcessTools;
41+
42+
import jdk.test.lib.Asserts;
43+
44+
public class TestEliminateBoxInDebugInfo {
45+
public static void runTest() throws Exception {
46+
final String[] arguments = {
47+
"-XX:CompileCommand=compileonly,compiler/c2/TestEliminateBoxInDebugInfo$Test.foo",
48+
"-XX:CompileCommand=dontinline,compiler/c2/TestEliminateBoxInDebugInfo$Test.black",
49+
"-Xbatch",
50+
"-XX:+PrintEliminateAllocations",
51+
Test.class.getName()
52+
};
53+
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(arguments);
54+
OutputAnalyzer output = new OutputAnalyzer(pb.start());
55+
System.out.println(output.getStdout());
56+
String pattern = ".*Eliminated.*";
57+
Pattern r = Pattern.compile(pattern);
58+
Matcher m = r.matcher(output.getStdout());
59+
if (!m.find()) {
60+
throw new RuntimeException("Could not find Elimination output");
61+
}
62+
}
63+
64+
public static void main(String[] args) throws Exception {
65+
runTest();
66+
}
67+
68+
static class Test {
69+
public static void main(String[] args) throws Exception {
70+
// warmup
71+
for(int i = 0; i < 100000; i++) {
72+
foo(1000 + (i % 1000));
73+
}
74+
}
75+
76+
public static int foo(int value) {
77+
Integer ii = Integer.valueOf(value);
78+
int sum = 0;
79+
if (value > 999) {
80+
sum += ii.intValue();
81+
}
82+
black();
83+
return sum;
84+
}
85+
86+
public static void black() {}
87+
}
88+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
/*
2+
* Copyright (c) 2021, Huawei Technologies Co., Ltd. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8261137
27+
* @requires vm.flavor == "server"
28+
* @summary Verify that box object identity matches after deoptimization When it is eliminated.
29+
* @library /test/lib
30+
*
31+
* @run main/othervm -Xbatch compiler.c2.TestIdentityWithEliminateBoxInDebugInfo
32+
*/
33+
package compiler.c2;
34+
35+
import jdk.test.lib.Asserts;
36+
37+
public class TestIdentityWithEliminateBoxInDebugInfo {
38+
interface TestF {
39+
void apply(boolean condition);
40+
}
41+
42+
public static void helper(TestF f) {
43+
// warmup
44+
for(int i = 0; i < 100000; i++) {
45+
f.apply(true);
46+
}
47+
// deoptimize
48+
f.apply(false);
49+
}
50+
51+
public static void runTest() throws Exception {
52+
helper((c) -> {
53+
Integer a = Integer.valueOf(42);
54+
Integer b = Integer.valueOf(-42);
55+
if (!c) {
56+
Asserts.assertTrue(a == Integer.valueOf(42));
57+
Asserts.assertTrue(b == Integer.valueOf(-42));
58+
}
59+
});
60+
61+
helper((c) -> {
62+
long highBitsOnly = 2L << 40;
63+
Long a = Long.valueOf(42L);
64+
Long b = Long.valueOf(-42L);
65+
Long h = Long.valueOf(highBitsOnly);
66+
if (!c) {
67+
Asserts.assertTrue(a == Long.valueOf(42L));
68+
Asserts.assertTrue(b == Long.valueOf(-42L));
69+
Asserts.assertFalse(h == Long.valueOf(highBitsOnly));
70+
}
71+
});
72+
73+
helper((c) -> {
74+
Character a = Character.valueOf('a');
75+
Character b = Character.valueOf('Z');
76+
if (!c) {
77+
Asserts.assertTrue(a == Character.valueOf('a'));
78+
Asserts.assertTrue(b == Character.valueOf('Z'));
79+
}
80+
});
81+
82+
helper((c) -> {
83+
Short a = Short.valueOf((short)42);
84+
Short b = Short.valueOf((short)-42);
85+
if (!c) {
86+
Asserts.assertTrue(a == Short.valueOf((short)42));
87+
Asserts.assertTrue(b == Short.valueOf((short)-42));
88+
}
89+
});
90+
91+
helper((c) -> {
92+
Byte a = Byte.valueOf((byte)42);
93+
Byte b = Byte.valueOf((byte)-42);
94+
if (!c) {
95+
Asserts.assertTrue(a == Byte.valueOf((byte)42));
96+
Asserts.assertTrue(b == Byte.valueOf((byte)-42));
97+
}
98+
});
99+
100+
helper((c) -> {
101+
Boolean a = Boolean.valueOf(true);
102+
Boolean b = Boolean.valueOf(false);
103+
if (!c) {
104+
Asserts.assertTrue(a == Boolean.valueOf(true));
105+
Asserts.assertTrue(b == Boolean.valueOf(false));
106+
}
107+
});
108+
}
109+
110+
public static void main(String[] args) throws Exception {
111+
runTest();
112+
}
113+
}

0 commit comments

Comments
 (0)
Please sign in to comment.