2017-11-06 17:48:00 -08:00
|
|
|
/*
|
2022-07-06 14:23:44 +00:00
|
|
|
* Copyright (c) 2017, 2022, Oracle and/or its affiliates. All rights reserved.
|
2017-11-06 17:48:00 -08:00
|
|
|
* 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
|
2017-12-05 08:27:54 +01:00
|
|
|
* @bug 8164512 8191360
|
2018-02-05 07:20:26 +01:00
|
|
|
* @requires vm.compMode != "Xcomp"
|
2020-10-13 09:35:58 +00:00
|
|
|
* @comment Under musl, dlclose is a no-op. The static variable 'count' in libnative.c
|
|
|
|
* keeps its value across a GC and the check in Test.java fails.
|
|
|
|
* @requires !vm.musl
|
2017-11-06 17:48:00 -08:00
|
|
|
* @summary verify if the native library is unloaded when the class loader is GC'ed
|
2020-12-08 18:01:04 +00:00
|
|
|
* @library /test/lib/
|
|
|
|
* @build jdk.test.lib.util.ForceGC
|
2017-11-06 17:48:00 -08:00
|
|
|
* @build p.Test
|
|
|
|
* @run main/othervm/native -Xcheck:jni NativeLibraryTest
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
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;
|
|
|
|
|
2020-12-08 18:01:04 +00:00
|
|
|
import jdk.test.lib.util.ForceGC;
|
|
|
|
|
2017-11-06 17:48:00 -08:00
|
|
|
public class NativeLibraryTest {
|
|
|
|
static final Path CLASSES = Paths.get("classes");
|
|
|
|
static int unloadedCount = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Called by JNI_OnUnload when the native library is unloaded
|
|
|
|
*/
|
|
|
|
static void nativeLibraryUnloaded() {
|
|
|
|
unloadedCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String... args) throws Exception {
|
|
|
|
setup();
|
|
|
|
|
|
|
|
for (int count=1; count <= 5; count++) {
|
2020-12-19 00:17:02 +00:00
|
|
|
System.out.println("count: " + count);
|
2017-11-06 17:48:00 -08:00
|
|
|
// create a class loader and load a native library
|
|
|
|
runTest();
|
2020-12-08 18:01:04 +00:00
|
|
|
// Unload the class loader and native library, and give the Cleaner
|
|
|
|
// thread a chance to unload the native library.
|
|
|
|
// unloadedCount is incremented when the native library is unloaded.
|
|
|
|
final int finalCount = count;
|
2022-07-06 14:23:44 +00:00
|
|
|
if (!ForceGC.wait(() -> finalCount == unloadedCount)) {
|
2017-11-06 17:48:00 -08:00
|
|
|
throw new RuntimeException("Expected unloaded=" + count +
|
|
|
|
" but got=" + unloadedCount);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Loads p.Test class with a new class loader and its static initializer
|
|
|
|
* will load a native library.
|
|
|
|
*
|
|
|
|
* The class loader becomes unreachable when this method returns and
|
|
|
|
* the native library should be unloaded at some point after the class
|
|
|
|
* loader is garbage collected.
|
|
|
|
*/
|
|
|
|
static void runTest() throws Exception {
|
|
|
|
// invoke p.Test.run() that loads the native library
|
|
|
|
Runnable r = newTestRunnable();
|
|
|
|
r.run();
|
|
|
|
|
|
|
|
// reload the native library by the same class loader
|
|
|
|
r.run();
|
|
|
|
|
|
|
|
// load the native library by another class loader
|
|
|
|
Runnable r1 = newTestRunnable();
|
|
|
|
try {
|
|
|
|
r1.run();
|
|
|
|
throw new RuntimeException("should fail to load the native library" +
|
|
|
|
" by another class loader");
|
|
|
|
} catch (UnsatisfiedLinkError e) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Loads p.Test class with a new class loader and returns
|
|
|
|
* a Runnable instance.
|
|
|
|
*/
|
|
|
|
static Runnable newTestRunnable() throws Exception {
|
|
|
|
TestLoader loader = new TestLoader();
|
|
|
|
Class<?> c = Class.forName("p.Test", true, loader);
|
|
|
|
return (Runnable) c.newInstance();
|
|
|
|
}
|
|
|
|
|
|
|
|
static class TestLoader extends URLClassLoader {
|
|
|
|
static URL[] toURLs() {
|
|
|
|
try {
|
|
|
|
return new URL[] { CLASSES.toUri().toURL() };
|
|
|
|
} catch (MalformedURLException e) {
|
|
|
|
throw new Error(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TestLoader() {
|
|
|
|
super("testloader", toURLs(), ClassLoader.getSystemClassLoader());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* move p/Test.class out from classpath to the scratch directory
|
|
|
|
*/
|
|
|
|
static void setup() throws IOException {
|
|
|
|
String dir = System.getProperty("test.classes", ".");
|
|
|
|
Path file = Paths.get("p", "Test.class");
|
|
|
|
Files.createDirectories(CLASSES.resolve("p"));
|
|
|
|
Files.move(Paths.get(dir).resolve(file),
|
|
|
|
CLASSES.resolve("p").resolve("Test.class"));
|
|
|
|
}
|
|
|
|
}
|