36e2ddad4d
Reviewed-by: ccheung, minqi
166 lines
5.9 KiB
Java
166 lines
5.9 KiB
Java
/*
|
|
* Copyright (c) 2017, 2021, 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.lang.instrument.ClassDefinition;
|
|
import java.lang.instrument.Instrumentation;
|
|
import java.lang.instrument.UnmodifiableClassException;
|
|
import java.net.URL;
|
|
import java.net.URLClassLoader;
|
|
import java.io.File;
|
|
import java.security.CodeSigner;
|
|
import java.security.CodeSource;
|
|
import java.security.ProtectionDomain;
|
|
import sun.hotspot.WhiteBox;
|
|
|
|
public class RedefineClassApp {
|
|
static WhiteBox wb = WhiteBox.getWhiteBox();
|
|
|
|
public static interface Intf { // Loaded from Boot class loader (-Xbootclasspath/a).
|
|
public String get();
|
|
}
|
|
public static class Bar implements Intf { // Loaded from Boot class loader.
|
|
public String get() {
|
|
return "buzz";
|
|
}
|
|
}
|
|
public static class Foo implements Intf { // Loaded from AppClassLoader
|
|
public String get() {
|
|
return "buzz";
|
|
}
|
|
}
|
|
|
|
static int numTests = 0;
|
|
static int failed = 0;
|
|
static Instrumentation instrumentation;
|
|
|
|
public static void main(String args[]) throws Throwable {
|
|
if (!wb.areSharedStringsMapped()) {
|
|
System.out.println("Shared strings are ignored.");
|
|
return;
|
|
}
|
|
|
|
File bootJar = new File(args[0]);
|
|
File appJar = new File(args[1]);
|
|
|
|
instrumentation = InstrumentationRegisterClassFileTransformer.getInstrumentation();
|
|
System.out.println("INFO: instrumentation = " + instrumentation);
|
|
|
|
testBootstrapCDS("Bootstrap Loader", bootJar);
|
|
testAppCDSv1("Application Loader", appJar);
|
|
|
|
if (failed > 0) {
|
|
throw new RuntimeException("FINAL RESULT: " + failed + " out of " + numTests + " test case(s) have failed");
|
|
} else {
|
|
System.out.println("FINAL RESULT: All " + numTests + " test case(s) have passed!");
|
|
}
|
|
|
|
// Full GC. The cached objects in adjustable archive heap regions are
|
|
// scanned. The archive regions are verified. No error should be
|
|
// reported.
|
|
wb.fullGC();
|
|
}
|
|
|
|
static void testBootstrapCDS(String group, File jar) throws Throwable {
|
|
doTest(group, new Bar(), jar);
|
|
}
|
|
|
|
static void testAppCDSv1(String group, File jar) throws Throwable {
|
|
doTest(group, new Foo(), jar);
|
|
}
|
|
|
|
static void checkArchivedMirrorObject(Class klass) {
|
|
if (wb.areOpenArchiveHeapObjectsMapped()) {
|
|
if (!wb.isShared(klass)) {
|
|
failed ++;
|
|
System.out.println("FAILED. " + klass + " mirror object is not archived");
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void doTest(String group, Intf object, File jar) throws Throwable {
|
|
numTests ++;
|
|
|
|
Class klass = object.getClass();
|
|
System.out.println();
|
|
System.out.println("++++++++++++++++++++++++++");
|
|
System.out.println("Test group: " + group);
|
|
System.out.println("Testing with classloader = " + klass.getClassLoader());
|
|
System.out.println("Testing with class = " + klass);
|
|
System.out.println("Test is shared = " + wb.isSharedClass(klass));
|
|
System.out.println("++++++++++++++++++++++++++");
|
|
|
|
// Check archived mirror object before redefine
|
|
checkArchivedMirrorObject(klass);
|
|
|
|
// Call get() before redefine. All strings in archived classes are shared.
|
|
String res = object.get();
|
|
System.out.println("get() returns " + res);
|
|
if (res.equals("buzz") && wb.isShared(res)) {
|
|
System.out.println("get() returns " + res + ", string is shared");
|
|
} else {
|
|
if (!res.equals("buzz")) {
|
|
System.out.println("FAILED. buzz is expected but got " + res);
|
|
} else {
|
|
System.out.println("FAILED. " + res + " is not shared");
|
|
}
|
|
failed ++;
|
|
return;
|
|
}
|
|
res = null; // release the local reference to the string
|
|
|
|
// Run GC
|
|
System.gc();
|
|
System.gc();
|
|
System.gc();
|
|
|
|
// Redefine the shared class
|
|
byte[] buff = Util.getClassFileFromJar(jar, klass.getName());
|
|
Util.replace(buff, "buzz", "huzz");
|
|
String f = "(failed)";
|
|
try {
|
|
instrumentation.redefineClasses(new ClassDefinition(klass, buff));
|
|
f = object.get();
|
|
} catch (UnmodifiableClassException|UnsupportedOperationException e) {
|
|
e.printStackTrace();
|
|
}
|
|
if (f.equals("huzz")) {
|
|
System.out.println("PASSED: object.get() after redefinition returns " + f);
|
|
} else {
|
|
System.out.println("FAILED: object.get() after redefinition returns " + f);
|
|
failed ++;
|
|
}
|
|
|
|
// Run GC. Should not crash.
|
|
System.gc();
|
|
System.gc();
|
|
System.gc();
|
|
|
|
// Check archived mirror object after redefine and GC
|
|
checkArchivedMirrorObject(klass);
|
|
|
|
System.out.println("++++++++++++++++++++++++++++++++++++++++++++++++ (done)\n\n");
|
|
}
|
|
}
|