File: | jdk/src/hotspot/share/classfile/classLoader.cpp |
Warning: | line 1187, column 7 Value stored to 'e' is never read |
Press '?' to see keyboard shortcuts
Keyboard shortcuts:
1 | /* |
2 | * Copyright (c) 1997, 2021, 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. |
8 | * |
9 | * This code is distributed in the hope that it will be useful, but WITHOUT |
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
12 | * version 2 for more details (a copy is included in the LICENSE file that |
13 | * accompanied this code). |
14 | * |
15 | * You should have received a copy of the GNU General Public License version |
16 | * 2 along with this work; if not, write to the Free Software Foundation, |
17 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
18 | * |
19 | * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
20 | * or visit www.oracle.com if you need additional information or have any |
21 | * questions. |
22 | * |
23 | */ |
24 | |
25 | #include "precompiled.hpp" |
26 | #include "jvm.h" |
27 | #include "jimage.hpp" |
28 | #include "cds/filemap.hpp" |
29 | #include "classfile/classFileStream.hpp" |
30 | #include "classfile/classLoader.inline.hpp" |
31 | #include "classfile/classLoaderData.inline.hpp" |
32 | #include "classfile/classLoaderExt.hpp" |
33 | #include "classfile/classLoadInfo.hpp" |
34 | #include "classfile/javaClasses.hpp" |
35 | #include "classfile/moduleEntry.hpp" |
36 | #include "classfile/modules.hpp" |
37 | #include "classfile/packageEntry.hpp" |
38 | #include "classfile/klassFactory.hpp" |
39 | #include "classfile/symbolTable.hpp" |
40 | #include "classfile/systemDictionary.hpp" |
41 | #include "classfile/systemDictionaryShared.hpp" |
42 | #include "classfile/vmClasses.hpp" |
43 | #include "classfile/vmSymbols.hpp" |
44 | #include "compiler/compileBroker.hpp" |
45 | #include "interpreter/bytecodeStream.hpp" |
46 | #include "interpreter/oopMapCache.hpp" |
47 | #include "logging/log.hpp" |
48 | #include "logging/logStream.hpp" |
49 | #include "logging/logTag.hpp" |
50 | #include "memory/allocation.inline.hpp" |
51 | #include "memory/oopFactory.hpp" |
52 | #include "memory/resourceArea.hpp" |
53 | #include "memory/universe.hpp" |
54 | #include "oops/instanceKlass.hpp" |
55 | #include "oops/instanceRefKlass.hpp" |
56 | #include "oops/klass.inline.hpp" |
57 | #include "oops/method.inline.hpp" |
58 | #include "oops/objArrayOop.inline.hpp" |
59 | #include "oops/oop.inline.hpp" |
60 | #include "oops/symbol.hpp" |
61 | #include "prims/jvm_misc.hpp" |
62 | #include "runtime/arguments.hpp" |
63 | #include "runtime/handles.inline.hpp" |
64 | #include "runtime/init.hpp" |
65 | #include "runtime/interfaceSupport.inline.hpp" |
66 | #include "runtime/java.hpp" |
67 | #include "runtime/javaCalls.hpp" |
68 | #include "runtime/os.hpp" |
69 | #include "runtime/perfData.hpp" |
70 | #include "runtime/threadCritical.hpp" |
71 | #include "runtime/timer.hpp" |
72 | #include "runtime/vm_version.hpp" |
73 | #include "services/management.hpp" |
74 | #include "services/threadService.hpp" |
75 | #include "utilities/classpathStream.hpp" |
76 | #include "utilities/events.hpp" |
77 | #include "utilities/hashtable.inline.hpp" |
78 | #include "utilities/macros.hpp" |
79 | #include "utilities/utf8.hpp" |
80 | |
81 | // Entry point in java.dll for path canonicalization |
82 | |
83 | typedef int (*canonicalize_fn_t)(const char *orig, char *out, int len); |
84 | |
85 | static canonicalize_fn_t CanonicalizeEntry = NULL__null; |
86 | |
87 | // Entry points in zip.dll for loading zip/jar file entries |
88 | |
89 | typedef void * * (*ZipOpen_t)(const char *name, char **pmsg); |
90 | typedef void (*ZipClose_t)(jzfile *zip); |
91 | typedef jzentry* (*FindEntry_t)(jzfile *zip, const char *name, jint *sizeP, jint *nameLen); |
92 | typedef jboolean (*ReadEntry_t)(jzfile *zip, jzentry *entry, unsigned char *buf, char *namebuf); |
93 | typedef jzentry* (*GetNextEntry_t)(jzfile *zip, jint n); |
94 | typedef jint (*Crc32_t)(jint crc, const jbyte *buf, jint len); |
95 | |
96 | static ZipOpen_t ZipOpen = NULL__null; |
97 | static ZipClose_t ZipClose = NULL__null; |
98 | static FindEntry_t FindEntry = NULL__null; |
99 | static ReadEntry_t ReadEntry = NULL__null; |
100 | static GetNextEntry_t GetNextEntry = NULL__null; |
101 | static Crc32_t Crc32 = NULL__null; |
102 | int ClassLoader::_libzip_loaded = 0; |
103 | |
104 | // Entry points for jimage.dll for loading jimage file entries |
105 | |
106 | static JImageOpen_t JImageOpen = NULL__null; |
107 | static JImageClose_t JImageClose = NULL__null; |
108 | static JImageFindResource_t JImageFindResource = NULL__null; |
109 | static JImageGetResource_t JImageGetResource = NULL__null; |
110 | |
111 | // JimageFile pointer, or null if exploded JDK build. |
112 | static JImageFile* JImage_file = NULL__null; |
113 | |
114 | // Globals |
115 | |
116 | PerfCounter* ClassLoader::_perf_accumulated_time = NULL__null; |
117 | PerfCounter* ClassLoader::_perf_classes_inited = NULL__null; |
118 | PerfCounter* ClassLoader::_perf_class_init_time = NULL__null; |
119 | PerfCounter* ClassLoader::_perf_class_init_selftime = NULL__null; |
120 | PerfCounter* ClassLoader::_perf_classes_verified = NULL__null; |
121 | PerfCounter* ClassLoader::_perf_class_verify_time = NULL__null; |
122 | PerfCounter* ClassLoader::_perf_class_verify_selftime = NULL__null; |
123 | PerfCounter* ClassLoader::_perf_classes_linked = NULL__null; |
124 | PerfCounter* ClassLoader::_perf_class_link_time = NULL__null; |
125 | PerfCounter* ClassLoader::_perf_class_link_selftime = NULL__null; |
126 | PerfCounter* ClassLoader::_perf_sys_class_lookup_time = NULL__null; |
127 | PerfCounter* ClassLoader::_perf_shared_classload_time = NULL__null; |
128 | PerfCounter* ClassLoader::_perf_sys_classload_time = NULL__null; |
129 | PerfCounter* ClassLoader::_perf_app_classload_time = NULL__null; |
130 | PerfCounter* ClassLoader::_perf_app_classload_selftime = NULL__null; |
131 | PerfCounter* ClassLoader::_perf_app_classload_count = NULL__null; |
132 | PerfCounter* ClassLoader::_perf_define_appclasses = NULL__null; |
133 | PerfCounter* ClassLoader::_perf_define_appclass_time = NULL__null; |
134 | PerfCounter* ClassLoader::_perf_define_appclass_selftime = NULL__null; |
135 | PerfCounter* ClassLoader::_perf_app_classfile_bytes_read = NULL__null; |
136 | PerfCounter* ClassLoader::_perf_sys_classfile_bytes_read = NULL__null; |
137 | PerfCounter* ClassLoader::_unsafe_defineClassCallCounter = NULL__null; |
138 | |
139 | GrowableArray<ModuleClassPathList*>* ClassLoader::_patch_mod_entries = NULL__null; |
140 | GrowableArray<ModuleClassPathList*>* ClassLoader::_exploded_entries = NULL__null; |
141 | ClassPathEntry* ClassLoader::_jrt_entry = NULL__null; |
142 | |
143 | ClassPathEntry* volatile ClassLoader::_first_append_entry_list = NULL__null; |
144 | ClassPathEntry* volatile ClassLoader::_last_append_entry = NULL__null; |
145 | #if INCLUDE_CDS1 |
146 | ClassPathEntry* ClassLoader::_app_classpath_entries = NULL__null; |
147 | ClassPathEntry* ClassLoader::_last_app_classpath_entry = NULL__null; |
148 | ClassPathEntry* ClassLoader::_module_path_entries = NULL__null; |
149 | ClassPathEntry* ClassLoader::_last_module_path_entry = NULL__null; |
150 | #endif |
151 | |
152 | // helper routines |
153 | bool string_starts_with(const char* str, const char* str_to_find) { |
154 | size_t str_len = strlen(str); |
155 | size_t str_to_find_len = strlen(str_to_find); |
156 | if (str_to_find_len > str_len) { |
157 | return false; |
158 | } |
159 | return (strncmp(str, str_to_find, str_to_find_len) == 0); |
160 | } |
161 | |
162 | static const char* get_jimage_version_string() { |
163 | static char version_string[10] = ""; |
164 | if (version_string[0] == '\0') { |
165 | jio_snprintf(version_string, sizeof(version_string), "%d.%d", |
166 | VM_Version::vm_major_version(), VM_Version::vm_minor_version()); |
167 | } |
168 | return (const char*)version_string; |
169 | } |
170 | |
171 | bool ClassLoader::string_ends_with(const char* str, const char* str_to_find) { |
172 | size_t str_len = strlen(str); |
173 | size_t str_to_find_len = strlen(str_to_find); |
174 | if (str_to_find_len > str_len) { |
175 | return false; |
176 | } |
177 | return (strncmp(str + (str_len - str_to_find_len), str_to_find, str_to_find_len) == 0); |
178 | } |
179 | |
180 | // Used to obtain the package name from a fully qualified class name. |
181 | Symbol* ClassLoader::package_from_class_name(const Symbol* name, bool* bad_class_name) { |
182 | if (name == NULL__null) { |
183 | if (bad_class_name != NULL__null) { |
184 | *bad_class_name = true; |
185 | } |
186 | return NULL__null; |
187 | } |
188 | |
189 | int utf_len = name->utf8_length(); |
190 | const jbyte* base = (const jbyte*)name->base(); |
191 | const jbyte* start = base; |
192 | const jbyte* end = UTF8::strrchr(start, utf_len, JVM_SIGNATURE_SLASH); |
193 | if (end == NULL__null) { |
194 | return NULL__null; |
195 | } |
196 | // Skip over '['s |
197 | if (*start == JVM_SIGNATURE_ARRAY) { |
198 | do { |
199 | start++; |
200 | } while (start < end && *start == JVM_SIGNATURE_ARRAY); |
201 | |
202 | // Fully qualified class names should not contain a 'L'. |
203 | // Set bad_class_name to true to indicate that the package name |
204 | // could not be obtained due to an error condition. |
205 | // In this situation, is_same_class_package returns false. |
206 | if (*start == JVM_SIGNATURE_CLASS) { |
207 | if (bad_class_name != NULL__null) { |
208 | *bad_class_name = true; |
209 | } |
210 | return NULL__null; |
211 | } |
212 | } |
213 | // A class name could have just the slash character in the name, |
214 | // in which case start > end |
215 | if (start >= end) { |
216 | // No package name |
217 | if (bad_class_name != NULL__null) { |
218 | *bad_class_name = true; |
219 | } |
220 | return NULL__null; |
221 | } |
222 | return SymbolTable::new_symbol(name, start - base, end - base); |
223 | } |
224 | |
225 | // Given a fully qualified package name, find its defining package in the class loader's |
226 | // package entry table. |
227 | PackageEntry* ClassLoader::get_package_entry(Symbol* pkg_name, ClassLoaderData* loader_data) { |
228 | if (pkg_name == NULL__null) { |
229 | return NULL__null; |
230 | } |
231 | PackageEntryTable* pkgEntryTable = loader_data->packages(); |
232 | return pkgEntryTable->lookup_only(pkg_name); |
233 | } |
234 | |
235 | const char* ClassPathEntry::copy_path(const char* path) { |
236 | char* copy = NEW_C_HEAP_ARRAY(char, strlen(path)+1, mtClass)(char*) (AllocateHeap((strlen(path)+1) * sizeof(char), mtClass )); |
237 | strcpy(copy, path); |
238 | return copy; |
239 | } |
240 | |
241 | ClassFileStream* ClassPathDirEntry::open_stream(JavaThread* current, const char* name) { |
242 | // construct full path name |
243 | assert((_dir != NULL) && (name != NULL), "sanity")do { if (!((_dir != __null) && (name != __null))) { ( *g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 243, "assert(" "(_dir != __null) && (name != __null)" ") failed", "sanity"); ::breakpoint(); } } while (0); |
244 | size_t path_len = strlen(_dir) + strlen(name) + strlen(os::file_separator()) + 1; |
245 | char* path = NEW_RESOURCE_ARRAY_IN_THREAD(current, char, path_len)(char*) resource_allocate_bytes(current, (path_len) * sizeof( char)); |
246 | int len = jio_snprintf(path, path_len, "%s%s%s", _dir, os::file_separator(), name); |
247 | assert(len == (int)(path_len - 1), "sanity")do { if (!(len == (int)(path_len - 1))) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 247, "assert(" "len == (int)(path_len - 1)" ") failed", "sanity" ); ::breakpoint(); } } while (0); |
248 | // check if file exists |
249 | struct stat st; |
250 | if (os::stat(path, &st) == 0) { |
251 | // found file, open it |
252 | int file_handle = os::open(path, 0, 0); |
253 | if (file_handle != -1) { |
254 | // read contents into resource array |
255 | u1* buffer = NEW_RESOURCE_ARRAY(u1, st.st_size)(u1*) resource_allocate_bytes((st.st_size) * sizeof(u1)); |
256 | size_t num_read = os::read(file_handle, (char*) buffer, st.st_size); |
257 | // close file |
258 | os::close(file_handle); |
259 | // construct ClassFileStream |
260 | if (num_read == (size_t)st.st_size) { |
261 | if (UsePerfData) { |
262 | ClassLoader::perf_sys_classfile_bytes_read()->inc(num_read); |
263 | } |
264 | FREE_RESOURCE_ARRAY(char, path, path_len)resource_free_bytes((char*)(path), (path_len) * sizeof(char)); |
265 | // Resource allocated |
266 | return new ClassFileStream(buffer, |
267 | st.st_size, |
268 | _dir, |
269 | ClassFileStream::verify); |
270 | } |
271 | } |
272 | } |
273 | FREE_RESOURCE_ARRAY(char, path, path_len)resource_free_bytes((char*)(path), (path_len) * sizeof(char)); |
274 | return NULL__null; |
275 | } |
276 | |
277 | ClassPathZipEntry::ClassPathZipEntry(jzfile* zip, const char* zip_name, |
278 | bool is_boot_append, bool from_class_path_attr) : ClassPathEntry() { |
279 | _zip = zip; |
280 | _zip_name = copy_path(zip_name); |
281 | _from_class_path_attr = from_class_path_attr; |
282 | } |
283 | |
284 | ClassPathZipEntry::~ClassPathZipEntry() { |
285 | (*ZipClose)(_zip); |
286 | FREE_C_HEAP_ARRAY(char, _zip_name)FreeHeap((char*)(_zip_name)); |
287 | } |
288 | |
289 | u1* ClassPathZipEntry::open_entry(JavaThread* current, const char* name, jint* filesize, bool nul_terminate) { |
290 | // enable call to C land |
291 | ThreadToNativeFromVM ttn(current); |
292 | // check whether zip archive contains name |
293 | jint name_len; |
294 | jzentry* entry = (*FindEntry)(_zip, name, filesize, &name_len); |
295 | if (entry == NULL__null) return NULL__null; |
296 | u1* buffer; |
297 | char name_buf[128]; |
298 | char* filename; |
299 | if (name_len < 128) { |
300 | filename = name_buf; |
301 | } else { |
302 | filename = NEW_RESOURCE_ARRAY(char, name_len + 1)(char*) resource_allocate_bytes((name_len + 1) * sizeof(char) ); |
303 | } |
304 | |
305 | // read contents into resource array |
306 | int size = (*filesize) + ((nul_terminate) ? 1 : 0); |
307 | buffer = NEW_RESOURCE_ARRAY(u1, size)(u1*) resource_allocate_bytes((size) * sizeof(u1)); |
308 | if (!(*ReadEntry)(_zip, entry, buffer, filename)) return NULL__null; |
309 | |
310 | // return result |
311 | if (nul_terminate) { |
312 | buffer[*filesize] = 0; |
313 | } |
314 | return buffer; |
315 | } |
316 | |
317 | ClassFileStream* ClassPathZipEntry::open_stream(JavaThread* current, const char* name) { |
318 | jint filesize; |
319 | u1* buffer = open_entry(current, name, &filesize, false); |
320 | if (buffer == NULL__null) { |
321 | return NULL__null; |
322 | } |
323 | if (UsePerfData) { |
324 | ClassLoader::perf_sys_classfile_bytes_read()->inc(filesize); |
325 | } |
326 | // Resource allocated |
327 | return new ClassFileStream(buffer, |
328 | filesize, |
329 | _zip_name, |
330 | ClassFileStream::verify); |
331 | } |
332 | |
333 | // invoke function for each entry in the zip file |
334 | void ClassPathZipEntry::contents_do(void f(const char* name, void* context), void* context) { |
335 | JavaThread* thread = JavaThread::current(); |
336 | HandleMark handle_mark(thread); |
337 | ThreadToNativeFromVM ttn(thread); |
338 | for (int n = 0; ; n++) { |
339 | jzentry * ze = ((*GetNextEntry)(_zip, n)); |
340 | if (ze == NULL__null) break; |
341 | (*f)(ze->name, context); |
342 | } |
343 | } |
344 | |
345 | DEBUG_ONLY(ClassPathImageEntry* ClassPathImageEntry::_singleton = NULL;)ClassPathImageEntry* ClassPathImageEntry::_singleton = __null ; |
346 | |
347 | JImageFile* ClassPathImageEntry::jimage() const { |
348 | return JImage_file; |
349 | } |
350 | |
351 | JImageFile* ClassPathImageEntry::jimage_non_null() const { |
352 | assert(ClassLoader::has_jrt_entry(), "must be")do { if (!(ClassLoader::has_jrt_entry())) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 352, "assert(" "ClassLoader::has_jrt_entry()" ") failed", "must be" ); ::breakpoint(); } } while (0); |
353 | assert(jimage() != NULL, "should have been opened by ClassLoader::lookup_vm_options "do { if (!(jimage() != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 354, "assert(" "jimage() != __null" ") failed", "should have been opened by ClassLoader::lookup_vm_options " "and remained throughout normal JVM lifetime"); ::breakpoint (); } } while (0) |
354 | "and remained throughout normal JVM lifetime")do { if (!(jimage() != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 354, "assert(" "jimage() != __null" ") failed", "should have been opened by ClassLoader::lookup_vm_options " "and remained throughout normal JVM lifetime"); ::breakpoint (); } } while (0); |
355 | return jimage(); |
356 | } |
357 | |
358 | void ClassPathImageEntry::close_jimage() { |
359 | if (jimage() != NULL__null) { |
360 | (*JImageClose)(jimage()); |
361 | JImage_file = NULL__null; |
362 | } |
363 | } |
364 | |
365 | ClassPathImageEntry::ClassPathImageEntry(JImageFile* jimage, const char* name) : |
366 | ClassPathEntry() { |
367 | guarantee(jimage != NULL, "jimage file is null")do { if (!(jimage != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 367, "guarantee(" "jimage != NULL" ") failed", "jimage file is null" ); ::breakpoint(); } } while (0); |
368 | guarantee(name != NULL, "jimage file name is null")do { if (!(name != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 368, "guarantee(" "name != NULL" ") failed", "jimage file name is null" ); ::breakpoint(); } } while (0); |
369 | assert(_singleton == NULL, "VM supports only one jimage")do { if (!(_singleton == __null)) { (*g_assert_poison) = 'X'; ; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 369, "assert(" "_singleton == __null" ") failed", "VM supports only one jimage" ); ::breakpoint(); } } while (0); |
370 | DEBUG_ONLY(_singleton = this)_singleton = this; |
371 | size_t len = strlen(name) + 1; |
372 | _name = copy_path(name); |
373 | } |
374 | |
375 | ClassFileStream* ClassPathImageEntry::open_stream(JavaThread* current, const char* name) { |
376 | return open_stream_for_loader(current, name, ClassLoaderData::the_null_class_loader_data()); |
377 | } |
378 | |
379 | // For a class in a named module, look it up in the jimage file using this syntax: |
380 | // /<module-name>/<package-name>/<base-class> |
381 | // |
382 | // Assumptions: |
383 | // 1. There are no unnamed modules in the jimage file. |
384 | // 2. A package is in at most one module in the jimage file. |
385 | // |
386 | ClassFileStream* ClassPathImageEntry::open_stream_for_loader(JavaThread* current, const char* name, ClassLoaderData* loader_data) { |
387 | jlong size; |
388 | JImageLocationRef location = (*JImageFindResource)(jimage_non_null(), "", get_jimage_version_string(), name, &size); |
389 | |
390 | if (location == 0) { |
391 | TempNewSymbol class_name = SymbolTable::new_symbol(name); |
392 | TempNewSymbol pkg_name = ClassLoader::package_from_class_name(class_name); |
393 | |
394 | if (pkg_name != NULL__null) { |
395 | if (!Universe::is_module_initialized()) { |
396 | location = (*JImageFindResource)(jimage_non_null(), JAVA_BASE_NAME"java.base", get_jimage_version_string(), name, &size); |
397 | } else { |
398 | PackageEntry* package_entry = ClassLoader::get_package_entry(pkg_name, loader_data); |
399 | if (package_entry != NULL__null) { |
400 | ResourceMark rm(current); |
401 | // Get the module name |
402 | ModuleEntry* module = package_entry->module(); |
403 | assert(module != NULL, "Boot classLoader package missing module")do { if (!(module != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 403, "assert(" "module != __null" ") failed", "Boot classLoader package missing module" ); ::breakpoint(); } } while (0); |
404 | assert(module->is_named(), "Boot classLoader package is in unnamed module")do { if (!(module->is_named())) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 404, "assert(" "module->is_named()" ") failed", "Boot classLoader package is in unnamed module" ); ::breakpoint(); } } while (0); |
405 | const char* module_name = module->name()->as_C_string(); |
406 | if (module_name != NULL__null) { |
407 | location = (*JImageFindResource)(jimage_non_null(), module_name, get_jimage_version_string(), name, &size); |
408 | } |
409 | } |
410 | } |
411 | } |
412 | } |
413 | if (location != 0) { |
414 | if (UsePerfData) { |
415 | ClassLoader::perf_sys_classfile_bytes_read()->inc(size); |
416 | } |
417 | char* data = NEW_RESOURCE_ARRAY(char, size)(char*) resource_allocate_bytes((size) * sizeof(char)); |
418 | (*JImageGetResource)(jimage_non_null(), location, data, size); |
419 | // Resource allocated |
420 | assert(this == (ClassPathImageEntry*)ClassLoader::get_jrt_entry(), "must be")do { if (!(this == (ClassPathImageEntry*)ClassLoader::get_jrt_entry ())) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 420, "assert(" "this == (ClassPathImageEntry*)ClassLoader::get_jrt_entry()" ") failed", "must be"); ::breakpoint(); } } while (0); |
421 | return new ClassFileStream((u1*)data, |
422 | (int)size, |
423 | _name, |
424 | ClassFileStream::verify, |
425 | true); // from_boot_loader_modules_image |
426 | } |
427 | |
428 | return NULL__null; |
429 | } |
430 | |
431 | JImageLocationRef ClassLoader::jimage_find_resource(JImageFile* jf, |
432 | const char* module_name, |
433 | const char* file_name, |
434 | jlong &size) { |
435 | return ((*JImageFindResource)(jf, module_name, get_jimage_version_string(), file_name, &size)); |
436 | } |
437 | |
438 | bool ClassPathImageEntry::is_modules_image() const { |
439 | assert(this == _singleton, "VM supports a single jimage")do { if (!(this == _singleton)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 439, "assert(" "this == _singleton" ") failed", "VM supports a single jimage" ); ::breakpoint(); } } while (0); |
440 | assert(this == (ClassPathImageEntry*)ClassLoader::get_jrt_entry(), "must be used for jrt entry")do { if (!(this == (ClassPathImageEntry*)ClassLoader::get_jrt_entry ())) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 440, "assert(" "this == (ClassPathImageEntry*)ClassLoader::get_jrt_entry()" ") failed", "must be used for jrt entry"); ::breakpoint(); } } while (0); |
441 | return true; |
442 | } |
443 | |
444 | #if INCLUDE_CDS1 |
445 | void ClassLoader::exit_with_path_failure(const char* error, const char* message) { |
446 | Arguments::assert_is_dumping_archive(); |
447 | tty->print_cr("Hint: enable -Xlog:class+path=info to diagnose the failure"); |
448 | vm_exit_during_initialization(error, message); |
449 | } |
450 | #endif |
451 | |
452 | ModuleClassPathList::ModuleClassPathList(Symbol* module_name) { |
453 | _module_name = module_name; |
454 | _module_first_entry = NULL__null; |
455 | _module_last_entry = NULL__null; |
456 | } |
457 | |
458 | ModuleClassPathList::~ModuleClassPathList() { |
459 | // Clean out each ClassPathEntry on list |
460 | ClassPathEntry* e = _module_first_entry; |
461 | while (e != NULL__null) { |
462 | ClassPathEntry* next_entry = e->next(); |
463 | delete e; |
464 | e = next_entry; |
465 | } |
466 | } |
467 | |
468 | void ModuleClassPathList::add_to_list(ClassPathEntry* new_entry) { |
469 | if (new_entry != NULL__null) { |
470 | if (_module_last_entry == NULL__null) { |
471 | _module_first_entry = _module_last_entry = new_entry; |
472 | } else { |
473 | _module_last_entry->set_next(new_entry); |
474 | _module_last_entry = new_entry; |
475 | } |
476 | } |
477 | } |
478 | |
479 | void ClassLoader::trace_class_path(const char* msg, const char* name) { |
480 | LogTarget(Info, class, path)LogTargetImpl<LogLevel::Info, (LogTag::_class), (LogTag::_path ), (LogTag::__NO_TAG), (LogTag::__NO_TAG), (LogTag::__NO_TAG) , (LogTag::__NO_TAG)> lt; |
481 | if (lt.is_enabled()) { |
482 | LogStream ls(lt); |
483 | if (msg) { |
484 | ls.print("%s", msg); |
485 | } |
486 | if (name) { |
487 | if (strlen(name) < 256) { |
488 | ls.print("%s", name); |
489 | } else { |
490 | // For very long paths, we need to print each character separately, |
491 | // as print_cr() has a length limit |
492 | while (name[0] != '\0') { |
493 | ls.print("%c", name[0]); |
494 | name++; |
495 | } |
496 | } |
497 | } |
498 | ls.cr(); |
499 | } |
500 | } |
501 | |
502 | void ClassLoader::setup_bootstrap_search_path(JavaThread* current) { |
503 | const char* sys_class_path = Arguments::get_sysclasspath(); |
504 | assert(sys_class_path != NULL, "System boot class path must not be NULL")do { if (!(sys_class_path != __null)) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 504, "assert(" "sys_class_path != __null" ") failed", "System boot class path must not be NULL" ); ::breakpoint(); } } while (0); |
505 | if (PrintSharedArchiveAndExit) { |
506 | // Don't print sys_class_path - this is the bootcp of this current VM process, not necessarily |
507 | // the same as the bootcp of the shared archive. |
508 | } else { |
509 | trace_class_path("bootstrap loader class path=", sys_class_path); |
510 | } |
511 | setup_bootstrap_search_path_impl(current, sys_class_path); |
512 | } |
513 | |
514 | #if INCLUDE_CDS1 |
515 | void ClassLoader::setup_app_search_path(JavaThread* current, const char *class_path) { |
516 | Arguments::assert_is_dumping_archive(); |
517 | |
518 | ResourceMark rm(current); |
519 | ClasspathStream cp_stream(class_path); |
520 | |
521 | while (cp_stream.has_next()) { |
522 | const char* path = cp_stream.get_next(); |
523 | update_class_path_entry_list(current, path, false, false, false); |
524 | } |
525 | } |
526 | |
527 | void ClassLoader::add_to_module_path_entries(const char* path, |
528 | ClassPathEntry* entry) { |
529 | assert(entry != NULL, "ClassPathEntry should not be NULL")do { if (!(entry != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 529, "assert(" "entry != __null" ") failed", "ClassPathEntry should not be NULL" ); ::breakpoint(); } } while (0); |
530 | Arguments::assert_is_dumping_archive(); |
531 | |
532 | // The entry does not exist, add to the list |
533 | if (_module_path_entries == NULL__null) { |
534 | assert(_last_module_path_entry == NULL, "Sanity")do { if (!(_last_module_path_entry == __null)) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 534, "assert(" "_last_module_path_entry == __null" ") failed" , "Sanity"); ::breakpoint(); } } while (0); |
535 | _module_path_entries = _last_module_path_entry = entry; |
536 | } else { |
537 | _last_module_path_entry->set_next(entry); |
538 | _last_module_path_entry = entry; |
539 | } |
540 | } |
541 | |
542 | // Add a module path to the _module_path_entries list. |
543 | void ClassLoader::setup_module_search_path(JavaThread* current, const char* path) { |
544 | Arguments::assert_is_dumping_archive(); |
545 | struct stat st; |
546 | if (os::stat(path, &st) != 0) { |
547 | tty->print_cr("os::stat error %d (%s). CDS dump aborted (path was \"%s\").", |
548 | errno(*__errno_location ()), os::errno_name(errno(*__errno_location ())), path); |
549 | vm_exit_during_initialization(); |
550 | } |
551 | // File or directory found |
552 | ClassPathEntry* new_entry = NULL__null; |
553 | new_entry = create_class_path_entry(current, path, &st, |
554 | false /*is_boot_append */, false /* from_class_path_attr */); |
555 | if (new_entry != NULL__null) { |
556 | add_to_module_path_entries(path, new_entry); |
557 | } |
558 | } |
559 | |
560 | #endif // INCLUDE_CDS |
561 | |
562 | void ClassLoader::close_jrt_image() { |
563 | // Not applicable for exploded builds |
564 | if (!ClassLoader::has_jrt_entry()) return; |
565 | _jrt_entry->close_jimage(); |
566 | } |
567 | |
568 | // Construct the array of module/path pairs as specified to --patch-module |
569 | // for the boot loader to search ahead of the jimage, if the class being |
570 | // loaded is defined to a module that has been specified to --patch-module. |
571 | void ClassLoader::setup_patch_mod_entries() { |
572 | JavaThread* current = JavaThread::current(); |
573 | GrowableArray<ModulePatchPath*>* patch_mod_args = Arguments::get_patch_mod_prefix(); |
574 | int num_of_entries = patch_mod_args->length(); |
575 | |
576 | // Set up the boot loader's _patch_mod_entries list |
577 | _patch_mod_entries = new (ResourceObj::C_HEAP, mtModule) GrowableArray<ModuleClassPathList*>(num_of_entries, mtModule); |
578 | |
579 | for (int i = 0; i < num_of_entries; i++) { |
580 | const char* module_name = (patch_mod_args->at(i))->module_name(); |
581 | Symbol* const module_sym = SymbolTable::new_symbol(module_name); |
582 | assert(module_sym != NULL, "Failed to obtain Symbol for module name")do { if (!(module_sym != __null)) { (*g_assert_poison) = 'X'; ; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 582, "assert(" "module_sym != __null" ") failed", "Failed to obtain Symbol for module name" ); ::breakpoint(); } } while (0); |
583 | ModuleClassPathList* module_cpl = new ModuleClassPathList(module_sym); |
584 | |
585 | char* class_path = (patch_mod_args->at(i))->path_string(); |
586 | ResourceMark rm(current); |
587 | ClasspathStream cp_stream(class_path); |
588 | |
589 | while (cp_stream.has_next()) { |
590 | const char* path = cp_stream.get_next(); |
591 | struct stat st; |
592 | if (os::stat(path, &st) == 0) { |
593 | // File or directory found |
594 | ClassPathEntry* new_entry = create_class_path_entry(current, path, &st, false, false); |
595 | // If the path specification is valid, enter it into this module's list |
596 | if (new_entry != NULL__null) { |
597 | module_cpl->add_to_list(new_entry); |
598 | } |
599 | } |
600 | } |
601 | |
602 | // Record the module into the list of --patch-module entries only if |
603 | // valid ClassPathEntrys have been created |
604 | if (module_cpl->module_first_entry() != NULL__null) { |
605 | _patch_mod_entries->push(module_cpl); |
606 | } |
607 | } |
608 | } |
609 | |
610 | // Determine whether the module has been patched via the command-line |
611 | // option --patch-module |
612 | bool ClassLoader::is_in_patch_mod_entries(Symbol* module_name) { |
613 | if (_patch_mod_entries != NULL__null && _patch_mod_entries->is_nonempty()) { |
614 | int table_len = _patch_mod_entries->length(); |
615 | for (int i = 0; i < table_len; i++) { |
616 | ModuleClassPathList* patch_mod = _patch_mod_entries->at(i); |
617 | if (module_name->fast_compare(patch_mod->module_name()) == 0) { |
618 | return true; |
619 | } |
620 | } |
621 | } |
622 | return false; |
623 | } |
624 | |
625 | // Set up the _jrt_entry if present and boot append path |
626 | void ClassLoader::setup_bootstrap_search_path_impl(JavaThread* current, const char *class_path) { |
627 | ResourceMark rm(current); |
628 | ClasspathStream cp_stream(class_path); |
629 | bool set_base_piece = true; |
630 | |
631 | #if INCLUDE_CDS1 |
632 | if (Arguments::is_dumping_archive()) { |
633 | if (!Arguments::has_jimage()) { |
634 | vm_exit_during_initialization("CDS is not supported in exploded JDK build", NULL__null); |
635 | } |
636 | } |
637 | #endif |
638 | |
639 | while (cp_stream.has_next()) { |
640 | const char* path = cp_stream.get_next(); |
641 | |
642 | if (set_base_piece) { |
643 | // The first time through the bootstrap_search setup, it must be determined |
644 | // what the base or core piece of the boot loader search is. Either a java runtime |
645 | // image is present or this is an exploded module build situation. |
646 | assert(string_ends_with(path, MODULES_IMAGE_NAME) || string_ends_with(path, JAVA_BASE_NAME),do { if (!(string_ends_with(path, "modules") || string_ends_with (path, "java.base"))) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 647, "assert(" "string_ends_with(path, \"modules\") || string_ends_with(path, \"java.base\")" ") failed", "Incorrect boot loader search path, no java runtime image or " "java.base" " exploded build"); ::breakpoint(); } } while (0 ) |
647 | "Incorrect boot loader search path, no java runtime image or " JAVA_BASE_NAME " exploded build")do { if (!(string_ends_with(path, "modules") || string_ends_with (path, "java.base"))) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 647, "assert(" "string_ends_with(path, \"modules\") || string_ends_with(path, \"java.base\")" ") failed", "Incorrect boot loader search path, no java runtime image or " "java.base" " exploded build"); ::breakpoint(); } } while (0 ); |
648 | struct stat st; |
649 | if (os::stat(path, &st) == 0) { |
650 | // Directory found |
651 | if (JImage_file != NULL__null) { |
652 | assert(Arguments::has_jimage(), "sanity check")do { if (!(Arguments::has_jimage())) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 652, "assert(" "Arguments::has_jimage()" ") failed", "sanity check" ); ::breakpoint(); } } while (0); |
653 | const char* canonical_path = get_canonical_path(path, current); |
654 | assert(canonical_path != NULL, "canonical_path issue")do { if (!(canonical_path != __null)) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 654, "assert(" "canonical_path != __null" ") failed", "canonical_path issue" ); ::breakpoint(); } } while (0); |
655 | |
656 | _jrt_entry = new ClassPathImageEntry(JImage_file, canonical_path); |
657 | assert(_jrt_entry != NULL && _jrt_entry->is_modules_image(), "No java runtime image present")do { if (!(_jrt_entry != __null && _jrt_entry->is_modules_image ())) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 657, "assert(" "_jrt_entry != __null && _jrt_entry->is_modules_image()" ") failed", "No java runtime image present"); ::breakpoint() ; } } while (0); |
658 | assert(_jrt_entry->jimage() != NULL, "No java runtime image")do { if (!(_jrt_entry->jimage() != __null)) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 658, "assert(" "_jrt_entry->jimage() != __null" ") failed" , "No java runtime image"); ::breakpoint(); } } while (0); |
659 | } else { |
660 | // It's an exploded build. |
661 | ClassPathEntry* new_entry = create_class_path_entry(current, path, &st, false, false); |
662 | } |
663 | } else { |
664 | // If path does not exist, exit |
665 | vm_exit_during_initialization("Unable to establish the boot loader search path", path); |
666 | } |
667 | set_base_piece = false; |
668 | } else { |
669 | // Every entry on the system boot class path after the initial base piece, |
670 | // which is set by os::set_boot_path(), is considered an appended entry. |
671 | update_class_path_entry_list(current, path, false, true, false); |
672 | } |
673 | } |
674 | } |
675 | |
676 | // During an exploded modules build, each module defined to the boot loader |
677 | // will be added to the ClassLoader::_exploded_entries array. |
678 | void ClassLoader::add_to_exploded_build_list(JavaThread* current, Symbol* module_sym) { |
679 | assert(!ClassLoader::has_jrt_entry(), "Exploded build not applicable")do { if (!(!ClassLoader::has_jrt_entry())) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 679, "assert(" "!ClassLoader::has_jrt_entry()" ") failed", "Exploded build not applicable" ); ::breakpoint(); } } while (0); |
680 | assert(_exploded_entries != NULL, "_exploded_entries was not initialized")do { if (!(_exploded_entries != __null)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 680, "assert(" "_exploded_entries != __null" ") failed", "_exploded_entries was not initialized" ); ::breakpoint(); } } while (0); |
681 | |
682 | // Find the module's symbol |
683 | ResourceMark rm(current); |
684 | const char *module_name = module_sym->as_C_string(); |
685 | const char *home = Arguments::get_java_home(); |
686 | const char file_sep = os::file_separator()[0]; |
687 | // 10 represents the length of "modules" + 2 file separators + \0 |
688 | size_t len = strlen(home) + strlen(module_name) + 10; |
689 | char *path = NEW_RESOURCE_ARRAY(char, len)(char*) resource_allocate_bytes((len) * sizeof(char)); |
690 | jio_snprintf(path, len, "%s%cmodules%c%s", home, file_sep, file_sep, module_name); |
691 | |
692 | struct stat st; |
693 | if (os::stat(path, &st) == 0) { |
694 | // Directory found |
695 | ClassPathEntry* new_entry = create_class_path_entry(current, path, &st, false, false); |
696 | |
697 | // If the path specification is valid, enter it into this module's list. |
698 | // There is no need to check for duplicate modules in the exploded entry list, |
699 | // since no two modules with the same name can be defined to the boot loader. |
700 | // This is checked at module definition time in Modules::define_module. |
701 | if (new_entry != NULL__null) { |
702 | ModuleClassPathList* module_cpl = new ModuleClassPathList(module_sym); |
703 | module_cpl->add_to_list(new_entry); |
704 | { |
705 | MutexLocker ml(current, Module_lock); |
706 | _exploded_entries->push(module_cpl); |
707 | } |
708 | log_info(class, load)(!(LogImpl<(LogTag::_class), (LogTag::_load), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG), (LogTag::__NO_TAG) >::is_level(LogLevel::Info))) ? (void)0 : LogImpl<(LogTag ::_class), (LogTag::_load), (LogTag::__NO_TAG), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG)>::write<LogLevel ::Info>("path: %s", path); |
709 | } |
710 | } |
711 | } |
712 | |
713 | jzfile* ClassLoader::open_zip_file(const char* canonical_path, char** error_msg, JavaThread* thread) { |
714 | // enable call to C land |
715 | ThreadToNativeFromVM ttn(thread); |
716 | HandleMark hm(thread); |
717 | load_zip_library_if_needed(); |
718 | return (*ZipOpen)(canonical_path, error_msg); |
719 | } |
720 | |
721 | ClassPathEntry* ClassLoader::create_class_path_entry(JavaThread* current, |
722 | const char *path, const struct stat* st, |
723 | bool is_boot_append, |
724 | bool from_class_path_attr) { |
725 | ClassPathEntry* new_entry = NULL__null; |
726 | if ((st->st_mode & S_IFMT0170000) == S_IFREG0100000) { |
727 | ResourceMark rm(current); |
728 | // Regular file, should be a zip file |
729 | // Canonicalized filename |
730 | const char* canonical_path = get_canonical_path(path, current); |
731 | if (canonical_path == NULL__null) { |
732 | return NULL__null; |
733 | } |
734 | char* error_msg = NULL__null; |
735 | jzfile* zip = open_zip_file(canonical_path, &error_msg, current); |
736 | if (zip != NULL__null && error_msg == NULL__null) { |
737 | new_entry = new ClassPathZipEntry(zip, path, is_boot_append, from_class_path_attr); |
738 | } else { |
739 | #if INCLUDE_CDS1 |
740 | ClassLoaderExt::set_has_non_jar_in_classpath(); |
741 | #endif |
742 | return NULL__null; |
743 | } |
744 | log_info(class, path)(!(LogImpl<(LogTag::_class), (LogTag::_path), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG), (LogTag::__NO_TAG) >::is_level(LogLevel::Info))) ? (void)0 : LogImpl<(LogTag ::_class), (LogTag::_path), (LogTag::__NO_TAG), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG)>::write<LogLevel ::Info>("opened: %s", path); |
745 | log_info(class, load)(!(LogImpl<(LogTag::_class), (LogTag::_load), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG), (LogTag::__NO_TAG) >::is_level(LogLevel::Info))) ? (void)0 : LogImpl<(LogTag ::_class), (LogTag::_load), (LogTag::__NO_TAG), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG)>::write<LogLevel ::Info>("opened: %s", path); |
746 | } else { |
747 | // Directory |
748 | new_entry = new ClassPathDirEntry(path); |
749 | log_info(class, load)(!(LogImpl<(LogTag::_class), (LogTag::_load), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG), (LogTag::__NO_TAG) >::is_level(LogLevel::Info))) ? (void)0 : LogImpl<(LogTag ::_class), (LogTag::_load), (LogTag::__NO_TAG), (LogTag::__NO_TAG ), (LogTag::__NO_TAG), (LogTag::__NO_TAG)>::write<LogLevel ::Info>("path: %s", path); |
750 | } |
751 | return new_entry; |
752 | } |
753 | |
754 | |
755 | // Create a class path zip entry for a given path (return NULL if not found |
756 | // or zip/JAR file cannot be opened) |
757 | ClassPathZipEntry* ClassLoader::create_class_path_zip_entry(const char *path, bool is_boot_append) { |
758 | // check for a regular file |
759 | struct stat st; |
760 | if (os::stat(path, &st) == 0) { |
761 | if ((st.st_mode & S_IFMT0170000) == S_IFREG0100000) { |
762 | JavaThread* thread = JavaThread::current(); |
763 | ResourceMark rm(thread); |
764 | const char* canonical_path = get_canonical_path(path, thread); |
765 | if (canonical_path != NULL__null) { |
766 | char* error_msg = NULL__null; |
767 | jzfile* zip = open_zip_file(canonical_path, &error_msg, thread); |
768 | if (zip != NULL__null && error_msg == NULL__null) { |
769 | // create using canonical path |
770 | return new ClassPathZipEntry(zip, canonical_path, is_boot_append, false); |
771 | } |
772 | } |
773 | } |
774 | } |
775 | return NULL__null; |
776 | } |
777 | |
778 | // The boot append entries are added with a lock, and read lock free. |
779 | void ClassLoader::add_to_boot_append_entries(ClassPathEntry *new_entry) { |
780 | if (new_entry != NULL__null) { |
781 | MutexLocker ml(Bootclasspath_lock, Mutex::_no_safepoint_check_flag); |
782 | if (_last_append_entry == NULL__null) { |
783 | _last_append_entry = new_entry; |
784 | assert(first_append_entry() == NULL, "boot loader's append class path entry list not empty")do { if (!(first_append_entry() == __null)) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 784, "assert(" "first_append_entry() == __null" ") failed", "boot loader's append class path entry list not empty"); ::breakpoint (); } } while (0); |
785 | Atomic::release_store(&_first_append_entry_list, new_entry); |
786 | } else { |
787 | _last_append_entry->set_next(new_entry); |
788 | _last_append_entry = new_entry; |
789 | } |
790 | } |
791 | } |
792 | |
793 | // Record the path entries specified in -cp during dump time. The recorded |
794 | // information will be used at runtime for loading the archived app classes. |
795 | // |
796 | // Note that at dump time, ClassLoader::_app_classpath_entries are NOT used for |
797 | // loading app classes. Instead, the app class are loaded by the |
798 | // jdk/internal/loader/ClassLoaders$AppClassLoader instance. |
799 | void ClassLoader::add_to_app_classpath_entries(JavaThread* current, |
800 | const char* path, |
801 | ClassPathEntry* entry, |
802 | bool check_for_duplicates) { |
803 | #if INCLUDE_CDS1 |
804 | assert(entry != NULL, "ClassPathEntry should not be NULL")do { if (!(entry != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 804, "assert(" "entry != __null" ") failed", "ClassPathEntry should not be NULL" ); ::breakpoint(); } } while (0); |
805 | ClassPathEntry* e = _app_classpath_entries; |
806 | if (check_for_duplicates) { |
807 | while (e != NULL__null) { |
808 | if (strcmp(e->name(), entry->name()) == 0) { |
809 | // entry already exists |
810 | return; |
811 | } |
812 | e = e->next(); |
813 | } |
814 | } |
815 | |
816 | // The entry does not exist, add to the list |
817 | if (_app_classpath_entries == NULL__null) { |
818 | assert(_last_app_classpath_entry == NULL, "Sanity")do { if (!(_last_app_classpath_entry == __null)) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 818, "assert(" "_last_app_classpath_entry == __null" ") failed" , "Sanity"); ::breakpoint(); } } while (0); |
819 | _app_classpath_entries = _last_app_classpath_entry = entry; |
820 | } else { |
821 | _last_app_classpath_entry->set_next(entry); |
822 | _last_app_classpath_entry = entry; |
823 | } |
824 | |
825 | if (entry->is_jar_file()) { |
826 | ClassLoaderExt::process_jar_manifest(current, entry, check_for_duplicates); |
827 | } |
828 | #endif |
829 | } |
830 | |
831 | // Returns true IFF the file/dir exists and the entry was successfully created. |
832 | bool ClassLoader::update_class_path_entry_list(JavaThread* current, |
833 | const char *path, |
834 | bool check_for_duplicates, |
835 | bool is_boot_append, |
836 | bool from_class_path_attr) { |
837 | struct stat st; |
838 | if (os::stat(path, &st) == 0) { |
839 | // File or directory found |
840 | ClassPathEntry* new_entry = NULL__null; |
841 | new_entry = create_class_path_entry(current, path, &st, is_boot_append, from_class_path_attr); |
842 | if (new_entry == NULL__null) { |
843 | return false; |
844 | } |
845 | |
846 | // Do not reorder the bootclasspath which would break get_system_package(). |
847 | // Add new entry to linked list |
848 | if (is_boot_append) { |
849 | add_to_boot_append_entries(new_entry); |
850 | } else { |
851 | add_to_app_classpath_entries(current, path, new_entry, check_for_duplicates); |
852 | } |
853 | return true; |
854 | } else { |
855 | return false; |
856 | } |
857 | } |
858 | |
859 | static void print_module_entry_table(const GrowableArray<ModuleClassPathList*>* const module_list) { |
860 | ResourceMark rm; |
861 | int num_of_entries = module_list->length(); |
862 | for (int i = 0; i < num_of_entries; i++) { |
863 | ClassPathEntry* e; |
864 | ModuleClassPathList* mpl = module_list->at(i); |
865 | tty->print("%s=", mpl->module_name()->as_C_string()); |
866 | e = mpl->module_first_entry(); |
867 | while (e != NULL__null) { |
868 | tty->print("%s", e->name()); |
869 | e = e->next(); |
870 | if (e != NULL__null) { |
871 | tty->print("%s", os::path_separator()); |
872 | } |
873 | } |
874 | tty->print(" ;"); |
875 | } |
876 | } |
877 | |
878 | void ClassLoader::print_bootclasspath() { |
879 | ClassPathEntry* e; |
880 | tty->print("[bootclasspath= "); |
881 | |
882 | // Print --patch-module module/path specifications first |
883 | if (_patch_mod_entries != NULL__null) { |
884 | print_module_entry_table(_patch_mod_entries); |
885 | } |
886 | |
887 | // [jimage | exploded modules build] |
888 | if (has_jrt_entry()) { |
889 | // Print the location of the java runtime image |
890 | tty->print("%s ;", _jrt_entry->name()); |
891 | } else { |
892 | // Print exploded module build path specifications |
893 | if (_exploded_entries != NULL__null) { |
894 | print_module_entry_table(_exploded_entries); |
895 | } |
896 | } |
897 | |
898 | // appended entries |
899 | e = first_append_entry(); |
900 | while (e != NULL__null) { |
901 | tty->print("%s ;", e->name()); |
902 | e = e->next(); |
903 | } |
904 | tty->print_cr("]"); |
905 | } |
906 | |
907 | void* ClassLoader::dll_lookup(void* lib, const char* name, const char* path) { |
908 | void* func = os::dll_lookup(lib, name); |
909 | if (func == NULL__null) { |
910 | char msg[256] = ""; |
911 | jio_snprintf(msg, sizeof(msg), "Could not resolve \"%s\"", name); |
912 | vm_exit_during_initialization(msg, path); |
913 | } |
914 | return func; |
915 | } |
916 | |
917 | void ClassLoader::load_java_library() { |
918 | assert(CanonicalizeEntry == NULL, "should not load java library twice")do { if (!(CanonicalizeEntry == __null)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 918, "assert(" "CanonicalizeEntry == __null" ") failed", "should not load java library twice" ); ::breakpoint(); } } while (0); |
919 | void *javalib_handle = os::native_java_library(); |
920 | if (javalib_handle == NULL__null) { |
921 | vm_exit_during_initialization("Unable to load java library", NULL__null); |
922 | } |
923 | |
924 | CanonicalizeEntry = CAST_TO_FN_PTR(canonicalize_fn_t, dll_lookup(javalib_handle, "JDK_Canonicalize", NULL))(reinterpret_cast<canonicalize_fn_t>(dll_lookup(javalib_handle , "JDK_Canonicalize", __null))); |
925 | } |
926 | |
927 | void ClassLoader::release_load_zip_library() { |
928 | MutexLocker locker(Zip_lock, Monitor::_no_safepoint_check_flag); |
929 | if (_libzip_loaded == 0) { |
930 | load_zip_library(); |
931 | Atomic::release_store(&_libzip_loaded, 1); |
932 | } |
933 | } |
934 | |
935 | void ClassLoader::load_zip_library() { |
936 | assert(ZipOpen == NULL, "should not load zip library twice")do { if (!(ZipOpen == __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 936, "assert(" "ZipOpen == __null" ") failed", "should not load zip library twice" ); ::breakpoint(); } } while (0); |
937 | char path[JVM_MAXPATHLEN4096 + 1]; |
938 | char ebuf[1024]; |
939 | void* handle = NULL__null; |
940 | if (os::dll_locate_lib(path, sizeof(path), Arguments::get_dll_dir(), "zip")) { |
941 | handle = os::dll_load(path, ebuf, sizeof ebuf); |
942 | } |
943 | if (handle == NULL__null) { |
944 | vm_exit_during_initialization("Unable to load zip library", path); |
945 | } |
946 | |
947 | ZipOpen = CAST_TO_FN_PTR(ZipOpen_t, dll_lookup(handle, "ZIP_Open", path))(reinterpret_cast<ZipOpen_t>(dll_lookup(handle, "ZIP_Open" , path))); |
948 | ZipClose = CAST_TO_FN_PTR(ZipClose_t, dll_lookup(handle, "ZIP_Close", path))(reinterpret_cast<ZipClose_t>(dll_lookup(handle, "ZIP_Close" , path))); |
949 | FindEntry = CAST_TO_FN_PTR(FindEntry_t, dll_lookup(handle, "ZIP_FindEntry", path))(reinterpret_cast<FindEntry_t>(dll_lookup(handle, "ZIP_FindEntry" , path))); |
950 | ReadEntry = CAST_TO_FN_PTR(ReadEntry_t, dll_lookup(handle, "ZIP_ReadEntry", path))(reinterpret_cast<ReadEntry_t>(dll_lookup(handle, "ZIP_ReadEntry" , path))); |
951 | GetNextEntry = CAST_TO_FN_PTR(GetNextEntry_t, dll_lookup(handle, "ZIP_GetNextEntry", path))(reinterpret_cast<GetNextEntry_t>(dll_lookup(handle, "ZIP_GetNextEntry" , path))); |
952 | Crc32 = CAST_TO_FN_PTR(Crc32_t, dll_lookup(handle, "ZIP_CRC32", path))(reinterpret_cast<Crc32_t>(dll_lookup(handle, "ZIP_CRC32" , path))); |
953 | } |
954 | |
955 | void ClassLoader::load_jimage_library() { |
956 | assert(JImageOpen == NULL, "should not load jimage library twice")do { if (!(JImageOpen == __null)) { (*g_assert_poison) = 'X'; ; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 956, "assert(" "JImageOpen == __null" ") failed", "should not load jimage library twice" ); ::breakpoint(); } } while (0); |
957 | char path[JVM_MAXPATHLEN4096 + 1]; |
958 | char ebuf[1024]; |
959 | void* handle = NULL__null; |
960 | if (os::dll_locate_lib(path, sizeof(path), Arguments::get_dll_dir(), "jimage")) { |
961 | handle = os::dll_load(path, ebuf, sizeof ebuf); |
962 | } |
963 | if (handle == NULL__null) { |
964 | vm_exit_during_initialization("Unable to load jimage library", path); |
965 | } |
966 | |
967 | JImageOpen = CAST_TO_FN_PTR(JImageOpen_t, dll_lookup(handle, "JIMAGE_Open", path))(reinterpret_cast<JImageOpen_t>(dll_lookup(handle, "JIMAGE_Open" , path))); |
968 | JImageClose = CAST_TO_FN_PTR(JImageClose_t, dll_lookup(handle, "JIMAGE_Close", path))(reinterpret_cast<JImageClose_t>(dll_lookup(handle, "JIMAGE_Close" , path))); |
969 | JImageFindResource = CAST_TO_FN_PTR(JImageFindResource_t, dll_lookup(handle, "JIMAGE_FindResource", path))(reinterpret_cast<JImageFindResource_t>(dll_lookup(handle , "JIMAGE_FindResource", path))); |
970 | JImageGetResource = CAST_TO_FN_PTR(JImageGetResource_t, dll_lookup(handle, "JIMAGE_GetResource", path))(reinterpret_cast<JImageGetResource_t>(dll_lookup(handle , "JIMAGE_GetResource", path))); |
971 | } |
972 | |
973 | int ClassLoader::crc32(int crc, const char* buf, int len) { |
974 | load_zip_library_if_needed(); |
975 | return (*Crc32)(crc, (const jbyte*)buf, len); |
976 | } |
977 | |
978 | oop ClassLoader::get_system_package(const char* name, TRAPSJavaThread* __the_thread__) { |
979 | // Look up the name in the boot loader's package entry table. |
980 | if (name != NULL__null) { |
981 | TempNewSymbol package_sym = SymbolTable::new_symbol(name); |
982 | // Look for the package entry in the boot loader's package entry table. |
983 | PackageEntry* package = |
984 | ClassLoaderData::the_null_class_loader_data()->packages()->lookup_only(package_sym); |
985 | |
986 | // Return NULL if package does not exist or if no classes in that package |
987 | // have been loaded. |
988 | if (package != NULL__null && package->has_loaded_class()) { |
989 | ModuleEntry* module = package->module(); |
990 | if (module->location() != NULL__null) { |
991 | ResourceMark rm(THREAD__the_thread__); |
992 | Handle ml = java_lang_String::create_from_str( |
993 | module->location()->as_C_string(), THREAD__the_thread__); |
994 | return ml(); |
995 | } |
996 | // Return entry on boot loader class path. |
997 | Handle cph = java_lang_String::create_from_str( |
998 | ClassLoader::classpath_entry(package->classpath_index())->name(), THREAD__the_thread__); |
999 | return cph(); |
1000 | } |
1001 | } |
1002 | return NULL__null; |
1003 | } |
1004 | |
1005 | objArrayOop ClassLoader::get_system_packages(TRAPSJavaThread* __the_thread__) { |
1006 | ResourceMark rm(THREAD__the_thread__); |
1007 | // List of pointers to PackageEntrys that have loaded classes. |
1008 | GrowableArray<PackageEntry*>* loaded_class_pkgs = new GrowableArray<PackageEntry*>(50); |
1009 | { |
1010 | MutexLocker ml(THREAD__the_thread__, Module_lock); |
1011 | |
1012 | PackageEntryTable* pe_table = |
1013 | ClassLoaderData::the_null_class_loader_data()->packages(); |
1014 | |
1015 | // Collect the packages that have at least one loaded class. |
1016 | for (int x = 0; x < pe_table->table_size(); x++) { |
1017 | for (PackageEntry* package_entry = pe_table->bucket(x); |
1018 | package_entry != NULL__null; |
1019 | package_entry = package_entry->next()) { |
1020 | if (package_entry->has_loaded_class()) { |
1021 | loaded_class_pkgs->append(package_entry); |
1022 | } |
1023 | } |
1024 | } |
1025 | } |
1026 | |
1027 | |
1028 | // Allocate objArray and fill with java.lang.String |
1029 | objArrayOop r = oopFactory::new_objArray(vmClasses::String_klass(), |
1030 | loaded_class_pkgs->length(), CHECK_NULL__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return __null; (void)(0); |
1031 | objArrayHandle result(THREAD__the_thread__, r); |
1032 | for (int x = 0; x < loaded_class_pkgs->length(); x++) { |
1033 | PackageEntry* package_entry = loaded_class_pkgs->at(x); |
1034 | Handle str = java_lang_String::create_from_symbol(package_entry->name(), CHECK_NULL__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return __null; (void)(0); |
1035 | result->obj_at_put(x, str()); |
1036 | } |
1037 | return result(); |
1038 | } |
1039 | |
1040 | // caller needs ResourceMark |
1041 | const char* ClassLoader::file_name_for_class_name(const char* class_name, |
1042 | int class_name_len) { |
1043 | assert(class_name != NULL, "invariant")do { if (!(class_name != __null)) { (*g_assert_poison) = 'X'; ; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1043, "assert(" "class_name != __null" ") failed", "invariant" ); ::breakpoint(); } } while (0); |
1044 | assert((int)strlen(class_name) == class_name_len, "invariant")do { if (!((int)strlen(class_name) == class_name_len)) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1044, "assert(" "(int)strlen(class_name) == class_name_len" ") failed", "invariant"); ::breakpoint(); } } while (0); |
1045 | |
1046 | static const char class_suffix[] = ".class"; |
1047 | size_t class_suffix_len = sizeof(class_suffix); |
1048 | |
1049 | char* const file_name = NEW_RESOURCE_ARRAY(char,(char*) resource_allocate_bytes((class_name_len + class_suffix_len ) * sizeof(char)) |
1050 | class_name_len +(char*) resource_allocate_bytes((class_name_len + class_suffix_len ) * sizeof(char)) |
1051 | class_suffix_len)(char*) resource_allocate_bytes((class_name_len + class_suffix_len ) * sizeof(char)); // includes term NULL |
1052 | |
1053 | strncpy(file_name, class_name, class_name_len); |
1054 | strncpy(&file_name[class_name_len], class_suffix, class_suffix_len); |
1055 | |
1056 | return file_name; |
1057 | } |
1058 | |
1059 | ClassPathEntry* find_first_module_cpe(ModuleEntry* mod_entry, |
1060 | const GrowableArray<ModuleClassPathList*>* const module_list) { |
1061 | int num_of_entries = module_list->length(); |
1062 | const Symbol* class_module_name = mod_entry->name(); |
1063 | |
1064 | // Loop through all the modules in either the patch-module or exploded entries looking for module |
1065 | for (int i = 0; i < num_of_entries; i++) { |
1066 | ModuleClassPathList* module_cpl = module_list->at(i); |
1067 | Symbol* module_cpl_name = module_cpl->module_name(); |
1068 | |
1069 | if (module_cpl_name->fast_compare(class_module_name) == 0) { |
1070 | // Class' module has been located. |
1071 | return module_cpl->module_first_entry(); |
1072 | } |
1073 | } |
1074 | return NULL__null; |
1075 | } |
1076 | |
1077 | |
1078 | // Search either the patch-module or exploded build entries for class. |
1079 | ClassFileStream* ClassLoader::search_module_entries(JavaThread* current, |
1080 | const GrowableArray<ModuleClassPathList*>* const module_list, |
1081 | const char* const class_name, |
1082 | const char* const file_name) { |
1083 | ClassFileStream* stream = NULL__null; |
1084 | |
1085 | // Find the class' defining module in the boot loader's module entry table |
1086 | TempNewSymbol class_name_symbol = SymbolTable::new_symbol(class_name); |
1087 | TempNewSymbol pkg_name = package_from_class_name(class_name_symbol); |
1088 | PackageEntry* pkg_entry = get_package_entry(pkg_name, ClassLoaderData::the_null_class_loader_data()); |
1089 | ModuleEntry* mod_entry = (pkg_entry != NULL__null) ? pkg_entry->module() : NULL__null; |
1090 | |
1091 | // If the module system has not defined java.base yet, then |
1092 | // classes loaded are assumed to be defined to java.base. |
1093 | // When java.base is eventually defined by the module system, |
1094 | // all packages of classes that have been previously loaded |
1095 | // are verified in ModuleEntryTable::verify_javabase_packages(). |
1096 | if (!Universe::is_module_initialized() && |
1097 | !ModuleEntryTable::javabase_defined() && |
1098 | mod_entry == NULL__null) { |
1099 | mod_entry = ModuleEntryTable::javabase_moduleEntry(); |
1100 | } |
1101 | |
1102 | // The module must be a named module |
1103 | ClassPathEntry* e = NULL__null; |
1104 | if (mod_entry != NULL__null && mod_entry->is_named()) { |
1105 | if (module_list == _exploded_entries) { |
1106 | // The exploded build entries can be added to at any time so a lock is |
1107 | // needed when searching them. |
1108 | assert(!ClassLoader::has_jrt_entry(), "Must be exploded build")do { if (!(!ClassLoader::has_jrt_entry())) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1108, "assert(" "!ClassLoader::has_jrt_entry()" ") failed", "Must be exploded build"); ::breakpoint(); } } while (0); |
1109 | MutexLocker ml(current, Module_lock); |
1110 | e = find_first_module_cpe(mod_entry, module_list); |
1111 | } else { |
1112 | e = find_first_module_cpe(mod_entry, module_list); |
1113 | } |
1114 | } |
1115 | |
1116 | // Try to load the class from the module's ClassPathEntry list. |
1117 | while (e != NULL__null) { |
1118 | stream = e->open_stream(current, file_name); |
1119 | // No context.check is required since CDS is not supported |
1120 | // for an exploded modules build or if --patch-module is specified. |
1121 | if (NULL__null != stream) { |
1122 | return stream; |
1123 | } |
1124 | e = e->next(); |
1125 | } |
1126 | // If the module was located, break out even if the class was not |
1127 | // located successfully from that module's ClassPathEntry list. |
1128 | // There will not be another valid entry for that module. |
1129 | return NULL__null; |
1130 | } |
1131 | |
1132 | // Called by the boot classloader to load classes |
1133 | InstanceKlass* ClassLoader::load_class(Symbol* name, bool search_append_only, TRAPSJavaThread* __the_thread__) { |
1134 | assert(name != NULL, "invariant")do { if (!(name != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1134, "assert(" "name != __null" ") failed", "invariant"); :: breakpoint(); } } while (0); |
1135 | |
1136 | ResourceMark rm(THREAD__the_thread__); |
1137 | HandleMark hm(THREAD__the_thread__); |
1138 | |
1139 | const char* const class_name = name->as_C_string(); |
1140 | |
1141 | EventMarkClassLoading m("Loading class %s", class_name); |
1142 | |
1143 | const char* const file_name = file_name_for_class_name(class_name, |
1144 | name->utf8_length()); |
1145 | assert(file_name != NULL, "invariant")do { if (!(file_name != __null)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1145, "assert(" "file_name != __null" ") failed", "invariant" ); ::breakpoint(); } } while (0); |
1146 | |
1147 | // Lookup stream for parsing .class file |
1148 | ClassFileStream* stream = NULL__null; |
1149 | s2 classpath_index = 0; |
1150 | ClassPathEntry* e = NULL__null; |
1151 | |
1152 | // If search_append_only is true, boot loader visibility boundaries are |
1153 | // set to be _first_append_entry to the end. This includes: |
1154 | // [-Xbootclasspath/a]; [jvmti appended entries] |
1155 | // |
1156 | // If search_append_only is false, boot loader visibility boundaries are |
1157 | // set to be the --patch-module entries plus the base piece. This includes: |
1158 | // [--patch-module=<module>=<file>(<pathsep><file>)*]; [jimage | exploded module build] |
1159 | // |
1160 | |
1161 | // Load Attempt #1: --patch-module |
1162 | // Determine the class' defining module. If it appears in the _patch_mod_entries, |
1163 | // attempt to load the class from those locations specific to the module. |
1164 | // Specifications to --patch-module can contain a partial number of classes |
1165 | // that are part of the overall module definition. So if a particular class is not |
1166 | // found within its module specification, the search should continue to Load Attempt #2. |
1167 | // Note: The --patch-module entries are never searched if the boot loader's |
1168 | // visibility boundary is limited to only searching the append entries. |
1169 | if (_patch_mod_entries != NULL__null && !search_append_only) { |
1170 | // At CDS dump time, the --patch-module entries are ignored. That means a |
1171 | // class is still loaded from the runtime image even if it might |
1172 | // appear in the _patch_mod_entries. The runtime shared class visibility |
1173 | // check will determine if a shared class is visible based on the runtime |
1174 | // environemnt, including the runtime --patch-module setting. |
1175 | // |
1176 | // DynamicDumpSharedSpaces requires UseSharedSpaces to be enabled. Since --patch-module |
1177 | // is not supported with UseSharedSpaces, it is not supported with DynamicDumpSharedSpaces. |
1178 | assert(!DynamicDumpSharedSpaces, "sanity")do { if (!(!DynamicDumpSharedSpaces)) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1178, "assert(" "!DynamicDumpSharedSpaces" ") failed", "sanity" ); ::breakpoint(); } } while (0); |
1179 | if (!DumpSharedSpaces) { |
1180 | stream = search_module_entries(THREAD__the_thread__, _patch_mod_entries, class_name, file_name); |
1181 | } |
1182 | } |
1183 | |
1184 | // Load Attempt #2: [jimage | exploded build] |
1185 | if (!search_append_only && (NULL__null == stream)) { |
1186 | if (has_jrt_entry()) { |
1187 | e = _jrt_entry; |
Value stored to 'e' is never read | |
1188 | stream = _jrt_entry->open_stream(THREAD__the_thread__, file_name); |
1189 | } else { |
1190 | // Exploded build - attempt to locate class in its defining module's location. |
1191 | assert(_exploded_entries != NULL, "No exploded build entries present")do { if (!(_exploded_entries != __null)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1191, "assert(" "_exploded_entries != __null" ") failed", "No exploded build entries present" ); ::breakpoint(); } } while (0); |
1192 | stream = search_module_entries(THREAD__the_thread__, _exploded_entries, class_name, file_name); |
1193 | } |
1194 | } |
1195 | |
1196 | // Load Attempt #3: [-Xbootclasspath/a]; [jvmti appended entries] |
1197 | if (search_append_only && (NULL__null == stream)) { |
1198 | // For the boot loader append path search, the starting classpath_index |
1199 | // for the appended piece is always 1 to account for either the |
1200 | // _jrt_entry or the _exploded_entries. |
1201 | assert(classpath_index == 0, "The classpath_index has been incremented incorrectly")do { if (!(classpath_index == 0)) { (*g_assert_poison) = 'X'; ; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1201, "assert(" "classpath_index == 0" ") failed", "The classpath_index has been incremented incorrectly" ); ::breakpoint(); } } while (0); |
1202 | classpath_index = 1; |
1203 | |
1204 | e = first_append_entry(); |
1205 | while (e != NULL__null) { |
1206 | stream = e->open_stream(THREAD__the_thread__, file_name); |
1207 | if (NULL__null != stream) { |
1208 | break; |
1209 | } |
1210 | e = e->next(); |
1211 | ++classpath_index; |
1212 | } |
1213 | } |
1214 | |
1215 | if (NULL__null == stream) { |
1216 | return NULL__null; |
1217 | } |
1218 | |
1219 | stream->set_verify(ClassLoaderExt::should_verify(classpath_index)); |
1220 | |
1221 | ClassLoaderData* loader_data = ClassLoaderData::the_null_class_loader_data(); |
1222 | Handle protection_domain; |
1223 | ClassLoadInfo cl_info(protection_domain); |
1224 | |
1225 | InstanceKlass* result = KlassFactory::create_from_stream(stream, |
1226 | name, |
1227 | loader_data, |
1228 | cl_info, |
1229 | CHECK_NULL__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return __null; (void)(0); |
1230 | result->set_classpath_index(classpath_index); |
1231 | return result; |
1232 | } |
1233 | |
1234 | #if INCLUDE_CDS1 |
1235 | char* ClassLoader::skip_uri_protocol(char* source) { |
1236 | if (strncmp(source, "file:", 5) == 0) { |
1237 | // file: protocol path could start with file:/ or file:/// |
1238 | // locate the char after all the forward slashes |
1239 | int offset = 5; |
1240 | while (*(source + offset) == '/') { |
1241 | offset++; |
1242 | } |
1243 | source += offset; |
1244 | // for non-windows platforms, move back one char as the path begins with a '/' |
1245 | #ifndef _WINDOWS |
1246 | source -= 1; |
1247 | #endif |
1248 | } else if (strncmp(source, "jrt:/", 5) == 0) { |
1249 | source += 5; |
1250 | } |
1251 | return source; |
1252 | } |
1253 | |
1254 | // Record the shared classpath index and loader type for classes loaded |
1255 | // by the builtin loaders at dump time. |
1256 | void ClassLoader::record_result(JavaThread* current, InstanceKlass* ik, |
1257 | const ClassFileStream* stream, bool redefined) { |
1258 | Arguments::assert_is_dumping_archive(); |
1259 | assert(stream != NULL, "sanity")do { if (!(stream != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1259, "assert(" "stream != __null" ") failed", "sanity"); :: breakpoint(); } } while (0); |
1260 | |
1261 | if (ik->is_hidden()) { |
1262 | // We do not archive hidden classes. |
1263 | return; |
1264 | } |
1265 | |
1266 | oop loader = ik->class_loader(); |
1267 | char* src = (char*)stream->source(); |
1268 | if (src == NULL__null) { |
1269 | if (loader == NULL__null) { |
1270 | // JFR classes |
1271 | ik->set_shared_classpath_index(0); |
1272 | ik->set_shared_class_loader_type(ClassLoader::BOOT_LOADER); |
1273 | } |
1274 | return; |
1275 | } |
1276 | |
1277 | assert(has_jrt_entry(), "CDS dumping does not support exploded JDK build")do { if (!(has_jrt_entry())) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1277, "assert(" "has_jrt_entry()" ") failed", "CDS dumping does not support exploded JDK build" ); ::breakpoint(); } } while (0); |
1278 | |
1279 | ResourceMark rm(current); |
1280 | int classpath_index = -1; |
1281 | PackageEntry* pkg_entry = ik->package(); |
1282 | |
1283 | if (FileMapInfo::get_number_of_shared_paths() > 0) { |
1284 | // Save the path from the file: protocol or the module name from the jrt: protocol |
1285 | // if no protocol prefix is found, path is the same as stream->source(). This path |
1286 | // must be valid since the class has been successfully parsed. |
1287 | char* path = skip_uri_protocol(src); |
1288 | assert(path != NULL, "sanity")do { if (!(path != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1288, "assert(" "path != __null" ") failed", "sanity"); ::breakpoint (); } } while (0); |
1289 | for (int i = 0; i < FileMapInfo::get_number_of_shared_paths(); i++) { |
1290 | SharedClassPathEntry* ent = FileMapInfo::shared_path(i); |
1291 | // A shared path has been validated during its creation in ClassLoader::create_class_path_entry(), |
1292 | // it must be valid here. |
1293 | assert(ent->name() != NULL, "sanity")do { if (!(ent->name() != __null)) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1293, "assert(" "ent->name() != __null" ") failed", "sanity" ); ::breakpoint(); } } while (0); |
1294 | // If the path (from the class stream source) is the same as the shared |
1295 | // class or module path, then we have a match. |
1296 | // src may come from the App/Platform class loaders, which would canonicalize |
1297 | // the file name. We cannot use strcmp to check for equality against ent->name(). |
1298 | // We must use os::same_files (which is faster than canonicalizing ent->name()). |
1299 | if (os::same_files(ent->name(), path)) { |
1300 | // NULL pkg_entry and pkg_entry in an unnamed module implies the class |
1301 | // is from the -cp or boot loader append path which consists of -Xbootclasspath/a |
1302 | // and jvmti appended entries. |
1303 | if ((pkg_entry == NULL__null) || (pkg_entry->in_unnamed_module())) { |
1304 | // Ensure the index is within the -cp range before assigning |
1305 | // to the classpath_index. |
1306 | if (SystemDictionary::is_system_class_loader(loader) && |
1307 | (i >= ClassLoaderExt::app_class_paths_start_index()) && |
1308 | (i < ClassLoaderExt::app_module_paths_start_index())) { |
1309 | classpath_index = i; |
1310 | break; |
1311 | } else { |
1312 | if ((i >= 1) && |
1313 | (i < ClassLoaderExt::app_class_paths_start_index())) { |
1314 | // The class must be from boot loader append path which consists of |
1315 | // -Xbootclasspath/a and jvmti appended entries. |
1316 | assert(loader == NULL, "sanity")do { if (!(loader == __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1316, "assert(" "loader == __null" ") failed", "sanity"); :: breakpoint(); } } while (0); |
1317 | classpath_index = i; |
1318 | break; |
1319 | } |
1320 | } |
1321 | } else { |
1322 | // A class from a named module from the --module-path. Ensure the index is |
1323 | // within the --module-path range before assigning to the classpath_index. |
1324 | if ((pkg_entry != NULL__null) && !(pkg_entry->in_unnamed_module()) && (i > 0)) { |
1325 | if (i >= ClassLoaderExt::app_module_paths_start_index() && |
1326 | i < FileMapInfo::get_number_of_shared_paths()) { |
1327 | classpath_index = i; |
1328 | break; |
1329 | } |
1330 | } |
1331 | } |
1332 | } |
1333 | // for index 0 and the stream->source() is the modules image or has the jrt: protocol. |
1334 | // The class must be from the runtime modules image. |
1335 | if (i == 0 && (stream->from_boot_loader_modules_image() || string_starts_with(src, "jrt:"))) { |
1336 | classpath_index = i; |
1337 | break; |
1338 | } |
1339 | } |
1340 | |
1341 | // No path entry found for this class: most likely a shared class loaded by the |
1342 | // user defined classloader. |
1343 | if (classpath_index < 0 && !SystemDictionaryShared::is_builtin_loader(ik->class_loader_data())) { |
1344 | assert(ik->shared_classpath_index() < 0, "not assigned yet")do { if (!(ik->shared_classpath_index() < 0)) { (*g_assert_poison ) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1344, "assert(" "ik->shared_classpath_index() < 0" ") failed" , "not assigned yet"); ::breakpoint(); } } while (0); |
1345 | ik->set_shared_classpath_index(UNREGISTERED_INDEX-9999); |
1346 | SystemDictionaryShared::set_shared_class_misc_info(ik, (ClassFileStream*)stream); |
1347 | return; |
1348 | } |
1349 | } else { |
1350 | // The shared path table is set up after module system initialization. |
1351 | // The path table contains no entry before that. Any classes loaded prior |
1352 | // to the setup of the shared path table must be from the modules image. |
1353 | assert(stream->from_boot_loader_modules_image(), "stream must be loaded by boot loader from modules image")do { if (!(stream->from_boot_loader_modules_image())) { (* g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1353, "assert(" "stream->from_boot_loader_modules_image()" ") failed", "stream must be loaded by boot loader from modules image" ); ::breakpoint(); } } while (0); |
1354 | assert(FileMapInfo::get_number_of_shared_paths() == 0, "shared path table must not have been setup")do { if (!(FileMapInfo::get_number_of_shared_paths() == 0)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1354, "assert(" "FileMapInfo::get_number_of_shared_paths() == 0" ") failed", "shared path table must not have been setup"); :: breakpoint(); } } while (0); |
1355 | classpath_index = 0; |
1356 | } |
1357 | |
1358 | const char* const class_name = ik->name()->as_C_string(); |
1359 | const char* const file_name = file_name_for_class_name(class_name, |
1360 | ik->name()->utf8_length()); |
1361 | assert(file_name != NULL, "invariant")do { if (!(file_name != __null)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1361, "assert(" "file_name != __null" ") failed", "invariant" ); ::breakpoint(); } } while (0); |
1362 | |
1363 | ClassLoaderExt::record_result(classpath_index, ik, redefined); |
1364 | } |
1365 | #endif // INCLUDE_CDS |
1366 | |
1367 | // Initialize the class loader's access to methods in libzip. Parse and |
1368 | // process the boot classpath into a list ClassPathEntry objects. Once |
1369 | // this list has been created, it must not change order (see class PackageInfo) |
1370 | // it can be appended to and is by jvmti. |
1371 | |
1372 | void ClassLoader::initialize(TRAPSJavaThread* __the_thread__) { |
1373 | if (UsePerfData) { |
1374 | // jvmstat performance counters |
1375 | NEWPERFTICKCOUNTER(_perf_accumulated_time, SUN_CLS, "time"){_perf_accumulated_time = PerfDataManager::create_counter(SUN_CLS , "time", PerfData::U_Ticks,__the_thread__); if ((((ThreadShadow *)__the_thread__)->has_pending_exception())) return ; (void )(0);}; |
1376 | NEWPERFTICKCOUNTER(_perf_class_init_time, SUN_CLS, "classInitTime"){_perf_class_init_time = PerfDataManager::create_counter(SUN_CLS , "classInitTime", PerfData::U_Ticks,__the_thread__); if (((( ThreadShadow*)__the_thread__)->has_pending_exception())) return ; (void)(0);}; |
1377 | NEWPERFTICKCOUNTER(_perf_class_init_selftime, SUN_CLS, "classInitTime.self"){_perf_class_init_selftime = PerfDataManager::create_counter( SUN_CLS, "classInitTime.self", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1378 | NEWPERFTICKCOUNTER(_perf_class_verify_time, SUN_CLS, "classVerifyTime"){_perf_class_verify_time = PerfDataManager::create_counter(SUN_CLS , "classVerifyTime", PerfData::U_Ticks,__the_thread__); if (( ((ThreadShadow*)__the_thread__)->has_pending_exception())) return ; (void)(0);}; |
1379 | NEWPERFTICKCOUNTER(_perf_class_verify_selftime, SUN_CLS, "classVerifyTime.self"){_perf_class_verify_selftime = PerfDataManager::create_counter (SUN_CLS, "classVerifyTime.self", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1380 | NEWPERFTICKCOUNTER(_perf_class_link_time, SUN_CLS, "classLinkedTime"){_perf_class_link_time = PerfDataManager::create_counter(SUN_CLS , "classLinkedTime", PerfData::U_Ticks,__the_thread__); if (( ((ThreadShadow*)__the_thread__)->has_pending_exception())) return ; (void)(0);}; |
1381 | NEWPERFTICKCOUNTER(_perf_class_link_selftime, SUN_CLS, "classLinkedTime.self"){_perf_class_link_selftime = PerfDataManager::create_counter( SUN_CLS, "classLinkedTime.self", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1382 | NEWPERFEVENTCOUNTER(_perf_classes_inited, SUN_CLS, "initializedClasses"){_perf_classes_inited = PerfDataManager::create_counter(SUN_CLS , "initializedClasses", PerfData::U_Events,__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception( ))) return ; (void)(0);}; |
1383 | NEWPERFEVENTCOUNTER(_perf_classes_linked, SUN_CLS, "linkedClasses"){_perf_classes_linked = PerfDataManager::create_counter(SUN_CLS , "linkedClasses", PerfData::U_Events,__the_thread__); if ((( (ThreadShadow*)__the_thread__)->has_pending_exception())) return ; (void)(0);}; |
1384 | NEWPERFEVENTCOUNTER(_perf_classes_verified, SUN_CLS, "verifiedClasses"){_perf_classes_verified = PerfDataManager::create_counter(SUN_CLS , "verifiedClasses", PerfData::U_Events,__the_thread__); if ( (((ThreadShadow*)__the_thread__)->has_pending_exception()) ) return ; (void)(0);}; |
1385 | |
1386 | NEWPERFTICKCOUNTER(_perf_sys_class_lookup_time, SUN_CLS, "lookupSysClassTime"){_perf_sys_class_lookup_time = PerfDataManager::create_counter (SUN_CLS, "lookupSysClassTime", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1387 | NEWPERFTICKCOUNTER(_perf_shared_classload_time, SUN_CLS, "sharedClassLoadTime"){_perf_shared_classload_time = PerfDataManager::create_counter (SUN_CLS, "sharedClassLoadTime", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1388 | NEWPERFTICKCOUNTER(_perf_sys_classload_time, SUN_CLS, "sysClassLoadTime"){_perf_sys_classload_time = PerfDataManager::create_counter(SUN_CLS , "sysClassLoadTime", PerfData::U_Ticks,__the_thread__); if ( (((ThreadShadow*)__the_thread__)->has_pending_exception()) ) return ; (void)(0);}; |
1389 | NEWPERFTICKCOUNTER(_perf_app_classload_time, SUN_CLS, "appClassLoadTime"){_perf_app_classload_time = PerfDataManager::create_counter(SUN_CLS , "appClassLoadTime", PerfData::U_Ticks,__the_thread__); if ( (((ThreadShadow*)__the_thread__)->has_pending_exception()) ) return ; (void)(0);}; |
1390 | NEWPERFTICKCOUNTER(_perf_app_classload_selftime, SUN_CLS, "appClassLoadTime.self"){_perf_app_classload_selftime = PerfDataManager::create_counter (SUN_CLS, "appClassLoadTime.self", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1391 | NEWPERFEVENTCOUNTER(_perf_app_classload_count, SUN_CLS, "appClassLoadCount"){_perf_app_classload_count = PerfDataManager::create_counter( SUN_CLS, "appClassLoadCount", PerfData::U_Events,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1392 | NEWPERFTICKCOUNTER(_perf_define_appclasses, SUN_CLS, "defineAppClasses"){_perf_define_appclasses = PerfDataManager::create_counter(SUN_CLS , "defineAppClasses", PerfData::U_Ticks,__the_thread__); if ( (((ThreadShadow*)__the_thread__)->has_pending_exception()) ) return ; (void)(0);}; |
1393 | NEWPERFTICKCOUNTER(_perf_define_appclass_time, SUN_CLS, "defineAppClassTime"){_perf_define_appclass_time = PerfDataManager::create_counter (SUN_CLS, "defineAppClassTime", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1394 | NEWPERFTICKCOUNTER(_perf_define_appclass_selftime, SUN_CLS, "defineAppClassTime.self"){_perf_define_appclass_selftime = PerfDataManager::create_counter (SUN_CLS, "defineAppClassTime.self", PerfData::U_Ticks,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1395 | NEWPERFBYTECOUNTER(_perf_app_classfile_bytes_read, SUN_CLS, "appClassBytes"){_perf_app_classfile_bytes_read = PerfDataManager::create_counter (SUN_CLS, "appClassBytes", PerfData::U_Bytes,__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1396 | NEWPERFBYTECOUNTER(_perf_sys_classfile_bytes_read, SUN_CLS, "sysClassBytes"){_perf_sys_classfile_bytes_read = PerfDataManager::create_counter (SUN_CLS, "sysClassBytes", PerfData::U_Bytes,__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1397 | |
1398 | NEWPERFEVENTCOUNTER(_unsafe_defineClassCallCounter, SUN_CLS, "unsafeDefineClassCalls"){_unsafe_defineClassCallCounter = PerfDataManager::create_counter (SUN_CLS, "unsafeDefineClassCalls", PerfData::U_Events,__the_thread__ ); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0);}; |
1399 | } |
1400 | |
1401 | // lookup java library entry points |
1402 | load_java_library(); |
1403 | // jimage library entry points are loaded below, in lookup_vm_options |
1404 | setup_bootstrap_search_path(THREAD__the_thread__); |
1405 | } |
1406 | |
1407 | char* lookup_vm_resource(JImageFile *jimage, const char *jimage_version, const char *path) { |
1408 | jlong size; |
1409 | JImageLocationRef location = (*JImageFindResource)(jimage, "java.base", jimage_version, path, &size); |
1410 | if (location == 0) |
1411 | return NULL__null; |
1412 | char *val = NEW_C_HEAP_ARRAY(char, size+1, mtClass)(char*) (AllocateHeap((size+1) * sizeof(char), mtClass)); |
1413 | (*JImageGetResource)(jimage, location, val, size); |
1414 | val[size] = '\0'; |
1415 | return val; |
1416 | } |
1417 | |
1418 | // Lookup VM options embedded in the modules jimage file |
1419 | char* ClassLoader::lookup_vm_options() { |
1420 | jint error; |
1421 | char modules_path[JVM_MAXPATHLEN4096 + 1]; |
1422 | const char* fileSep = os::file_separator(); |
1423 | |
1424 | // Initialize jimage library entry points |
1425 | load_jimage_library(); |
1426 | |
1427 | jio_snprintf(modules_path, JVM_MAXPATHLEN4096 + 1, "%s%slib%smodules", Arguments::get_java_home(), fileSep, fileSep); |
1428 | JImage_file =(*JImageOpen)(modules_path, &error); |
1429 | if (JImage_file == NULL__null) { |
1430 | return NULL__null; |
1431 | } |
1432 | |
1433 | const char *jimage_version = get_jimage_version_string(); |
1434 | char *options = lookup_vm_resource(JImage_file, jimage_version, "jdk/internal/vm/options"); |
1435 | return options; |
1436 | } |
1437 | |
1438 | #if INCLUDE_CDS1 |
1439 | void ClassLoader::initialize_shared_path(JavaThread* current) { |
1440 | if (Arguments::is_dumping_archive()) { |
1441 | ClassLoaderExt::setup_search_paths(current); |
1442 | } |
1443 | } |
1444 | |
1445 | void ClassLoader::initialize_module_path(TRAPSJavaThread* __the_thread__) { |
1446 | if (Arguments::is_dumping_archive()) { |
1447 | ClassLoaderExt::setup_module_paths(THREAD__the_thread__); |
1448 | FileMapInfo::allocate_shared_path_table(CHECK__the_thread__); if ((((ThreadShadow*)__the_thread__)->has_pending_exception ())) return ; (void)(0); |
1449 | } |
1450 | } |
1451 | |
1452 | // Helper function used by CDS code to get the number of module path |
1453 | // entries during shared classpath setup time. |
1454 | int ClassLoader::num_module_path_entries() { |
1455 | Arguments::assert_is_dumping_archive(); |
1456 | int num_entries = 0; |
1457 | ClassPathEntry* e= ClassLoader::_module_path_entries; |
1458 | while (e != NULL__null) { |
1459 | num_entries ++; |
1460 | e = e->next(); |
1461 | } |
1462 | return num_entries; |
1463 | } |
1464 | #endif |
1465 | |
1466 | jlong ClassLoader::classloader_time_ms() { |
1467 | return UsePerfData ? |
1468 | Management::ticks_to_ms(_perf_accumulated_time->get_value()) : -1; |
1469 | } |
1470 | |
1471 | jlong ClassLoader::class_init_count() { |
1472 | return UsePerfData ? _perf_classes_inited->get_value() : -1; |
1473 | } |
1474 | |
1475 | jlong ClassLoader::class_init_time_ms() { |
1476 | return UsePerfData ? |
1477 | Management::ticks_to_ms(_perf_class_init_time->get_value()) : -1; |
1478 | } |
1479 | |
1480 | jlong ClassLoader::class_verify_time_ms() { |
1481 | return UsePerfData ? |
1482 | Management::ticks_to_ms(_perf_class_verify_time->get_value()) : -1; |
1483 | } |
1484 | |
1485 | jlong ClassLoader::class_link_count() { |
1486 | return UsePerfData ? _perf_classes_linked->get_value() : -1; |
1487 | } |
1488 | |
1489 | jlong ClassLoader::class_link_time_ms() { |
1490 | return UsePerfData ? |
1491 | Management::ticks_to_ms(_perf_class_link_time->get_value()) : -1; |
1492 | } |
1493 | |
1494 | int ClassLoader::compute_Object_vtable() { |
1495 | // hardwired for JDK1.2 -- would need to duplicate class file parsing |
1496 | // code to determine actual value from file |
1497 | // Would be value '11' if finals were in vtable |
1498 | int JDK_1_2_Object_vtable_size = 5; |
1499 | return JDK_1_2_Object_vtable_size * vtableEntry::size(); |
1500 | } |
1501 | |
1502 | |
1503 | void classLoader_init1() { |
1504 | EXCEPTION_MARKExceptionMark __em; JavaThread* __the_thread__ = __em.thread( );; |
1505 | ClassLoader::initialize(THREAD__the_thread__); |
1506 | if (HAS_PENDING_EXCEPTION(((ThreadShadow*)__the_thread__)->has_pending_exception())) { |
1507 | vm_exit_during_initialization("ClassLoader::initialize() failed unexpectedly"); |
1508 | } |
1509 | } |
1510 | |
1511 | // Complete the ClassPathEntry setup for the boot loader |
1512 | void ClassLoader::classLoader_init2(JavaThread* current) { |
1513 | // Setup the list of module/path pairs for --patch-module processing |
1514 | // This must be done after the SymbolTable is created in order |
1515 | // to use fast_compare on module names instead of a string compare. |
1516 | if (Arguments::get_patch_mod_prefix() != NULL__null) { |
1517 | setup_patch_mod_entries(); |
1518 | } |
1519 | |
1520 | // Create the ModuleEntry for java.base (must occur after setup_patch_mod_entries |
1521 | // to successfully determine if java.base has been patched) |
1522 | create_javabase(); |
1523 | |
1524 | // Setup the initial java.base/path pair for the exploded build entries. |
1525 | // As more modules are defined during module system initialization, more |
1526 | // entries will be added to the exploded build array. |
1527 | if (!has_jrt_entry()) { |
1528 | assert(!DumpSharedSpaces, "DumpSharedSpaces not supported with exploded module builds")do { if (!(!DumpSharedSpaces)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1528, "assert(" "!DumpSharedSpaces" ") failed", "DumpSharedSpaces not supported with exploded module builds" ); ::breakpoint(); } } while (0); |
1529 | assert(!DynamicDumpSharedSpaces, "DynamicDumpSharedSpaces not supported with exploded module builds")do { if (!(!DynamicDumpSharedSpaces)) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1529, "assert(" "!DynamicDumpSharedSpaces" ") failed", "DynamicDumpSharedSpaces not supported with exploded module builds" ); ::breakpoint(); } } while (0); |
1530 | assert(!UseSharedSpaces, "UsedSharedSpaces not supported with exploded module builds")do { if (!(!UseSharedSpaces)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1530, "assert(" "!UseSharedSpaces" ") failed", "UsedSharedSpaces not supported with exploded module builds" ); ::breakpoint(); } } while (0); |
1531 | // Set up the boot loader's _exploded_entries list. Note that this gets |
1532 | // done before loading any classes, by the same thread that will |
1533 | // subsequently do the first class load. So, no lock is needed for this. |
1534 | assert(_exploded_entries == NULL, "Should only get initialized once")do { if (!(_exploded_entries == __null)) { (*g_assert_poison) = 'X';; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1534, "assert(" "_exploded_entries == __null" ") failed", "Should only get initialized once" ); ::breakpoint(); } } while (0); |
1535 | _exploded_entries = new (ResourceObj::C_HEAP, mtModule) |
1536 | GrowableArray<ModuleClassPathList*>(EXPLODED_ENTRY_SIZE, mtModule); |
1537 | add_to_exploded_build_list(current, vmSymbols::java_base()); |
1538 | } |
1539 | } |
1540 | |
1541 | char* ClassLoader::get_canonical_path(const char* orig, Thread* thread) { |
1542 | assert(orig != NULL, "bad arguments")do { if (!(orig != __null)) { (*g_assert_poison) = 'X';; report_vm_error ("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1542, "assert(" "orig != __null" ") failed", "bad arguments" ); ::breakpoint(); } } while (0); |
1543 | // caller needs to allocate ResourceMark for the following output buffer |
1544 | char* canonical_path = NEW_RESOURCE_ARRAY_IN_THREAD(thread, char, JVM_MAXPATHLEN)(char*) resource_allocate_bytes(thread, (4096 + 1) * sizeof(char )); |
1545 | ResourceMark rm(thread); |
1546 | // os::native_path writes into orig_copy |
1547 | char* orig_copy = NEW_RESOURCE_ARRAY_IN_THREAD(thread, char, strlen(orig)+1)(char*) resource_allocate_bytes(thread, (strlen(orig)+1) * sizeof (char)); |
1548 | strcpy(orig_copy, orig); |
1549 | if ((CanonicalizeEntry)(os::native_path(orig_copy), canonical_path, JVM_MAXPATHLEN4096 + 1) < 0) { |
1550 | return NULL__null; |
1551 | } |
1552 | return canonical_path; |
1553 | } |
1554 | |
1555 | void ClassLoader::create_javabase() { |
1556 | JavaThread* current = JavaThread::current(); |
1557 | |
1558 | // Create java.base's module entry for the boot |
1559 | // class loader prior to loading j.l.Ojbect. |
1560 | ClassLoaderData* null_cld = ClassLoaderData::the_null_class_loader_data(); |
1561 | |
1562 | // Get module entry table |
1563 | ModuleEntryTable* null_cld_modules = null_cld->modules(); |
1564 | if (null_cld_modules == NULL__null) { |
1565 | vm_exit_during_initialization("No ModuleEntryTable for the boot class loader"); |
1566 | } |
1567 | |
1568 | { |
1569 | MutexLocker ml(current, Module_lock); |
1570 | if (ModuleEntryTable::javabase_moduleEntry() == NULL__null) { // may have been inited by CDS. |
1571 | ModuleEntry* jb_module = null_cld_modules->locked_create_entry(Handle(), |
1572 | false, vmSymbols::java_base(), NULL__null, NULL__null, null_cld); |
1573 | if (jb_module == NULL__null) { |
1574 | vm_exit_during_initialization("Unable to create ModuleEntry for " JAVA_BASE_NAME"java.base"); |
1575 | } |
1576 | ModuleEntryTable::set_javabase_moduleEntry(jb_module); |
1577 | } |
1578 | } |
1579 | } |
1580 | |
1581 | // Please keep following two functions at end of this file. With them placed at top or in middle of the file, |
1582 | // they could get inlined by agressive compiler, an unknown trick, see bug 6966589. |
1583 | void PerfClassTraceTime::initialize() { |
1584 | if (!UsePerfData) return; |
1585 | |
1586 | if (_eventp != NULL__null) { |
1587 | // increment the event counter |
1588 | _eventp->inc(); |
1589 | } |
1590 | |
1591 | // stop the current active thread-local timer to measure inclusive time |
1592 | _prev_active_event = -1; |
1593 | for (int i=0; i < EVENT_TYPE_COUNT; i++) { |
1594 | if (_timers[i].is_active()) { |
1595 | assert(_prev_active_event == -1, "should have only one active timer")do { if (!(_prev_active_event == -1)) { (*g_assert_poison) = 'X' ;; report_vm_error("/home/daniel/Projects/java/jdk/src/hotspot/share/classfile/classLoader.cpp" , 1595, "assert(" "_prev_active_event == -1" ") failed", "should have only one active timer" ); ::breakpoint(); } } while (0); |
1596 | _prev_active_event = i; |
1597 | _timers[i].stop(); |
1598 | } |
1599 | } |
1600 | |
1601 | if (_recursion_counters == NULL__null || (_recursion_counters[_event_type])++ == 0) { |
1602 | // start the inclusive timer if not recursively called |
1603 | _t.start(); |
1604 | } |
1605 | |
1606 | // start thread-local timer of the given event type |
1607 | if (!_timers[_event_type].is_active()) { |
1608 | _timers[_event_type].start(); |
1609 | } |
1610 | } |
1611 | |
1612 | PerfClassTraceTime::~PerfClassTraceTime() { |
1613 | if (!UsePerfData) return; |
1614 | |
1615 | // stop the thread-local timer as the event completes |
1616 | // and resume the thread-local timer of the event next on the stack |
1617 | _timers[_event_type].stop(); |
1618 | jlong selftime = _timers[_event_type].ticks(); |
1619 | |
1620 | if (_prev_active_event >= 0) { |
1621 | _timers[_prev_active_event].start(); |
1622 | } |
1623 | |
1624 | if (_recursion_counters != NULL__null && --(_recursion_counters[_event_type]) > 0) return; |
1625 | |
1626 | // increment the counters only on the leaf call |
1627 | _t.stop(); |
1628 | _timep->inc(_t.ticks()); |
1629 | if (_selftimep != NULL__null) { |
1630 | _selftimep->inc(selftime); |
1631 | } |
1632 | // add all class loading related event selftime to the accumulated time counter |
1633 | ClassLoader::perf_accumulated_time()->inc(selftime); |
1634 | |
1635 | // reset the timer |
1636 | _timers[_event_type].reset(); |
1637 | } |