c8cc7b67db
Reviewed-by: rriggs
244 lines
9.0 KiB
Java
244 lines
9.0 KiB
Java
/*
|
|
* Copyright (c) 2020, 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
|
|
* @summary verify if the hidden class is unloaded when the class loader is GC'ed
|
|
* @modules jdk.compiler
|
|
* @library /test/lib/
|
|
* @build jdk.test.lib.util.ForceGC
|
|
* @run testng/othervm UnloadingTest
|
|
*/
|
|
|
|
import java.io.IOException;
|
|
import java.lang.invoke.MethodHandles.Lookup;
|
|
import java.lang.ref.Reference;
|
|
import java.lang.ref.WeakReference;
|
|
import java.lang.reflect.Method;
|
|
import java.net.MalformedURLException;
|
|
import java.net.URL;
|
|
import java.net.URLClassLoader;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.nio.file.Paths;
|
|
import java.util.concurrent.atomic.AtomicInteger;
|
|
|
|
import jdk.test.lib.util.ForceGC;
|
|
|
|
import jdk.test.lib.compiler.CompilerUtils;
|
|
import jdk.test.lib.Utils;
|
|
|
|
import org.testng.annotations.BeforeTest;
|
|
import org.testng.annotations.Test;
|
|
|
|
import static java.lang.invoke.MethodHandles.lookup;
|
|
import static java.lang.invoke.MethodHandles.Lookup.ClassOption.*;
|
|
import static org.testng.Assert.*;
|
|
|
|
public class UnloadingTest {
|
|
private static final Path CLASSES_DIR = Paths.get("classes");
|
|
private static byte[] hiddenClassBytes;
|
|
|
|
@BeforeTest
|
|
static void setup() throws IOException {
|
|
Path src = Paths.get(Utils.TEST_SRC, "src", "LookupHelper.java");
|
|
if (!CompilerUtils.compile(src, CLASSES_DIR)) {
|
|
throw new RuntimeException("Compilation of the test failed: " + src);
|
|
}
|
|
|
|
hiddenClassBytes = Files.readAllBytes(CLASSES_DIR.resolve("LookupHelper.class"));
|
|
}
|
|
|
|
/*
|
|
* Test that a hidden class is unloaded while the loader remains strongly reachable
|
|
*/
|
|
@Test
|
|
public void unloadable() throws Exception {
|
|
TestLoader loader = new TestLoader();
|
|
Class<?> helper = Class.forName("LookupHelper", true, loader);
|
|
Method m = helper.getMethod("getLookup");
|
|
Lookup lookup = (Lookup)m.invoke(null);
|
|
HiddenClassUnloader unloader = createHiddenClass(lookup, false);
|
|
// the hidden class should be unloaded
|
|
unloader.unload();
|
|
|
|
// loader is strongly reachable
|
|
Reference.reachabilityFence(loader);
|
|
}
|
|
|
|
/*
|
|
* Test that a hidden class is not unloaded when the loader is strongly reachable
|
|
*/
|
|
@Test
|
|
public void notUnloadable() throws Exception {
|
|
TestLoader loader = new TestLoader();
|
|
Class<?> helper = Class.forName("LookupHelper", true, loader);
|
|
Method m = helper.getMethod("getLookup");
|
|
Lookup lookup = (Lookup)m.invoke(null);
|
|
HiddenClassUnloader unloader = createHiddenClass(lookup, true);
|
|
assertFalse(unloader.tryUnload()); // hidden class is not unloaded
|
|
|
|
// loader is strongly reachable
|
|
Reference.reachabilityFence(loader);
|
|
}
|
|
|
|
/*
|
|
* Create a nest of two hidden classes.
|
|
* They can be unloaded even the loader is strongly reachable
|
|
*/
|
|
@Test
|
|
public void hiddenClassNest() throws Exception {
|
|
TestLoader loader = new TestLoader();
|
|
Class<?> helper = Class.forName("LookupHelper", true, loader);
|
|
Method m = helper.getMethod("getLookup");
|
|
Lookup lookup = (Lookup)m.invoke(null);
|
|
HiddenClassUnloader[] unloaders = createNestOfTwoHiddenClasses(lookup, false, false);
|
|
|
|
// keep a strong reference to the nest member class
|
|
Class<?> member = unloaders[1].weakRef.get();
|
|
assertTrue(member != null);
|
|
// nest host and member will not be unloaded
|
|
assertFalse(unloaders[0].tryUnload());
|
|
assertFalse(unloaders[1].tryUnload());
|
|
|
|
// clear the reference to the nest member
|
|
Reference.reachabilityFence(member);
|
|
member = null;
|
|
|
|
// nest host and member will be unloaded
|
|
unloaders[0].unload();
|
|
unloaders[1].unload();
|
|
|
|
// loader is strongly reachable
|
|
Reference.reachabilityFence(loader);
|
|
}
|
|
|
|
/*
|
|
* Create a nest with a hidden class nest host and strong nest member.
|
|
* Test that both are not unloaded
|
|
*/
|
|
@Test
|
|
public void hiddenClassNestStrongMember() throws Exception {
|
|
TestLoader loader = new TestLoader();
|
|
Class<?> helper = Class.forName("LookupHelper", true, loader);
|
|
Method m = helper.getMethod("getLookup");
|
|
Lookup lookup = (Lookup)m.invoke(null);
|
|
HiddenClassUnloader[] unloaders = createNestOfTwoHiddenClasses(lookup, false, true);
|
|
assertFalse(unloaders[0].tryUnload()); // nest host cannot be unloaded
|
|
assertFalse(unloaders[1].tryUnload()); // nest member cannot be unloaded
|
|
|
|
// loader is strongly reachable
|
|
Reference.reachabilityFence(loader);
|
|
}
|
|
|
|
/*
|
|
* Create a nest with a strong hidden nest host and a hidden class member.
|
|
* The nest member can be unloaded whereas the nest host will not be unloaded.
|
|
*/
|
|
@Test
|
|
public void hiddenClassNestStrongHost() throws Exception {
|
|
TestLoader loader = new TestLoader();
|
|
Class<?> helper = Class.forName("LookupHelper", true, loader);
|
|
Method m = helper.getMethod("getLookup");
|
|
Lookup lookup = (Lookup)m.invoke(null);
|
|
HiddenClassUnloader[] unloaders = createNestOfTwoHiddenClasses(lookup, true, false);
|
|
assertFalse(unloaders[0].tryUnload()); // nest host cannot be unloaded
|
|
unloaders[1].unload();
|
|
|
|
// loader is strongly reachable
|
|
Reference.reachabilityFence(loader);
|
|
}
|
|
|
|
/*
|
|
* Create a HiddenClassUnloader that holds a weak reference to the newly created
|
|
* hidden class.
|
|
*/
|
|
static HiddenClassUnloader createHiddenClass(Lookup lookup, boolean strong) throws Exception {
|
|
Class<?> hc;
|
|
if (strong) {
|
|
hc = lookup.defineHiddenClass(hiddenClassBytes, false, STRONG).lookupClass();
|
|
} else {
|
|
hc = lookup.defineHiddenClass(hiddenClassBytes, false).lookupClass();
|
|
}
|
|
assertTrue(hc.getClassLoader() == lookup.lookupClass().getClassLoader());
|
|
return new HiddenClassUnloader(hc);
|
|
}
|
|
|
|
/*
|
|
* Create an array of HiddenClassUnloader with two elements: the first element
|
|
* is for the nest host and the second element is for the nest member.
|
|
*/
|
|
static HiddenClassUnloader[] createNestOfTwoHiddenClasses(Lookup lookup, boolean strongHost, boolean strongMember) throws Exception {
|
|
Lookup hostLookup;
|
|
if (strongHost) {
|
|
hostLookup = lookup.defineHiddenClass(hiddenClassBytes, false, STRONG);
|
|
} else {
|
|
hostLookup = lookup.defineHiddenClass(hiddenClassBytes, false);
|
|
}
|
|
Class<?> host = hostLookup.lookupClass();
|
|
Class<?> member;
|
|
if (strongMember) {
|
|
member = hostLookup.defineHiddenClass(hiddenClassBytes, false, NESTMATE, STRONG).lookupClass();
|
|
} else {
|
|
member = hostLookup.defineHiddenClass(hiddenClassBytes, false, NESTMATE).lookupClass();
|
|
}
|
|
assertTrue(member.getNestHost() == host);
|
|
return new HiddenClassUnloader[] { new HiddenClassUnloader(host), new HiddenClassUnloader(member) };
|
|
}
|
|
|
|
static class HiddenClassUnloader {
|
|
private final WeakReference<Class<?>> weakRef;
|
|
private HiddenClassUnloader(Class<?> hc) {
|
|
assertTrue(hc.isHidden());
|
|
this.weakRef = new WeakReference<>(hc);
|
|
}
|
|
|
|
void unload() {
|
|
// Force garbage collection to trigger unloading of class loader
|
|
// and native library.
|
|
if (!ForceGC.wait(() -> weakRef.refersTo(null))) {
|
|
throw new RuntimeException("loader " + " not unloaded!");
|
|
}
|
|
}
|
|
|
|
boolean tryUnload() {
|
|
return ForceGC.waitFor(() -> weakRef.refersTo(null), 2000L);
|
|
}
|
|
}
|
|
|
|
static class TestLoader extends URLClassLoader {
|
|
static URL[] toURLs() {
|
|
try {
|
|
return new URL[] { CLASSES_DIR.toUri().toURL() };
|
|
} catch (MalformedURLException e) {
|
|
throw new Error(e);
|
|
}
|
|
}
|
|
|
|
static AtomicInteger counter = new AtomicInteger();
|
|
TestLoader() {
|
|
super("testloader-" + counter.addAndGet(1), toURLs(), ClassLoader.getSystemClassLoader());
|
|
}
|
|
}
|
|
}
|