From af0b8d46d22b6fcce57d089eba34ccd4558760d9 Mon Sep 17 00:00:00 2001 From: Coleen Phillimore Date: Fri, 16 Jun 2017 09:13:56 -0400 Subject: [PATCH] 8181450: assert in BasicHashtable::verify_table Remove assert as it has small probability of happening and added logging Reviewed-by: kbarrett, sspitsyn --- .../src/share/vm/classfile/placeholders.cpp | 16 ++----- hotspot/src/share/vm/utilities/hashtable.cpp | 42 +++++++++++-------- hotspot/src/share/vm/utilities/hashtable.hpp | 4 -- 3 files changed, 28 insertions(+), 34 deletions(-) diff --git a/hotspot/src/share/vm/classfile/placeholders.cpp b/hotspot/src/share/vm/classfile/placeholders.cpp index fbb9728991f..168942cbaa7 100644 --- a/hotspot/src/share/vm/classfile/placeholders.cpp +++ b/hotspot/src/share/vm/classfile/placeholders.cpp @@ -218,27 +218,19 @@ void PlaceholderEntry::verify() const { } void PlaceholderTable::verify() { - int element_count = 0; - for (int pindex = 0; pindex < table_size(); pindex++) { - for (PlaceholderEntry* probe = bucket(pindex); - probe != NULL; - probe = probe->next()) { - probe->verify(); - element_count++; // both klasses and place holders count - } - } - guarantee(number_of_entries() == element_count, - "Verify of system dictionary failed"); + verify_table("Placeholder Table"); } #ifndef PRODUCT void PlaceholderTable::print() { + tty->print_cr("Placeholder table table_size=%d, entries=%d", + table_size(), number_of_entries()); for (int pindex = 0; pindex < table_size(); pindex++) { for (PlaceholderEntry* probe = bucket(pindex); probe != NULL; probe = probe->next()) { - if (Verbose) tty->print("%4d: ", pindex); + tty->print("%4d: ", pindex); tty->print(" place holder "); probe->print(); diff --git a/hotspot/src/share/vm/utilities/hashtable.cpp b/hotspot/src/share/vm/utilities/hashtable.cpp index 766e8b133d2..98bba3d5910 100644 --- a/hotspot/src/share/vm/utilities/hashtable.cpp +++ b/hotspot/src/share/vm/utilities/hashtable.cpp @@ -28,6 +28,7 @@ #include "classfile/javaClasses.inline.hpp" #include "classfile/moduleEntry.hpp" #include "classfile/packageEntry.hpp" +#include "classfile/placeholders.hpp" #include "classfile/protectionDomainCache.hpp" #include "classfile/stringTable.hpp" #include "memory/allocation.inline.hpp" @@ -307,11 +308,11 @@ template void Hashtable::print() { } } - template template void BasicHashtable::verify_table(const char* table_name) { int element_count = 0; int max_bucket_count = 0; + int max_bucket_number = 0; for (int index = 0; index < table_size(); index++) { int bucket_count = 0; for (T* probe = (T*)bucket(index); probe != NULL; probe = probe->next()) { @@ -319,29 +320,32 @@ template void BasicHashtable::verify_table(const char* table_name) bucket_count++; } element_count += bucket_count; - max_bucket_count = MAX2(max_bucket_count, bucket_count); + if (bucket_count > max_bucket_count) { + max_bucket_count = bucket_count; + max_bucket_number = index; + } } guarantee(number_of_entries() == element_count, "Verify of %s failed", table_name); - DEBUG_ONLY(verify_lookup_length(max_bucket_count, table_name)); + + // Log some statistics about the hashtable + log_info(hashtables)("%s max bucket size %d bucket %d element count %d table size %d", table_name, + max_bucket_count, max_bucket_number, _number_of_entries, _table_size); + if (_number_of_entries > 0 && log_is_enabled(Debug, hashtables)) { + for (int index = 0; index < table_size(); index++) { + int bucket_count = 0; + for (T* probe = (T*)bucket(index); probe != NULL; probe = probe->next()) { + log_debug(hashtables)("bucket %d hash " INTPTR_FORMAT, index, (intptr_t)probe->hash()); + bucket_count++; + } + if (bucket_count > 0) { + log_debug(hashtables)("bucket %d count %d", index, bucket_count); + } + } + } } - - #endif // PRODUCT -#ifdef ASSERT - -// Assert if the longest bucket is 10x longer than the average bucket size. -// Could change back to a warning, but warnings are not noticed. -template void BasicHashtable::verify_lookup_length(int max_bucket_count, const char *table_name) { - log_info(hashtables)("%s max bucket size %d element count %d table size %d", table_name, - max_bucket_count, _number_of_entries, _table_size); - assert (max_bucket_count < ((1 + number_of_entries()/table_size())*10), "Table is unbalanced"); -} - -#endif - - // Explicitly instantiate these types #if INCLUDE_ALL_GCS template class Hashtable; @@ -383,3 +387,5 @@ template void BasicHashtable::verify_table(char const* template void BasicHashtable::verify_table(char const*); template void BasicHashtable::verify_table(char const*); template void BasicHashtable::verify_table(char const*); +template void BasicHashtable::verify_table(char const*); + diff --git a/hotspot/src/share/vm/utilities/hashtable.hpp b/hotspot/src/share/vm/utilities/hashtable.hpp index 3703422ca09..2ef258d5d32 100644 --- a/hotspot/src/share/vm/utilities/hashtable.hpp +++ b/hotspot/src/share/vm/utilities/hashtable.hpp @@ -170,10 +170,6 @@ private: protected: -#ifdef ASSERT - void verify_lookup_length(int max_bucket_count, const char *table_name); -#endif - void initialize(int table_size, int entry_size, int number_of_entries); // Accessor