1
1
/*
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.
3
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
4
*
5
5
* This code is free software; you can redistribute it and/or modify it
36
36
import java .util .Deque ;
37
37
import java .util .function .BiFunction ;
38
38
import java .util .function .Function ;
39
- import java .util .Objects ;
40
39
import java .util .Map ;
41
40
import java .util .Set ;
42
41
import java .util .concurrent .ConcurrentHashMap ;
@@ -62,8 +61,6 @@ public final class NativeLibraries {
62
61
// unless specified
63
62
private final Class <?> caller ; // may be null
64
63
private final boolean searchJavaLibraryPath ;
65
- // loading JNI native libraries
66
- private final boolean isJNI ;
67
64
68
65
/**
69
66
* Creates a NativeLibraries instance for loading JNI native libraries
@@ -81,48 +78,14 @@ public final class NativeLibraries {
81
78
* @see <a href="${docroot}/specs/jni/invocation.html##library-and-version-management">
82
79
* JNI Specification: Library and Version Management</a>
83
80
*/
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 );
86
83
}
87
84
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 ) {
107
86
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 ();
123
87
this .caller = caller ;
124
88
this .searchJavaLibraryPath = searchJavaLibraryPath ;
125
- this .isJNI = false ;
126
89
}
127
90
128
91
/*
@@ -227,7 +190,7 @@ private NativeLibrary loadLibrary(Class<?> fromClass, String name, boolean isBui
227
190
}
228
191
}
229
192
230
- NativeLibraryImpl lib = new NativeLibraryImpl (fromClass , name , isBuiltin , isJNI );
193
+ NativeLibraryImpl lib = new NativeLibraryImpl (fromClass , name , isBuiltin , true );
231
194
// load the native library
232
195
NativeLibraryContext .push (lib );
233
196
try {
@@ -237,8 +200,7 @@ private NativeLibrary loadLibrary(Class<?> fromClass, String name, boolean isBui
237
200
// auto unloading is only supported for JNI native libraries
238
201
// loaded by custom class loaders that can be unloaded.
239
202
// 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 ();
242
204
if (autoUnload ) {
243
205
// register the loaded native library for auto unloading
244
206
// when the class loader is reclaimed, all native libraries
@@ -269,8 +231,6 @@ private NativeLibrary loadLibrary(Class<?> fromClass, String name, boolean isBui
269
231
*/
270
232
public NativeLibrary loadLibrary (String name ) {
271
233
assert name .indexOf (File .separatorChar ) < 0 ;
272
- assert caller != null ;
273
-
274
234
return loadLibrary (caller , name );
275
235
}
276
236
@@ -293,29 +253,6 @@ public NativeLibrary loadLibrary(Class<?> fromClass, String name) {
293
253
return lib ;
294
254
}
295
255
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
-
319
256
private NativeLibrary findFromPaths (String [] paths , Class <?> fromClass , String name ) {
320
257
for (String path : paths ) {
321
258
File libfile = new File (path , System .mapLibraryName (name ));
@@ -380,7 +317,11 @@ public long find(String name) {
380
317
return findEntry0 (this , name );
381
318
}
382
319
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 () {
384
325
return new Unloader (name , handle , isBuiltin , isJNI );
385
326
}
386
327
@@ -394,6 +335,13 @@ boolean open() {
394
335
395
336
return load (this , name , isBuiltin , isJNI , loadLibraryOnlyIfPresent );
396
337
}
338
+
339
+ /*
340
+ * Close this native library.
341
+ */
342
+ void close () {
343
+ unload (name , isBuiltin , isJNI , handle );
344
+ }
397
345
}
398
346
399
347
/*
@@ -429,7 +377,7 @@ public void run() {
429
377
acquireNativeLibraryLock (name );
430
378
try {
431
379
/* remove the native library name */
432
- if (!loadedLibraryNames .remove (name )) {
380
+ if (isJNI && !loadedLibraryNames .remove (name )) {
433
381
throw new IllegalStateException (name + " has already been unloaded" );
434
382
}
435
383
NativeLibraryContext .push (UNLOADER );
0 commit comments