| /* |
| * Copyright (c) 2014, 2023, Oracle and/or its affiliates. All rights reserved. |
| * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
| * |
| * This code is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License version 2 only, as |
| * published by the Free Software Foundation. |
| * |
| * This code is distributed in the hope that it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
| * version 2 for more details (a copy is included in the LICENSE file that |
| * accompanied this code). |
| * |
| * You should have received a copy of the GNU General Public License version |
| * 2 along with this work; if not, write to the Free Software Foundation, |
| * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
| * |
| * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
| * or visit www.oracle.com if you need additional information or have any |
| * questions. |
| */ |
| |
| /* |
| * @test |
| * @bug 4981566 5028634 5094412 6304984 7025786 7025789 8001112 8028545 |
| * 8000961 8030610 8028546 8188870 8173382 8173382 8193290 8205619 8028563 |
| * 8245147 8245586 8257453 8286035 |
| * @summary Check interpretation of -target and -source options |
| * @modules java.compiler |
| * jdk.compiler |
| * @run main Versions |
| */ |
| |
| import java.io.*; |
| import java.nio.*; |
| import java.nio.channels.*; |
| |
| import javax.tools.JavaCompiler; |
| import javax.tools.ToolProvider; |
| import javax.tools.JavaFileObject; |
| import javax.tools.StandardJavaFileManager; |
| import java.util.List; |
| import java.util.ArrayList; |
| import java.util.Set; |
| import java.util.function.Consumer; |
| |
| /* |
| * If not explicitly specified the latest source and latest target |
| * values are the defaults. If explicitly specified, the target value |
| * has to be greater than or equal to the source value. |
| */ |
| public class Versions { |
| |
| protected JavaCompiler javacompiler; |
| protected int failedCases; |
| |
| public Versions() throws IOException { |
| javacompiler = ToolProvider.getSystemJavaCompiler(); |
| genSourceFiles(); |
| failedCases = 0; |
| } |
| |
| public static void main(String... args) throws IOException { |
| Versions versions = new Versions(); |
| versions.run(); |
| } |
| |
| public static final Set<String> RETIRED_SOURCES = |
| Set.of("1.2", "1.3", "1.4", "1.5", "1.6", "1.7"); |
| |
| public static final Set<String> VALID_SOURCES = |
| Set.of("1.8", "1.9", "1.10", "11", "12", "13", "14", |
| "15", "16", "17", "18", "19", "20", "21"); |
| |
| public static final String LATEST_MAJOR_VERSION = "65.0"; |
| |
| static enum SourceTarget { |
| EIGHT(true, "52.0", "8"), |
| NINE(true, "53.0", "9"), |
| TEN(true, "54.0", "10"), |
| ELEVEN(false, "55.0", "11"), |
| TWELVE(false, "56.0", "12"), |
| THIRTEEN(false, "57.0", "13"), |
| FOURTEEN(false, "58.0", "14"), |
| FIFTEEN(false, "59.0", "15"), |
| SIXTEEN(false, "60.0", "16"), |
| SEVENTEEN(false, "61.0", "17"), |
| EIGHTEEN(false, "62.0", "18"), |
| NINETEEN(false, "63.0", "19"), |
| TWENTY(false, "64.0", "20"), |
| TWENTY_ONE(false,"65.0", "21"), |
| ; // Reduce code churn when appending new constants |
| |
| private final boolean dotOne; |
| private final String classFileVer; |
| private final String target; |
| private final int intTarget; |
| |
| private SourceTarget(boolean dotOne, String classFileVer, String target) { |
| this.dotOne = dotOne; |
| this.classFileVer = classFileVer; |
| this.target = target; |
| this.intTarget = Integer.parseInt(target); |
| } |
| |
| public void checksrc(Versions versions, List<String> args) { |
| // checker.accept(version, args); |
| versions.printargs("checksrc" + target, args); |
| List<String> expectedPassFiles = new ArrayList<>(); |
| List<String> expectedFailFiles = new ArrayList<>(); |
| |
| for (SourceExample srcEg : SourceExample.values()) { |
| var x = (srcEg.sourceLevel <= this.intTarget) ? |
| expectedPassFiles.add(srcEg.fileName()): |
| expectedFailFiles.add(srcEg.fileName()); |
| } |
| |
| versions.expectedPass(args, expectedPassFiles); |
| versions.expectedFail(args, expectedFailFiles); |
| } |
| |
| public boolean dotOne() { |
| return dotOne; |
| } |
| |
| public String classFileVer() { |
| return classFileVer; |
| } |
| |
| public String target() { |
| return target; |
| } |
| |
| public int intTarget() { |
| return intTarget; |
| } |
| } |
| |
| void run() { |
| String TC = ""; |
| System.out.println("Version.java: Starting"); |
| |
| check(LATEST_MAJOR_VERSION); |
| for (String source : VALID_SOURCES) { |
| check(LATEST_MAJOR_VERSION, List.of("-source " + source)); |
| } |
| |
| // Verify that a -source value less than a -target value is |
| // accepted and that the resulting class files are dependent |
| // on the target setting alone. |
| SourceTarget[] sourceTargets = SourceTarget.values(); |
| for (int i = 0; i < sourceTargets.length; i++) { |
| SourceTarget st = sourceTargets[i]; |
| String classFileVer = st.classFileVer(); |
| String target = st.target(); |
| boolean dotOne = st.dotOne(); |
| check_source_target(dotOne, List.of(classFileVer, target, target)); |
| for (int j = i - 1; j >= 0; j--) { |
| String source = sourceTargets[j].target(); |
| check_source_target(dotOne, List.of(classFileVer, source, target)); |
| } |
| } |
| |
| // Verify acceptance of different combinations of -source N, |
| // -target M; N <= M |
| for (int i = 0; i < sourceTargets.length; i++) { |
| SourceTarget st = sourceTargets[i]; |
| |
| st.checksrc(this, List.of("-source " + st.target())); |
| st.checksrc(this, List.of("-source " + st.target(), "-target " + st.target())); |
| |
| if (st.dotOne()) { |
| st.checksrc(this, List.of("-source 1." + st.target())); |
| st.checksrc(this, List.of("-source 1." + st.target(), "-target 1." + st.target())); |
| } |
| |
| if (i == sourceTargets.length - 1) { |
| // Can use -target without -source setting only for |
| // most recent target since the most recent source is |
| // the default. |
| st.checksrc(this, List.of("-target " + st.target())); |
| |
| if (!st.classFileVer().equals(LATEST_MAJOR_VERSION)) { |
| throw new RuntimeException(st + |
| "does not have class file version" + |
| LATEST_MAJOR_VERSION); |
| } |
| } |
| } |
| |
| // Verify that -source N -target (N-1) is rejected |
| for (int i = 1 /* Skip zeroth value */; i < sourceTargets.length; i++) { |
| fail(List.of("-source " + sourceTargets[i].target(), |
| "-target " + sourceTargets[i-1].target(), |
| "Base.java")); |
| } |
| |
| // Previously supported source/target values |
| for (String source : RETIRED_SOURCES) { |
| fail(List.of("-source " + source, "-target " + source, "Base.java")); |
| } |
| |
| if (failedCases > 0) { |
| System.err.println("failedCases = " + String.valueOf(failedCases)); |
| throw new Error("Test failed"); |
| } |
| |
| } |
| |
| protected void printargs(String fname, List<String> args) { |
| System.out.printf("test: %s", fname); |
| for (String onearg : args) { |
| System.out.printf(" %s", onearg); |
| } |
| System.out.printf("\n", fname); |
| } |
| |
| protected void check_source_target(boolean dotOne, List<String> args) { |
| printargs("check_source_target", args); |
| check_target(dotOne, List.of(args.get(0), args.get(1), args.get(2))); |
| if (dotOne) { |
| check_target(dotOne, List.of(args.get(0), "1." + args.get(1), args.get(2))); |
| } |
| } |
| |
| protected void check_target(boolean dotOne, List<String> args) { |
| check(args.get(0), List.of("-source " + args.get(1), "-target " + args.get(2))); |
| if (dotOne) { |
| check(args.get(0), List.of("-source " + args.get(1), "-target 1." + args.get(2))); |
| } |
| } |
| |
| protected void check(String major) { |
| check(major, List.of()); |
| } |
| |
| protected void check(String major, List<String> args) { |
| printargs("check", args); |
| |
| List<String> jcargs = javaCompilerOptions(args); |
| |
| boolean creturn = compile("Base.java", jcargs); |
| if (!creturn) { |
| // compilation errors note and return.. assume no class file |
| System.err.println("check: Compilation Failed"); |
| System.err.println("\t classVersion:\t" + major); |
| System.err.println("\t arguments:\t" + jcargs); |
| failedCases++; |
| |
| } else if (!checkClassFileVersion("Base.class", major)) { |
| failedCases++; |
| } |
| } |
| |
| /** |
| * Create a list of options suitable for use with {@link JavaCompiler} |
| * @param args a list of space-delimited options, such as "-source 11" |
| * @return a list of arguments suitable for use with {@link JavaCompiler} |
| */ |
| private static List<String> javaCompilerOptions(List<String> args) { |
| List<String> jcargs = new ArrayList<>(); |
| jcargs.add("-Xlint:-options"); |
| |
| // add in args conforming to List requirements of JavaCompiler |
| for (String onearg : args) { |
| String[] fields = onearg.split(" "); |
| for (String onefield : fields) { |
| jcargs.add(onefield); |
| } |
| } |
| return jcargs; |
| } |
| |
| /** |
| * The BASE source example is expected to compile on all source |
| * levels. Otherwise, an example is expected to compile on its |
| * declared source level and later, but to _not_ compile on |
| * earlier source levels. (This enum is _not_ intended to capture |
| * the uncommon program that is accepted in one version of the |
| * language and rejected in a later version.) |
| * |
| * When version of the language get a new, non-preview feature, a |
| * new source example enum constant should be added. |
| */ |
| enum SourceExample { |
| BASE(7, "Base.java", "public class Base { }\n"), |
| |
| |
| SOURCE_8(8, "New8.java", |
| // New feature in 8: lambda |
| """ |
| public class New8 { |
| void m() { |
| new Thread(() -> { }); |
| } |
| } |
| """), |
| |
| SOURCE_10(10, "New10.java", |
| // New feature in 10: var |
| """ |
| public class New10 { |
| void m() { |
| var tmp = new Thread(() -> { }); |
| } |
| } |
| """), |
| |
| SOURCE_11(11, "New11.java", |
| // New feature in 11: var for lambda parameters |
| """ |
| public class New11 { |
| static java.util.function.Function<String,String> f = (var x) -> x.substring(0); |
| void m(String name) { |
| var tmp = new Thread(() -> { }, f.apply(name)); |
| } |
| } |
| """), |
| |
| SOURCE_14(14, "New14.java", |
| // New feature in 14: switch expressions |
| """ |
| public class New14 { |
| static { |
| int i = 5; |
| System.out.println( |
| switch(i) { |
| case 0 -> false; |
| default -> true; |
| } |
| ); |
| } |
| } |
| """), |
| |
| SOURCE_15(15, "New15.java", |
| // New feature in 15: text blocks |
| """ |
| public class New15 { |
| public static final String s = |
| \"\"\" |
| Hello, World. |
| \"\"\" |
| ; |
| } |
| """), |
| |
| SOURCE_16(16, "New16.java", |
| // New feature in 16: records |
| """ |
| public class New16 { |
| public record Record(double rpm) { |
| public static final Record LONG_PLAY = new Record(100.0/3.0); |
| } |
| } |
| """), |
| |
| SOURCE_17(17, "New17.java", |
| // New feature in 17: sealed classes |
| """ |
| public class New17 { |
| public static sealed class Seal {} |
| |
| public static final class Pinniped extends Seal {} |
| public static final class TaperedThread extends Seal {} |
| public static final class Wax extends Seal {} |
| } |
| """), |
| |
| SOURCE_21(21, "New21.java", |
| // New feature in 21: pattern matching for switch |
| """ |
| public class New21 { |
| public static void main(String... args) { |
| Object o = new Object(){}; |
| |
| System.out.println(switch (o) { |
| case Integer i -> String.format("%d", i); |
| default -> o.toString(); |
| }); |
| } |
| } |
| """), |
| |
| ; // Reduce code churn when appending new constants |
| |
| private int sourceLevel; |
| private String fileName; |
| private String fileContents; |
| |
| private SourceExample(int sourceLevel, String fileName, String fileContents) { |
| this.sourceLevel = sourceLevel; |
| this.fileName = fileName; |
| this.fileContents = fileContents; |
| } |
| |
| public String fileName() {return fileName;} |
| public String fileContents() {return fileContents;} |
| } |
| |
| protected void expected(List<String> args, List<String> fileNames, |
| Consumer<List<String>> passOrFail) { |
| ArrayList<String> fullArguments = new ArrayList<>(args); |
| // Issue compile with each filename in turn. |
| for(String fileName : fileNames) { |
| fullArguments.add(fileName); |
| passOrFail.accept(fullArguments); |
| fullArguments.remove(fullArguments.size() - 1); |
| } |
| } |
| |
| protected void expectedPass(List<String> args, List<String> fileNames) { |
| expected(args, fileNames, this::pass); |
| } |
| |
| protected void expectedFail(List<String> args, List<String> fileNames) { |
| expected(args, fileNames, this::fail); |
| } |
| |
| protected void pass(List<String> args) { |
| printargs("pass", args); |
| |
| List<String> jcargs = javaCompilerOptions(args); |
| |
| // empty list is error |
| if (jcargs.isEmpty()) { |
| System.err.println("error: test error in pass() - No arguments"); |
| System.err.println("\t arguments:\t" + jcargs); |
| failedCases++; |
| return; |
| } |
| |
| // the last argument is the filename *.java |
| String filename = jcargs.get(jcargs.size() - 1); |
| jcargs.remove(jcargs.size() - 1); |
| |
| boolean creturn = compile(filename, jcargs); |
| // expect a compilation failure, failure if otherwise |
| if (!creturn) { |
| System.err.println("pass: Compilation erroneously failed"); |
| System.err.println("\t arguments:\t" + jcargs); |
| System.err.println("\t file :\t" + filename); |
| failedCases++; |
| |
| } |
| } |
| |
| protected void fail(List<String> args) { |
| printargs("fail", args); |
| |
| List<String> jcargs = javaCompilerOptions(args); |
| |
| // empty list is error |
| if (jcargs.isEmpty()) { |
| System.err.println("error: test error in fail()- No arguments"); |
| System.err.println("\t arguments:\t" + jcargs); |
| failedCases++; |
| return; |
| } |
| |
| // the last argument is the filename *.java |
| String filename = jcargs.get(jcargs.size() - 1); |
| jcargs.remove(jcargs.size() - 1); |
| |
| boolean creturn = compile(filename, jcargs); |
| // expect a compilation failure, failure if otherwise |
| if (creturn) { |
| System.err.println("fail: Compilation erroneously succeeded"); |
| System.err.println("\t arguments:\t" + jcargs); |
| System.err.println("\t file :\t" + filename); |
| failedCases++; |
| } |
| } |
| |
| protected boolean compile(String sourceFile, List<String> options) { |
| JavaCompiler.CompilationTask jctask; |
| try (StandardJavaFileManager fm = javacompiler.getStandardFileManager(null, null, null)) { |
| Iterable<? extends JavaFileObject> files = fm.getJavaFileObjects(sourceFile); |
| |
| jctask = javacompiler.getTask( |
| null, // Writer |
| fm, // JavaFileManager |
| null, // DiagnosticListener |
| options, // Iterable<String> |
| null, // Iterable<String> classes |
| files); // Iterable<? extends JavaFileObject> |
| |
| try { |
| return jctask.call(); |
| } catch (IllegalStateException e) { |
| System.err.println(e); |
| return false; |
| } |
| } catch (IOException e) { |
| throw new Error(e); |
| } |
| } |
| |
| protected void writeSourceFile(String fileName, String body) throws IOException{ |
| try (Writer fw = new FileWriter(fileName)) { |
| fw.write(body); |
| } |
| } |
| |
| protected void genSourceFiles() throws IOException{ |
| for (SourceExample srcEg : SourceExample.values()) { |
| writeSourceFile(srcEg.fileName(), srcEg.fileContents()); |
| } |
| } |
| |
| protected boolean checkClassFileVersion |
| (String filename,String classVersionNumber) { |
| ByteBuffer bb = ByteBuffer.allocate(1024); |
| try (FileChannel fc = new FileInputStream(filename).getChannel()) { |
| bb.clear(); |
| if (fc.read(bb) < 0) |
| throw new IOException("Could not read from file : " + filename); |
| bb.flip(); |
| int minor = bb.getShort(4); |
| int major = bb.getShort(6); |
| String fileVersion = major + "." + minor; |
| if (fileVersion.equals(classVersionNumber)) { |
| return true; |
| } else { |
| System.err.println("checkClassFileVersion : Failed"); |
| System.err.println("\tclassfile version mismatch"); |
| System.err.println("\texpected : " + classVersionNumber); |
| System.err.println("\tfound : " + fileVersion); |
| return false; |
| } |
| } |
| catch (IOException e) { |
| System.err.println("checkClassFileVersion : Failed"); |
| System.err.println("\terror :\t" + e.getMessage()); |
| System.err.println("\tfile:\tfilename"); |
| } |
| return false; |
| } |
| } |
| |