< prev index next >

src/hotspot/share/ci/ciEnv.cpp

Print this page

   1 /*
   2  * Copyright (c) 1999, 2025, 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  *

  46 #include "compiler/compilerEvent.hpp"
  47 #include "compiler/compileTask.hpp"
  48 #include "compiler/disassembler.hpp"
  49 #include "gc/shared/collectedHeap.inline.hpp"
  50 #include "interpreter/bytecodeStream.hpp"
  51 #include "interpreter/linkResolver.hpp"
  52 #include "jfr/jfrEvents.hpp"
  53 #include "jvm.h"
  54 #include "logging/log.hpp"
  55 #include "memory/allocation.inline.hpp"
  56 #include "memory/oopFactory.hpp"
  57 #include "memory/resourceArea.hpp"
  58 #include "memory/universe.hpp"
  59 #include "oops/constantPool.inline.hpp"
  60 #include "oops/cpCache.inline.hpp"
  61 #include "oops/method.inline.hpp"
  62 #include "oops/methodData.hpp"
  63 #include "oops/objArrayKlass.hpp"
  64 #include "oops/objArrayOop.inline.hpp"
  65 #include "oops/oop.inline.hpp"

  66 #include "oops/resolvedIndyEntry.hpp"
  67 #include "oops/symbolHandle.hpp"
  68 #include "prims/jvmtiExport.hpp"
  69 #include "prims/methodHandles.hpp"
  70 #include "runtime/fieldDescriptor.inline.hpp"
  71 #include "runtime/handles.inline.hpp"
  72 #include "runtime/init.hpp"
  73 #include "runtime/javaThread.hpp"
  74 #include "runtime/jniHandles.inline.hpp"
  75 #include "runtime/reflection.hpp"
  76 #include "runtime/safepointVerifiers.hpp"
  77 #include "runtime/sharedRuntime.hpp"
  78 #include "utilities/dtrace.hpp"
  79 #include "utilities/macros.hpp"
  80 #ifdef COMPILER1
  81 #include "c1/c1_Runtime1.hpp"
  82 #endif
  83 #ifdef COMPILER2
  84 #include "opto/runtime.hpp"
  85 #endif

 472   // If we fail to find an array klass, look again for its element type.
 473   // The element type may be available either locally or via constraints.
 474   // In either case, if we can find the element type in the system dictionary,
 475   // we must build an array type around it.  The CI requires array klasses
 476   // to be loaded if their element klasses are loaded, except when memory
 477   // is exhausted.
 478   if (Signature::is_array(sym) &&
 479       (sym->char_at(1) == JVM_SIGNATURE_ARRAY || sym->char_at(1) == JVM_SIGNATURE_CLASS)) {
 480     // We have an unloaded array.
 481     // Build it on the fly if the element class exists.
 482     SignatureStream ss(sym, false);
 483     ss.skip_array_prefix(1);
 484     // Get element ciKlass recursively.
 485     ciKlass* elem_klass =
 486       get_klass_by_name_impl(accessing_klass,
 487                              cpool,
 488                              get_symbol(ss.as_symbol()),
 489                              require_local);
 490     if (elem_klass != nullptr && elem_klass->is_loaded()) {
 491       // Now make an array for it
 492       return ciObjArrayKlass::make_impl(elem_klass);
 493     }
 494   }
 495 
 496   if (found_klass == nullptr && !cpool.is_null() && cpool->has_preresolution()) {
 497     // Look inside the constant pool for pre-resolved class entries.
 498     for (int i = cpool->length() - 1; i >= 1; i--) {
 499       if (cpool->tag_at(i).is_klass()) {
 500         Klass* kls = cpool->resolved_klass_at(i);
 501         if (kls->name() == sym) {
 502           found_klass = kls;
 503           break;
 504         }
 505       }
 506     }
 507   }
 508 
 509   if (found_klass != nullptr) {
 510     // Found it.  Build a CI handle.
 511     return get_klass(found_klass);
 512   }

1338   }
1339 }
1340 
1341 // Read an object field.  Lookup is done by name only.
1342 static inline oop obj_field(oop obj, const char* name) {
1343     return ciReplay::obj_field(obj, name);
1344 }
1345 
1346 // Process a java.lang.invoke.LambdaForm object and record any dynamic locations.
1347 void ciEnv::record_lambdaform(Thread* thread, oop form) {
1348   assert(java_lang_invoke_LambdaForm::is_instance(form), "!");
1349 
1350   {
1351     // Check LambdaForm.vmentry field
1352     oop member = java_lang_invoke_LambdaForm::vmentry(form);
1353     RecordLocation fp0(this, "vmentry");
1354     record_member(thread, member);
1355   }
1356 
1357   // Check LambdaForm.names array
1358   objArrayOop names = (objArrayOop)obj_field(form, "names");


1359   if (names != nullptr) {
1360     RecordLocation lp0(this, "names");
1361     int len = names->length();
1362     for (int i = 0; i < len; ++i) {
1363       oop name = names->obj_at(i);
1364       RecordLocation lp1(this, "%d", i);
1365      // Check LambdaForm.names[i].function field
1366       RecordLocation lp2(this, "function");
1367       oop function = obj_field(name, "function");
1368       if (function != nullptr) {
1369         // Check LambdaForm.names[i].function.member field
1370         oop member = obj_field(function, "member");
1371         if (member != nullptr) {
1372           RecordLocation lp3(this, "member");
1373           record_member(thread, member);
1374         }
1375         // Check LambdaForm.names[i].function.resolvedHandle field
1376         oop mh = obj_field(function, "resolvedHandle");
1377         if (mh != nullptr) {
1378           RecordLocation lp3(this, "resolvedHandle");

   1 /*
   2  * Copyright (c) 1999, 2026, 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  *

  46 #include "compiler/compilerEvent.hpp"
  47 #include "compiler/compileTask.hpp"
  48 #include "compiler/disassembler.hpp"
  49 #include "gc/shared/collectedHeap.inline.hpp"
  50 #include "interpreter/bytecodeStream.hpp"
  51 #include "interpreter/linkResolver.hpp"
  52 #include "jfr/jfrEvents.hpp"
  53 #include "jvm.h"
  54 #include "logging/log.hpp"
  55 #include "memory/allocation.inline.hpp"
  56 #include "memory/oopFactory.hpp"
  57 #include "memory/resourceArea.hpp"
  58 #include "memory/universe.hpp"
  59 #include "oops/constantPool.inline.hpp"
  60 #include "oops/cpCache.inline.hpp"
  61 #include "oops/method.inline.hpp"
  62 #include "oops/methodData.hpp"
  63 #include "oops/objArrayKlass.hpp"
  64 #include "oops/objArrayOop.inline.hpp"
  65 #include "oops/oop.inline.hpp"
  66 #include "oops/oopCast.inline.hpp"
  67 #include "oops/resolvedIndyEntry.hpp"
  68 #include "oops/symbolHandle.hpp"
  69 #include "prims/jvmtiExport.hpp"
  70 #include "prims/methodHandles.hpp"
  71 #include "runtime/fieldDescriptor.inline.hpp"
  72 #include "runtime/handles.inline.hpp"
  73 #include "runtime/init.hpp"
  74 #include "runtime/javaThread.hpp"
  75 #include "runtime/jniHandles.inline.hpp"
  76 #include "runtime/reflection.hpp"
  77 #include "runtime/safepointVerifiers.hpp"
  78 #include "runtime/sharedRuntime.hpp"
  79 #include "utilities/dtrace.hpp"
  80 #include "utilities/macros.hpp"
  81 #ifdef COMPILER1
  82 #include "c1/c1_Runtime1.hpp"
  83 #endif
  84 #ifdef COMPILER2
  85 #include "opto/runtime.hpp"
  86 #endif

 473   // If we fail to find an array klass, look again for its element type.
 474   // The element type may be available either locally or via constraints.
 475   // In either case, if we can find the element type in the system dictionary,
 476   // we must build an array type around it.  The CI requires array klasses
 477   // to be loaded if their element klasses are loaded, except when memory
 478   // is exhausted.
 479   if (Signature::is_array(sym) &&
 480       (sym->char_at(1) == JVM_SIGNATURE_ARRAY || sym->char_at(1) == JVM_SIGNATURE_CLASS)) {
 481     // We have an unloaded array.
 482     // Build it on the fly if the element class exists.
 483     SignatureStream ss(sym, false);
 484     ss.skip_array_prefix(1);
 485     // Get element ciKlass recursively.
 486     ciKlass* elem_klass =
 487       get_klass_by_name_impl(accessing_klass,
 488                              cpool,
 489                              get_symbol(ss.as_symbol()),
 490                              require_local);
 491     if (elem_klass != nullptr && elem_klass->is_loaded()) {
 492       // Now make an array for it
 493       return ciArrayKlass::make(elem_klass);
 494     }
 495   }
 496 
 497   if (found_klass == nullptr && !cpool.is_null() && cpool->has_preresolution()) {
 498     // Look inside the constant pool for pre-resolved class entries.
 499     for (int i = cpool->length() - 1; i >= 1; i--) {
 500       if (cpool->tag_at(i).is_klass()) {
 501         Klass* kls = cpool->resolved_klass_at(i);
 502         if (kls->name() == sym) {
 503           found_klass = kls;
 504           break;
 505         }
 506       }
 507     }
 508   }
 509 
 510   if (found_klass != nullptr) {
 511     // Found it.  Build a CI handle.
 512     return get_klass(found_klass);
 513   }

1339   }
1340 }
1341 
1342 // Read an object field.  Lookup is done by name only.
1343 static inline oop obj_field(oop obj, const char* name) {
1344     return ciReplay::obj_field(obj, name);
1345 }
1346 
1347 // Process a java.lang.invoke.LambdaForm object and record any dynamic locations.
1348 void ciEnv::record_lambdaform(Thread* thread, oop form) {
1349   assert(java_lang_invoke_LambdaForm::is_instance(form), "!");
1350 
1351   {
1352     // Check LambdaForm.vmentry field
1353     oop member = java_lang_invoke_LambdaForm::vmentry(form);
1354     RecordLocation fp0(this, "vmentry");
1355     record_member(thread, member);
1356   }
1357 
1358   // Check LambdaForm.names array
1359   // The type of the array is Name[] and Name is an identity class,
1360   // so the array is always an array of references
1361   refArrayOop names = oop_cast<refArrayOop>(obj_field(form, "names"));
1362   if (names != nullptr) {
1363     RecordLocation lp0(this, "names");
1364     int len = names->length();
1365     for (int i = 0; i < len; ++i) {
1366       oop name = names->obj_at(i);
1367       RecordLocation lp1(this, "%d", i);
1368      // Check LambdaForm.names[i].function field
1369       RecordLocation lp2(this, "function");
1370       oop function = obj_field(name, "function");
1371       if (function != nullptr) {
1372         // Check LambdaForm.names[i].function.member field
1373         oop member = obj_field(function, "member");
1374         if (member != nullptr) {
1375           RecordLocation lp3(this, "member");
1376           record_member(thread, member);
1377         }
1378         // Check LambdaForm.names[i].function.resolvedHandle field
1379         oop mh = obj_field(function, "resolvedHandle");
1380         if (mh != nullptr) {
1381           RecordLocation lp3(this, "resolvedHandle");
< prev index next >