< prev index next >

src/hotspot/share/cds/heapShared.cpp

Print this page
*** 21,12 ***
--- 21,14 ---
   * questions.
   *
   */
  
  #include "cds/aotArtifactFinder.hpp"
+ #include "cds/aotCacheAccess.hpp"
  #include "cds/aotClassInitializer.hpp"
  #include "cds/aotClassLocation.hpp"
+ #include "cds/aotConstantPoolResolver.hpp"
  #include "cds/aotLogging.hpp"
  #include "cds/aotMetaspace.hpp"
  #include "cds/aotOopChecker.hpp"
  #include "cds/aotReferenceObjSupport.hpp"
  #include "cds/archiveBuilder.hpp"

*** 89,10 ***
--- 91,21 ---
    bool valid() {
      return klass_name != nullptr;
    }
  };
  
+ class HeapShared::ContextMark : public StackObj {
+   ResourceMark rm;
+ public:
+   ContextMark(const char* c) : rm{} {
+     _context->push(c);
+   }
+   ~ContextMark() {
+     _context->pop();
+   }
+ };
+ 
  DumpedInternedStrings *HeapShared::_dumped_interned_strings = nullptr;
  
  size_t HeapShared::_alloc_count[HeapShared::ALLOC_STAT_SLOTS];
  size_t HeapShared::_alloc_size[HeapShared::ALLOC_STAT_SLOTS];
  size_t HeapShared::_total_obj_count;

*** 122,10 ***
--- 135,11 ---
    {"jdk/internal/module/ArchivedModuleGraph",     "archivedModuleGraph"},
    {"java/util/ImmutableCollections",              "archivedObjects"},
    {"java/lang/ModuleLayer",                       "EMPTY_LAYER"},
    {"java/lang/module/Configuration",              "EMPTY_CONFIGURATION"},
    {"jdk/internal/math/FDBigInteger",              "archivedCaches"},
+   {"java/lang/reflect/Proxy$ProxyBuilder",        "archivedData"},    // FIXME -- requires AOTClassLinking
  
  #ifndef PRODUCT
    {nullptr, nullptr}, // Extra slot for -XX:ArchiveHeapTestClass
  #endif
    {nullptr, nullptr},

*** 139,11 ***
    {nullptr, nullptr},
  };
  
  KlassSubGraphInfo* HeapShared::_dump_time_special_subgraph;
  ArchivedKlassSubGraphInfoRecord* HeapShared::_run_time_special_subgraph;
! GrowableArrayCHeap<oop, mtClassShared>* HeapShared::_pending_roots = nullptr;
  GrowableArrayCHeap<OopHandle, mtClassShared>* HeapShared::_root_segments = nullptr;
  int HeapShared::_root_segment_max_size_elems;
  OopHandle HeapShared::_scratch_basic_type_mirrors[T_VOID+1];
  MetaspaceObjToOopHandleTable* HeapShared::_scratch_objects_table = nullptr;
  
--- 153,12 ---
    {nullptr, nullptr},
  };
  
  KlassSubGraphInfo* HeapShared::_dump_time_special_subgraph;
  ArchivedKlassSubGraphInfoRecord* HeapShared::_run_time_special_subgraph;
! GrowableArrayCHeap<OopHandle, mtClassShared>* HeapShared::_pending_roots = nullptr;
+ GrowableArrayCHeap<const char*, mtClassShared>* HeapShared::_context = nullptr;
  GrowableArrayCHeap<OopHandle, mtClassShared>* HeapShared::_root_segments = nullptr;
  int HeapShared::_root_segment_max_size_elems;
  OopHandle HeapShared::_scratch_basic_type_mirrors[T_VOID+1];
  MetaspaceObjToOopHandleTable* HeapShared::_scratch_objects_table = nullptr;
  

*** 251,25 ***
      assert(has_been_archived(obj), "must be");
    }
    // No GC should happen since we aren't scanning _pending_roots.
    assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
  
!   return _pending_roots->append(obj);
  }
  
  objArrayOop HeapShared::root_segment(int segment_idx) {
!   if (CDSConfig::is_dumping_heap()) {
      assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
    } else {
      assert(CDSConfig::is_using_archive(), "must be");
    }
  
    objArrayOop segment = (objArrayOop)_root_segments->at(segment_idx).resolve();
    assert(segment != nullptr, "should have been initialized");
    return segment;
  }
  
  void HeapShared::get_segment_indexes(int idx, int& seg_idx, int& int_idx) {
    assert(_root_segment_max_size_elems > 0, "sanity");
  
    // Try to avoid divisions for the common case.
    if (idx < _root_segment_max_size_elems) {
--- 266,181 ---
      assert(has_been_archived(obj), "must be");
    }
    // No GC should happen since we aren't scanning _pending_roots.
    assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
  
!   OopHandle oh(Universe::vm_global(), obj);
+   return _pending_roots->append(oh);
  }
  
  objArrayOop HeapShared::root_segment(int segment_idx) {
!   if (CDSConfig::is_dumping_heap() && !CDSConfig::is_dumping_final_static_archive()) {
      assert(Thread::current() == (Thread*)VMThread::vm_thread(), "should be in vm thread");
    } else {
      assert(CDSConfig::is_using_archive(), "must be");
    }
  
    objArrayOop segment = (objArrayOop)_root_segments->at(segment_idx).resolve();
    assert(segment != nullptr, "should have been initialized");
    return segment;
  }
  
+ class OrigToScratchObjectTable: public HashTable<OopHandle, OopHandle,
+     36137, // prime number
+     AnyObj::C_HEAP,
+     mtClassShared,
+     HeapShared::oop_handle_hash,
+     HeapShared::oop_handle_equals> {};
+ 
+ static OrigToScratchObjectTable* _orig_to_scratch_object_table = nullptr;
+ 
+ void HeapShared::track_scratch_object(oop orig_obj, oop scratch_obj) {
+   MutexLocker ml(ArchivedObjectTables_lock, Mutex::_no_safepoint_check_flag);
+   if (_orig_to_scratch_object_table == nullptr) {
+     _orig_to_scratch_object_table = new (mtClass)OrigToScratchObjectTable();
+   }
+ 
+   OopHandle orig_h(Universe::vm_global(), orig_obj);
+   OopHandle scratch_h(Universe::vm_global(), scratch_obj);
+   _orig_to_scratch_object_table->put_when_absent(orig_h, scratch_h);
+ }
+ 
+ oop HeapShared::orig_to_scratch_object(oop orig_obj) {
+   MutexLocker ml(ArchivedObjectTables_lock, Mutex::_no_safepoint_check_flag);
+   if (_orig_to_scratch_object_table != nullptr) {
+     OopHandle orig(&orig_obj);
+     OopHandle* v = _orig_to_scratch_object_table->get(orig);
+     if (v != nullptr) {
+       return v->resolve();
+     }
+   }
+   return nullptr;
+ }
+ 
+ // Permanent oops are used to support AOT-compiled methods, which may have in-line references
+ // to Strings and MH oops.
+ //
+ // At runtime, these oops are stored in _runtime_permanent_oops (which keeps them alive forever)
+ // and are accssed vis AOTCacheAccess::get_archived_object(int).
+ struct PermanentOopInfo {
+   int _index;       // Gets assigned only if HeapShared::get_archived_object_permanent_index() has been called on the object
+   int _heap_offset; // Offset of the object from the bottom of the archived heap.
+   PermanentOopInfo(int index, int heap_offset) : _index(index), _heap_offset(heap_offset) {}
+ };
+ 
+ class PermanentOopTable: public HashTable<OopHandle, PermanentOopInfo,
+     36137, // prime number
+     AnyObj::C_HEAP,
+     mtClassShared,
+     HeapShared::oop_handle_hash,
+     HeapShared::oop_handle_equals> {};
+ 
+ static int _dumptime_permanent_oop_count = 0;
+ static PermanentOopTable* _dumptime_permanent_oop_table = nullptr;
+ static GrowableArrayCHeap<OopHandle, mtClassShared>* _runtime_permanent_oops = nullptr;
+ 
+ // ArchiveHeapWriter adds each archived heap object to _dumptime_permanent_oop_table,
+ // so we can remember their offset (from the bottom of the archived heap).
+ void HeapShared::add_to_permanent_oop_table(oop obj, int offset) {
+   assert_at_safepoint();
+   if (_dumptime_permanent_oop_table == nullptr) {
+     _dumptime_permanent_oop_table = new (mtClass)PermanentOopTable();
+   }
+ 
+   PermanentOopInfo info(-1, offset);
+   OopHandle oh(Universe::vm_global(), obj);
+   _dumptime_permanent_oop_table->put_when_absent(oh, info);
+ }
+ 
+ // A permanent index is assigned to an archived object ONLY when
+ // the AOT compiler calls this function.
+ int HeapShared::get_archived_object_permanent_index(oop obj) {
+   MutexLocker ml(ArchivedObjectTables_lock, Mutex::_no_safepoint_check_flag);
+ 
+   if (!CDSConfig::is_dumping_heap()) {
+     return -1; // Called by the Leyden old workflow
+   }
+   if (_dumptime_permanent_oop_table == nullptr) {
+     return -1;
+   }
+ 
+   if (_orig_to_scratch_object_table != nullptr) {
+     OopHandle orig(&obj);
+     OopHandle* v = _orig_to_scratch_object_table->get(orig);
+     if (v != nullptr) {
+       obj = v->resolve();
+     }
+   }
+ 
+   OopHandle tmp(&obj);
+   PermanentOopInfo* info = _dumptime_permanent_oop_table->get(tmp);
+   if (info == nullptr) {
+     return -1;
+   } else {
+     if (info->_index < 0) {
+       info->_index = _dumptime_permanent_oop_count++;
+     }
+     return info->_index;
+   }
+ }
+ 
+ oop HeapShared::get_archived_object(int permanent_index) {
+   assert(permanent_index >= 0, "sanity");
+   assert(ArchiveHeapLoader::is_in_use(), "sanity");
+   assert(_runtime_permanent_oops != nullptr, "sanity");
+ 
+   return _runtime_permanent_oops->at(permanent_index).resolve();
+ }
+ 
+ // Remember all archived heap objects that have a permanent index.
+ //   table[i] = offset of oop whose permanent index is i.
+ void CachedCodeDirectoryInternal::dumptime_init_internal() {
+   const int count = _dumptime_permanent_oop_count;
+   if (count == 0) {
+     // Avoid confusing CDS code with zero-sized tables, just return.
+     log_info(cds)("No permanent oops");
+     _permanent_oop_count = count;
+     _permanent_oop_offsets = nullptr;
+     return;
+   }
+ 
+   int* table = (int*)AOTCacheAccess::allocate_aot_code_region(count * sizeof(int));
+   for (int i = 0; i < count; i++) {
+     table[count] = -1;
+   }
+   _dumptime_permanent_oop_table->iterate([&](OopHandle o, PermanentOopInfo& info) {
+     int index = info._index;
+     if (index >= 0) {
+       assert(index < count, "sanity");
+       table[index] = info._heap_offset;
+     }
+     return true; // continue
+   });
+ 
+   for (int i = 0; i < count; i++) {
+     assert(table[i] >= 0, "must be");
+   }
+ 
+   log_info(cds)("Dumped %d permanent oops", count);
+ 
+   _permanent_oop_count = count;
+   AOTCacheAccess::set_pointer(&_permanent_oop_offsets, table);
+ }
+ 
+ // This is called during the bootstrap of the production run, before any GC can happen.
+ // Record each permanent oop in a OopHandle for GC safety.
+ void CachedCodeDirectoryInternal::runtime_init_internal() {
+   int count = _permanent_oop_count;
+   int* table = _permanent_oop_offsets;
+   _runtime_permanent_oops = new GrowableArrayCHeap<OopHandle, mtClassShared>();
+   for (int i = 0; i < count; i++) {
+     oop obj = ArchiveHeapLoader::oop_from_offset(table[i]);
+     OopHandle oh(Universe::vm_global(), obj);
+     _runtime_permanent_oops->append(oh);
+   }
+ };
+ 
  void HeapShared::get_segment_indexes(int idx, int& seg_idx, int& int_idx) {
    assert(_root_segment_max_size_elems > 0, "sanity");
  
    // Try to avoid divisions for the common case.
    if (idx < _root_segment_max_size_elems) {

*** 437,19 ***
    return (objArrayOop)_scratch_objects_table->get_oop(src);
  }
  
  void HeapShared::init_dumping() {
    _scratch_objects_table = new (mtClass)MetaspaceObjToOopHandleTable();
!   _pending_roots = new GrowableArrayCHeap<oop, mtClassShared>(500);
  }
  
  void HeapShared::init_scratch_objects_for_basic_type_mirrors(TRAPS) {
    for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
      BasicType bt = (BasicType)i;
      if (!is_reference_type(bt)) {
        oop m = java_lang_Class::create_basic_type_mirror(type2name(bt), bt, CHECK);
        _scratch_basic_type_mirrors[i] = OopHandle(Universe::vm_global(), m);
      }
    }
  }
  
  // Given java_mirror that represents a (primitive or reference) type T,
--- 608,20 ---
    return (objArrayOop)_scratch_objects_table->get_oop(src);
  }
  
  void HeapShared::init_dumping() {
    _scratch_objects_table = new (mtClass)MetaspaceObjToOopHandleTable();
!   _pending_roots = new GrowableArrayCHeap<OopHandle, mtClassShared>(500);
  }
  
  void HeapShared::init_scratch_objects_for_basic_type_mirrors(TRAPS) {
    for (int i = T_BOOLEAN; i < T_VOID+1; i++) {
      BasicType bt = (BasicType)i;
      if (!is_reference_type(bt)) {
        oop m = java_lang_Class::create_basic_type_mirror(type2name(bt), bt, CHECK);
        _scratch_basic_type_mirrors[i] = OopHandle(Universe::vm_global(), m);
+       track_scratch_object(Universe::java_mirror(bt), m);
      }
    }
  }
  
  // Given java_mirror that represents a (primitive or reference) type T,

*** 486,10 ***
--- 658,11 ---
  oop HeapShared::scratch_java_mirror(Klass* k) {
    return _scratch_objects_table->get_oop(k);
  }
  
  void HeapShared::set_scratch_java_mirror(Klass* k, oop mirror) {
+   track_scratch_object(k->java_mirror(), mirror);
    _scratch_objects_table->set_oop(k, mirror);
  }
  
  void HeapShared::remove_scratch_objects(Klass* k) {
    // Klass is being deallocated. Java mirror can still be alive, and it should not

*** 501,10 ***
--- 674,19 ---
    }
    _scratch_objects_table->remove_oop(k);
    if (k->is_instance_klass()) {
      _scratch_objects_table->remove(InstanceKlass::cast(k)->constants());
    }
+   if (mirror != nullptr) {
+     OopHandle tmp(&mirror);
+     OopHandle* v = _orig_to_scratch_object_table->get(tmp);
+     if (v != nullptr) {
+       oop scratch_mirror = v->resolve();
+       java_lang_Class::set_klass(scratch_mirror, nullptr);
+       _orig_to_scratch_object_table->remove(tmp);
+     }
+   }
  }
  
  //TODO: we eventually want a more direct test for these kinds of things.
  //For example the JVM could record some bit of context from the creation
  //of the klass, such as who called the hidden class factory.  Using

*** 555,15 ***
        switch (fd.field_type()) {
        case T_OBJECT:
        case T_ARRAY:
          {
            oop field_obj = orig_mirror->obj_field(offset);
-           if (offset == java_lang_Class::reflection_data_offset()) {
-             // Class::reflectData use SoftReference, which cannot be archived. Set it
-             // to null and it will be recreated at runtime.
-             field_obj = nullptr;
-           }
            m->obj_field_put(offset, field_obj);
            if (field_obj != nullptr) {
              bool success = archive_reachable_objects_from(1, _dump_time_special_subgraph, field_obj);
              assert(success, "sanity");
            }
--- 737,10 ---

*** 634,10 ***
--- 811,18 ---
  
    if (CDSConfig::is_dumping_aot_linked_classes()) {
      java_lang_Class::set_module(scratch_m, java_lang_Class::module(orig_mirror));
      java_lang_Class::set_protection_domain(scratch_m, java_lang_Class::protection_domain(orig_mirror));
    }
+ 
+   Klass* k = java_lang_Class::as_Klass(orig_mirror); // is null Universe::void_mirror();
+   if (CDSConfig::is_dumping_reflection_data() &&
+       k != nullptr && k->is_instance_klass() &&
+       java_lang_Class::reflection_data(orig_mirror) != nullptr &&
+       AOTConstantPoolResolver::can_archive_reflection_data(InstanceKlass::cast(k))) {
+     java_lang_Class::set_reflection_data(scratch_m, java_lang_Class::reflection_data(orig_mirror));
+   }
  }
  
  static objArrayOop get_archived_resolved_references(InstanceKlass* src_ik) {
    if (SystemDictionaryShared::is_builtin_loader(src_ik->class_loader_data())) {
      objArrayOop rr = src_ik->constants()->resolved_references_or_null();

*** 692,10 ***
--- 877,11 ---
      NoSafepointVerifier nsv;
  
      // The special subgraph doesn't belong to any class. We use Object_klass() here just
      // for convenience.
      _dump_time_special_subgraph = init_subgraph_info(vmClasses::Object_klass(), false);
+     _context = new GrowableArrayCHeap<const char*, mtClassShared>(250);
  
      // Cache for recording where the archived objects are copied to
      create_archived_object_cache();
  
      if (UseCompressedOops || UseG1GC) {

*** 719,16 ***
  }
  
  void HeapShared::write_heap(ArchiveHeapInfo *heap_info) {
    {
      NoSafepointVerifier nsv;
!     CDSHeapVerifier::verify();
      check_special_subgraph_classes();
    }
  
    StringTable::write_shared_table();
!   ArchiveHeapWriter::write(_pending_roots, heap_info);
  
    ArchiveBuilder::OtherROAllocMark mark;
    write_subgraph_info_table();
  }
  
--- 905,23 ---
  }
  
  void HeapShared::write_heap(ArchiveHeapInfo *heap_info) {
    {
      NoSafepointVerifier nsv;
!     if (!SkipArchiveHeapVerification) {
+       CDSHeapVerifier::verify();
+     }
      check_special_subgraph_classes();
    }
  
    StringTable::write_shared_table();
!   GrowableArrayCHeap<oop, mtClassShared>* roots = new GrowableArrayCHeap<oop, mtClassShared>(_pending_roots->length());
+   for (int i = 0; i < _pending_roots->length(); i++) {
+     roots->append(_pending_roots->at(i).resolve());
+   }
+   ArchiveHeapWriter::write(roots, heap_info);
+   delete roots;
  
    ArchiveBuilder::OtherROAllocMark mark;
    write_subgraph_info_table();
  }
  

*** 736,10 ***
--- 929,16 ---
    oop m = scratch_java_mirror(orig_mirror);
    if (m != nullptr) { // nullptr if for custom class loader
      copy_java_mirror(orig_mirror, m);
      bool success = archive_reachable_objects_from(1, _dump_time_special_subgraph, m);
      assert(success, "sanity");
+ 
+     oop extra;
+     if ((extra = java_lang_Class::reflection_data(m)) != nullptr) {
+       success = archive_reachable_objects_from(1, _dump_time_special_subgraph, extra);
+       assert(success, "sanity");
+     }
    }
  }
  
  void HeapShared::scan_java_class(Klass* orig_k) {
    scan_java_mirror(orig_k->java_mirror());

*** 1130,10 ***
--- 1329,23 ---
  void HeapShared::resolve_classes(JavaThread* current) {
    assert(CDSConfig::is_using_archive(), "runtime only!");
    if (!ArchiveHeapLoader::is_in_use()) {
      return; // nothing to do
    }
+ 
+   if (!CDSConfig::is_using_aot_linked_classes()) {
+     assert( _run_time_special_subgraph != nullptr, "must be");
+     Array<Klass*>* klasses = _run_time_special_subgraph->subgraph_object_klasses();
+     if (klasses != nullptr) {
+       for (int i = 0; i < klasses->length(); i++) {
+         Klass* k = klasses->at(i);
+         ExceptionMark em(current); // no exception can happen here
+         resolve_or_init(k, /*do_init*/false, current);
+       }
+     }
+   }
+ 
    resolve_classes_for_subgraphs(current, archive_subgraph_entry_fields);
    resolve_classes_for_subgraphs(current, fmg_archive_subgraph_entry_fields);
  }
  
  void HeapShared::resolve_classes_for_subgraphs(JavaThread* current, ArchivableStaticFieldInfo fields[]) {

*** 1157,20 ***
    if (record == nullptr) {
     clear_archived_roots_of(k);
    }
  }
  
  void HeapShared::initialize_java_lang_invoke(TRAPS) {
    if (CDSConfig::is_using_aot_linked_classes() || CDSConfig::is_dumping_method_handles()) {
!     resolve_or_init("java/lang/invoke/Invokers$Holder", true, CHECK);
!     resolve_or_init("java/lang/invoke/MethodHandle", true, CHECK);
!     resolve_or_init("java/lang/invoke/MethodHandleNatives", true, CHECK);
!     resolve_or_init("java/lang/invoke/DirectMethodHandle$Holder", true, CHECK);
!     resolve_or_init("java/lang/invoke/DelegatingMethodHandle$Holder", true, CHECK);
!     resolve_or_init("java/lang/invoke/LambdaForm$Holder", true, CHECK);
!     resolve_or_init("java/lang/invoke/BoundMethodHandle$Species_L", true, CHECK);
    }
  }
  
  // Initialize the InstanceKlasses of objects that are reachable from the following roots:
  //   - interned strings
  //   - Klass::java_mirror() -- including aot-initialized mirrors such as those of Enum klasses.
--- 1369,40 ---
    if (record == nullptr) {
     clear_archived_roots_of(k);
    }
  }
  
+ static const char* java_lang_invoke_core_klasses[] = {
+   "java/lang/invoke/Invokers$Holder",
+   "java/lang/invoke/MethodHandle",
+   "java/lang/invoke/MethodHandleNatives",
+   "java/lang/invoke/DirectMethodHandle$Holder",
+   "java/lang/invoke/DelegatingMethodHandle$Holder",
+   "java/lang/invoke/LambdaForm$Holder",
+   "java/lang/invoke/BoundMethodHandle$Species_L",
+ };
+ 
  void HeapShared::initialize_java_lang_invoke(TRAPS) {
    if (CDSConfig::is_using_aot_linked_classes() || CDSConfig::is_dumping_method_handles()) {
!     int len = sizeof(java_lang_invoke_core_klasses)/sizeof(char*);
!     for (int i = 0; i < len; i++) {
!       resolve_or_init(java_lang_invoke_core_klasses[i], true, CHECK);
!     }
!   }
! }
! 
+ bool HeapShared::is_core_java_lang_invoke_klass(InstanceKlass* klass) {
+   // TODO: Crude, rewrite using Symbols or vmClasses instead
+   ResourceMark rm;
+   char* s2 = klass->name()->as_C_string();
+   int len = sizeof(java_lang_invoke_core_klasses)/sizeof(char*);
+   for (int i = 0; i < len; i++) {
+     if (strcmp(java_lang_invoke_core_klasses[i], s2) == 0) {
+       return true;
+     }
    }
+   return false;
  }
  
  // Initialize the InstanceKlasses of objects that are reachable from the following roots:
  //   - interned strings
  //   - Klass::java_mirror() -- including aot-initialized mirrors such as those of Enum klasses.

*** 1505,10 ***
--- 1737,24 ---
      vmClasses::Long_klass()->initialize(CHECK);
      vmClasses::Void_klass()->initialize(CHECK);
    }
  }
  
+ void HeapShared::exit_on_error() {
+   if (_context != nullptr) {
+     ResourceMark rm;
+     LogStream ls(Log(cds, heap)::error());
+     ls.print_cr("Context");
+     for (int i = 0; i < _context->length(); i++) {
+       const char* s = _context->at(i);
+       ls.print_cr("- %s", s);
+     }
+   }
+   debug_trace();
+   AOTMetaspace::unrecoverable_writing_error();
+ }
+ 
  // (1) If orig_obj has not been archived yet, archive it.
  // (2) If orig_obj has not been seen yet (since start_recording_subgraph() was called),
  //     trace all  objects that are reachable from it, and make sure these objects are archived.
  // (3) Record the klasses of all objects that are reachable from orig_obj (including those that
  //     were already archived when this function is called)

*** 1771,10 ***
--- 2017,15 ---
  }
  
  void HeapShared::verify_reachable_objects_from(oop obj) {
    _num_total_verifications ++;
    if (java_lang_Class::is_instance(obj)) {
+     Klass* k = java_lang_Class::as_Klass(obj);
+     if (RegeneratedClasses::has_been_regenerated(k)) {
+       k = RegeneratedClasses::get_regenerated_object(k);
+       obj = k->java_mirror();
+     }
      obj = scratch_java_mirror(obj);
      assert(obj != nullptr, "must be");
    }
    if (!has_been_seen_during_subgraph_recording(obj)) {
      set_has_been_seen_during_subgraph_recording(obj);

*** 2083,20 ***
--- 2334,22 ---
    for (int i = 0; fields[i].valid(); ) {
      ArchivableStaticFieldInfo* info = &fields[i];
      const char* klass_name = info->klass_name;
      start_recording_subgraph(info->klass, klass_name, is_full_module_graph);
  
+     ContextMark cm(klass_name);
      // If you have specified consecutive fields of the same klass in
      // fields[], these will be archived in the same
      // {start_recording_subgraph ... done_recording_subgraph} pass to
      // save time.
      for (; fields[i].valid(); i++) {
        ArchivableStaticFieldInfo* f = &fields[i];
        if (f->klass_name != klass_name) {
          break;
        }
  
+       ContextMark cm(f->field_name);
        archive_reachable_objects_from_static_field(f->klass, f->klass_name,
                                                    f->offset, f->field_name);
      }
      done_recording_subgraph(info->klass, klass_name);
    }
< prev index next >