2015-08-07 17:58:01 +03:00
|
|
|
/*
|
2020-04-28 19:57:02 -07:00
|
|
|
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
|
2015-08-07 17:58:01 +03:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package gc.g1.humongousObjects;
|
|
|
|
|
2015-12-10 20:14:00 +03:00
|
|
|
import gc.testlibrary.Helpers;
|
2015-08-07 17:58:01 +03:00
|
|
|
import jdk.test.lib.Asserts;
|
|
|
|
import sun.hotspot.WhiteBox;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test TestHumongousThreshold
|
|
|
|
* @summary Checks that objects larger than half a region are allocated as humongous
|
2016-06-24 19:52:31 +04:00
|
|
|
* @requires vm.gc.G1
|
2016-08-19 10:06:30 -04:00
|
|
|
* @library /test/lib /
|
2016-05-01 12:47:00 +03:00
|
|
|
* @modules java.base/jdk.internal.misc
|
2015-08-07 17:58:01 +03:00
|
|
|
* @modules java.management
|
|
|
|
* @build sun.hotspot.WhiteBox
|
|
|
|
* @run driver ClassFileInstaller sun.hotspot.WhiteBox
|
|
|
|
*
|
|
|
|
* @run main/othervm -XX:+UseG1GC -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:.
|
|
|
|
* -XX:G1HeapRegionSize=1M
|
|
|
|
* gc.g1.humongousObjects.TestHumongousThreshold
|
|
|
|
*
|
|
|
|
* @run main/othervm -XX:+UseG1GC -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:.
|
|
|
|
* -XX:G1HeapRegionSize=2M
|
|
|
|
* gc.g1.humongousObjects.TestHumongousThreshold
|
|
|
|
*
|
|
|
|
* @run main/othervm -XX:+UseG1GC -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:.
|
|
|
|
* -XX:G1HeapRegionSize=4M
|
|
|
|
* gc.g1.humongousObjects.TestHumongousThreshold
|
|
|
|
*
|
|
|
|
* @run main/othervm -XX:+UseG1GC -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:.
|
|
|
|
* -XX:G1HeapRegionSize=8M
|
|
|
|
* gc.g1.humongousObjects.TestHumongousThreshold
|
|
|
|
*
|
|
|
|
* @run main/othervm -XX:+UseG1GC -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:.
|
2015-09-09 15:14:05 +03:00
|
|
|
* -Xms128M -XX:G1HeapRegionSize=16M
|
2015-08-07 17:58:01 +03:00
|
|
|
* gc.g1.humongousObjects.TestHumongousThreshold
|
|
|
|
*
|
|
|
|
* @run main/othervm -XX:+UseG1GC -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:.
|
2015-09-09 15:14:05 +03:00
|
|
|
* -Xms200M -XX:G1HeapRegionSize=32M
|
2015-08-07 17:58:01 +03:00
|
|
|
* gc.g1.humongousObjects.TestHumongousThreshold
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
public class TestHumongousThreshold {
|
|
|
|
private static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
|
|
|
|
private static final int REGION_SIZE = WHITE_BOX.g1RegionSize();
|
|
|
|
private static final int MAX_CONTINUOUS_SIZE_CHECK = 129;
|
2016-03-14 20:00:57 +03:00
|
|
|
private static final int NON_HUMONGOUS_STEPS = 10;
|
2015-08-07 17:58:01 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The method allocates byte[] with specified size and checks that:
|
|
|
|
* 1. byte[] is allocated as we specified in expectedHumongous.
|
|
|
|
* 2. byte[] is allocated as humongous if its size is large than a half of region and non-humongous otherwise.
|
|
|
|
* It uses WB to obtain the size of created byte[]. Only objects larger than half of region are expected
|
|
|
|
* to be humongous.
|
|
|
|
*
|
|
|
|
* @param arraySize size of allocation
|
|
|
|
* @param expectedHumongous expected humongous/non-humongous allocation
|
|
|
|
* @return allocated byte array
|
|
|
|
*/
|
|
|
|
|
2016-03-14 20:00:57 +03:00
|
|
|
private static void allocateAndCheck(int arraySize, boolean expectedHumongous) {
|
2015-08-07 17:58:01 +03:00
|
|
|
byte[] storage = new byte[arraySize];
|
|
|
|
long objectSize = WHITE_BOX.getObjectSize(storage);
|
|
|
|
boolean shouldBeHumongous = objectSize > (REGION_SIZE / 2);
|
|
|
|
|
|
|
|
Asserts.assertEquals(expectedHumongous, shouldBeHumongous, "Despite we expected this object to be "
|
|
|
|
+ (expectedHumongous ? "humongous" : "non-humongous") + " it appeared otherwise when we checked "
|
|
|
|
+ "object size - likely test bug; Allocation size = " + arraySize + "; Object size = " + objectSize
|
|
|
|
+ "; region size = " + REGION_SIZE);
|
|
|
|
|
|
|
|
Asserts.assertEquals(WHITE_BOX.g1IsHumongous(storage), shouldBeHumongous,
|
|
|
|
"Object should be allocated as " + (shouldBeHumongous ? "humongous"
|
|
|
|
: "non-humongous") + " but it wasn't; Allocation size = " + arraySize + "; Object size = "
|
|
|
|
+ objectSize + "; region size = " + REGION_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String[] args) {
|
|
|
|
int byteArrayMemoryOverhead = Helpers.detectByteArrayAllocationOverhead();
|
|
|
|
|
|
|
|
// Largest non-humongous byte[]
|
|
|
|
int maxByteArrayNonHumongousSize = (REGION_SIZE / 2) - byteArrayMemoryOverhead;
|
|
|
|
|
|
|
|
// Increment for non-humongous testing
|
2016-03-14 20:00:57 +03:00
|
|
|
int nonHumongousStep = maxByteArrayNonHumongousSize / NON_HUMONGOUS_STEPS;
|
2015-08-07 17:58:01 +03:00
|
|
|
|
|
|
|
// Maximum byte[] that takes one region
|
|
|
|
int maxByteArrayOneRegionSize = REGION_SIZE - byteArrayMemoryOverhead;
|
|
|
|
|
|
|
|
// Sizes in regions
|
|
|
|
// i,e, 1.0f means one region, 1.5f means one and half region etc
|
|
|
|
float[] humongousFactors = {0.8f, 1.0f, 1.2f, 1.5f, 1.7f, 2.0f, 2.5f};
|
|
|
|
|
|
|
|
// Some diagnostic output
|
|
|
|
System.out.format("%s started%n", TestHumongousThreshold.class.getName());
|
|
|
|
System.out.format("Actual G1 region size %d%n", REGION_SIZE);
|
|
|
|
System.out.format("byte[] memory overhead %d%n", byteArrayMemoryOverhead);
|
|
|
|
|
|
|
|
// Non-humongous allocations
|
|
|
|
System.out.format("Doing non-humongous allocations%n");
|
|
|
|
|
|
|
|
// Testing allocations with byte[] with length from 0 to MAX_CONTINUOUS_SIZE_CHECK
|
|
|
|
System.out.format("Testing allocations with byte[] with length from 0 to %d%n", MAX_CONTINUOUS_SIZE_CHECK);
|
|
|
|
for (int i = 0; i < MAX_CONTINUOUS_SIZE_CHECK; ++i) {
|
|
|
|
allocateAndCheck(i, false);
|
|
|
|
}
|
|
|
|
|
2016-03-14 20:00:57 +03:00
|
|
|
// Testing allocations with byte[] with length from 0 to nonHumongousStep * NON_HUMONGOUS_STEPS
|
2015-08-07 17:58:01 +03:00
|
|
|
System.out.format("Testing allocations with byte[] with length from 0 to %d with step %d%n",
|
2016-03-14 20:00:57 +03:00
|
|
|
nonHumongousStep * NON_HUMONGOUS_STEPS, nonHumongousStep);
|
|
|
|
for (int i = 0; i < NON_HUMONGOUS_STEPS; ++i) {
|
2015-08-07 17:58:01 +03:00
|
|
|
allocateAndCheck(i * nonHumongousStep, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Testing allocations with byte[] of maximum non-humongous length
|
|
|
|
System.out.format("Testing allocations with byte[] of maximum non-humongous length %d%n",
|
|
|
|
maxByteArrayNonHumongousSize);
|
|
|
|
allocateAndCheck(maxByteArrayNonHumongousSize, false);
|
|
|
|
|
|
|
|
// Humongous allocations
|
|
|
|
System.out.format("Doing humongous allocations%n");
|
|
|
|
// Testing with minimum humongous object
|
|
|
|
System.out.format("Testing with byte[] of minimum humongous object %d%n", maxByteArrayNonHumongousSize + 1);
|
|
|
|
allocateAndCheck(maxByteArrayNonHumongousSize + 1, true);
|
|
|
|
|
|
|
|
// Testing allocations with byte[] with length from (maxByteArrayNonHumongousSize + 1) to
|
|
|
|
// (maxByteArrayNonHumongousSize + 1 + MAX_CONTINUOUS_SIZE_CHECK)
|
|
|
|
System.out.format("Testing allocations with byte[] with length from %d to %d%n",
|
|
|
|
maxByteArrayNonHumongousSize + 1, maxByteArrayNonHumongousSize + 1 + MAX_CONTINUOUS_SIZE_CHECK);
|
|
|
|
for (int i = 0; i < MAX_CONTINUOUS_SIZE_CHECK; ++i) {
|
|
|
|
allocateAndCheck(maxByteArrayNonHumongousSize + 1 + i, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checking that large (more than a half of region size) objects are humongous
|
|
|
|
System.out.format("Checking that large (more than a half of region size) objects are humongous%n");
|
|
|
|
for (float factor : humongousFactors) {
|
|
|
|
allocateAndCheck((int) (maxByteArrayOneRegionSize * factor), true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|