8296329: jar validator doesn't account for minor class file version

Reviewed-by: jvernee
This commit is contained in:
Bo Zhang 2022-11-23 03:09:12 +00:00 committed by Jorn Vernee
parent 09f70dad2f
commit faf48e61be
3 changed files with 126 additions and 4 deletions

View File

@ -105,7 +105,7 @@ final class FingerPrint {
} }
public boolean isCompatibleVersion(FingerPrint that) { public boolean isCompatibleVersion(FingerPrint that) {
return attrs.version >= that.attrs.version; return attrs.majorVersion >= that.attrs.majorVersion;
} }
public boolean isSameAPI(FingerPrint that) { public boolean isSameAPI(FingerPrint that) {
@ -236,7 +236,7 @@ final class FingerPrint {
private String name; private String name;
private String outerClassName; private String outerClassName;
private String superName; private String superName;
private int version; private int majorVersion;
private int access; private int access;
private boolean publicClass; private boolean publicClass;
private boolean nestedClass; private boolean nestedClass;
@ -255,7 +255,7 @@ final class FingerPrint {
@Override @Override
public void visit(int version, int access, String name, String signature, public void visit(int version, int access, String name, String signature,
String superName, String[] interfaces) { String superName, String[] interfaces) {
this.version = version; this.majorVersion = version & 0xFFFF; // JDK-8296329: extract major version only
this.access = access; this.access = access;
this.name = name; this.name = name;
this.nestedClass = name.contains("$"); this.nestedClass = name.contains("$");

View File

@ -102,6 +102,10 @@ public class MRTestBase {
} }
void javac(Path dest, Path... sourceFiles) throws Throwable { void javac(Path dest, Path... sourceFiles) throws Throwable {
javac(dest, List.of(), sourceFiles);
}
void javac(Path dest, List<String> extraParameters, Path... sourceFiles) throws Throwable {
List<String> commands = new ArrayList<>(); List<String> commands = new ArrayList<>();
String opts = System.getProperty("test.compiler.opts"); String opts = System.getProperty("test.compiler.opts");
@ -113,6 +117,7 @@ public class MRTestBase {
Stream.of(sourceFiles) Stream.of(sourceFiles)
.map(Object::toString) .map(Object::toString)
.forEach(x -> commands.add(x)); .forEach(x -> commands.add(x));
commands.addAll(extraParameters);
StringWriter sw = new StringWriter(); StringWriter sw = new StringWriter();
try (PrintWriter pw = new PrintWriter(sw)) { try (PrintWriter pw = new PrintWriter(sw)) {
@ -121,7 +126,6 @@ public class MRTestBase {
throw new RuntimeException(sw.toString()); throw new RuntimeException(sw.toString());
} }
} }
} }
OutputAnalyzer jarWithStdin(File stdinSource, OutputAnalyzer jarWithStdin(File stdinSource,

View File

@ -0,0 +1,118 @@
/*
* Copyright (c) 2022, 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 8296329
* @summary Tests for version validator.
* @library /test/lib
* @modules java.base/jdk.internal.misc
* jdk.compiler
* jdk.jartool
* @build jdk.test.lib.Utils
* jdk.test.lib.Asserts
* jdk.test.lib.JDKToolFinder
* jdk.test.lib.JDKToolLauncher
* jdk.test.lib.Platform
* jdk.test.lib.process.*
* MRTestBase
* @run testng/timeout=1200 VersionValidatorTest
*/
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
import java.lang.reflect.Method;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.List;
public class VersionValidatorTest extends MRTestBase {
private Path root;
@BeforeMethod
void testInit(Method method) {
root = Paths.get(method.getName());
}
@Test(dataProvider = "differentMajorVersions")
public void onlyCompatibleVersionIsAllowedInMultiReleaseJar(String baseMajorVersion, String otherMajorVersion,
boolean enablePreviewForBaseVersion, boolean enablePreviewForOtherVersion, boolean isAcceptable)
throws Throwable {
Path baseVersionClassesDir = compileLibClass(baseMajorVersion, enablePreviewForBaseVersion);
Path otherVersionClassesDir = compileLibClass(otherMajorVersion, enablePreviewForOtherVersion);
var result = jar("--create", "--file", "lib.jar", "-C", baseVersionClassesDir.toString(), "Lib.class",
"--release", otherMajorVersion, "-C", otherVersionClassesDir.toString(), "Lib.class");
if (isAcceptable) {
result.shouldHaveExitValue(SUCCESS)
.shouldBeEmptyIgnoreVMWarnings();
} else {
result.shouldNotHaveExitValue(SUCCESS)
.shouldContain("has a class version incompatible with an earlier version");
}
}
private Path compileLibClass(String majorVersion, boolean enablePreview) throws Throwable {
String classTemplate = """
public class Lib {
public static int version = $VERSION;
}
""";
Path sourceFile = Files.createDirectories(root.resolve("src").resolve(majorVersion)).resolve("Lib.java");
Files.write(sourceFile, classTemplate.replace("$VERSION", majorVersion).getBytes());
Path classesDir = root.resolve("classes").resolve(majorVersion);
javac(classesDir, List.of("--release", majorVersion), sourceFile);
if (enablePreview) {
rewriteMinorVersionForEnablePreviewClass(classesDir.resolve("Lib.class"));
}
return classesDir;
}
private void rewriteMinorVersionForEnablePreviewClass(Path classFile) throws Throwable {
byte[] classBytes = Files.readAllBytes(classFile);
classBytes[4] = -1;
classBytes[5] = -1;
Files.write(classFile, classBytes);
}
@DataProvider
Object[][] differentMajorVersions() {
return new Object[][] {
{ "19", "20", false, true, true },
{ "19", "20", false, false, true },
{ "19", "20", true, true, true },
{ "19", "20", true, false, true },
{ "20", "19", false, true, false },
{ "20", "19", false, false, false },
{ "20", "19", true, true, false },
{ "20", "19", true, false, false },
};
}
}