Skip to content

Commit 94eda53

Browse files
committedMar 7, 2023
8201516: DebugNonSafepoints generates incorrect information
Reviewed-by: kvn, roland
1 parent c51d40c commit 94eda53

File tree

3 files changed

+152
-2
lines changed

3 files changed

+152
-2
lines changed
 

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

+4-2
Original file line numberDiff line numberDiff line change
@@ -612,8 +612,6 @@ Parse::Parse(JVMState* caller, ciMethod* parse_method, float expected_uses)
612612
// Parse all the basic blocks.
613613
do_all_blocks();
614614

615-
C->set_default_node_notes(caller_nn);
616-
617615
// Check for bailouts during conversion to graph
618616
if (failing()) {
619617
if (log) log->done("parse");
@@ -624,6 +622,10 @@ Parse::Parse(JVMState* caller, ciMethod* parse_method, float expected_uses)
624622
set_map(entry_map);
625623
do_exits();
626624

625+
// Only reset this now, to make sure that debug information emitted
626+
// for exiting control flow still refers to the inlined method.
627+
C->set_default_node_notes(caller_nn);
628+
627629
if (log) log->done("parse nodes='%d' live='%d' memory='" SIZE_FORMAT "'",
628630
C->unique(), C->live_nodes(), C->node_arena()->used());
629631
}

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

+13
Original file line numberDiff line numberDiff line change
@@ -469,6 +469,14 @@ PhaseRenumberLive::PhaseRenumberLive(PhaseGVN* gvn,
469469

470470
uint worklist_size = worklist->size();
471471

472+
GrowableArray<Node_Notes*>* old_node_note_array = C->node_note_array();
473+
if (old_node_note_array != nullptr) {
474+
int new_size = (_useful.size() >> 8) + 1; // The node note array uses blocks, see C->_log2_node_notes_block_size
475+
new_size = MAX2(8, new_size);
476+
C->set_node_note_array(new (C->comp_arena()) GrowableArray<Node_Notes*> (C->comp_arena(), new_size, 0, nullptr));
477+
C->grow_node_notes(C->node_note_array(), new_size);
478+
}
479+
472480
// Iterate over the set of live nodes.
473481
for (uint current_idx = 0; current_idx < _useful.size(); current_idx++) {
474482
Node* n = _useful.at(current_idx);
@@ -484,6 +492,11 @@ PhaseRenumberLive::PhaseRenumberLive(PhaseGVN* gvn,
484492
assert(_old2new_map.at(n->_idx) == -1, "already seen");
485493
_old2new_map.at_put(n->_idx, current_idx);
486494

495+
if (old_node_note_array != nullptr) {
496+
Node_Notes* nn = C->locate_node_notes(old_node_note_array, n->_idx);
497+
C->set_node_notes_at(current_idx, nn);
498+
}
499+
487500
n->set_idx(current_idx); // Update node ID.
488501

489502
if (in_worklist) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,135 @@
1+
/*
2+
* Copyright (c) 2023, Oracle and/or its affiliates. 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+
package compiler.c2.irTests;
25+
26+
import compiler.lib.ir_framework.*;
27+
28+
/*
29+
* @test
30+
* @bug 8201516
31+
* @summary Verify that debug information in C2 compiled code is correct.
32+
* @library /test/lib /
33+
* @requires vm.compiler2.enabled
34+
* @run driver compiler.c2.irTests.TestDebugInfo
35+
*/
36+
public class TestDebugInfo {
37+
38+
public static void main(String[] args) {
39+
TestFramework.runWithFlags("-XX:+UnlockDiagnosticVMOptions", "-XX:+DebugNonSafepoints");
40+
}
41+
42+
static class MyClass {
43+
final int val;
44+
45+
@ForceInline
46+
public MyClass(int val) {
47+
this.val = val;
48+
}
49+
50+
@ForceInline
51+
synchronized void synchronizedMethod(boolean throwIt) {
52+
if (throwIt) {
53+
throw new RuntimeException(); // Make sure there is an exception state
54+
}
55+
}
56+
}
57+
58+
static Object[] array = new Object[3];
59+
static MyClass myVal = new MyClass(42);
60+
61+
// Verify that the MemBarRelease emitted at the MyClass constructor exit
62+
// does not incorrectly reference the caller method in its debug information.
63+
@Test
64+
@IR(failOn = {"MemBarRelease.*testFinalFieldInit.*bci:-1"}, phase = CompilePhase.BEFORE_MATCHING)
65+
public static void testFinalFieldInit() {
66+
array[0] = new MyClass(42);
67+
array[1] = new MyClass(42);
68+
array[2] = new MyClass(42);
69+
}
70+
71+
// Verify that the MemBarReleaseLock emitted at the synchronizedMethod exit
72+
// does not incorrectly reference the caller method in its debug information.
73+
@Test
74+
@IR(failOn = {"MemBarReleaseLock.*testSynchronized.*bci:-1"}, phase = CompilePhase.BEFORE_MATCHING)
75+
public static void testSynchronized() {
76+
try {
77+
myVal.synchronizedMethod(false);
78+
myVal.synchronizedMethod(true);
79+
} catch (Exception e) {
80+
// Ignore
81+
}
82+
}
83+
84+
static byte b0 = 0;
85+
static byte b1 = 0;
86+
static byte b2 = 0;
87+
static byte b3 = 0;
88+
89+
@ForceInline
90+
public static Integer useless3(Integer val) {
91+
return ++val;
92+
}
93+
94+
@ForceInline
95+
public static Integer useless2(Integer val) {
96+
return useless3(useless3(useless3(useless3(useless3(useless3(useless3(useless3(val))))))));
97+
}
98+
99+
@ForceInline
100+
public static Integer useless1(Integer val) {
101+
return useless2(useless2(useless2(useless2(useless2(useless2(useless2(useless2(val))))))));
102+
}
103+
104+
@ForceInline
105+
public static void useful3() {
106+
b3 = 3;
107+
}
108+
109+
@ForceInline
110+
public static void useful2() {
111+
useful3();
112+
b2 = 2;
113+
}
114+
115+
@ForceInline
116+
public static void useful1() {
117+
useful2();
118+
b1 = 1;
119+
}
120+
121+
// Verify that RenumberLiveNodes preserves the debug information side table.
122+
@Test
123+
@IR(counts = {"StoreB.*name=b3.*useful3.*bci:1.*useful2.*bci:0.*useful1.*bci:0.*testRenumberLiveNodes.*bci:9", "= 1"}, phase = CompilePhase.BEFORE_MATCHING)
124+
@IR(counts = {"StoreB.*name=b2.*useful2.*bci:4.*useful1.*bci:0.*testRenumberLiveNodes.*bci:9", "= 1"}, phase = CompilePhase.BEFORE_MATCHING)
125+
@IR(counts = {"StoreB.*name=b1.*useful1.*bci:4.*testRenumberLiveNodes.*bci:9", "= 1"}, phase = CompilePhase.BEFORE_MATCHING)
126+
@IR(counts = {"StoreB.*name=b0.*testRenumberLiveNodes.*bci:13", "= 1"}, phase = CompilePhase.BEFORE_MATCHING)
127+
public static void testRenumberLiveNodes() {
128+
// This generates ~3700 useless nodes to trigger RenumberLiveNodes
129+
useless1(42);
130+
131+
// Do something useful
132+
useful1();
133+
b0 = 0;
134+
}
135+
}

0 commit comments

Comments
 (0)
Please sign in to comment.