jdk-24/test/hotspot/jtreg/runtime/FieldLayout/FieldDensityTest.java
Frederic Parain 9886cb401c 8237767: Field layout computation overhaul
Reviewed-by: dholmes, coleenp, lfoltan, shade
2020-02-10 09:49:12 -05:00

158 lines
4.5 KiB
Java

/*
* Copyright (c) 2020, Oracle and/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
* @bug 8237767
* @summary Verify behaviour of field layout algorithm
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.management
* @run main/othervm FieldDensityTest
*/
/*
* @test
* @requires vm.bits == "64"
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.management
* @run main/othervm -XX:+UseCompressedOops -XX:+UseCompressedClassPointers FieldDensityTest
* @run main/othervm -XX:+UseCompressedOops -XX:-UseCompressedClassPointers FieldDensityTest
*/
import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.Comparator;
import jdk.internal.misc.Unsafe;
import jdk.test.lib.Asserts;
public class FieldDensityTest {
static int OOP_SIZE_IN_BYTES = 0;
static {
if (System.getProperty("sun.arch.data.model").equals("64")) {
if (System.getProperty("java.vm.compressedOopsMode") == null) {
OOP_SIZE_IN_BYTES = 8;
} else {
OOP_SIZE_IN_BYTES = 4;
}
} else {
OOP_SIZE_IN_BYTES = 4;
}
}
static class FieldInfo {
public Field field;
public long offset;
FieldInfo(Field field, long offset) {
this.field = field;
this.offset = offset;
}
static void checkFieldsContiguity(FieldInfo[] fieldInfo) {
Arrays.sort(fieldInfo, new SortByOffset());
for (int i = 0 ; i < fieldInfo.length - 2; i++) {
int size = sizeInBytesFromType(fieldInfo[i].field.getType());
Asserts.assertEquals((int)(fieldInfo[i].offset + size), (int)fieldInfo[i+1].offset,
"Empty slot between fields, should not happen");
}
}
}
static int sizeInBytesFromType(Class type) {
if (!type.isPrimitive()) {
return OOP_SIZE_IN_BYTES;
}
switch(type.getTypeName()) {
case "boolean":
case "byte": return 1;
case "char":
case "short": return 2;
case "int":
case "float": return 4;
case "long":
case "double": return 8;
default:
throw new RuntimeException("Unrecognized signature");
}
}
static class SortByOffset implements Comparator<FieldInfo> {
public int compare(FieldInfo a, FieldInfo b)
{
return (int)(a.offset - b.offset);
}
}
static class E {
public byte b0;
}
static class F extends E {
public byte b1;
}
static class G extends F {
public byte b2;
}
static class H extends G {
public byte b3;
}
public static class A {
public int i;
public byte b;
public long l;
public Object o;
}
public static class B extends A {
public byte b0, b1, b2;
}
static void testFieldsContiguity(Class c) {
Unsafe unsafe = Unsafe.getUnsafe();
Field[] fields = c.getFields();
FieldInfo[] fieldsInfo = new FieldInfo[fields.length];
int i = 0;
for (Field f : fields) {
long offset = unsafe.objectFieldOffset(f);
fieldsInfo[i] = new FieldInfo(f, offset);
i++;
}
FieldInfo.checkFieldsContiguity(fieldsInfo);
}
public static void main(String[] args) {
H h = new H();
testFieldsContiguity(h.getClass());
B b = new B();
testFieldsContiguity(b.getClass());
}
}