< prev index next >

src/hotspot/share/cds/archiveUtils.cpp

Print this page
@@ -31,33 +31,37 @@
  #include "cds/classListWriter.hpp"
  #include "cds/dynamicArchive.hpp"
  #include "cds/filemap.hpp"
  #include "cds/heapShared.hpp"
  #include "cds/metaspaceShared.hpp"
+ #include "classfile/classLoader.hpp"
  #include "classfile/systemDictionaryShared.hpp"
  #include "classfile/vmClasses.hpp"
  #include "interpreter/bootstrapInfo.hpp"
  #include "memory/metaspaceUtils.hpp"
  #include "memory/resourceArea.hpp"
  #include "oops/compressedOops.inline.hpp"
+ #include "oops/klass.inline.hpp"
  #include "runtime/arguments.hpp"
  #include "utilities/bitMap.inline.hpp"
  #include "utilities/debug.hpp"
  #include "utilities/formatBuffer.hpp"
  #include "utilities/globalDefinitions.hpp"
  
  CHeapBitMap* ArchivePtrMarker::_ptrmap = nullptr;
  CHeapBitMap* ArchivePtrMarker::_rw_ptrmap = nullptr;
  CHeapBitMap* ArchivePtrMarker::_ro_ptrmap = nullptr;
+ CHeapBitMap* ArchivePtrMarker::_cc_ptrmap = nullptr;
  VirtualSpace* ArchivePtrMarker::_vs;
  
  bool ArchivePtrMarker::_compacted;
  
  void ArchivePtrMarker::initialize(CHeapBitMap* ptrmap, VirtualSpace* vs) {
    assert(_ptrmap == nullptr, "initialize only once");
    assert(_rw_ptrmap == nullptr, "initialize only once");
    assert(_ro_ptrmap == nullptr, "initialize only once");
+   assert(_cc_ptrmap == nullptr, "initialize only once");
    _vs = vs;
    _compacted = false;
    _ptrmap = ptrmap;
  
    // Use this as initial guesstimate. We should need less space in the

@@ -69,39 +73,57 @@
  
    // We need one bit per pointer in the archive.
    _ptrmap->initialize(estimated_archive_size / sizeof(intptr_t));
  }
  
- void ArchivePtrMarker::initialize_rw_ro_maps(CHeapBitMap* rw_ptrmap, CHeapBitMap* ro_ptrmap) {
+ void ArchivePtrMarker::initialize_rw_ro_cc_maps(CHeapBitMap* rw_ptrmap, CHeapBitMap* ro_ptrmap, CHeapBitMap* cc_ptrmap) {
    address* rw_bottom = (address*)ArchiveBuilder::current()->rw_region()->base();
    address* ro_bottom = (address*)ArchiveBuilder::current()->ro_region()->base();
+   address* cc_bottom = (address*)ArchiveBuilder::current()->cc_region()->base();
  
    _rw_ptrmap = rw_ptrmap;
    _ro_ptrmap = ro_ptrmap;
+   _cc_ptrmap = cc_ptrmap;
  
    size_t rw_size = ArchiveBuilder::current()->rw_region()->used() / sizeof(address);
    size_t ro_size = ArchiveBuilder::current()->ro_region()->used() / sizeof(address);
+   size_t cc_size = ArchiveBuilder::current()->cc_region()->used() / sizeof(address);
    // ro_start is the first bit in _ptrmap that covers the pointer that would sit at ro_bottom.
    // E.g., if rw_bottom = (address*)100
    //          ro_bottom = (address*)116
    //       then for 64-bit platform:
    //          ro_start = ro_bottom - rw_bottom = (116 - 100) / sizeof(address) = 2;
    size_t ro_start = ro_bottom - rw_bottom;
+   size_t cc_start = cc_bottom - rw_bottom;
  
-   // Note: ptrmap is big enough only to cover the last pointer in ro_region.
+   // Note: ptrmap is big enough only to cover the last pointer in cc_region or ro_region.
    // See ArchivePtrMarker::compact()
+   if (ro_start + ro_size >_ptrmap->size()) {
+     ro_size = _ptrmap->size() - ro_start; // ro is smaller than we thought
+     cc_size = 0;                          // cc is empty
+   } else if (cc_size != 0 && cc_start + cc_size > _ptrmap->size()) {
+     cc_size = _ptrmap->size() - cc_start; // ro is smaller than we thought
+   }
+ 
+   assert(rw_size < _ptrmap->size(), "sanity");
+   assert(ro_size < _ptrmap->size(), "sanity");
+   assert(cc_size < _ptrmap->size(), "sanity");
+   assert(rw_size + ro_size + cc_size <= _ptrmap->size(), "sanity");
+ 
    _rw_ptrmap->initialize(rw_size);
-   _ro_ptrmap->initialize(_ptrmap->size() - ro_start);
+   _ro_ptrmap->initialize(ro_size);
+   _cc_ptrmap->initialize(cc_size);
  
-   for (size_t rw_bit = 0; rw_bit < _rw_ptrmap->size(); rw_bit++) {
-     _rw_ptrmap->at_put(rw_bit, _ptrmap->at(rw_bit));
+   for (size_t i = 0; i < rw_size; i++) {
+     _rw_ptrmap->at_put(i, _ptrmap->at(i));
    }
- 
-   for(size_t ro_bit = ro_start; ro_bit < _ptrmap->size(); ro_bit++) {
-     _ro_ptrmap->at_put(ro_bit-ro_start, _ptrmap->at(ro_bit));
+   for (size_t i = 0; i < ro_size; i++) {
+     _ro_ptrmap->at_put(i, _ptrmap->at(ro_start + i));
+   }
+   for (size_t i = 0; i < cc_size; i++) {
+     _cc_ptrmap->at_put(i, _ptrmap->at(cc_start + i));
    }
-   assert(_ptrmap->size() - ro_start == _ro_ptrmap->size(), "must be");
  }
  
  void ArchivePtrMarker::mark_pointer(address* ptr_loc) {
    assert(_ptrmap != nullptr, "not initialized");
    assert(!_compacted, "cannot mark anymore");

@@ -260,13 +282,14 @@
      ArchivePtrMarker::mark_pointer(p);
    }
  }
  
  void DumpRegion::print(size_t total_bytes) const {
+   char* base = used() > 0 ? ArchiveBuilder::current()->to_requested(_base) : nullptr;
    log_debug(cds)("%s space: " SIZE_FORMAT_W(9) " [ %4.1f%% of total] out of " SIZE_FORMAT_W(9) " bytes [%5.1f%% used] at " INTPTR_FORMAT,
                   _name, used(), percent_of(used(), total_bytes), reserved(), percent_of(used(), reserved()),
-                  p2i(ArchiveBuilder::current()->to_requested(_base)));
+                  p2i(base));
  }
  
  void DumpRegion::print_out_of_space_msg(const char* failing_region, size_t needed_bytes) {
    log_error(cds)("[%-8s] " PTR_FORMAT " - " PTR_FORMAT " capacity =%9d, allocated =%9d",
                   _name, p2i(_base), p2i(_top), int(_end - _base), int(_top - _base));

@@ -285,13 +308,14 @@
    _base = _top = _rs->base();
    _end = _rs->end();
  }
  
  void DumpRegion::pack(DumpRegion* next) {
-   assert(!is_packed(), "sanity");
-   _end = (char*)align_up(_top, MetaspaceShared::core_region_alignment());
-   _is_packed = true;
+   if (!is_packed()) {
+     _end = (char*)align_up(_top, MetaspaceShared::core_region_alignment());
+     _is_packed = true;
+   }
    if (next != nullptr) {
      next->_rs = _rs;
      next->_vs = _vs;
      next->_base = next->_top = this->_end;
      next->_end = _rs->end();

@@ -368,10 +392,89 @@
        }
      }
    }
  }
  
+ // Used in logging: "boot", "boot2", "plat", "app" and "unreg";
+ const char* ArchiveUtils::class_category(Klass* k) {
+   if (ArchiveBuilder::is_active() && ArchiveBuilder::current()->is_in_buffer_space(k)) {
+     k = ArchiveBuilder::current()->get_source_addr(k);
+   }
+ 
+   if (k->is_array_klass()) {
+     return "array";
+   } else {
+     oop loader = k->class_loader();
+     if (loader == nullptr) {
+       if (k->module() != nullptr &&
+           k->module()->name() != nullptr &&
+           k->module()->name()->equals("java.base")) {
+         return "boot"; // boot classes in java.base
+       } else {
+         return "boot2"; // boot classes outside of java.base
+       }
+     } else {
+       if (loader == SystemDictionary::java_platform_loader()) {
+         return "plat";
+       } else if (loader == SystemDictionary::java_system_loader()) {
+         return "app";
+       } else {
+         return "unreg";
+       }
+     }
+   }
+ }
+ 
+ // "boot", "platform", "app" or nullptr
+ const char* ArchiveUtils::builtin_loader_name_or_null(oop loader) {
+   if (loader == nullptr) {
+     return "boot";
+   } else if (loader == SystemDictionary::java_platform_loader()) {
+     return "platform";
+   } else if (loader == SystemDictionary::java_system_loader()) {
+     return "app";
+   } else {
+     return nullptr;
+   }
+ }
+ 
+ // "boot", "platform", "app". Asserts if not a built-in-loader
+ const char* ArchiveUtils::builtin_loader_name(oop loader) {
+   const char* name = builtin_loader_name_or_null(loader);
+   assert(name != nullptr, "must be a built-in loader");
+   return name;
+ }
+ 
+ bool ArchiveUtils::builtin_loader_from_type(const char* loader_type, oop* value_ret) {
+   if (strcmp(loader_type, "boot") == 0) {
+     *value_ret = nullptr;
+     return true;
+   } else if (strcmp(loader_type, "platform") == 0) {
+     *value_ret = SystemDictionary::java_platform_loader();
+     return true;
+   } else if (strcmp(loader_type, "app") == 0) {
+     *value_ret = SystemDictionary::java_system_loader();
+     return true;
+   } else {
+     DEBUG_ONLY(*value_ret = cast_to_oop((void*)badOopVal));
+     return false;
+   }
+ }
+ 
+ oop ArchiveUtils::builtin_loader_from_type(int loader_type) {
+   if (loader_type == ClassLoader::BOOT_LOADER) {
+     return nullptr;
+   } else if (loader_type == ClassLoader::PLATFORM_LOADER)  {
+     return SystemDictionary::java_platform_loader();
+   } else if (loader_type == ClassLoader::APP_LOADER) {
+     return SystemDictionary::java_system_loader();
+   } else {
+     ShouldNotReachHere();
+     return nullptr;
+   }
+ }
+ 
  size_t HeapRootSegments::size_in_bytes(size_t seg_idx) {
    assert(seg_idx < _count, "In range");
    return objArrayOopDesc::object_size(size_in_elems(seg_idx)) * HeapWordSize;
  }
  

@@ -387,6 +490,5 @@
  
  size_t HeapRootSegments::segment_offset(size_t seg_idx) {
    assert(seg_idx < _count, "In range");
    return _base_offset + seg_idx * _max_size_in_bytes;
  }
- 
< prev index next >