/* * 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> 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()); } } }