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