Skip to content

Commit 3984253

Browse files
committedMay 17, 2022
8284115: [IR Framework] Compilation is not found due to rare safepoint while dumping PrintIdeal/PrintOptoAssembly
Reviewed-by: kvn, thartmann
1 parent b434b1f commit 3984253

File tree

6 files changed

+120
-51
lines changed

6 files changed

+120
-51
lines changed
 

‎test/hotspot/jtreg/compiler/lib/ir_framework/driver/irmatching/parser/AbstractLine.java

+38-1
Original file line numberDiff line numberDiff line change
@@ -25,16 +25,20 @@
2525

2626
import java.io.BufferedReader;
2727
import java.io.IOException;
28+
import java.util.regex.Matcher;
29+
import java.util.regex.Pattern;
2830

2931
/**
3032
* Base class of a read line from the hotspot_pid* file.
3133
*/
3234
abstract class AbstractLine {
3335
private final BufferedReader reader;
3436
protected String line;
37+
private final Pattern compileIdPatternForTestClass;
3538

36-
public AbstractLine(BufferedReader reader) {
39+
public AbstractLine(BufferedReader reader, Pattern compileIdPatternForTestClass) {
3740
this.reader = reader;
41+
this.compileIdPatternForTestClass = compileIdPatternForTestClass;
3842
}
3943

4044
public String getLine() {
@@ -48,4 +52,37 @@ public boolean readLine() throws IOException {
4852
line = reader.readLine();
4953
return line != null;
5054
}
55+
56+
/**
57+
* Is this line a start of a method in the test class? We only care about test class entries. There might be non-class
58+
* entries as well if the user specified additional compile commands. Ignore these.
59+
*/
60+
public boolean isTestClassCompilation() {
61+
if (isCompilation()) {
62+
Matcher matcher = compileIdPatternForTestClass.matcher(line);
63+
return matcher.find();
64+
}
65+
return false;
66+
}
67+
68+
/**
69+
* Is this header a C2 non-OSR compilation header entry?
70+
*/
71+
public boolean isCompilation() {
72+
return line.startsWith("<task_queued") && notOSRCompilation() && notC2Compilation();
73+
}
74+
75+
/**
76+
* OSR compilations have compile_kind set.
77+
*/
78+
protected boolean notOSRCompilation() {
79+
return !line.contains("compile_kind='");
80+
}
81+
82+
/**
83+
* Non-C2 compilations have level set.
84+
*/
85+
private boolean notC2Compilation() {
86+
return !line.contains("level='");
87+
}
5188
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
/*
2+
* Copyright (c) 2022, 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.lib.ir_framework.driver.irmatching.parser;
25+
26+
import java.util.List;
27+
28+
/**
29+
* Class representing a PrintIdeal or PrintOptoAssembly output block read from the hotspot_pid* file.
30+
*/
31+
record Block(String output, List<String> testClassCompilations) {
32+
public String getOutput() {
33+
return output;
34+
}
35+
36+
public boolean containsTestClassCompilations() {
37+
return !testClassCompilations.isEmpty();
38+
}
39+
40+
public List<String> getTestClassCompilations() {
41+
return testClassCompilations;
42+
}
43+
}

‎test/hotspot/jtreg/compiler/lib/ir_framework/driver/irmatching/parser/BlockLine.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -24,14 +24,15 @@
2424
package compiler.lib.ir_framework.driver.irmatching.parser;
2525

2626
import java.io.BufferedReader;
27+
import java.util.regex.Pattern;
2728

2829
/**
2930
* Class representing a block line inside a PrintIdeal or PrintOptoAssembly output block read from the hotspot_pid* file.
3031
*/
3132
class BlockLine extends AbstractLine {
3233

33-
public BlockLine(BufferedReader reader) {
34-
super(reader);
34+
public BlockLine(BufferedReader reader, Pattern compileIdPatternForTestClass) {
35+
super(reader, compileIdPatternForTestClass);
3536
}
3637

3738
/**

‎test/hotspot/jtreg/compiler/lib/ir_framework/driver/irmatching/parser/BlockOutputReader.java

+15-6
Original file line numberDiff line numberDiff line change
@@ -25,27 +25,36 @@
2525

2626
import java.io.BufferedReader;
2727
import java.io.IOException;
28+
import java.util.ArrayList;
29+
import java.util.List;
30+
import java.util.regex.Pattern;
2831

2932
/**
3033
* Class to read all lines of a PrintIdeal or PrintOptoAssembly block.
3134
*/
3235
class BlockOutputReader {
33-
private final BufferedReader reader;
36+
private final BlockLine line;
3437

35-
public BlockOutputReader(BufferedReader reader) {
36-
this.reader = reader;
38+
public BlockOutputReader(BufferedReader reader, Pattern compileIdPatternForTestClass) {
39+
this.line = new BlockLine(reader, compileIdPatternForTestClass);
3740
}
3841

3942
/**
4043
* Read all lines belonging to a PrintIdeal or PrintOptoAssembly output block.
4144
*/
42-
public String readBlock() throws IOException {
43-
BlockLine line = new BlockLine(reader);
45+
public Block readBlock() throws IOException {
4446
StringBuilder builder = new StringBuilder();
47+
List<String> testClassCompilations = new ArrayList<>();
4548
while (line.readLine() && !line.isBlockEnd()) {
49+
if (line.isTestClassCompilation()) {
50+
// Could have safepointed while writing the block (see IRMatcher.SAFEPOINT_WHILE_PRINTING_MESSAGE)
51+
// and enqueuing the next test class method for compilation during the interruption. Record this
52+
// method to ensure that we read the PrintIdeal/PrintOptoAssembly blocks for that method later.
53+
testClassCompilations.add(line.getLine());
54+
}
4655
builder.append(escapeXML(line.getLine())).append(System.lineSeparator());
4756
}
48-
return builder.toString();
57+
return new Block(builder.toString(), testClassCompilations);
4958
}
5059

5160
/**

‎test/hotspot/jtreg/compiler/lib/ir_framework/driver/irmatching/parser/HotSpotPidFileParser.java

+20-5
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ public HotSpotPidFileParser(String testClass) {
5656
public void setCompilationsMap(Map<String, IRMethod> compilationsMap) {
5757
this.compilationsMap = compilationsMap;
5858
}
59+
5960
/**
6061
* Parse the hotspot_pid*.log file from the test VM. Read the PrintIdeal and PrintOptoAssembly outputs for all
6162
* methods of the test class that need to be IR matched (found in compilations map).
@@ -75,18 +76,28 @@ private void processFileLines(String hotspotPidFileName) throws IOException {
7576
Map<Integer, IRMethod> compileIdMap = new HashMap<>();
7677
try (var reader = Files.newBufferedReader(Paths.get(hotspotPidFileName))) {
7778
Line line = new Line(reader, compileIdPatternForTestClass);
78-
BlockOutputReader blockOutputReader = new BlockOutputReader(reader);
79+
BlockOutputReader blockOutputReader = new BlockOutputReader(reader, compileIdPatternForTestClass);
7980
while (line.readLine()) {
8081
if (line.isTestClassCompilation()) {
8182
parseTestMethodCompileId(compileIdMap, line.getLine());
8283
} else if (isTestMethodBlockStart(line, compileIdMap)) {
83-
String blockOutput = blockOutputReader.readBlock();
84-
setIRMethodOutput(blockOutput, line, compileIdMap);
84+
processMethodBlock(compileIdMap, line, blockOutputReader);
8585
}
8686
}
8787
}
8888
}
8989

90+
private void processMethodBlock(Map<Integer, IRMethod> compileIdMap, Line line, BlockOutputReader blockOutputReader)
91+
throws IOException {
92+
Block block = blockOutputReader.readBlock();
93+
if (block.containsTestClassCompilations()) {
94+
// Register all test method compilations that could have been emitted during a rare safepoint while
95+
// dumping the PrintIdeal/PrintOptoAssembly output.
96+
block.getTestClassCompilations().forEach(l -> parseTestMethodCompileId(compileIdMap, l));
97+
}
98+
setIRMethodOutput(block.getOutput(), line, compileIdMap);
99+
}
100+
90101
private void parseTestMethodCompileId(Map<Integer, IRMethod> compileIdMap, String line) {
91102
String methodName = parseMethodName(line);
92103
if (isTestAnnotatedMethod(methodName)) {
@@ -101,6 +112,9 @@ private String parseMethodName(String line) {
101112
return matcher.group(2);
102113
}
103114

115+
/**
116+
* Is this a @Test method?
117+
*/
104118
private boolean isTestAnnotatedMethod(String testMethodName) {
105119
return compilationsMap.containsKey(testMethodName);
106120
}
@@ -109,8 +123,6 @@ private IRMethod getIrMethod(String testMethodName) {
109123
return compilationsMap.get(testMethodName);
110124
}
111125

112-
113-
114126
private int getCompileId(String line) {
115127
Matcher matcher = COMPILE_ID_PATTERN.matcher(line);
116128
if (!matcher.find()) {
@@ -119,6 +131,9 @@ private int getCompileId(String line) {
119131
return Integer.parseInt(matcher.group(1));
120132
}
121133

134+
/**
135+
* Is this line the start of a PrintIdeal/PrintOptoAssembly output block of a @Test method?
136+
*/
122137
private boolean isTestMethodBlockStart(Line line, Map<Integer, IRMethod> compileIdMap) {
123138
return line.isBlockStart() && isTestClassMethodBlock(line, compileIdMap);
124139
}

‎test/hotspot/jtreg/compiler/lib/ir_framework/driver/irmatching/parser/Line.java

+1-37
Original file line numberDiff line numberDiff line change
@@ -31,44 +31,8 @@
3131
* Class representing a normal line read from the hotspot_pid* file.
3232
*/
3333
class Line extends AbstractLine {
34-
private final Pattern compileIdPatternForTestClass;
35-
3634
public Line(BufferedReader reader, Pattern compileIdPatternForTestClass) {
37-
super(reader);
38-
this.compileIdPatternForTestClass = compileIdPatternForTestClass;
39-
}
40-
41-
/**
42-
* Is this line a start of a @Test annotated method? We only care about test class entries. There might be non-class
43-
* entries as well if user specified additional compile commands. Ignore these.
44-
*/
45-
public boolean isTestClassCompilation() {
46-
if (isCompilation()) {
47-
Matcher matcher = compileIdPatternForTestClass.matcher(line);
48-
return matcher.find();
49-
}
50-
return false;
51-
}
52-
53-
/**
54-
* Is this header a C2 non-OSR compilation header entry?
55-
*/
56-
public boolean isCompilation() {
57-
return line.startsWith("<task_queued") && notOSRCompilation() && notC2Compilation();
58-
}
59-
60-
/**
61-
* OSR compilations have compile_kind set.
62-
*/
63-
private boolean notOSRCompilation() {
64-
return !line.contains("compile_kind='");
65-
}
66-
67-
/**
68-
* Non-C2 compilations have level set.
69-
*/
70-
private boolean notC2Compilation() {
71-
return !line.contains("level='");
35+
super(reader, compileIdPatternForTestClass);
7236
}
7337

7438
/**

0 commit comments

Comments
 (0)
Please sign in to comment.