Skip to content
This repository has been archived by the owner on Feb 2, 2023. It is now read-only.

Commit

Permalink
8286277: CDS VerifyError when calling clone() on object array
Browse files Browse the repository at this point in the history
Reviewed-by: yan
Backport-of: 646c8aaeeccb494c72ff84c6e0f303f790be0ba9
  • Loading branch information
jankratochvil authored and Yuri Nesterenko committed Jul 22, 2022
1 parent 88e7d24 commit 3a5ebd1
Show file tree
Hide file tree
Showing 7 changed files with 346 additions and 9 deletions.
18 changes: 9 additions & 9 deletions src/hotspot/share/classfile/verifier.cpp
Expand Up @@ -2962,15 +2962,15 @@ void ClassVerifier::verify_invoke_instructions(
_klass, ref_class, method_name, method_sig, true)) {
// It's protected access, check if stack object is
// assignable to current class.
bool is_assignable = current_type().is_assignable_from(
stack_object_type, this, true, CHECK_VERIFY(this));
if (!is_assignable) {
if (ref_class_type.name() == vmSymbols::java_lang_Object()
&& stack_object_type.is_array()
&& method_name == vmSymbols::clone_name()) {
// Special case: arrays pretend to implement public Object
// clone().
} else {
if (ref_class_type.name() == vmSymbols::java_lang_Object()
&& stack_object_type.is_array()
&& method_name == vmSymbols::clone_name()) {
// Special case: arrays pretend to implement public Object
// clone().
} else {
bool is_assignable = current_type().is_assignable_from(
stack_object_type, this, true, CHECK_VERIFY(this));
if (!is_assignable) {
verify_error(ErrorContext::bad_type(bci,
current_frame->stack_top_ctx(),
TypeOrigin::implicit(current_type())),
Expand Down
78 changes: 78 additions & 0 deletions test/hotspot/jtreg/runtime/appcds/VerifyObjArrayCloneTest.java
@@ -0,0 +1,78 @@
/*
* 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 Verifier handling of invoking java/lang/Object::clone() on object arrays.
* @bug 8286277
* @requires vm.cds
* @library /test/lib /test/hotspot/jtreg/runtime/verifier /test/hotspot/jtreg/runtime/appcds/test-classes
* @build InvokeCloneValid InvokeCloneInvalid VerifyObjArrayCloneTestApp
* @run driver ClassFileInstaller -jar app.jar VerifyObjArrayCloneTestApp
* @run driver ClassFileInstaller -jar tests.jar InvokeCloneValid InvokeCloneInvalid
* @run driver VerifyObjArrayCloneTest
*/

import java.io.File;
import jdk.test.lib.Platform;

public class VerifyObjArrayCloneTest {
private static String appJar = ClassFileInstaller.getJarPath("app.jar");
private static String testsJar = ClassFileInstaller.getJarPath("tests.jar");
private static String mainAppClass = "VerifyObjArrayCloneTestApp";

public static void main(String... args) throws Exception {
testInAppPath();
if (Platform.areCustomLoadersSupportedForCDS()) {
testInCustomLoader();
}
}

// Try to load InvokeCloneValid and InvokeCloneInvalid from the AppClassLoader
static void testInAppPath() throws Exception {
String cp = appJar + File.pathSeparator + testsJar;
TestCommon.dump(cp, TestCommon.list(mainAppClass,
"InvokeCloneValid",
"InvokeCloneInvalid"));

TestCommon.run("-cp", cp, "-Xlog:cds+verification=trace",
mainAppClass)
.assertNormalExit();
}

// Try to load InvokeCloneValid and InvokeCloneInvalid from a custom class loader
static void testInCustomLoader() throws Exception {
String cp = appJar;

String classlist[] = new String[] {
mainAppClass,
"java/lang/Object id: 1",
"InvokeCloneValid id: 2 super: 1 source: " + testsJar,
"InvokeCloneInvalid id: 3 super: 1 source: " + testsJar,
};

TestCommon.dump(cp, classlist);
TestCommon.run("-cp", cp, "-Xlog:cds+verification=trace",
mainAppClass, testsJar)
.assertNormalExit();
}
}
@@ -0,0 +1,81 @@
/*
* 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 Verifier handling of invoking java/lang/Object::clone() on object arrays.
* @bug 8286277
* @requires vm.cds
* @library /test/lib /test/hotspot/jtreg/runtime/verifier
* /test/hotspot/jtreg/runtime/appcds
* /test/hotspot/jtreg/runtime/appcds/test-classes
* @build sun.hotspot.WhiteBox
* @build InvokeCloneValid InvokeCloneInvalid VerifyObjArrayCloneTestApp
* @run driver ClassFileInstaller -jar app.jar VerifyObjArrayCloneTestApp
* @run driver ClassFileInstaller -jar tests.jar InvokeCloneValid InvokeCloneInvalid
* @run driver ClassFileInstaller -jar WhiteBox.jar sun.hotspot.WhiteBox
* @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:./WhiteBox.jar VerifyObjArrayCloneTest
*/

import java.io.File;

public class VerifyObjArrayCloneTest extends DynamicArchiveTestBase {
private static String appJar = ClassFileInstaller.getJarPath("app.jar");
private static String testsJar = ClassFileInstaller.getJarPath("tests.jar");
private static String mainAppClass = "VerifyObjArrayCloneTestApp";

public static void main(String[] args) throws Exception {
runTest(VerifyObjArrayCloneTest::testInAppPath);
runTest(VerifyObjArrayCloneTest::testInCustomLoader);
}

// Try to load InvokeCloneValid and InvokeCloneInvalid from the AppClassLoader
private static void testInAppPath() throws Exception {
String cp = appJar + File.pathSeparator + testsJar;
String topArchiveName = getNewArchiveName("top");
dump(topArchiveName,
"-cp", cp,
mainAppClass)
.assertNormalExit();

run(topArchiveName,
"-cp", cp,
mainAppClass)
.assertNormalExit();
}

// Try to load InvokeCloneValid and InvokeCloneInvalid from a custom class loader
private static void testInCustomLoader() throws Exception {
String cp = appJar;
String topArchiveName = getNewArchiveName("top");
dump(topArchiveName,
"-cp", cp,
"-Xlog:cds+class=debug",
mainAppClass, testsJar)
.assertNormalExit();

run(topArchiveName,
"-cp", cp,
mainAppClass, testsJar)
.assertNormalExit();
}
}
@@ -0,0 +1,56 @@
/*
* 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.
*/

import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;

public class VerifyObjArrayCloneTestApp {
public static void main(String args[]) throws Exception {
ClassLoader appLoader = VerifyObjArrayCloneTestApp.class.getClassLoader();
if (args.length == 0) {
// Load the test classes from the classpath
doTest(appLoader);
} else {
File f = new File(args[0]);
URL[] classLoaderUrls = new URL[] {f.getAbsoluteFile().toURI().toURL()};
URLClassLoader customLoader = new URLClassLoader(classLoaderUrls, appLoader);
doTest(customLoader);
}
}

public static void doTest(ClassLoader loader) throws Exception {
try {
Class.forName("InvokeCloneValid", /*initialize=*/true, loader);
} catch (VerifyError e) {
throw new RuntimeException("Unexpected VerifyError", e);
}

try {
Class.forName("InvokeCloneInvalid", /*initialize=*/true, loader);
throw new RuntimeException("VerifyError expected but not thrown");
} catch (VerifyError e) {
System.out.println("Expected: " + e);
}
}
}
46 changes: 46 additions & 0 deletions test/hotspot/jtreg/runtime/verifier/InvokeClone.java
@@ -0,0 +1,46 @@
/*
* 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 Verifier handling of invoking java/lang/Object::clone() on object arrays.
* @bug 8286277
* @build InvokeCloneValid InvokeCloneInvalid
* @run main/othervm -Xverify InvokeClone
*/

public class InvokeClone {
public static void main(String[] args) throws ClassNotFoundException {
try {
Class.forName("InvokeCloneValid");
} catch (VerifyError e) {
throw new RuntimeException("Unexpected VerifyError", e);
}

try {
Class.forName("InvokeCloneInvalid");
throw new RuntimeException("VerifyError expected but not thrown");
} catch (VerifyError e) {
System.out.println("Expected: " + e);
}
}
}
34 changes: 34 additions & 0 deletions test/hotspot/jtreg/runtime/verifier/InvokeCloneInvalid.jasm
@@ -0,0 +1,34 @@
/*
* 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.
*/

super public class InvokeCloneInvalid
version 52:0
{
public static Method test:"(Ljava/lang/String;)V"
stack 1 locals 1
{
aload_0;
invokevirtual Method "java/lang/Object".clone:"()Ljava/lang/Object;";
return;
}
}
42 changes: 42 additions & 0 deletions test/hotspot/jtreg/runtime/verifier/InvokeCloneValid.jasm
@@ -0,0 +1,42 @@
/*
* 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.
*/

super public class InvokeCloneValid
version 52:0
{
public static Method test:"([Ljava/lang/Object;)V"
stack 1 locals 1
{
aload_0;
invokevirtual Method "java/lang/Object".clone:"()Ljava/lang/Object;";
return;
}

public static Method test2:"([Ljava/lang/String;)V"
stack 1 locals 1
{
aload_0;
invokevirtual Method "java/lang/Object".clone:"()Ljava/lang/Object;";
return;
}
}

1 comment on commit 3a5ebd1

@openjdk-notifier
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please sign in to comment.