From 66514251c1acc7f4db92903eba516dd793addf37 Mon Sep 17 00:00:00 2001 From: Thomas Stuefe Date: Wed, 20 May 2020 15:56:39 +0200 Subject: [PATCH] 8244733: Add ResourceHashtable::xxx_if_absent Reviewed-by: coleenp, iklam, rehn, dholmes --- .../share/classfile/bytecodeAssembler.cpp | 10 ++--- .../share/classfile/classLoaderStats.cpp | 41 ++++++++----------- .../share/classfile/classLoaderStats.hpp | 4 +- .../classfile/systemDictionaryShared.cpp | 23 +++++------ .../share/jfr/periodic/jfrPeriodic.cpp | 24 +++++------ src/hotspot/share/utilities/resourceHash.hpp | 40 ++++++++++++++++++ .../gtest/utilities/test_resourceHash.cpp | 41 ++++++++++++++++++- 7 files changed, 127 insertions(+), 56 deletions(-) diff --git a/src/hotspot/share/classfile/bytecodeAssembler.cpp b/src/hotspot/share/classfile/bytecodeAssembler.cpp index 3d91cec69d3..399f9ee65a9 100644 --- a/src/hotspot/share/classfile/bytecodeAssembler.cpp +++ b/src/hotspot/share/classfile/bytecodeAssembler.cpp @@ -32,12 +32,12 @@ #include "utilities/bytes.hpp" u2 BytecodeConstantPool::find_or_add(BytecodeCPEntry const& bcpe) { - u2 index; - u2* probe = _indices.get(bcpe); - if (probe == NULL) { - index = _entries.length(); + + u2 index = _entries.length(); + bool created = false; + u2* probe = _indices.put_if_absent(bcpe, index, &created); + if (created) { _entries.append(bcpe); - _indices.put(bcpe, index); } else { index = *probe; } diff --git a/src/hotspot/share/classfile/classLoaderStats.cpp b/src/hotspot/share/classfile/classLoaderStats.cpp index 27ffd826c5a..89c001d0bec 100644 --- a/src/hotspot/share/classfile/classLoaderStats.cpp +++ b/src/hotspot/share/classfile/classLoaderStats.cpp @@ -44,27 +44,23 @@ public: } }; - void ClassLoaderStatsClosure::do_cld(ClassLoaderData* cld) { oop cl = cld->class_loader(); - ClassLoaderStats* cls; // The hashtable key is the ClassLoader oop since we want to account // for "real" classes and anonymous classes together - ClassLoaderStats** cls_ptr = _stats->get(cl); - if (cls_ptr == NULL) { - cls = new ClassLoaderStats(); - _stats->put(cl, cls); + bool added = false; + ClassLoaderStats* cls = _stats->put_if_absent(cl, &added); + if (added) { + cls->_class_loader = cl; _total_loaders++; - } else { - cls = *cls_ptr; } + assert(cls->_class_loader == cl, "Sanity"); if (!cld->has_class_mirror_holder()) { cls->_cld = cld; } - cls->_class_loader = cl; if (cl != NULL) { cls->_parent = java_lang_ClassLoader::parent(cl); addEmptyParents(cls->_parent); @@ -105,25 +101,25 @@ void ClassLoaderStatsClosure::do_cld(ClassLoaderData* cld) { #endif -bool ClassLoaderStatsClosure::do_entry(oop const& key, ClassLoaderStats* const& cls) { - Klass* class_loader_klass = (cls->_class_loader == NULL ? NULL : cls->_class_loader->klass()); - Klass* parent_klass = (cls->_parent == NULL ? NULL : cls->_parent->klass()); +bool ClassLoaderStatsClosure::do_entry(oop const& key, ClassLoaderStats const& cls) { + Klass* class_loader_klass = (cls._class_loader == NULL ? NULL : cls._class_loader->klass()); + Klass* parent_klass = (cls._parent == NULL ? NULL : cls._parent->klass()); _out->print(INTPTR_FORMAT " " INTPTR_FORMAT " " INTPTR_FORMAT " " UINTX_FORMAT_W(6) " " SIZE_FORMAT_W(8) " " SIZE_FORMAT_W(8) " ", - p2i(class_loader_klass), p2i(parent_klass), p2i(cls->_cld), - cls->_classes_count, - cls->_chunk_sz, cls->_block_sz); + p2i(class_loader_klass), p2i(parent_klass), p2i(cls._cld), + cls._classes_count, + cls._chunk_sz, cls._block_sz); if (class_loader_klass != NULL) { _out->print("%s", class_loader_klass->external_name()); } else { _out->print(""); } _out->cr(); - if (cls->_hidden_classes_count > 0) { + if (cls._hidden_classes_count > 0) { _out->print_cr(SPACE SPACE SPACE " " UINTX_FORMAT_W(6) " " SIZE_FORMAT_W(8) " " SIZE_FORMAT_W(8) " + hidden classes", "", "", "", - cls->_hidden_classes_count, - cls->_hidden_chunk_sz, cls->_hidden_block_sz); + cls._hidden_classes_count, + cls._hidden_chunk_sz, cls._hidden_block_sz); } return true; } @@ -146,15 +142,14 @@ void ClassLoaderStatsClosure::print() { void ClassLoaderStatsClosure::addEmptyParents(oop cl) { while (cl != NULL && java_lang_ClassLoader::loader_data_acquire(cl) == NULL) { // This classloader has not loaded any classes - ClassLoaderStats** cls_ptr = _stats->get(cl); - if (cls_ptr == NULL) { - // It does not exist in our table - add it - ClassLoaderStats* cls = new ClassLoaderStats(); + bool added = false; + ClassLoaderStats* cls = _stats->put_if_absent(cl, &added); + if (added) { cls->_class_loader = cl; cls->_parent = java_lang_ClassLoader::parent(cl); - _stats->put(cl, cls); _total_loaders++; } + assert(cls->_class_loader == cl, "Sanity"); cl = java_lang_ClassLoader::parent(cl); } diff --git a/src/hotspot/share/classfile/classLoaderStats.hpp b/src/hotspot/share/classfile/classLoaderStats.hpp index 6970b5fc957..c9f602721d4 100644 --- a/src/hotspot/share/classfile/classLoaderStats.hpp +++ b/src/hotspot/share/classfile/classLoaderStats.hpp @@ -115,7 +115,7 @@ protected: return hash; } - typedef ResourceHashtable StatsTable; outputStream* _out; @@ -136,7 +136,7 @@ public: } virtual void do_cld(ClassLoaderData* cld); - virtual bool do_entry(oop const& key, ClassLoaderStats* const& cls); + virtual bool do_entry(oop const& key, ClassLoaderStats const& cls); void print(); private: diff --git a/src/hotspot/share/classfile/systemDictionaryShared.cpp b/src/hotspot/share/classfile/systemDictionaryShared.cpp index ad1d7f7b3a2..63ee66d5e06 100644 --- a/src/hotspot/share/classfile/systemDictionaryShared.cpp +++ b/src/hotspot/share/classfile/systemDictionaryShared.cpp @@ -198,14 +198,14 @@ class DumpTimeSharedClassTable: public ResourceHashtable< int _unregistered_count; public: DumpTimeSharedClassInfo* find_or_allocate_info_for(InstanceKlass* k) { - DumpTimeSharedClassInfo* p = get(k); - if (p == NULL) { + bool created = false; + DumpTimeSharedClassInfo* p = put_if_absent(k, &created); + if (created) { assert(!SystemDictionaryShared::no_class_loading_should_happen(), "no new classes can be loaded while dumping archive"); - put(k, DumpTimeSharedClassInfo()); - p = get(k); - assert(p != NULL, "sanity"); p->_klass = k; + } else { + assert(p->_klass == k, "Sanity"); } return p; } @@ -1041,19 +1041,16 @@ static ResourceHashtable< ResourceObj::C_HEAP> _loaded_unregistered_classes; bool SystemDictionaryShared::add_unregistered_class(InstanceKlass* k, TRAPS) { + // We don't allow duplicated unregistered classes of the same name. assert(DumpSharedSpaces, "only when dumping"); - Symbol* name = k->name(); - if (_loaded_unregistered_classes.get(name) != NULL) { - // We don't allow duplicated unregistered classes of the same name. - return false; - } else { - bool isnew = _loaded_unregistered_classes.put(name, true); - assert(isnew, "sanity"); + bool created = false; + _loaded_unregistered_classes.put_if_absent(name, true, &created); + if (created) { MutexLocker mu_r(THREAD, Compile_lock); // add_to_hierarchy asserts this. SystemDictionary::add_to_hierarchy(k, CHECK_false); - return true; } + return created; } // This function is called to resolve the super/interfaces of shared classes for diff --git a/src/hotspot/share/jfr/periodic/jfrPeriodic.cpp b/src/hotspot/share/jfr/periodic/jfrPeriodic.cpp index 2c333b28d0f..72f102d0bf4 100644 --- a/src/hotspot/share/jfr/periodic/jfrPeriodic.cpp +++ b/src/hotspot/share/jfr/periodic/jfrPeriodic.cpp @@ -468,21 +468,21 @@ class JfrClassLoaderStatsClosure : public ClassLoaderStatsClosure { public: JfrClassLoaderStatsClosure() : ClassLoaderStatsClosure(NULL) {} - bool do_entry(oop const& key, ClassLoaderStats* const& cls) { - const ClassLoaderData* this_cld = cls->_class_loader != NULL ? - java_lang_ClassLoader::loader_data_acquire(cls->_class_loader) : NULL; - const ClassLoaderData* parent_cld = cls->_parent != NULL ? - java_lang_ClassLoader::loader_data_acquire(cls->_parent) : NULL; + bool do_entry(oop const& key, ClassLoaderStats const& cls) { + const ClassLoaderData* this_cld = cls._class_loader != NULL ? + java_lang_ClassLoader::loader_data_acquire(cls._class_loader) : NULL; + const ClassLoaderData* parent_cld = cls._parent != NULL ? + java_lang_ClassLoader::loader_data_acquire(cls._parent) : NULL; EventClassLoaderStatistics event; event.set_classLoader(this_cld); event.set_parentClassLoader(parent_cld); - event.set_classLoaderData((intptr_t)cls->_cld); - event.set_classCount(cls->_classes_count); - event.set_chunkSize(cls->_chunk_sz); - event.set_blockSize(cls->_block_sz); - event.set_hiddenClassCount(cls->_hidden_classes_count); - event.set_hiddenChunkSize(cls->_hidden_chunk_sz); - event.set_hiddenBlockSize(cls->_hidden_block_sz); + event.set_classLoaderData((intptr_t)cls._cld); + event.set_classCount(cls._classes_count); + event.set_chunkSize(cls._chunk_sz); + event.set_blockSize(cls._block_sz); + event.set_hiddenClassCount(cls._hidden_classes_count); + event.set_hiddenChunkSize(cls._hidden_chunk_sz); + event.set_hiddenBlockSize(cls._hidden_block_sz); event.commit(); return true; } diff --git a/src/hotspot/share/utilities/resourceHash.hpp b/src/hotspot/share/utilities/resourceHash.hpp index be5c22f3c6e..40e04f0d64f 100644 --- a/src/hotspot/share/utilities/resourceHash.hpp +++ b/src/hotspot/share/utilities/resourceHash.hpp @@ -51,6 +51,11 @@ class ResourceHashtable : public ResourceObj { Node(unsigned hash, K const& key, V const& value) : _hash(hash), _key(key), _value(value), _next(NULL) {} + + // Create a node with a default-constructed value. + Node(unsigned hash, K const& key) : + _hash(hash), _key(key), _value(), _next(NULL) {} + }; Node* _table[SIZE]; @@ -124,6 +129,41 @@ class ResourceHashtable : public ResourceObj { } } + // Look up the key. + // If an entry for the key exists, leave map unchanged and return a pointer to its value. + // If no entry for the key exists, create a new entry from key and a default-created value + // and return a pointer to the value. + // *p_created is new if entry was created, false if entry pre-existed. + V* put_if_absent(K const& key, bool* p_created) { + unsigned hv = HASH(key); + Node** ptr = lookup_node(hv, key); + if (*ptr == NULL) { + *ptr = new (ALLOC_TYPE, MEM_TYPE) Node(hv, key); + *p_created = true; + } else { + *p_created = false; + } + return &(*ptr)->_value; + } + + // Look up the key. + // If an entry for the key exists, leave map unchanged and return a pointer to its value. + // If no entry for the key exists, create a new entry from key and value and return a + // pointer to the value. + // *p_created is new if entry was created, false if entry pre-existed. + V* put_if_absent(K const& key, V const& value, bool* p_created) { + unsigned hv = HASH(key); + Node** ptr = lookup_node(hv, key); + if (*ptr == NULL) { + *ptr = new (ALLOC_TYPE, MEM_TYPE) Node(hv, key, value); + *p_created = true; + } else { + *p_created = false; + } + return &(*ptr)->_value; + } + + bool remove(K const& key) { unsigned hv = HASH(key); Node** ptr = lookup_node(hv, key); diff --git a/test/hotspot/gtest/utilities/test_resourceHash.cpp b/test/hotspot/gtest/utilities/test_resourceHash.cpp index bca28ea0754..a10241a84f3 100644 --- a/test/hotspot/gtest/utilities/test_resourceHash.cpp +++ b/test/hotspot/gtest/utilities/test_resourceHash.cpp @@ -26,12 +26,13 @@ #include "memory/resourceArea.hpp" #include "unittest.hpp" #include "utilities/debug.hpp" +#include "utilities/globalDefinitions.hpp" #include "utilities/resourceHash.hpp" class CommonResourceHashtableTest : public ::testing::Test { protected: typedef void* K; - typedef int V; + typedef uintx V; const static MEMFLAGS MEM_TYPE = mtInternal; static unsigned identity_hash(const K& k) { @@ -58,6 +59,7 @@ class CommonResourceHashtableTest : public ::testing::Test { } } }; + }; class SmallResourceHashtableTest : public CommonResourceHashtableTest { @@ -96,7 +98,44 @@ class SmallResourceHashtableTest : public CommonResourceHashtableTest { } ASSERT_TRUE(rh.remove(as_K(step))); + ASSERT_FALSE(rh.contains(as_K(step))); rh.iterate(&et); + + + // Test put_if_absent(key) (creating a default-created value) + bool created = false; + V* v = rh.put_if_absent(as_K(step), &created); + ASSERT_TRUE(rh.contains(as_K(step))); + ASSERT_TRUE(created); + *v = (V)step; + + // Calling this function a second time should yield the same value pointer + V* v2 = rh.put_if_absent(as_K(step), &created); + ASSERT_EQ(v, v2); + ASSERT_EQ(*v2, *v); + ASSERT_FALSE(created); + + ASSERT_TRUE(rh.remove(as_K(step))); + ASSERT_FALSE(rh.contains(as_K(step))); + rh.iterate(&et); + + // Test put_if_absent(key, value) + v = rh.put_if_absent(as_K(step), step, &created); + ASSERT_EQ(*v, step); + ASSERT_TRUE(rh.contains(as_K(step))); + ASSERT_TRUE(created); + + v2 = rh.put_if_absent(as_K(step), step, &created); + // Calling this function a second time should yield the same value pointer + ASSERT_EQ(v, v2); + ASSERT_EQ(*v2, (V)step); + ASSERT_FALSE(created); + + ASSERT_TRUE(rh.remove(as_K(step))); + ASSERT_FALSE(rh.contains(as_K(step))); + rh.iterate(&et); + + } }; };