Skip to content

Commit 980d187

Browse files
author
Mandy Chung
committedFeb 16, 2022
8281335: Allow a library already loaded via System::loadLibrary to be loaded as a raw library
Reviewed-by: sundar, mcimadamore
1 parent 8164552 commit 980d187

File tree

7 files changed

+215
-98
lines changed

7 files changed

+215
-98
lines changed
 

‎src/java.base/share/classes/java/lang/ClassLoader.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2013, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2013, 2022, Oracle and/or its affiliates. All rights reserved.
33
* Copyright (c) 2019, Azul Systems, Inc. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
@@ -2385,7 +2385,7 @@ protected String findLibrary(String libname) {
23852385
return null;
23862386
}
23872387

2388-
private final NativeLibraries libraries = NativeLibraries.jniNativeLibraries(this);
2388+
private final NativeLibraries libraries = NativeLibraries.newInstance(this);
23892389

23902390
// Invoked in the java.lang.Runtime class to implement load and loadLibrary.
23912391
static NativeLibrary loadLibrary(Class<?> fromClass, File file) {

‎src/java.base/share/classes/jdk/internal/loader/BootLoader.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -74,7 +74,7 @@ private BootLoader() { }
7474

7575
// native libraries loaded by the boot class loader
7676
private static final NativeLibraries NATIVE_LIBS
77-
= NativeLibraries.jniNativeLibraries(null);
77+
= NativeLibraries.newInstance(null);
7878

7979
/**
8080
* Returns the unnamed module for the boot loader.

‎src/java.base/share/classes/jdk/internal/loader/NativeLibraries.java

+19-71
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -36,7 +36,6 @@
3636
import java.util.Deque;
3737
import java.util.function.BiFunction;
3838
import java.util.function.Function;
39-
import java.util.Objects;
4039
import java.util.Map;
4140
import java.util.Set;
4241
import java.util.concurrent.ConcurrentHashMap;
@@ -62,8 +61,6 @@ public final class NativeLibraries {
6261
// unless specified
6362
private final Class<?> caller; // may be null
6463
private final boolean searchJavaLibraryPath;
65-
// loading JNI native libraries
66-
private final boolean isJNI;
6764

6865
/**
6966
* Creates a NativeLibraries instance for loading JNI native libraries
@@ -81,48 +78,14 @@ public final class NativeLibraries {
8178
* @see <a href="${docroot}/specs/jni/invocation.html##library-and-version-management">
8279
* JNI Specification: Library and Version Management</a>
8380
*/
84-
public static NativeLibraries jniNativeLibraries(ClassLoader loader) {
85-
return new NativeLibraries(loader);
81+
public static NativeLibraries newInstance(ClassLoader loader) {
82+
return new NativeLibraries(loader, loader != null ? null : NativeLibraries.class, loader != null);
8683
}
8784

88-
/**
89-
* Creates a raw NativeLibraries instance that has the following properties:
90-
* 1. Native libraries loaded in this raw NativeLibraries instance are
91-
* not JNI native libraries. Hence JNI_OnLoad and JNI_OnUnload will
92-
* be ignored. No support for linking of native method.
93-
* 2. Native libraries not auto-unloaded. They may be explicitly unloaded
94-
* via NativeLibraries::unload.
95-
* 3. No relationship with class loaders.
96-
*
97-
* This static factory method is restricted for JDK trusted class use.
98-
*/
99-
public static NativeLibraries rawNativeLibraries(Class<?> trustedCaller,
100-
boolean searchJavaLibraryPath) {
101-
return new NativeLibraries(trustedCaller, searchJavaLibraryPath);
102-
}
103-
104-
private NativeLibraries(ClassLoader loader) {
105-
// for null loader, default the caller to this class and
106-
// do not search java.library.path
85+
private NativeLibraries(ClassLoader loader, Class<?> caller, boolean searchJavaLibraryPath) {
10786
this.loader = loader;
108-
this.caller = loader != null ? null : NativeLibraries.class;
109-
this.searchJavaLibraryPath = loader != null ? true : false;
110-
this.isJNI = true;
111-
}
112-
113-
/*
114-
* Constructs a NativeLibraries instance of no relationship with class loaders
115-
* and disabled auto unloading.
116-
*/
117-
private NativeLibraries(Class<?> caller, boolean searchJavaLibraryPath) {
118-
Objects.requireNonNull(caller);
119-
if (!VM.isSystemDomainLoader(caller.getClassLoader())) {
120-
throw new IllegalArgumentException("must be JDK trusted class");
121-
}
122-
this.loader = caller.getClassLoader();
12387
this.caller = caller;
12488
this.searchJavaLibraryPath = searchJavaLibraryPath;
125-
this.isJNI = false;
12689
}
12790

12891
/*
@@ -227,7 +190,7 @@ private NativeLibrary loadLibrary(Class<?> fromClass, String name, boolean isBui
227190
}
228191
}
229192

230-
NativeLibraryImpl lib = new NativeLibraryImpl(fromClass, name, isBuiltin, isJNI);
193+
NativeLibraryImpl lib = new NativeLibraryImpl(fromClass, name, isBuiltin, true);
231194
// load the native library
232195
NativeLibraryContext.push(lib);
233196
try {
@@ -237,8 +200,7 @@ private NativeLibrary loadLibrary(Class<?> fromClass, String name, boolean isBui
237200
// auto unloading is only supported for JNI native libraries
238201
// loaded by custom class loaders that can be unloaded.
239202
// built-in class loaders are never unloaded.
240-
boolean autoUnload = isJNI && !VM.isSystemDomainLoader(loader)
241-
&& loader != ClassLoaders.appClassLoader();
203+
boolean autoUnload = !VM.isSystemDomainLoader(loader) && loader != ClassLoaders.appClassLoader();
242204
if (autoUnload) {
243205
// register the loaded native library for auto unloading
244206
// when the class loader is reclaimed, all native libraries
@@ -269,8 +231,6 @@ private NativeLibrary loadLibrary(Class<?> fromClass, String name, boolean isBui
269231
*/
270232
public NativeLibrary loadLibrary(String name) {
271233
assert name.indexOf(File.separatorChar) < 0;
272-
assert caller != null;
273-
274234
return loadLibrary(caller, name);
275235
}
276236

@@ -293,29 +253,6 @@ public NativeLibrary loadLibrary(Class<?> fromClass, String name) {
293253
return lib;
294254
}
295255

296-
/**
297-
* Unloads the given native library
298-
*
299-
* @param lib native library
300-
*/
301-
public void unload(NativeLibrary lib) {
302-
if (isJNI) {
303-
throw new UnsupportedOperationException("explicit unloading cannot be used with auto unloading");
304-
}
305-
Objects.requireNonNull(lib);
306-
acquireNativeLibraryLock(lib.name());
307-
try {
308-
NativeLibraryImpl nl = libraries.remove(lib.name());
309-
if (nl != lib) {
310-
throw new IllegalArgumentException(lib.name() + " not loaded by this NativeLibraries instance");
311-
}
312-
// unload the native library and also remove from the global name registry
313-
nl.unloader().run();
314-
} finally {
315-
releaseNativeLibraryLock(lib.name());
316-
}
317-
}
318-
319256
private NativeLibrary findFromPaths(String[] paths, Class<?> fromClass, String name) {
320257
for (String path : paths) {
321258
File libfile = new File(path, System.mapLibraryName(name));
@@ -380,7 +317,11 @@ public long find(String name) {
380317
return findEntry0(this, name);
381318
}
382319

383-
Runnable unloader() {
320+
/*
321+
* Unloader::run method is invoked to unload the native library
322+
* when this class loader becomes phantom reachable.
323+
*/
324+
private Runnable unloader() {
384325
return new Unloader(name, handle, isBuiltin, isJNI);
385326
}
386327

@@ -394,6 +335,13 @@ boolean open() {
394335

395336
return load(this, name, isBuiltin, isJNI, loadLibraryOnlyIfPresent);
396337
}
338+
339+
/*
340+
* Close this native library.
341+
*/
342+
void close() {
343+
unload(name, isBuiltin, isJNI, handle);
344+
}
397345
}
398346

399347
/*
@@ -429,7 +377,7 @@ public void run() {
429377
acquireNativeLibraryLock(name);
430378
try {
431379
/* remove the native library name */
432-
if (!loadedLibraryNames.remove(name)) {
380+
if (isJNI && !loadedLibraryNames.remove(name)) {
433381
throw new IllegalStateException(name + " has already been unloaded");
434382
}
435383
NativeLibraryContext.push(UNLOADER);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,139 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation. Oracle designates this
8+
* particular file as subject to the "Classpath" exception as provided
9+
* by Oracle in the LICENSE file that accompanied this code.
10+
*
11+
* This code is distributed in the hope that it will be useful, but WITHOUT
12+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14+
* version 2 for more details (a copy is included in the LICENSE file that
15+
* accompanied this code).
16+
*
17+
* You should have received a copy of the GNU General Public License version
18+
* 2 along with this work; if not, write to the Free Software Foundation,
19+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20+
*
21+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22+
* or visit www.oracle.com if you need additional information or have any
23+
* questions.
24+
*/
25+
package jdk.internal.loader;
26+
27+
import jdk.internal.misc.VM;
28+
29+
import java.io.IOException;
30+
import java.lang.invoke.MethodHandles;
31+
import java.nio.file.Path;
32+
import java.security.AccessController;
33+
import java.security.PrivilegedAction;
34+
import java.util.Map;
35+
import java.util.Objects;
36+
import java.util.concurrent.ConcurrentHashMap;
37+
38+
import static jdk.internal.loader.NativeLibraries.*;
39+
40+
41+
/**
42+
* RawNativeLibraries has the following properties:
43+
* 1. Native libraries loaded in this RawNativeLibraries instance are
44+
* not JNI native libraries. Hence JNI_OnLoad and JNI_OnUnload will
45+
* be ignored. No support for linking of native method.
46+
* 2. Native libraries not auto-unloaded. They may be explicitly unloaded
47+
* via NativeLibraries::unload.
48+
* 3. No relationship with class loaders.
49+
*/
50+
public final class RawNativeLibraries {
51+
final Map<String, NativeLibraryImpl> libraries = new ConcurrentHashMap<>();
52+
final Class<?> caller;
53+
54+
private RawNativeLibraries(MethodHandles.Lookup trustedCaller) {
55+
this.caller = trustedCaller.lookupClass();
56+
}
57+
58+
/**
59+
* Creates a RawNativeLibraries instance that has no relationship with
60+
* any class loaders and disabled auto unloading.
61+
*
62+
* This static factory method is restricted for JDK trusted class use.
63+
*/
64+
public static RawNativeLibraries newInstance(MethodHandles.Lookup trustedCaller) {
65+
if (!trustedCaller.hasFullPrivilegeAccess() ||
66+
!VM.isSystemDomainLoader(trustedCaller.lookupClass().getClassLoader())) {
67+
throw new InternalError(trustedCaller + " does not have access to raw native library loading");
68+
}
69+
return new RawNativeLibraries(trustedCaller);
70+
}
71+
72+
/*
73+
* Load a native library from the given path. Returns null if the given
74+
* library is determined to be non-loadable, which is system-dependent.
75+
*
76+
* @param path the path of the native library
77+
*/
78+
@SuppressWarnings("removal")
79+
public NativeLibrary load(Path path) {
80+
String name = AccessController.doPrivileged(new PrivilegedAction<>() {
81+
public String run() {
82+
try {
83+
return path.toRealPath().toString();
84+
} catch (IOException e) {
85+
return null;
86+
}
87+
}
88+
});
89+
if (name == null) {
90+
return null;
91+
}
92+
return load(name);
93+
}
94+
95+
/**
96+
* Load a native library of the given pathname, which is platform-specific.
97+
* Returns null if it fails to load the given pathname.
98+
*
99+
* If the given pathname does not contain a name-separator character,
100+
* for example on Unix a slash character, the library search strategy
101+
* is system-dependent for example on Unix, see dlopen.
102+
*
103+
* @apiNote
104+
* The {@code pathname} argument is platform-specific.
105+
* {@link System#mapLibraryName} can be used to convert a name to
106+
* a platform-specific pathname:
107+
* {@snippet
108+
* RawNativeLibraries libs = RawNativeLibraries.newInstance(MethodHandles.lookup());
109+
* NativeLibrary lib = libs.load(System.mapLibraryName("blas"));
110+
* }
111+
*
112+
* @param pathname the pathname of the native library
113+
* @see System#mapLibraryName(String)
114+
*/
115+
public NativeLibrary load(String pathname) {
116+
return libraries.computeIfAbsent(pathname, this::get);
117+
}
118+
119+
private NativeLibraryImpl get(String pathname) {
120+
NativeLibraryImpl lib = new NativeLibraryImpl(caller, pathname, false, false);
121+
if (!lib.open()) {
122+
return null; // fail to open the native library
123+
}
124+
return lib;
125+
}
126+
127+
/*
128+
* Unloads the given native library.
129+
*/
130+
public void unload(NativeLibrary lib) {
131+
Objects.requireNonNull(lib);
132+
if (!libraries.remove(lib.name(), lib)) {
133+
throw new IllegalArgumentException(lib.name() + " not loaded by this RawNativeLibraries instance");
134+
}
135+
NativeLibraryImpl nl = (NativeLibraryImpl)lib;
136+
nl.close();
137+
}
138+
}
139+

0 commit comments

Comments
 (0)
Please sign in to comment.