| File: | jdk/src/hotspot/share/classfile/classLoader.cpp |
| Warning: | line 371, column 10 Value stored to 'len' during its initialization 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; |
Value stored to 'len' during its initialization is never read | |
| 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; |
| 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 | } |