Skip to content

Commit

Permalink
8331744: java.lang.classfile.TypeKind improvements
Browse files Browse the repository at this point in the history
Reviewed-by: asotona, redestad
  • Loading branch information
liach authored and cl4es committed May 10, 2024
1 parent dea8076 commit 784b8fc
Show file tree
Hide file tree
Showing 5 changed files with 183 additions and 16 deletions.
35 changes: 23 additions & 12 deletions src/java.base/share/classes/java/lang/classfile/TypeKind.java
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2022, 2023, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2022, 2024, 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
Expand Down Expand Up @@ -58,17 +58,20 @@ public enum TypeKind {

private final String name;
private final String descriptor;
private final int newarraycode;
private final int newarrayCode;

/** {@return the human-readable name corresponding to this type} */
public String typeName() { return name; }

/** {@return the field descriptor character corresponding to this type} */
public String descriptor() { return descriptor; }

/** {@return the code used by the {@code newarray} opcode corresponding to this type} */
public int newarraycode() {
return newarraycode;
/**
* {@return the code used by the {@code newarray} opcode corresponding to this type}
* @since 23
*/
public int newarrayCode() {
return newarrayCode;
}

/**
Expand All @@ -94,19 +97,21 @@ public TypeKind asLoadable() {
};
}

TypeKind(String name, String descriptor, int newarraycode) {
TypeKind(String name, String descriptor, int newarrayCode) {
this.name = name;
this.descriptor = descriptor;
this.newarraycode = newarraycode;
this.newarrayCode = newarrayCode;
}

/**
* {@return the type kind associated with the array type described by the
* array code used as an operand to {@code newarray}}
* @param newarraycode the operand of the {@code newarray} instruction
* @param newarrayCode the operand of the {@code newarray} instruction
* @throws IllegalArgumentException if the code is invalid
* @since 23
*/
public static TypeKind fromNewArrayCode(int newarraycode) {
return switch (newarraycode) {
public static TypeKind fromNewarrayCode(int newarrayCode) {
return switch (newarrayCode) {
case 4 -> TypeKind.BooleanType;
case 5 -> TypeKind.CharType;
case 6 -> TypeKind.FloatType;
Expand All @@ -115,15 +120,19 @@ public static TypeKind fromNewArrayCode(int newarraycode) {
case 9 -> TypeKind.ShortType;
case 10 -> TypeKind.IntType;
case 11 -> TypeKind.LongType;
default -> throw new IllegalArgumentException("Bad new array code: " + newarraycode);
default -> throw new IllegalArgumentException("Bad newarray code: " + newarrayCode);
};
}

/**
* {@return the type kind associated with the specified field descriptor}
* @param s the field descriptor
* @throws IllegalArgumentException only if the descriptor is not valid
*/
public static TypeKind fromDescriptor(CharSequence s) {
if (s.isEmpty()) { // implicit null check
throw new IllegalArgumentException("Empty descriptor");
}
return switch (s.charAt(0)) {
case '[', 'L' -> TypeKind.ReferenceType;
case 'B' -> TypeKind.ByteType;
Expand All @@ -144,6 +153,8 @@ public static TypeKind fromDescriptor(CharSequence s) {
* @param descriptor the field descriptor
*/
public static TypeKind from(TypeDescriptor.OfField<?> descriptor) {
return fromDescriptor(descriptor.descriptorString());
return descriptor.isPrimitive() // implicit null check
? fromDescriptor(descriptor.descriptorString())
: TypeKind.ReferenceType;
}
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2022, 2024, 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
Expand Down Expand Up @@ -56,7 +56,7 @@ public sealed interface NewPrimitiveArrayInstruction extends Instruction
*/
static NewPrimitiveArrayInstruction of(TypeKind typeKind) {
// Implicit null-check:
if (typeKind.newarraycode() < 0) {
if (typeKind.newarrayCode() < 0) {
throw new IllegalArgumentException("Illegal component type: " + typeKind.typeName());
}
return new AbstractInstruction.UnboundNewPrimitiveArrayInstruction(typeKind);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -557,7 +557,7 @@ public BoundNewPrimitiveArrayInstruction(Opcode op, CodeImpl code, int pos) {

@Override
public TypeKind typeKind() {
return TypeKind.fromNewArrayCode(code.classReader.readU1(pos + 1));
return TypeKind.fromNewarrayCode(code.classReader.readU1(pos + 1));
}

@Override
Expand Down Expand Up @@ -1149,7 +1149,7 @@ public TypeKind typeKind() {

@Override
public void writeTo(DirectCodeBuilder writer) {
writer.writeNewPrimitiveArray(typeKind.newarraycode());
writer.writeNewPrimitiveArray(typeKind.newarrayCode());
}

@Override
Expand Down
48 changes: 48 additions & 0 deletions test/jdk/jdk/classfile/TypeKindTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
/*
* Copyright (c) 2024, 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
* @summary Testing TypeKind.
* @bug 8331744
* @run junit TypeKindTest
*/
import org.junit.jupiter.api.Test;

import java.lang.classfile.TypeKind;

import static org.junit.Assert.assertThrows;

class TypeKindTest {
@Test
void testContracts() {
assertThrows(NullPointerException.class, () -> TypeKind.from(null));

assertThrows(NullPointerException.class, () -> TypeKind.fromDescriptor(null));
assertThrows(IllegalArgumentException.class, () -> TypeKind.fromDescriptor(""));
assertThrows(IllegalArgumentException.class, () -> TypeKind.fromDescriptor("int"));

assertThrows(IllegalArgumentException.class, () -> TypeKind.fromNewarrayCode(-1));
assertThrows(IllegalArgumentException.class, () -> TypeKind.fromNewarrayCode(21));
}
}
108 changes: 108 additions & 0 deletions test/micro/org/openjdk/bench/java/lang/classfile/TypeKindBench.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
/*
* Copyright (c) 2024, 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.
*/
package org.openjdk.bench.java.lang.classfile;

import org.openjdk.jmh.annotations.Benchmark;
import org.openjdk.jmh.annotations.BenchmarkMode;
import org.openjdk.jmh.annotations.Fork;
import org.openjdk.jmh.annotations.Measurement;
import org.openjdk.jmh.annotations.Mode;
import org.openjdk.jmh.annotations.OutputTimeUnit;
import org.openjdk.jmh.annotations.Param;
import org.openjdk.jmh.annotations.Scope;
import org.openjdk.jmh.annotations.Setup;
import org.openjdk.jmh.annotations.State;
import org.openjdk.jmh.annotations.Warmup;
import org.openjdk.jmh.infra.Blackhole;

import java.lang.classfile.TypeKind;
import java.lang.constant.ClassDesc;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Random;
import java.util.concurrent.TimeUnit;

/**
* Performance of conversion from type descriptor objects to type kind.
*/
@BenchmarkMode(Mode.AverageTime)
@OutputTimeUnit(TimeUnit.NANOSECONDS)
@Warmup(iterations = 3, time = 2)
@Measurement(iterations = 6, time = 1)
@Fork(jvmArgsAppend = "--enable-preview", value = 1)
@State(Scope.Thread)
public class TypeKindBench {

public enum ClassType {
PRIMITIVE, REFERENCE, MIXED;
}

@Param
ClassType type;
Class<?>[] classes;
ClassDesc[] classDescs;

@Setup
public void setup() {
var references = List.of(Character.class, String.class, Integer.class,
Long.class, Object.class, int[].class, TypeKindBench.class,
Byte[].class, boolean[][].class);
var primitives = List.of(int.class, long.class, void.class, double.class,
float.class, boolean.class, char.class, short.class, byte.class);
final List<Class<?>> candidates = switch (type) {
case REFERENCE -> references;
case PRIMITIVE -> primitives;
case MIXED -> {
var list = new ArrayList<Class<?>>(references.size() + primitives.size());
list.addAll(references);
list.addAll(primitives);
yield list;
}
};

// Use fixed seed to ensure results are comparable across
// different JVMs
classes = new Random(0xbf5fe40dd887d9e2L)
.ints(100, 0, candidates.size())
.mapToObj(candidates::get)
.toArray(Class<?>[]::new);
classDescs = Arrays.stream(classes)
.map(cl -> cl.describeConstable().orElseThrow())
.toArray(ClassDesc[]::new);
}

@Benchmark
public void fromClasses(Blackhole bh) {
for (var clz : classes) {
bh.consume(TypeKind.from(clz));
}
}

@Benchmark
public void fromClassDescs(Blackhole bh) {
for (var clz : classDescs) {
bh.consume(TypeKind.from(clz));
}
}
}

1 comment on commit 784b8fc

@openjdk-notifier
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please sign in to comment.