Skip to content

Commit c1b4212

Browse files
committedJan 18, 2023
8300195: Fall-through issue occurs when using record pattern in switch statements
Reviewed-by: vromero
1 parent 0f925fe commit c1b4212

File tree

2 files changed

+122
-11
lines changed

2 files changed

+122
-11
lines changed
 

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/TransPatterns.java

+13-8
Original file line numberDiff line numberDiff line change
@@ -590,14 +590,10 @@ private void handleSwitch(JCTree tree,
590590
}
591591
}
592592
c.labels = translatedLabels.toList();
593-
if (c.caseKind == CaseTree.CaseKind.STATEMENT) {
594-
previousCompletesNormally = c.completesNormally;
595-
} else {
596-
previousCompletesNormally = false;
597-
JCBreak brk = make.at(TreeInfo.endPos(c.stats.last())).Break(null);
598-
brk.target = tree;
599-
c.stats = c.stats.append(brk);
600-
}
593+
previousCompletesNormally =
594+
c.caseKind == CaseTree.CaseKind.STATEMENT &&
595+
c.completesNormally;
596+
appendBreakIfNeeded(tree, c);
601597
}
602598

603599
if (tree.hasTag(Tag.SWITCH)) {
@@ -642,6 +638,14 @@ public void visitCase(JCCase c) {
642638
}.scan(c.stats);
643639
}
644640

641+
private void appendBreakIfNeeded(JCTree switchTree, JCCase c) {
642+
if (c.caseKind == CaseTree.CaseKind.RULE) {
643+
JCBreak brk = make.at(TreeInfo.endPos(c.stats.last())).Break(null);
644+
brk.target = switchTree;
645+
c.stats = c.stats.append(brk);
646+
}
647+
}
648+
645649
JCMethodInvocation makeApply(JCExpression selector, Name name, List<JCExpression> args) {
646650
MethodSymbol method = rs.resolveInternalMethod(
647651
currentClassTree.pos(), env,
@@ -740,6 +744,7 @@ public void resolve(VarSymbol commonBinding,
740744
} else {
741745
newLabel = List.of(make.PatternCaseLabel(binding, newGuard));
742746
}
747+
appendBreakIfNeeded(currentSwitch, accummulated);
743748
nestedCases.add(make.Case(CaseKind.STATEMENT, newLabel, accummulated.stats, null));
744749
}
745750
if (!hasUnconditional) {

‎test/langtools/tools/javac/patterns/PatternDesugaring.java

+109-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2022, 2023, 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
@@ -23,8 +23,9 @@
2323

2424
/**
2525
* @test
26-
* @bug 8291769
26+
* @bug 8291769 8300195
2727
* @summary Verify the compiled code does not have unwanted constructs.
28+
* @enablePreview
2829
* @library /tools/lib
2930
* @modules jdk.compiler/com.sun.tools.javac.api
3031
* jdk.compiler/com.sun.tools.javac.main
@@ -34,11 +35,17 @@
3435
* @run main PatternDesugaring
3536
*/
3637

38+
import java.io.ByteArrayOutputStream;
3739
import java.io.IOException;
40+
import java.io.PrintStream;
41+
import java.lang.reflect.Method;
42+
import java.net.URL;
43+
import java.net.URLClassLoader;
44+
import java.nio.charset.StandardCharsets;
3845
import java.nio.file.Files;
3946
import java.nio.file.Path;
4047
import java.nio.file.Paths;
41-
import java.util.List;
48+
import java.util.Objects;
4249
import java.util.function.Consumer;
4350

4451
import toolbox.TestRunner;
@@ -198,4 +205,103 @@ private void doTest(Path base, String[] libraryCode, String testCode, Consumer<S
198205
validate.accept(decompiled);
199206
}
200207

208+
@Test
209+
public void testRuleCases(Path base) throws Exception {
210+
doTestRun(base,
211+
new String[0],
212+
"""
213+
package test;
214+
public class Test {
215+
public static void main(String... args) {
216+
System.out.println(test(new R("a")));
217+
System.out.println(test(new R(3)));
218+
System.out.println(test(new R(new R("a"))));
219+
System.out.println(test(new R(new R(3))));
220+
}
221+
public static int test(Object obj) {
222+
int res;
223+
switch (obj) {
224+
case R(String s) -> res = s.length();
225+
case R(Integer i) -> res = i;
226+
case R(R(String s)) -> res = 10 + s.length();
227+
case R(R(Integer i)) -> res = 10 + i;
228+
default -> res = -1;
229+
}
230+
return res;
231+
}
232+
record R(Object o) {}
233+
}
234+
""",
235+
output -> {
236+
String expectedOutput = """
237+
1
238+
3
239+
11
240+
13
241+
""";
242+
if (!Objects.equals(output, expectedOutput)) {
243+
throw new AssertionError("Unexpected output," +
244+
" expected: " + expectedOutput +
245+
" actual: " + output);
246+
}
247+
});
248+
}
249+
250+
private void doTestRun(Path base, String[] libraryCode, String testCode, Consumer<String> validate) throws Exception {
251+
Path current = base.resolve(".");
252+
Path libClasses = current.resolve("libClasses");
253+
254+
Files.createDirectories(libClasses);
255+
256+
if (libraryCode.length != 0) {
257+
Path libSrc = current.resolve("lib-src");
258+
259+
for (String code : libraryCode) {
260+
tb.writeJavaFiles(libSrc, code);
261+
}
262+
263+
new JavacTask(tb)
264+
.options("--enable-preview",
265+
"-source", JAVA_VERSION)
266+
.outdir(libClasses)
267+
.files(tb.findJavaFiles(libSrc))
268+
.run();
269+
}
270+
271+
Path src = current.resolve("src");
272+
tb.writeJavaFiles(src, testCode);
273+
274+
Path classes = current.resolve("libClasses");
275+
276+
Files.createDirectories(libClasses);
277+
278+
var log =
279+
new JavacTask(tb)
280+
.options("--enable-preview",
281+
"-source", JAVA_VERSION,
282+
"-XDrawDiagnostics",
283+
"-Xlint:-preview",
284+
"--class-path", libClasses.toString(),
285+
"-XDshould-stop.at=FLOW")
286+
.outdir(classes)
287+
.files(tb.findJavaFiles(src))
288+
.run(Task.Expect.SUCCESS)
289+
.writeAll();
290+
291+
ClassLoader cl = new URLClassLoader(new URL[] {classes.toUri().toURL()});
292+
Class<?> testClass = cl.loadClass("test.Test");
293+
Method main = testClass.getMethod("main", String[].class);
294+
PrintStream prevOut = System.out;
295+
var data = new ByteArrayOutputStream();
296+
try (var outStream = new PrintStream(data, true, StandardCharsets.UTF_8)) {
297+
System.setOut(outStream);
298+
main.invoke(null, (Object) new String[0]);
299+
} finally {
300+
System.setOut(prevOut);
301+
}
302+
String output = new String(data.toByteArray(), StandardCharsets.UTF_8);
303+
output = output.replaceAll("\\R", "\n");
304+
validate.accept(output);
305+
}
306+
201307
}

0 commit comments

Comments
 (0)
Please sign in to comment.