diff --git a/src/hotspot/share/gc/g1/g1CardSet.cpp b/src/hotspot/share/gc/g1/g1CardSet.cpp index 4b91dc54092..cabe8573f88 100644 --- a/src/hotspot/share/gc/g1/g1CardSet.cpp +++ b/src/hotspot/share/gc/g1/g1CardSet.cpp @@ -44,20 +44,36 @@ G1CardSet::CardSetPtr G1CardSet::FullCardSet = (G1CardSet::CardSetPtr)-1; +static uint default_log2_card_region_per_region() { + uint log2_card_region_per_heap_region = 0; + + const uint card_container_limit = G1CardSetContainer::LogCardsPerRegionLimit; + if (card_container_limit < (uint)HeapRegion::LogCardsPerRegion) { + log2_card_region_per_heap_region = (uint)HeapRegion::LogCardsPerRegion - card_container_limit; + } + + return log2_card_region_per_heap_region; +} + G1CardSetConfiguration::G1CardSetConfiguration() : G1CardSetConfiguration(HeapRegion::LogCardsPerRegion, /* inline_ptr_bits_per_card */ G1RemSetArrayOfCardsEntries, /* num_cards_in_array */ (double)G1RemSetCoarsenHowlBitmapToHowlFullPercent / 100, /* cards_in_bitmap_threshold_percent */ G1RemSetHowlNumBuckets, /* num_buckets_in_howl */ (double)G1RemSetCoarsenHowlToFullPercent / 100, /* cards_in_howl_threshold_percent */ - (uint)HeapRegion::CardsPerRegion) /* max_cards_in_cardset */ - { } + (uint)HeapRegion::CardsPerRegion, /* max_cards_in_cardset */ + default_log2_card_region_per_region()) /* log2_card_region_per_region */ +{ + assert((_log2_card_region_per_heap_region + _log2_cards_per_card_region) == (uint)HeapRegion::LogCardsPerRegion, + "inconsistent heap region virtualization setup"); +} G1CardSetConfiguration::G1CardSetConfiguration(uint num_cards_in_array, double cards_in_bitmap_threshold_percent, uint max_buckets_in_howl, double cards_in_howl_threshold_percent, - uint max_cards_in_card_set) : + uint max_cards_in_card_set, + uint log2_card_region_per_region) : G1CardSetConfiguration(log2i_exact(max_cards_in_card_set), /* inline_ptr_bits_per_card */ num_cards_in_array, /* num_cards_in_array */ cards_in_bitmap_threshold_percent, /* cards_in_bitmap_threshold_percent */ @@ -65,15 +81,17 @@ G1CardSetConfiguration::G1CardSetConfiguration(uint num_cards_in_array, num_cards_in_array, max_buckets_in_howl), cards_in_howl_threshold_percent, /* cards_in_howl_threshold_percent */ - max_cards_in_card_set) /* max_cards_in_cardset */ - { } + max_cards_in_card_set, /* max_cards_in_cardset */ + log2_card_region_per_region) +{ } G1CardSetConfiguration::G1CardSetConfiguration(uint inline_ptr_bits_per_card, uint num_cards_in_array, double cards_in_bitmap_threshold_percent, uint num_buckets_in_howl, double cards_in_howl_threshold_percent, - uint max_cards_in_card_set) : + uint max_cards_in_card_set, + uint log2_card_region_per_heap_region) : _inline_ptr_bits_per_card(inline_ptr_bits_per_card), _num_cards_in_array(num_cards_in_array), _num_buckets_in_howl(num_buckets_in_howl), @@ -82,13 +100,14 @@ G1CardSetConfiguration::G1CardSetConfiguration(uint inline_ptr_bits_per_card, _num_cards_in_howl_bitmap(G1CardSetHowl::bitmap_size(_max_cards_in_card_set, _num_buckets_in_howl)), _cards_in_howl_bitmap_threshold(_num_cards_in_howl_bitmap * cards_in_bitmap_threshold_percent), _log2_num_cards_in_howl_bitmap(log2i_exact(_num_cards_in_howl_bitmap)), - _bitmap_hash_mask(~(~(0) << _log2_num_cards_in_howl_bitmap)) { + _bitmap_hash_mask(~(~(0) << _log2_num_cards_in_howl_bitmap)), + _log2_card_region_per_heap_region(log2_card_region_per_heap_region), + _log2_cards_per_card_region(log2i_exact(_max_cards_in_card_set) - _log2_card_region_per_heap_region) { assert(is_power_of_2(_max_cards_in_card_set), "max_cards_in_card_set must be a power of 2: %u", _max_cards_in_card_set); init_card_set_alloc_options(); - log_configuration(); } @@ -109,11 +128,14 @@ void G1CardSetConfiguration::log_configuration() { "InlinePtr #elems %u size %zu " "Array Of Cards #elems %u size %zu " "Howl #buckets %u coarsen threshold %u " - "Howl Bitmap #elems %u size %zu coarsen threshold %u", + "Howl Bitmap #elems %u size %zu coarsen threshold %u " + "Card regions per heap region %u cards per card region %u", num_cards_in_inline_ptr(), sizeof(void*), num_cards_in_array(), G1CardSetArray::size_in_bytes(num_cards_in_array()), num_buckets_in_howl(), cards_in_howl_threshold(), - num_cards_in_howl_bitmap(), G1CardSetBitMap::size_in_bytes(num_cards_in_howl_bitmap()), cards_in_howl_bitmap_threshold()); + num_cards_in_howl_bitmap(), G1CardSetBitMap::size_in_bytes(num_cards_in_howl_bitmap()), cards_in_howl_bitmap_threshold(), + (uint)1 << log2_card_region_per_heap_region(), + (uint)1 << log2_cards_per_card_region()); } uint G1CardSetConfiguration::num_cards_in_inline_ptr() const { @@ -828,8 +850,26 @@ public: } }; +template class CardOrRanges> +class G1CardSetIterateCardsIterator : public G1CardSet::G1CardSetPtrIterator { + G1CardSet* _card_set; + Closure& _iter; + +public: + + G1CardSetIterateCardsIterator(G1CardSet* card_set, + Closure& iter) : + _card_set(card_set), + _iter(iter) { } + + void do_cardsetptr(uint region_idx, size_t num_occupied, G1CardSet::CardSetPtr card_set) override { + CardOrRanges cl(_iter, region_idx); + _card_set->iterate_cards_or_ranges_in_container(card_set, cl); + } +}; + void G1CardSet::iterate_cards(G1CardSetCardIterator& iter) { - G1CardSetMergeCardIterator cl(this, iter); + G1CardSetIterateCardsIterator cl(this, iter); iterate_containers(&cl); } diff --git a/src/hotspot/share/gc/g1/g1CardSet.hpp b/src/hotspot/share/gc/g1/g1CardSet.hpp index 3589ef4988e..937bc99a919 100644 --- a/src/hotspot/share/gc/g1/g1CardSet.hpp +++ b/src/hotspot/share/gc/g1/g1CardSet.hpp @@ -58,6 +58,8 @@ class G1CardSetConfiguration { uint _cards_in_howl_bitmap_threshold; uint _log2_num_cards_in_howl_bitmap; size_t _bitmap_hash_mask; + uint _log2_card_region_per_heap_region; + uint _log2_cards_per_card_region; G1CardSetAllocOptions* _card_set_alloc_options; @@ -66,7 +68,8 @@ class G1CardSetConfiguration { double cards_in_bitmap_threshold_percent, uint num_buckets_in_howl, double cards_in_howl_threshold_percent, - uint max_cards_in_card_set); + uint max_cards_in_card_set, + uint log2_card_region_per_heap_region); void init_card_set_alloc_options(); void log_configuration(); @@ -75,12 +78,13 @@ public: // Initialize card set configuration from globals. G1CardSetConfiguration(); // Initialize card set configuration from parameters. - // Only for test + // Testing only. G1CardSetConfiguration(uint num_cards_in_array, double cards_in_bitmap_threshold_percent, uint max_buckets_in_howl, double cards_in_howl_threshold_percent, - uint max_cards_in_card_set); + uint max_cards_in_cardset, + uint log2_card_region_per_region); ~G1CardSetConfiguration(); @@ -115,6 +119,20 @@ public: // with more entries per region are coarsened to Full. uint max_cards_in_region() const { return _max_cards_in_card_set; } + // Heap region virtualization: there are some limitations to how many cards the + // containers can cover to save memory for the common case. Heap region virtualization + // allows to use multiple entries in the G1CardSet hash table per area covered + // by the remembered set (e.g. heap region); each such entry is called "card_region". + // + // The next two members give information about how many card regions are there + // per area (heap region) and how many cards each card region has. + + // The log2 of the amount of card regions per heap region configured. + uint log2_card_region_per_heap_region() const { return _log2_card_region_per_heap_region; } + // The log2 of the number of cards per card region. This is calculated from max_cards_in_region() + // and above. + uint log2_cards_per_card_region() const { return _log2_cards_per_card_region; } + // Memory object types configuration // Number of distinctly sized memory objects on the card set heap. // Currently contains CHT-Nodes, ArrayOfCards, BitMaps, Howl @@ -171,9 +189,6 @@ class G1CardSet : public CHeapObj { friend class G1CardSetTest; friend class G1CardSetMtTestTask; - template class CardorRanges> - friend class G1CardSetMergeCardIterator; - friend class G1TransferCard; friend class G1ReleaseCardsets; @@ -278,23 +293,6 @@ private: template void iterate_cards_during_transfer(CardSetPtr const card_set, CardVisitor& found); - // Iterate over the container, calling a method on every card or card range contained - // in the card container. - // For every container, first calls - // - // void start_iterate(uint tag, uint region_idx); - // - // Then for every card or card range it calls - // - // void do_card(uint card_idx); - // void do_card_range(uint card_idx, uint length); - // - // where card_idx is the card index within that region_idx passed before in - // start_iterate(). - // - template - void iterate_cards_or_ranges_in_container(CardSetPtr const card_set, CardOrRangeVisitor& found); - uint card_set_type_to_mem_object_type(uintptr_t type) const; uint8_t* allocate_mem_object(uintptr_t type); void free_mem_object(CardSetPtr card_set); @@ -340,7 +338,23 @@ public: void print(outputStream* os); - // Various iterators - should be made inlineable somehow. + // Iterate over the container, calling a method on every card or card range contained + // in the card container. + // For every container, first calls + // + // void start_iterate(uint tag, uint region_idx); + // + // Then for every card or card range it calls + // + // void do_card(uint card_idx); + // void do_card_range(uint card_idx, uint length); + // + // where card_idx is the card index within that region_idx passed before in + // start_iterate(). + // + template + void iterate_cards_or_ranges_in_container(CardSetPtr const card_set, CardOrRangeVisitor& found); + class G1CardSetPtrIterator { public: virtual void do_cardsetptr(uint region_idx, size_t num_occupied, CardSetPtr card_set) = 0; @@ -354,11 +368,6 @@ public: }; void iterate_cards(G1CardSetCardIterator& iter); - - // Iterate all cards for card set merging. Must be a CardOrRangeVisitor as - // explained above. - template - void iterate_for_merge(CardOrRangeVisitor& cl); }; class G1CardSetHashTableValue { diff --git a/src/hotspot/share/gc/g1/g1CardSet.inline.hpp b/src/hotspot/share/gc/g1/g1CardSet.inline.hpp index 879f6f1eaff..4ebc8f664a9 100644 --- a/src/hotspot/share/gc/g1/g1CardSet.inline.hpp +++ b/src/hotspot/share/gc/g1/g1CardSet.inline.hpp @@ -80,46 +80,4 @@ inline void G1CardSet::iterate_cards_or_ranges_in_container(CardSetPtr const car ShouldNotReachHere(); } -template -class G1ContainerCardsOrRanges { - Closure& _iter; - uint _region_idx; - -public: - G1ContainerCardsOrRanges(Closure& iter, uint region_idx) : _iter(iter), _region_idx(region_idx) { } - - bool start_iterate(uint tag) { - return _iter.start_iterate(tag, _region_idx); - } - - void operator()(uint card_idx) { - _iter.do_card(card_idx); - } - - void operator()(uint card_idx, uint length) { - _iter.do_card_range(card_idx, length); - } -}; - -template class CardOrRanges> -class G1CardSetMergeCardIterator : public G1CardSet::G1CardSetPtrIterator { - G1CardSet* _card_set; - Closure& _iter; - -public: - - G1CardSetMergeCardIterator(G1CardSet* card_set, Closure& iter) : _card_set(card_set), _iter(iter) { } - - void do_cardsetptr(uint region_idx, size_t num_occupied, G1CardSet::CardSetPtr card_set) override { - CardOrRanges cl(_iter, region_idx); - _card_set->iterate_cards_or_ranges_in_container(card_set, cl); - } -}; - -template -inline void G1CardSet::iterate_for_merge(CardOrRangeVisitor& cl) { - G1CardSetMergeCardIterator cl2(this, cl); - iterate_containers(&cl2, true /* at_safepoint */); -} - #endif // SHARE_GC_G1_G1CARDSET_INLINE_HPP diff --git a/src/hotspot/share/gc/g1/g1RemSet.cpp b/src/hotspot/share/gc/g1/g1RemSet.cpp index 123de75f916..b689e649762 100644 --- a/src/hotspot/share/gc/g1/g1RemSet.cpp +++ b/src/hotspot/share/gc/g1/g1RemSet.cpp @@ -106,14 +106,14 @@ class G1RemSetScanState : public CHeapObj { // within a region to claim. Dependent on the region size as proxy for the heap // size, we limit the total number of chunks to limit memory usage and maintenance // effort of that table vs. granularity of distributing scanning work. - // Testing showed that 8 for 1M/2M region, 16 for 4M/8M regions, 32 for 16/32M regions - // seems to be such a good trade-off. + // Testing showed that 8 for 1M/2M region, 16 for 4M/8M regions, 32 for 16/32M regions, + // and so on seems to be such a good trade-off. static uint get_chunks_per_region(uint log_region_size) { // Limit the expected input values to current known possible values of the // (log) region size. Adjust as necessary after testing if changing the permissible // values for region size. - assert(log_region_size >= 20 && log_region_size <= 25, - "expected value in [20,25], but got %u", log_region_size); + assert(log_region_size >= 20 && log_region_size <= 29, + "expected value in [20,29], but got %u", log_region_size); return 1u << (log_region_size / 2 - 7); } diff --git a/src/hotspot/share/gc/g1/g1_globals.hpp b/src/hotspot/share/gc/g1/g1_globals.hpp index 34dbabe4fec..ed71b6d45b2 100644 --- a/src/hotspot/share/gc/g1/g1_globals.hpp +++ b/src/hotspot/share/gc/g1/g1_globals.hpp @@ -257,7 +257,7 @@ \ product(size_t, G1HeapRegionSize, 0, \ "Size of the G1 regions.") \ - range(0, 32*M) \ + range(0, NOT_LP64(32*M) LP64_ONLY(512*M)) \ constraint(G1HeapRegionSizeConstraintFunc,AfterMemoryInit) \ \ product(uint, G1ConcRefinementThreads, 0, \ diff --git a/src/hotspot/share/gc/g1/heapRegion.cpp b/src/hotspot/share/gc/g1/heapRegion.cpp index baf618c2cc4..78ca4ad8bf9 100644 --- a/src/hotspot/share/gc/g1/heapRegion.cpp +++ b/src/hotspot/share/gc/g1/heapRegion.cpp @@ -65,8 +65,9 @@ void HeapRegion::setup_heap_region_size(size_t max_heap_size) { size_t region_size = G1HeapRegionSize; // G1HeapRegionSize = 0 means decide ergonomically. if (region_size == 0) { - region_size = MAX2(max_heap_size / HeapRegionBounds::target_number(), - HeapRegionBounds::min_size()); + region_size = clamp(max_heap_size / HeapRegionBounds::target_number(), + HeapRegionBounds::min_size(), + HeapRegionBounds::max_ergonomics_size()); } // Make sure region size is a power of 2. Rounding up since this diff --git a/src/hotspot/share/gc/g1/heapRegionBounds.hpp b/src/hotspot/share/gc/g1/heapRegionBounds.hpp index 5a3f269749f..5e748de5928 100644 --- a/src/hotspot/share/gc/g1/heapRegionBounds.hpp +++ b/src/hotspot/share/gc/g1/heapRegionBounds.hpp @@ -34,12 +34,14 @@ private: // heaps a bit more efficiently. static const size_t MIN_REGION_SIZE = 1024 * 1024; + // Maximum region size determined ergonomically. + static const size_t MAX_ERGONOMICS_SIZE = 32 * 1024 * 1024; // Maximum region size; we don't go higher than that. There's a good // reason for having an upper bound. We don't want regions to get too // large, otherwise cleanup's effectiveness would decrease as there // will be fewer opportunities to find totally empty regions after // marking. - static const size_t MAX_REGION_SIZE = 32 * 1024 * 1024; + static const size_t MAX_REGION_SIZE = 512 * 1024 * 1024; // The automatic region size calculation will try to have around this // many regions in the heap. @@ -47,6 +49,7 @@ private: public: static inline size_t min_size(); + static inline size_t max_ergonomics_size(); static inline size_t max_size(); static inline size_t target_number(); }; diff --git a/src/hotspot/share/gc/g1/heapRegionBounds.inline.hpp b/src/hotspot/share/gc/g1/heapRegionBounds.inline.hpp index 20d81f6a62e..73a44bcf179 100644 --- a/src/hotspot/share/gc/g1/heapRegionBounds.inline.hpp +++ b/src/hotspot/share/gc/g1/heapRegionBounds.inline.hpp @@ -31,6 +31,10 @@ size_t HeapRegionBounds::min_size() { return MIN_REGION_SIZE; } +size_t HeapRegionBounds::max_ergonomics_size() { + return MAX_ERGONOMICS_SIZE; +} + size_t HeapRegionBounds::max_size() { return MAX_REGION_SIZE; } diff --git a/src/hotspot/share/gc/g1/heapRegionRemSet.cpp b/src/hotspot/share/gc/g1/heapRegionRemSet.cpp index c7633e77235..a0e740ad267 100644 --- a/src/hotspot/share/gc/g1/heapRegionRemSet.cpp +++ b/src/hotspot/share/gc/g1/heapRegionRemSet.cpp @@ -26,6 +26,7 @@ #include "precompiled.hpp" #include "gc/g1/g1BlockOffsetTable.inline.hpp" +#include "gc/g1/g1CardSetContainers.inline.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/g1ConcurrentRefine.hpp" #include "gc/g1/heapRegionManager.inline.hpp" diff --git a/src/hotspot/share/gc/g1/heapRegionRemSet.inline.hpp b/src/hotspot/share/gc/g1/heapRegionRemSet.inline.hpp index b6c4ee9c1bb..ff1f75df602 100644 --- a/src/hotspot/share/gc/g1/heapRegionRemSet.inline.hpp +++ b/src/hotspot/share/gc/g1/heapRegionRemSet.inline.hpp @@ -55,10 +55,64 @@ void HeapRegionRemSet::set_state_complete() { _state = Complete; } +template +class G1ContainerCardsOrRanges { + Closure& _iter; + uint _region_idx; + uint _offset; + +public: + G1ContainerCardsOrRanges(Closure& iter, uint region_idx, uint offset) : _iter(iter), _region_idx(region_idx), _offset(offset) { } + + bool start_iterate(uint tag) { + return _iter.start_iterate(tag, _region_idx); + } + + void operator()(uint card_idx) { + _iter.do_card(card_idx + _offset); + } + + void operator()(uint card_idx, uint length) { + _iter.do_card_range(card_idx + _offset, length); + } +}; + +template class CardOrRanges> +class G1HeapRegionRemSetMergeCardIterator : public G1CardSet::G1CardSetPtrIterator { + G1CardSet* _card_set; + Closure& _iter; + uint _log_card_regions_per_region; + uint _card_regions_per_region_mask; + uint _log_card_region_size; + +public: + + G1HeapRegionRemSetMergeCardIterator(G1CardSet* card_set, + Closure& iter, + uint log_card_regions_per_region, + uint log_card_region_size) : + _card_set(card_set), + _iter(iter), + _log_card_regions_per_region(log_card_regions_per_region), + _card_regions_per_region_mask((1 << log_card_regions_per_region) - 1), + _log_card_region_size(log_card_region_size) { + } + + void do_cardsetptr(uint card_region_idx, size_t num_occupied, G1CardSet::CardSetPtr card_set) override { + CardOrRanges cl(_iter, + card_region_idx >> _log_card_regions_per_region, + (card_region_idx & _card_regions_per_region_mask) << _log_card_region_size); + _card_set->iterate_cards_or_ranges_in_container(card_set, cl); + } +}; template inline void HeapRegionRemSet::iterate_for_merge(CardOrRangeVisitor& cl) { - _card_set.iterate_for_merge(cl); + G1HeapRegionRemSetMergeCardIterator cl2(&_card_set, + cl, + _card_set.config()->log2_card_region_per_heap_region(), + _card_set.config()->log2_cards_per_card_region()); + _card_set.iterate_containers(&cl2, true /* at_safepoint */); } void HeapRegionRemSet::split_card(OopOrNarrowOopStar from, uint& card_region, uint& card_within_region) const { diff --git a/test/hotspot/gtest/gc/g1/test_g1CardSet.cpp b/test/hotspot/gtest/gc/g1/test_g1CardSet.cpp index d535248c39d..58e6f3e3c24 100644 --- a/test/hotspot/gtest/gc/g1/test_g1CardSet.cpp +++ b/test/hotspot/gtest/gc/g1/test_g1CardSet.cpp @@ -210,7 +210,8 @@ void G1CardSetTest::cardset_basic_test() { BitmapCoarsenThreshold, 8, FullCardSetThreshold, - CardsPerRegion); + CardsPerRegion, + 0); G1CardSetFreePool free_pool(config.num_mem_object_types()); G1CardSetMemoryManager mm(&config, &free_pool); @@ -428,7 +429,8 @@ void G1CardSetTest::cardset_mt_test() { BitmapCoarsenThreshold, 8, FullCardSetThreshold, - CardsPerRegion); + CardsPerRegion, + 0); G1CardSetFreePool free_pool(config.num_mem_object_types()); G1CardSetMemoryManager mm(&config, &free_pool); diff --git a/test/hotspot/jtreg/gc/arguments/TestG1HeapRegionSize.java b/test/hotspot/jtreg/gc/arguments/TestG1HeapRegionSize.java index bb8a0473c24..d571e2ae5b5 100644 --- a/test/hotspot/jtreg/gc/arguments/TestG1HeapRegionSize.java +++ b/test/hotspot/jtreg/gc/arguments/TestG1HeapRegionSize.java @@ -42,6 +42,7 @@ import java.util.ArrayList; import java.util.Arrays; import jdk.test.lib.process.OutputAnalyzer; +import jdk.test.lib.Platform; import jdk.test.lib.process.ProcessTools; public class TestG1HeapRegionSize { @@ -82,6 +83,13 @@ public class TestG1HeapRegionSize { checkG1HeapRegionSize(new String[] { "-Xmx64m", "-XX:G1HeapRegionSize=2m" }, 2*M, 0); checkG1HeapRegionSize(new String[] { "-Xmx64m", "-XX:G1HeapRegionSize=3m" }, 4*M, 0); checkG1HeapRegionSize(new String[] { "-Xmx256m", "-XX:G1HeapRegionSize=32m" }, 32*M, 0); - checkG1HeapRegionSize(new String[] { "-Xmx256m", "-XX:G1HeapRegionSize=64m" }, 32*M, 1); + if (Platform.is64bit()) { + checkG1HeapRegionSize(new String[] { "-Xmx4096m", "-XX:G1HeapRegionSize=64m" }, 64*M, 0); + checkG1HeapRegionSize(new String[] { "-Xmx4096m", "-XX:G1HeapRegionSize=512m" }, 512*M, 0); + checkG1HeapRegionSize(new String[] { "-Xmx4096m", "-XX:G1HeapRegionSize=1024m" }, 512*M, 1); + checkG1HeapRegionSize(new String[] { "-Xmx128g" }, 32*M, 0); + } else { + checkG1HeapRegionSize(new String[] { "-Xmx256m", "-XX:G1HeapRegionSize=64m" }, 64*M, 1); + } } }