7cc1371059
Co-authored-by: Lois Foltan <lois.foltan@oracle.com> Co-authored-by: David Holmes <david.holmes@oracle.com> Co-authored-by: Harold Seigel <harold.seigel@oracle.com> Co-authored-by: Serguei Spitsyn <serguei.spitsyn@oracle.com> Co-authored-by: Alex Buckley <alex.buckley@oracle.com> Co-authored-by: Jamsheed Mohammed C M <jamsheed.c.m@oracle.com> Co-authored-by: Jan Lahoda <jan.lahoda@oracle.com> Co-authored-by: Amy Lu <amy.lu@oracle.com> Reviewed-by: alanb, cjplummer, coleenp, dholmes, dlong, forax, jlahoda, psandoz, plevart, sspitsyn, vromero
247 lines
9.1 KiB
Java
247 lines
9.1 KiB
Java
/*
|
|
* Copyright (c) 2020, 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
|
|
ForceGC gc = new ForceGC();
|
|
assertTrue(gc.await(() -> weakRef.get() == null));
|
|
|
|
if (weakRef.get() != null) {
|
|
throw new RuntimeException("loader " + " not unloaded!");
|
|
}
|
|
}
|
|
|
|
boolean tryUnload() {
|
|
ForceGC gc = new ForceGC();
|
|
return gc.await(() -> weakRef.get() == null);
|
|
}
|
|
}
|
|
|
|
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());
|
|
}
|
|
}
|
|
}
|