Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: openjdk/shenandoah
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: 6d095910
Choose a base ref
...
head repository: openjdk/shenandoah
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: 7e708569
Choose a head ref

Commits on Feb 26, 2024

  1. Verified

    This commit was signed with the committer’s verified signature.
    Copy the full SHA
    7bb1d38 View commit details
  2. Revert "Remove dead code for inelastic plabs"

    This reverts commit 7bb1d38.
    kdnilsen committed Feb 26, 2024
    Copy the full SHA
    8bc4367 View commit details
  3. Round LAB sizes down rather than up to force alignment

    When we round up, we introduce the risk that the new size exceeds the
    maximum LAB size, resulting in an assertion error.
    kdnilsen committed Feb 26, 2024
    Copy the full SHA
    99cce53 View commit details
  4. Copy the full SHA
    11b26bb View commit details

Commits on Feb 27, 2024

  1. Copy the full SHA
    941d8aa View commit details

Commits on Mar 13, 2024

  1. Copy the full SHA
    39c5885 View commit details
  2. Copy the full SHA
    28a382b View commit details

Commits on Mar 16, 2024

  1. Re-enable old-growth trigger

    kdnilsen committed Mar 16, 2024
    Copy the full SHA
    42ade6e View commit details

Commits on Mar 17, 2024

  1. Add a jtreg test for old growth trigger

    kdnilsen committed Mar 17, 2024
    Copy the full SHA
    aed9633 View commit details

Commits on Mar 27, 2024

  1. Respond to reviewer feedback

    kdnilsen committed Mar 27, 2024
    Copy the full SHA
    c950050 View commit details

Commits on Mar 28, 2024

  1. Fix typo introduced during refactoring

    kdnilsen committed Mar 28, 2024
    Copy the full SHA
    052d381 View commit details
  2. Merge branch 'master' of https://git.openjdk.org/shenandoah into enab…

    …le-old-growth-triggers
    kdnilsen committed Mar 28, 2024
    Copy the full SHA
    f99a71a View commit details
  3. Fix another typo

    kdnilsen committed Mar 28, 2024
    Copy the full SHA
    e9fcdc1 View commit details

Commits on Apr 4, 2024

  1. Move trigger code to ShenandoahOldHeuristics from ShenandoahOldGenera…

    …tion
    kdnilsen committed Apr 4, 2024
    Copy the full SHA
    b333414 View commit details

Commits on Apr 10, 2024

  1. Copy the full SHA
    a43675a View commit details

Commits on Apr 11, 2024

  1. Change behavior of max_old and min_old

    kdnilsen committed Apr 11, 2024
    Copy the full SHA
    d881300 View commit details
  2. Revert "Change behavior of max_old and min_old"

    This reverts commit d881300.
    kdnilsen committed Apr 11, 2024
    Copy the full SHA
    c2cb1b7 View commit details

Commits on Apr 15, 2024

  1. Merge branch 'master' of https://git.openjdk.org/shenandoah into enab…

    …le-old-growth-triggers
    kdnilsen committed Apr 15, 2024
    Copy the full SHA
    ef967a5 View commit details
  2. Refactoring for reviewer feedback

    kdnilsen committed Apr 15, 2024
    Copy the full SHA
    e4e79b0 View commit details
  3. Copy the full SHA
    141fec1 View commit details
  4. Copy the full SHA
    93731d0 View commit details
  5. Increase workload to force OLD collections

    kdnilsen committed Apr 15, 2024
    Copy the full SHA
    901dbe0 View commit details
  6. Fix zero build

    kdnilsen committed Apr 15, 2024
    Copy the full SHA
    b767ab9 View commit details

Commits on Apr 17, 2024

  1. Correct typo in symbolic constant

    FRACTIONAL_DENOMINATOR should be a power of 2.  Small round-off errors
    were observed because there was an error in one digit of this contant.
    kdnilsen committed Apr 17, 2024
    Copy the full SHA
    8e5164c View commit details
  2. Minor refinements to old-growth heuristics

    1. If old-marking finds old-gen is entirely empty, reset the heuristics
       to initial state, so we can restart the search for appropriate
       old-gen size
    2. When testing whether old-gen growth "still" exceeds the trigger
       threshold following mixed evacuations, include both old usage and
       old humongous waste, as that is the quantity that was used for
       initial trigger.
    kdnilsen committed Apr 17, 2024
    Copy the full SHA
    0dc8aba View commit details

Commits on Apr 23, 2024

  1. Copy the full SHA
    bac08f0 View commit details
  2. Merge remote-tracking branch 'origin/master' into enable-old-growth-t…

    …riggers
    kdnilsen committed Apr 23, 2024
    Copy the full SHA
    7e70856 View commit details
Original file line number Diff line number Diff line change
@@ -56,8 +56,10 @@ int ShenandoahOldHeuristics::compare_by_index(RegionData a, RegionData b) {
}
}

ShenandoahOldHeuristics::ShenandoahOldHeuristics(ShenandoahOldGeneration* generation) :
ShenandoahOldHeuristics::ShenandoahOldHeuristics(ShenandoahOldGeneration* generation, ShenandoahGenerationalHeap* gen_heap) :
ShenandoahHeuristics(generation),
_heap(gen_heap),
_old_gen(generation),
_first_pinned_candidate(NOT_FOUND),
_last_old_collection_candidate(0),
_next_old_collection_candidate(0),
@@ -549,7 +551,63 @@ void ShenandoahOldHeuristics::clear_triggers() {
_cannot_expand_trigger = false;
_fragmentation_trigger = false;
_growth_trigger = false;
}
}

void ShenandoahOldHeuristics::trigger_collection_if_fragmented(size_t first_old_region, size_t last_old_region, size_t old_region_count, size_t num_regions) {
if (ShenandoahGenerationalHumongousReserve > 0) {
size_t old_region_span = (first_old_region <= last_old_region)? (last_old_region + 1 - first_old_region): 0;
size_t allowed_old_gen_span = num_regions - (ShenandoahGenerationalHumongousReserve * num_regions) / 100;

// Tolerate lower density if total span is small. Here's the implementation:
// if old_gen spans more than 100% and density < 75%, trigger old-defrag
// else if old_gen spans more than 87.5% and density < 62.5%, trigger old-defrag
// else if old_gen spans more than 75% and density < 50%, trigger old-defrag
// else if old_gen spans more than 62.5% and density < 37.5%, trigger old-defrag
// else if old_gen spans more than 50% and density < 25%, trigger old-defrag
//
// A previous implementation was more aggressive in triggering, resulting in degraded throughput when
// humongous allocation was not required.

size_t old_available = _old_gen->available();
size_t region_size_bytes = ShenandoahHeapRegion::region_size_bytes();
size_t old_unaffiliated_available = _old_gen->free_unaffiliated_regions() * region_size_bytes;
assert(old_available >= old_unaffiliated_available, "sanity");
size_t old_fragmented_available = old_available - old_unaffiliated_available;

size_t old_bytes_consumed = old_region_count * region_size_bytes - old_fragmented_available;
size_t old_bytes_spanned = old_region_span * region_size_bytes;
double old_density = ((double) old_bytes_consumed) / old_bytes_spanned;

uint eighths = 8;
for (uint i = 0; i < 5; i++) {
size_t span_threshold = eighths * allowed_old_gen_span / 8;
double density_threshold = (eighths - 2) / 8.0;
if ((old_region_span >= span_threshold) && (old_density < density_threshold)) {
trigger_old_is_fragmented(old_density, first_old_region, last_old_region);
return;
}
eighths--;
}
}
}

void ShenandoahOldHeuristics::trigger_collection_if_overgrown() {
size_t old_used = _old_gen->used() + _old_gen->get_humongous_waste();
size_t trigger_threshold = _old_gen->usage_trigger_threshold();
// Detects unsigned arithmetic underflow
assert(old_used <= _heap->capacity(),
"Old used (" SIZE_FORMAT ", " SIZE_FORMAT") must not be more than heap capacity (" SIZE_FORMAT ")",
_old_gen->used(), _old_gen->get_humongous_waste(), _heap->capacity());
if (old_used > trigger_threshold) {
trigger_old_has_grown();
}
}

void ShenandoahOldHeuristics::trigger_maybe(size_t first_old_region, size_t last_old_region,
size_t old_region_count, size_t num_regions) {
trigger_collection_if_fragmented(first_old_region, last_old_region, old_region_count, num_regions);
trigger_collection_if_overgrown();
}

bool ShenandoahOldHeuristics::should_start_gc() {
// Cannot start a new old-gen GC until previous one has finished.
@@ -595,7 +653,7 @@ bool ShenandoahOldHeuristics::should_start_gc() {
if (_growth_trigger) {
// Growth may be falsely triggered during mixed evacuations, before the mixed-evacuation candidates have been
// evacuated. Before acting on a false trigger, we check to confirm the trigger condition is still satisfied.
const size_t current_usage = _old_generation->used();
const size_t current_usage = _old_generation->used() + _old_generation->humongous_waste();
const size_t trigger_threshold = _old_generation->usage_trigger_threshold();
const size_t heap_size = heap->capacity();
const size_t ignore_threshold = (ShenandoahIgnoreOldGrowthBelowPercentage * heap_size) / 100;
@@ -618,6 +676,7 @@ bool ShenandoahOldHeuristics::should_start_gc() {
byte_size_in_proper_unit(current_usage), proper_unit_for_byte_size(current_usage), percent_growth);
return true;
} else {
// Mixed evacuations have decreased current_usage such that old-growth trigger is no longer relevant.
_growth_trigger = false;
}
}
Original file line number Diff line number Diff line change
@@ -27,6 +27,7 @@


#include "gc/shenandoah/heuristics/shenandoahHeuristics.hpp"
#include "gc/shenandoah/shenandoahGenerationalHeap.hpp"

class ShenandoahCollectionSet;
class ShenandoahHeapRegion;
@@ -51,6 +52,9 @@ class ShenandoahOldHeuristics : public ShenandoahHeuristics {

static uint NOT_FOUND;

ShenandoahGenerationalHeap* _heap;
ShenandoahOldGeneration* _old_gen;

// After final marking of the old generation, this heuristic will select
// a set of candidate regions to be included in subsequent mixed collections.
// The regions are sorted into a `_region_data` array (declared in base
@@ -109,11 +113,22 @@ class ShenandoahOldHeuristics : public ShenandoahHeuristics {

static int compare_by_index(RegionData a, RegionData b);

inline void trigger_old_is_fragmented(double density, size_t first_old_index, size_t last_old_index) {
_fragmentation_trigger = true;
_fragmentation_density = density;
_fragmentation_first_old_region = first_old_index;
_fragmentation_last_old_region = last_old_index;
}
inline void trigger_old_has_grown() { _growth_trigger = true; }

void trigger_collection_if_fragmented(size_t first_old_region, size_t last_old_region, size_t old_region_count, size_t num_regions);
void trigger_collection_if_overgrown();

protected:
void choose_collection_set_from_regiondata(ShenandoahCollectionSet* set, RegionData* data, size_t data_size, size_t free) override;

public:
explicit ShenandoahOldHeuristics(ShenandoahOldGeneration* generation);
explicit ShenandoahOldHeuristics(ShenandoahOldGeneration* generation, ShenandoahGenerationalHeap* gen_heap);

// Prepare for evacuation of old-gen regions by capturing the mark results of a recently completed concurrent mark pass.
void prepare_for_old_collections();
@@ -160,14 +175,6 @@ class ShenandoahOldHeuristics : public ShenandoahHeuristics {

void trigger_cannot_expand() { _cannot_expand_trigger = true; };

inline void trigger_old_is_fragmented(double density, size_t first_old_index, size_t last_old_index) {
_fragmentation_trigger = true;
_fragmentation_density = density;
_fragmentation_first_old_region = first_old_index;
_fragmentation_last_old_region = last_old_index;
}
void trigger_old_has_grown() { _growth_trigger = true; }

inline void get_fragmentation_trigger_reason_for_log_message(double &density, size_t &first_index, size_t &last_index) {
density = _fragmentation_density;
first_index = _fragmentation_first_old_region;
@@ -176,6 +183,8 @@ class ShenandoahOldHeuristics : public ShenandoahHeuristics {

void clear_triggers();

void trigger_maybe(size_t first_old_region, size_t last_old_region, size_t old_region_count, size_t num_regions);

void record_cycle_end() override;

bool should_start_gc() override;
7 changes: 5 additions & 2 deletions src/hotspot/share/gc/shenandoah/shenandoahHeap.cpp
Original file line number Diff line number Diff line change
@@ -2832,8 +2832,11 @@ void ShenandoahHeap::rebuild_free_set(bool concurrent) {
// Rebuild free set based on adjusted generation sizes.
_free_set->rebuild(young_cset_regions, old_cset_regions);

if (mode()->is_generational() && (ShenandoahGenerationalHumongousReserve > 0)) {
old_generation()->maybe_trigger_collection(first_old_region, last_old_region, old_region_count);
if (mode()->is_generational()) {
ShenandoahGenerationalHeap* gen_heap = ShenandoahGenerationalHeap::heap();
ShenandoahOldGeneration* old_gen = gen_heap->old_generation();
ShenandoahOldHeuristics* old_heuristics = (ShenandoahOldHeuristics*) old_gen->heuristics();
old_heuristics->trigger_maybe(first_old_region, last_old_region, old_region_count, num_regions());
}
}

64 changes: 11 additions & 53 deletions src/hotspot/share/gc/shenandoah/shenandoahOldGeneration.cpp
Original file line number Diff line number Diff line change
@@ -227,10 +227,16 @@ size_t ShenandoahOldGeneration::get_live_bytes_after_last_mark() const {
}

void ShenandoahOldGeneration::set_live_bytes_after_last_mark(size_t bytes) {
_live_bytes_after_last_mark = bytes;
_growth_before_compaction /= 2;
if (_growth_before_compaction < _min_growth_before_compaction) {
_growth_before_compaction = _min_growth_before_compaction;
if (bytes == 0) {
// Restart search for best old-gen size to the initial state
_live_bytes_after_last_mark = ShenandoahHeap::heap()->capacity() * INITIAL_LIVE_FRACTION / FRACTIONAL_DENOMINATOR;
_growth_before_compaction = INITIAL_GROWTH_BEFORE_COMPACTION;
} else {
_live_bytes_after_last_mark = bytes;
_growth_before_compaction /= 2;
if (_growth_before_compaction < _min_growth_before_compaction) {
_growth_before_compaction = _min_growth_before_compaction;
}
}
}

@@ -520,7 +526,7 @@ bool ShenandoahOldGeneration::validate_waiting_for_bootstrap() {
#endif

ShenandoahHeuristics* ShenandoahOldGeneration::initialize_heuristics(ShenandoahMode* gc_mode) {
_old_heuristics = new ShenandoahOldHeuristics(this);
_old_heuristics = new ShenandoahOldHeuristics(this, ShenandoahGenerationalHeap::heap());
_old_heuristics->set_guaranteed_gc_interval(ShenandoahGuaranteedOldGCInterval);
_heuristics = _old_heuristics;
return _heuristics;
@@ -615,54 +621,6 @@ void ShenandoahOldGeneration::prepare_for_mixed_collections_after_global_gc() {
_old_heuristics->coalesce_and_fill_candidates_count());
}

void ShenandoahOldGeneration::maybe_trigger_collection(size_t first_old_region, size_t last_old_region, size_t old_region_count) {
ShenandoahHeap* heap = ShenandoahHeap::heap();
const size_t old_region_span = (first_old_region <= last_old_region)? (last_old_region + 1 - first_old_region): 0;
const size_t allowed_old_gen_span = heap->num_regions() - (ShenandoahGenerationalHumongousReserve * heap->num_regions() / 100);

// Tolerate lower density if total span is small. Here's the implementation:
// if old_gen spans more than 100% and density < 75%, trigger old-defrag
// else if old_gen spans more than 87.5% and density < 62.5%, trigger old-defrag
// else if old_gen spans more than 75% and density < 50%, trigger old-defrag
// else if old_gen spans more than 62.5% and density < 37.5%, trigger old-defrag
// else if old_gen spans more than 50% and density < 25%, trigger old-defrag
//
// A previous implementation was more aggressive in triggering, resulting in degraded throughput when
// humongous allocation was not required.

const size_t old_available = available();
const size_t region_size_bytes = ShenandoahHeapRegion::region_size_bytes();
const size_t old_unaffiliated_available = free_unaffiliated_regions() * region_size_bytes;
assert(old_available >= old_unaffiliated_available, "sanity");
const size_t old_fragmented_available = old_available - old_unaffiliated_available;

const size_t old_bytes_consumed = old_region_count * region_size_bytes - old_fragmented_available;
const size_t old_bytes_spanned = old_region_span * region_size_bytes;
const double old_density = ((double) old_bytes_consumed) / old_bytes_spanned;

uint eighths = 8;
for (uint i = 0; i < 5; i++) {
size_t span_threshold = eighths * allowed_old_gen_span / 8;
double density_threshold = (eighths - 2) / 8.0;
if ((old_region_span >= span_threshold) && (old_density < density_threshold)) {
heuristics()->trigger_old_is_fragmented(old_density, first_old_region, last_old_region);
break;
}
eighths--;
}

const size_t old_used = used() + get_humongous_waste();
const size_t trigger_threshold = usage_trigger_threshold();
// Detects unsigned arithmetic underflow
assert(old_used <= heap->free_set()->capacity(),
"Old used (" SIZE_FORMAT ", " SIZE_FORMAT") must not be more than heap capacity (" SIZE_FORMAT ")",
used(), get_humongous_waste(), heap->free_set()->capacity());

if (old_used > trigger_threshold) {
heuristics()->trigger_old_has_grown();
}
}

void ShenandoahOldGeneration::parallel_region_iterate_free(ShenandoahHeapRegionClosure* cl) {
// Iterate over old and free regions (exclude young).
ShenandoahExcludeRegionClosure<YOUNG_GENERATION> exclude_cl(cl);
4 changes: 2 additions & 2 deletions src/hotspot/share/gc/shenandoah/shenandoahOldGeneration.hpp
Original file line number Diff line number Diff line change
@@ -27,6 +27,7 @@

#include "gc/shenandoah/heuristics/shenandoahOldHeuristics.hpp"
#include "gc/shenandoah/shenandoahGeneration.hpp"
#include "gc/shenandoah/shenandoahGenerationalHeap.hpp"
#include "gc/shenandoah/shenandoahSharedVariables.hpp"

class ShenandoahHeapRegion;
@@ -221,7 +222,6 @@ class ShenandoahOldGeneration : public ShenandoahGeneration {
// Abandon any regions waiting for mixed collections
void abandon_collection_candidates();

void maybe_trigger_collection(size_t first_old_region, size_t last_old_region, size_t old_region_count);
public:
enum State {
FILLING, WAITING_FOR_BOOTSTRAP, BOOTSTRAPPING, MARKING, EVACUATING, EVACUATING_AFTER_GLOBAL
@@ -234,7 +234,7 @@ class ShenandoahOldGeneration : public ShenandoahGeneration {
private:
State _state;

static const size_t FRACTIONAL_DENOMINATOR = 64536;
static const size_t FRACTIONAL_DENOMINATOR = 65536;

// During initialization of the JVM, we search for the correct old-gen size by initially performing old-gen
// collection when old-gen usage is 50% more (INITIAL_GROWTH_BEFORE_COMPACTION) than the initial old-gen size
Original file line number Diff line number Diff line change
@@ -0,0 +1,113 @@
/*
* Copyright Amazon.com Inc. or its affiliates. All Rights Reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
*/

/*
* @test
* @summary Test that growth of old-gen triggers old-gen marking
* @requires vm.gc.Shenandoah
* @library /test/lib
* @run driver TestOldGrowthTriggers
*/

import java.util.*;
import java.math.BigInteger;

import jdk.test.lib.Asserts;
import jdk.test.lib.process.ProcessTools;
import jdk.test.lib.process.OutputAnalyzer;

public class TestOldGrowthTriggers {

public static void makeOldAllocations() {
// Expect most of the BigInteger entries placed into array to be promoted, and most will eventually become garbage within old

final int array_size = 512 * 1024; // 512K entries
BigInteger array[] = new BigInteger[array_size];
Random r = new Random(46);

for (int i = 0; i < array_size; i++) {
array[i] = new BigInteger(128, r);
}

for (int refill_count = 0; refill_count < 192; refill_count++) {
// Each refill repopulates array_size randomly selected elements within array
for (int i = 0; i < array_size; i++) {
int replace_index = r.nextInt(array_size);
int derive_index = r.nextInt(array_size);
switch (i & 0x3) {
case 0:
// 50% chance of creating garbage
array[replace_index] = array[replace_index].max(array[derive_index]);
break;
case 1:
// 50% chance of creating garbage
array[replace_index] = array[replace_index].min(array[derive_index]);
break;
case 2:
// creates new old BigInteger, releases old BigInteger,
// may create ephemeral data while computing gcd
array[replace_index] = array[replace_index].gcd(array[derive_index]);
break;
case 3:
// creates new old BigInteger, releases old BigInteger
array[replace_index] = array[replace_index].multiply(array[derive_index]);
break;
}
if ((i & 0x3) == 0x3) {
} else {
}
}
}
}

public static void testOld(String... args) throws Exception {
String[] cmds = Arrays.copyOf(args, args.length + 2);
cmds[args.length] = TestOldGrowthTriggers.class.getName();
cmds[args.length + 1] = "test";
ProcessBuilder pb = ProcessTools.createLimitedTestJavaProcessBuilder(cmds);
OutputAnalyzer output = new OutputAnalyzer(pb.start());
output.shouldHaveExitValue(0);
if (!output.getOutput().contains("Trigger (OLD): Old has overgrown")) {
throw new AssertionError("Generational mode: Should experience OLD triggers due to growth");
}
}

public static void main(String[] args) throws Exception {
if (args.length > 0 && args[0].equals("test")) {
makeOldAllocations();
return;
}

testOld("-Xlog:gc",
"-Xms256m",
"-Xmx256m",
"-XX:+UnlockDiagnosticVMOptions",
"-XX:+UnlockExperimentalVMOptions",
"-XX:+UseShenandoahGC",
"-XX:ShenandoahGCMode=generational",
"-XX:ShenandoahGuaranteedYoungGCInterval=0",
"-XX:ShenandoahGuaranteedOldGCInterval=0"
);
}
}