Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

8331744: java.lang.classfile.TypeKind improvements #19109

Closed
wants to merge 7 commits into from
Closed
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
39 changes: 13 additions & 26 deletions src/java.base/share/classes/java/lang/classfile/TypeKind.java
Original file line number Diff line number Diff line change
@@ -27,7 +27,6 @@

import java.lang.invoke.TypeDescriptor;
import jdk.internal.javac.PreviewFeature;
import jdk.internal.vm.annotation.Stable;

/**
* Describes the types that can be part of a field or method descriptor.
@@ -59,7 +58,6 @@ public enum TypeKind {

private final String name;
private final String descriptor;
private final char descriptorChar;
private final int newarrayCode;

/** {@return the human-readable name corresponding to this type} */
@@ -103,7 +101,6 @@ public TypeKind asLoadable() {
this.name = name;
this.descriptor = descriptor;
this.newarrayCode = newarrayCode;
this.descriptorChar = descriptor.charAt(0);
}

/**
@@ -136,11 +133,19 @@ public static TypeKind fromDescriptor(CharSequence s) {
if (s.isEmpty()) { // implicit null check
throw new IllegalArgumentException("Empty descriptor");
}
var ret = fromChar(s.charAt(0));
if (ret == null) {
throw new IllegalArgumentException("Bad type: " + s);
}
return ret;
return switch (s.charAt(0)) {
case '[', 'L' -> TypeKind.ReferenceType;
case 'B' -> TypeKind.ByteType;
case 'C' -> TypeKind.CharType;
case 'Z' -> TypeKind.BooleanType;
case 'S' -> TypeKind.ShortType;
case 'I' -> TypeKind.IntType;
case 'F' -> TypeKind.FloatType;
case 'J' -> TypeKind.LongType;
case 'D' -> TypeKind.DoubleType;
case 'V' -> TypeKind.VoidType;
default -> throw new IllegalArgumentException("Bad type: " + s);
};
}

/**
@@ -152,22 +157,4 @@ public static TypeKind from(TypeDescriptor.OfField<?> descriptor) {
? fromDescriptor(descriptor.descriptorString())
: TypeKind.ReferenceType;
}

private static final @Stable TypeKind[] TABLE = new TypeKind[16];

static {
for (TypeKind kind : values()) {
char c = kind.descriptorChar;
TABLE[(c + (c >> 1)) & 0xf] = kind;
}
TABLE[8] = TypeKind.ReferenceType; // hash for '['
}

private static TypeKind fromChar(char c) {
var tk = TABLE[(c + (c >> 1)) & 0xf];
if (tk.descriptorChar == c || c == '[') {
return tk;
}
return null;
}
}
30 changes: 0 additions & 30 deletions test/jdk/jdk/classfile/TypeKindTest.java
Original file line number Diff line number Diff line change
@@ -32,7 +32,6 @@
import java.lang.classfile.TypeKind;

import static org.junit.Assert.assertThrows;
import static org.junit.jupiter.api.Assertions.assertEquals;

class TypeKindTest {
@Test
@@ -46,33 +45,4 @@ void testContracts() {
assertThrows(IllegalArgumentException.class, () -> TypeKind.fromNewarrayCode(-1));
assertThrows(IllegalArgumentException.class, () -> TypeKind.fromNewarrayCode(21));
}

@Test
void testMapping() {
for (char c : "BCDFIJSZVL[".toCharArray()) {
var s = String.valueOf(c);
assertEquals(fromDescriptorSwitch(s), TypeKind.fromDescriptor(s));
}

assertThrows(IllegalArgumentException.class, () -> TypeKind.fromDescriptor("E"));
// hash of P same as [
assertThrows(IllegalArgumentException.class, () -> TypeKind.fromDescriptor("P"));
}

// The known correct switch-based version of fromDescriptor
private static TypeKind fromDescriptorSwitch(String s) {
return switch (s.charAt(0)) {
case '[', 'L' -> TypeKind.ReferenceType;
case 'B' -> TypeKind.ByteType;
case 'C' -> TypeKind.CharType;
case 'Z' -> TypeKind.BooleanType;
case 'S' -> TypeKind.ShortType;
case 'I' -> TypeKind.IntType;
case 'F' -> TypeKind.FloatType;
case 'J' -> TypeKind.LongType;
case 'D' -> TypeKind.DoubleType;
case 'V' -> TypeKind.VoidType;
default -> throw new IllegalArgumentException("Bad type: " + s);
};
}
}