9886cb401c
Reviewed-by: dholmes, coleenp, lfoltan, shade
158 lines
4.5 KiB
Java
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());
|
|
}
|
|
}
|