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

8294960: Convert java.base/java.lang.invoke package to use the Classfile API to generate lambdas and method handles #17108

Closed
wants to merge 41 commits into from
Closed
Changes from 1 commit
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
d193947
8294960: Convert java.base/java.lang.invoke package to use the Classf…
asotona Dec 13, 2023
57461c5
InvokerBytecodeGenerator::emit... improvements
asotona Dec 14, 2023
c1b413b
consolidation of descriptors handling
asotona Dec 14, 2023
52f08e1
performance improvements
asotona Dec 14, 2023
dc53d75
fixed InnerClassLambdaMetafactory for hidden caller classes
asotona Dec 15, 2023
0f356eb
added missing comment
asotona Dec 15, 2023
8fee456
Apply suggestions from code review
asotona Dec 18, 2023
098df10
Apply suggestions from code review
asotona Dec 18, 2023
1baa867
Merge branch 'master' into JDK-8294960-invoke
asotona Apr 4, 2024
500bb8f
Reversion of ClassBuilder changes
asotona Apr 4, 2024
9eade33
Merge branch 'master' into JDK-8294960-invoke
asotona Apr 17, 2024
304054b
applied suggested changes
asotona Apr 17, 2024
bae31c6
Merge branch 'master' into JDK-8294960-invoke
asotona Apr 18, 2024
d5cbbc6
Merge branch 'master' into JDK-8294960-invoke
asotona Apr 29, 2024
803c804
Deferred initialization of attributes map by moving into a holder class
asotona Apr 29, 2024
c2776be
Reduce init overhead of InvokeBytecodeGenerator and StackMapGenerator
cl4es Apr 29, 2024
ee3a70a
Remove stray MRE_LF_interpretWithArguments
cl4es Apr 29, 2024
75d4a09
Merge pull request #3 from cl4es/minor_init_improvements
asotona Apr 29, 2024
1717d0a
Only create box/unbox MethodRefEntries on request
cl4es Apr 29, 2024
1099de7
Merge pull request #4 from cl4es/boxunbox_holder
asotona Apr 29, 2024
107507b
Merge branch 'master' into JDK-8294960-invoke
asotona May 2, 2024
eea3652
fixed CodeBuilder use in j.l.invoke
asotona May 2, 2024
e1dbabc
Merge branch 'master' into JDK-8294960-invoke
asotona May 24, 2024
9360b0e
Merge branch 'master' into JDK-8294960-invoke
asotona Jun 5, 2024
019633b
Apply suggestions from code review
asotona Jun 6, 2024
902b02e
fixed imports
asotona Jun 6, 2024
e814749
use of jdk.internal.constant to improve performance
asotona Jun 6, 2024
c3d345c
fixed naming conventions
asotona Jun 6, 2024
f870a8d
reverted static initialization of ConstantPoolBuilder and CP entries
asotona Jun 6, 2024
9d10569
Update src/java.base/share/classes/java/lang/invoke/InvokerBytecodeGe…
asotona Jun 17, 2024
cfd2d5a
applied suggested changes
asotona Jun 17, 2024
3aaf246
Merge branch 'master' into JDK-8294960-invoke
asotona Jun 17, 2024
80170d3
SerializationHostileMethod
cl4es Jun 18, 2024
9a63310
Inline Consumer<MethodBuilder> into generateSer.. method, move seldom…
cl4es Jun 18, 2024
1ce5360
Reduce gratuitous code movement
cl4es Jun 18, 2024
d336748
Merge pull request #8 from cl4es/serialization_hostile
asotona Jun 18, 2024
857b882
Inlined condy construction directly into CP entries
asotona Jun 18, 2024
ac20f1f
Update src/java.base/share/classes/java/lang/invoke/InnerClassLambdaM…
asotona Jun 19, 2024
257d66e
fixed sneaky completion typo
asotona Jun 19, 2024
16f6565
problem-listed runtime/ClassInitErrors/TestStackOverflowDuringInit.java
asotona Jun 19, 2024
6e851a5
removed empty line
asotona Jun 19, 2024
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
22 changes: 15 additions & 7 deletions src/java.base/share/classes/java/lang/invoke/ClassSpecializer.java
Original file line number Diff line number Diff line change
@@ -41,9 +41,12 @@
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function;

import jdk.internal.constant.MethodTypeDescImpl;
import jdk.internal.constant.ReferenceClassDescImpl;
import jdk.internal.loader.BootLoader;
import jdk.internal.vm.annotation.Stable;
import sun.invoke.util.BytecodeName;
import sun.invoke.util.Wrapper;

import static java.lang.classfile.ClassFile.*;
import static java.lang.constant.ConstantDescs.*;
@@ -61,8 +64,8 @@
/*non-public*/
abstract class ClassSpecializer<T,K,S extends ClassSpecializer<T,K,S>.SpeciesData> {

private static final ClassDesc CD_LambdaForm = ClassDesc.ofDescriptor("Ljava/lang/invoke/LambdaForm;");
private static final ClassDesc CD_BoundMethodHandle = ClassDesc.ofDescriptor("Ljava/lang/invoke/BoundMethodHandle;");
private static final ClassDesc CD_LambdaForm = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/LambdaForm;");
private static final ClassDesc CD_BoundMethodHandle = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/BoundMethodHandle;");

private final Class<T> topClass;
private final Class<K> keyType;
@@ -581,7 +584,7 @@ Class<? extends T> generateConcreteSpeciesCode(String className, ClassSpecialize
// These are named like constants because there is only one per specialization scheme:

private final ClassDesc CD_SPECIES_DATA = classDesc(metaType);
private final MethodTypeDesc MTD_SPECIES_DATA = MethodTypeDesc.of(CD_SPECIES_DATA);
private final MethodTypeDesc MTD_SPECIES_DATA = MethodTypeDescImpl.ofValidated(CD_SPECIES_DATA);
private final String SPECIES_DATA_NAME = sdAccessor.getName();
private final int SPECIES_DATA_MODS = sdAccessor.getModifiers();
private final List<String> TRANSFORM_NAMES; // derived from transformMethods
@@ -603,7 +606,7 @@ Class<? extends T> generateConcreteSpeciesCode(String className, ClassSpecialize
TRANSFORM_TYPES = List.of(tts.toArray(new MethodType[0]));
TRANSFORM_MODS = List.of(tms.toArray(new Integer[0]));
}
private static final MethodTypeDesc MTD_TRANFORM_HELPER = MethodTypeDesc.of(CD_MethodHandle, CD_int);
private static final MethodTypeDesc MTD_TRANFORM_HELPER = MethodTypeDescImpl.ofValidated(CD_MethodHandle, CD_int);
private static final int ACC_PPP = ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED;

/*non-public*/
@@ -945,14 +948,19 @@ static String classBCName(String str) {
}

static ClassDesc classDesc(Class<?> cls) {
return cls == Object.class ? CD_Object
return cls.isPrimitive() ? Wrapper.forPrimitiveType(cls).classDescriptor()
: cls == Object.class ? CD_Object
: cls == MethodType.class ? CD_MethodType
: cls == LambdaForm.class ? CD_LambdaForm
: cls == BoundMethodHandle.class ? CD_BoundMethodHandle
: ClassDesc.ofDescriptor(cls.descriptorString());
: ReferenceClassDescImpl.ofValidated(cls.descriptorString());
}

static MethodTypeDesc methodDesc(MethodType mt) {
return MethodTypeDesc.ofDescriptor(mt.descriptorString());
var params = new ClassDesc[mt.parameterCount()];
for (int i = 0; i < params.length; i++) {
params[i] = classDesc(mt.parameterType(i));
}
return MethodTypeDescImpl.ofValidated(classDesc(mt.returnType()), params);
}
}
Original file line number Diff line number Diff line change
@@ -52,6 +52,9 @@
import static java.lang.invoke.MethodHandles.Lookup.ClassOption.NESTMATE;
import static java.lang.invoke.MethodHandles.Lookup.ClassOption.STRONG;
import static java.lang.invoke.MethodType.methodType;
import jdk.internal.constant.MethodTypeDescImpl;
import jdk.internal.constant.ReferenceClassDescImpl;
import sun.invoke.util.Wrapper;

/**
* Lambda metafactory implementation which dynamically creates an
@@ -63,22 +66,22 @@
private static final String LAMBDA_INSTANCE_FIELD = "LAMBDA_INSTANCE$";

// Serialization support
private static final ClassDesc CD_SERIALIZED_LAMBDA = ClassDesc.ofDescriptor("Ljava/lang/invoke/SerializedLambda;");
private static final ClassDesc CD_NOT_SERIALIZABLE_EXCEPTION = ClassDesc.ofDescriptor("Ljava/io/NotSerializableException;");
private static final ClassDesc CD_OBJECTOUTPUTSTREAM = ClassDesc.ofDescriptor("Ljava/io/ObjectOutputStream;");
private static final ClassDesc CD_OBJECTINPUTSTREAM = ClassDesc.ofDescriptor("Ljava/io/ObjectInputStream;");
private static final MethodTypeDesc MTD_METHOD_WRITE_REPLACE = MethodTypeDesc.of(CD_Object);
private static final MethodTypeDesc MTD_METHOD_WRITE_OBJECT = MethodTypeDesc.of(CD_void, CD_OBJECTOUTPUTSTREAM);
private static final MethodTypeDesc MTD_METHOD_READ_OBJECT = MethodTypeDesc.of(CD_void, CD_OBJECTINPUTSTREAM);
private static final ClassDesc CD_SERIALIZED_LAMBDA = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/SerializedLambda;");
private static final ClassDesc CD_NOT_SERIALIZABLE_EXCEPTION = ReferenceClassDescImpl.ofValidated("Ljava/io/NotSerializableException;");
private static final ClassDesc CD_OBJECTOUTPUTSTREAM = ReferenceClassDescImpl.ofValidated("Ljava/io/ObjectOutputStream;");
private static final ClassDesc CD_OBJECTINPUTSTREAM = ReferenceClassDescImpl.ofValidated("Ljava/io/ObjectInputStream;");
private static final MethodTypeDesc MTD_METHOD_WRITE_REPLACE = MethodTypeDescImpl.ofValidated(CD_Object);
private static final MethodTypeDesc MTD_METHOD_WRITE_OBJECT = MethodTypeDescImpl.ofValidated(CD_void, CD_OBJECTOUTPUTSTREAM);
private static final MethodTypeDesc MTD_METHOD_READ_OBJECT = MethodTypeDescImpl.ofValidated(CD_void, CD_OBJECTINPUTSTREAM);

private static final String NAME_METHOD_WRITE_REPLACE = "writeReplace";
private static final String NAME_METHOD_READ_OBJECT = "readObject";
private static final String NAME_METHOD_WRITE_OBJECT = "writeObject";

private static final MethodTypeDesc MTD_CTOR_SERIALIZED_LAMBDA = MethodTypeDesc.of(CD_void,
CD_Class, CD_String, CD_String, CD_String, CD_int, CD_String, CD_String, CD_String, CD_String, CD_Object.arrayType());
private static final MethodTypeDesc MTD_CTOR_SERIALIZED_LAMBDA = MethodTypeDescImpl.ofValidated(CD_void,
CD_Class, CD_String, CD_String, CD_String, CD_int, CD_String, CD_String, CD_String, CD_String, ReferenceClassDescImpl.ofValidated("[Ljava/lang/Object;"));

private static final MethodTypeDesc MTD_CTOR_NOT_SERIALIZABLE_EXCEPTION = MethodTypeDesc.of(CD_void, CD_String);
private static final MethodTypeDesc MTD_CTOR_NOT_SERIALIZABLE_EXCEPTION = MethodTypeDescImpl.ofValidated(CD_void, CD_String);

private static final String[] EMPTY_STRING_ARRAY = new String[0];
private static final ClassDesc[] EMPTY_CLASSDESC_ARRAY = new ClassDesc[0];
@@ -560,15 +563,20 @@ private Opcode invocationOpcode() throws InternalError {
}

static ClassDesc implClassDesc(Class<?> cls) {
return cls.isHidden() ? ClassDesc.ofInternalName(cls.getName().replace('.', '/'))
: classDesc(cls);
return cls.isHidden() ? ReferenceClassDescImpl.ofValidatedBinaryName(cls.getName())
: ReferenceClassDescImpl.ofValidated(cls.descriptorString());
}

static ClassDesc classDesc(Class<?> cls) {
return ClassDesc.ofDescriptor(cls.descriptorString());
return cls.isPrimitive() ? Wrapper.forPrimitiveType(cls).classDescriptor()
: ReferenceClassDescImpl.ofValidated(cls.descriptorString());
}

static MethodTypeDesc methodDesc(MethodType mt) {
return MethodTypeDesc.ofDescriptor(mt.descriptorString());
var params = new ClassDesc[mt.parameterCount()];
for (int i = 0; i < params.length; i++) {
params[i] = classDesc(mt.parameterType(i));
}
return MethodTypeDescImpl.ofValidated(classDesc(mt.returnType()), params);
}
}
Original file line number Diff line number Diff line change
@@ -51,6 +51,8 @@
import java.util.Set;
import java.util.function.Consumer;
import java.util.stream.Stream;
import jdk.internal.constant.MethodTypeDescImpl;
import jdk.internal.constant.ReferenceClassDescImpl;

import static java.lang.classfile.ClassFile.*;
import static java.lang.constant.ConstantDescs.*;
@@ -67,28 +69,27 @@
*/
class InvokerBytecodeGenerator {
/** Define class names for convenience. */
private static final ClassDesc CD_DMH = ClassDesc.ofDescriptor("Ljava/lang/invoke/DirectMethodHandle;");
private static final ClassDesc CD_MHI = ClassDesc.ofDescriptor("Ljava/lang/invoke/MethodHandleImpl;");
private static final ClassDesc CD_LF = ClassDesc.ofDescriptor("Ljava/lang/invoke/LambdaForm;");
private static final ClassDesc CD_LFN = ClassDesc.ofDescriptor("Ljava/lang/invoke/LambdaForm$Name;");
private static final ClassDesc CD_OBJARY = ClassDesc.ofDescriptor("[Ljava/lang/Object;");

private static final ClassDesc CD_LOOP_CLAUSES = ClassDesc.ofDescriptor("Ljava/lang/invoke/MethodHandleImpl$LoopClauses;");

private static final ClassDesc CD_MHARY2 = ClassDesc.ofDescriptor("[[Ljava/lang/invoke/MethodHandle;");

private static final ClassDesc CD_CasesHolder = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/MethodHandleImpl$CasesHolder;");
private static final ClassDesc CD_DirectMethodHandle = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/DirectMethodHandle;");
private static final ClassDesc CD_MethodHandleImpl = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/MethodHandleImpl;");
private static final ClassDesc CD_LambdaForm = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/LambdaForm;");
private static final ClassDesc CD_LambdaForm_Name = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/LambdaForm$Name;");
private static final ClassDesc CD_Object_array = ReferenceClassDescImpl.ofValidated("[Ljava/lang/Object;");
private static final ClassDesc CD_LoopClauses_array = ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/MethodHandleImpl$LoopClauses;");
private static final ClassDesc CD_MethodHandle_array = ReferenceClassDescImpl.ofValidated("[Ljava/lang/invoke/MethodHandle;");
private static final ClassDesc CD_MethodHandle_array2 = ReferenceClassDescImpl.ofValidated("[[Ljava/lang/invoke/MethodHandle;");

private static final String CLASS_PREFIX = "java/lang/invoke/LambdaForm$";
private static final String SOURCE_PREFIX = "LambdaForm$";

private static final ConstantPoolBuilder CP = ConstantPoolBuilder.of();
private static final ClassEntry CE_Object = CP.classEntry(CD_Object);
private static final ClassEntry CE_OBJARY = CP.classEntry(CD_OBJARY);
private static final ClassEntry CE_OBJARY = CP.classEntry(CD_Object_array);
private static final ClassEntry CE_MethodHandle = CP.classEntry(CD_MethodHandle);
private static final InterfaceMethodRefEntry MRE_List_get = CP.interfaceMethodRefEntry(CD_List, "get", MethodTypeDesc.of(CD_Object, CD_int));
private static final MethodRefEntry MRE_MethodHandles_classData = CP.methodRefEntry(CD_MethodHandles, "classData", MethodTypeDesc.of(CD_Object, CD_Class));
private static final MethodRefEntry MRE_Class_cast = CP.methodRefEntry(CD_Class, "cast", MethodTypeDesc.of(CD_Object, CD_Object));
private static final MethodRefEntry MRE_Class_isInstance = CP.methodRefEntry(CD_Class, "isInstance", MethodTypeDesc.of(CD_boolean, CD_Object));
private static final InterfaceMethodRefEntry MRE_List_get = CP.interfaceMethodRefEntry(CD_List, "get", MethodTypeDescImpl.ofValidated(CD_Object, CD_int));
private static final MethodRefEntry MRE_MethodHandles_classData = CP.methodRefEntry(CD_MethodHandles, "classData", MethodTypeDescImpl.ofValidated(CD_Object, CD_Class));
private static final MethodRefEntry MRE_Class_cast = CP.methodRefEntry(CD_Class, "cast", MethodTypeDescImpl.ofValidated(CD_Object, CD_Object));
private static final MethodRefEntry MRE_Class_isInstance = CP.methodRefEntry(CD_Class, "isInstance", MethodTypeDescImpl.ofValidated(CD_boolean, CD_Object));

// Static builders to avoid lambdas
private static final Consumer<FieldBuilder> STATIC_FINAL_FIELD = new Consumer<FieldBuilder>() {
@@ -232,7 +233,7 @@ String classData(Object arg) {
} else if (arg instanceof MethodHandle) {
desc = CD_MethodHandle;
} else if (arg instanceof LambdaForm) {
desc = CD_LF;
desc = CD_LambdaForm;
} else {
desc = CD_Object;
}
@@ -544,15 +545,15 @@ private boolean checkActualReceiver(CodeBuilder cob) {
// Expects MethodHandle on the stack and actual receiver MethodHandle in slot #0
cob.dup()
.aload(0)
Copy link
Member

Choose a reason for hiding this comment

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

Note for other reviewers: localsMap[0] is 0 in constructor

.invokestatic(CD_MHI, "assertSame", MethodTypeDesc.of(CD_void, CD_Object, CD_Object));
.invokestatic(CD_MethodHandleImpl, "assertSame", MethodTypeDescImpl.ofValidated(CD_void, CD_Object, CD_Object));
return true;
}

static final Annotation DONTINLINE = Annotation.of(ClassDesc.ofDescriptor("Ljdk/internal/vm/annotation/DontInline;"));
static final Annotation FORCEINLINE = Annotation.of(ClassDesc.ofDescriptor("Ljdk/internal/vm/annotation/ForceInline;"));
static final Annotation HIDDEN = Annotation.of(ClassDesc.ofDescriptor("Ljdk/internal/vm/annotation/Hidden;"));
static final Annotation INJECTEDPROFILE = Annotation.of(ClassDesc.ofDescriptor("Ljava/lang/invoke/InjectedProfile;"));
static final Annotation LF_COMPILED = Annotation.of(ClassDesc.ofDescriptor("Ljava/lang/invoke/LambdaForm$Compiled;"));
static final Annotation DONTINLINE = Annotation.of(ReferenceClassDescImpl.ofValidated("Ljdk/internal/vm/annotation/DontInline;"));
static final Annotation FORCEINLINE = Annotation.of(ReferenceClassDescImpl.ofValidated("Ljdk/internal/vm/annotation/ForceInline;"));
static final Annotation HIDDEN = Annotation.of(ReferenceClassDescImpl.ofValidated("Ljdk/internal/vm/annotation/Hidden;"));
static final Annotation INJECTEDPROFILE = Annotation.of(ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/InjectedProfile;"));
static final Annotation LF_COMPILED = Annotation.of(ReferenceClassDescImpl.ofValidated("Ljava/lang/invoke/LambdaForm$Compiled;"));

/**
* Generate an invoker method for the passed {@link LambdaForm}.
@@ -740,8 +741,8 @@ void emitInvoke(CodeBuilder cob, Name name) {
// load receiver
cob.aload(0);
emitReferenceCast(cob, MethodHandle.class, null);
cob.getfield(CD_MethodHandle, "form", CD_LF)
.getfield(CD_LF, "names", CD_LFN);
cob.getfield(CD_MethodHandle, "form", CD_LambdaForm)
.getfield(CD_LambdaForm, "names", CD_LambdaForm_Name);
// TODO more to come
}

@@ -1170,8 +1171,7 @@ private Name emitTableSwitch(CodeBuilder cob, int pos, int numCases) {
MethodTypeDesc caseDescriptor = methodDesc(caseType.basicType());

emitPushArgument(cob, invoker, 2); // push cases
cob.getfield(ClassDesc.ofInternalName("java/lang/invoke/MethodHandleImpl$CasesHolder"), "cases",
CD_MethodHandle.arrayType());
cob.getfield(CD_CasesHolder, "cases", CD_MethodHandle_array);
int casesLocal = extendLocalsMap(new Class<?>[] { MethodHandle[].class });
Copy link
Member

Choose a reason for hiding this comment

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

We should look into replacing the local map with CodeBuilder.allocateLocal etc. in the future.

Copy link
Member Author

Choose a reason for hiding this comment

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

Yes, however actual API does not match directly.
It would require an extension of the API or more significant refactoring of InvokerBytecodeGenerator.

emitStoreInsn(cob, TypeKind.ReferenceType, casesLocal);

@@ -1332,7 +1332,7 @@ private Name emitLoop(CodeBuilder cob, int pos) {

// PREINIT:
emitPushArgument(cob, MethodHandleImpl.LoopClauses.class, invoker.arguments[1]);
cob.getfield(CD_LOOP_CLAUSES, "clauses", CD_MHARY2);
cob.getfield(CD_LoopClauses_array, "clauses", CD_MethodHandle_array2);
emitStoreInsn(cob, TypeKind.ReferenceType, clauseDataIndex);

// INIT:
@@ -1572,9 +1572,9 @@ public void accept(CodeBuilder cob) {
}
// invoke
cob.aload(0);
cob.getfield(CD_MethodHandle, "form", CD_LF);
cob.getfield(CD_MethodHandle, "form", CD_LambdaForm);
cob.swap(); // swap form and array; avoid local variable
cob.invokevirtual(CD_LF, "interpretWithArguments", MethodTypeDesc.of(CD_Object, CD_OBJARY));
cob.invokevirtual(CD_LambdaForm, "interpretWithArguments", MethodTypeDescImpl.ofValidated(CD_Object, CD_Object_array));

// maybe unbox
Class<?> rtype = invokerType.returnType();
@@ -1691,13 +1691,18 @@ public void accept(CodeBuilder cob) {

static ClassDesc classDesc(Class<?> cls) {
// assert(VerifyAccess.isTypeVisible(cls, Object.class)) : cls.getName();
return cls == MethodHandle.class ? CD_MethodHandle
: cls == DirectMethodHandle.class ? CD_DMH
return cls.isPrimitive() ? Wrapper.forPrimitiveType(cls).classDescriptor()
: cls == MethodHandle.class ? CD_MethodHandle
: cls == DirectMethodHandle.class ? CD_DirectMethodHandle
: cls == Object.class ? CD_Object
: ClassDesc.ofDescriptor(cls.descriptorString());
: ReferenceClassDescImpl.ofValidated(cls.descriptorString());
}

static MethodTypeDesc methodDesc(MethodType mt) {
return MethodTypeDesc.ofDescriptor(mt.descriptorString());
var params = new ClassDesc[mt.parameterCount()];
for (int i = 0; i < params.length; i++) {
params[i] = classDesc(mt.parameterType(i));
}
return MethodTypeDescImpl.ofValidated(classDesc(mt.returnType()), params);
}
}
Loading