mirror of
https://github.com/i-net-software/JWebAssembly.git
synced 2025-03-25 07:27:52 +01:00
InvokeDynamic, WIP
This commit is contained in:
parent
4872dd137a
commit
45b5a4a955
56
src/de/inetsoftware/classparser/BootstrapMethod.java
Normal file
56
src/de/inetsoftware/classparser/BootstrapMethod.java
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2020 Volker Berlin (i-net software)
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
|
||||||
|
*/
|
||||||
|
package de.inetsoftware.classparser;
|
||||||
|
|
||||||
|
import java.io.DataInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* https://docs.oracle.com/javase/specs/jvms/se9/html/jvms-4.html#jvms-4.7.23
|
||||||
|
*
|
||||||
|
* @author Volker Berlin
|
||||||
|
*/
|
||||||
|
public class BootstrapMethod {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Signature and return type of method to be implemented by the function object.
|
||||||
|
*/
|
||||||
|
private String samMethodType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A direct method handle describing the implementation method which should be called
|
||||||
|
*/
|
||||||
|
private ConstantMethodRef implMethod;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The signature and return type that should be enforced dynamically at invocation time. This may be the same as
|
||||||
|
* {@code samMethodType}, or may be a specialization of it.
|
||||||
|
*/
|
||||||
|
private String instantiatedMethodType;
|
||||||
|
|
||||||
|
BootstrapMethod( DataInputStream input, ConstantPool constantPool ) throws IOException {
|
||||||
|
int ref = input.readUnsignedShort();
|
||||||
|
//ConstantMethodRef method = (ConstantMethodRef)constantPool.get( ref ); // ever: java/lang/invoke/LambdaMetafactory.metafactory(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
|
||||||
|
|
||||||
|
int argCount = input.readUnsignedShort(); // ever: 3 parameters
|
||||||
|
|
||||||
|
// the 3 values
|
||||||
|
samMethodType = (String)constantPool.get( input.readUnsignedShort() );
|
||||||
|
implMethod = (ConstantMethodRef)constantPool.get( input.readUnsignedShort() );
|
||||||
|
instantiatedMethodType = (String)constantPool.get( input.readUnsignedShort() );
|
||||||
|
}
|
||||||
|
}
|
@ -62,6 +62,8 @@ public class ClassFile {
|
|||||||
|
|
||||||
private Map<String,Map<String,Object>> annotations;
|
private Map<String,Map<String,Object>> annotations;
|
||||||
|
|
||||||
|
private BootstrapMethod[] bootstrapMethods;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load a class file and create a model of the class.
|
* Load a class file and create a model of the class.
|
||||||
*
|
*
|
||||||
@ -200,6 +202,30 @@ public class ClassFile {
|
|||||||
return annotations.get( annotation );
|
return annotations.get( annotation );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the x-the BootstrapMethod. Bootstrap methods are used for creating an lambda object.
|
||||||
|
*
|
||||||
|
* @param methodIdx
|
||||||
|
* the index of the method
|
||||||
|
* @return the method
|
||||||
|
* @throws IOException
|
||||||
|
* if any error occur
|
||||||
|
*/
|
||||||
|
public BootstrapMethod getBootstrapMethod( int methodIdx ) throws IOException {
|
||||||
|
if( bootstrapMethods == null ) {
|
||||||
|
AttributeInfo data = attributes.get( "BootstrapMethods" );
|
||||||
|
if( data != null ) {
|
||||||
|
DataInputStream input = data.getDataInputStream();
|
||||||
|
int count = input.readUnsignedShort();
|
||||||
|
bootstrapMethods = new BootstrapMethod[count];
|
||||||
|
for( int i = 0; i < count; i++ ) {
|
||||||
|
bootstrapMethods[i] = new BootstrapMethod( input, constantPool );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return bootstrapMethods[methodIdx];
|
||||||
|
}
|
||||||
|
|
||||||
public ConstantPool getConstantPool() {
|
public ConstantPool getConstantPool() {
|
||||||
return constantPool;
|
return constantPool;
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright 2019 Volker Berlin (i-net software)
|
Copyright 2019 - 2020 Volker Berlin (i-net software)
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
@ -19,10 +19,12 @@ package de.inetsoftware.classparser;
|
|||||||
/**
|
/**
|
||||||
* @author Volker Berlin
|
* @author Volker Berlin
|
||||||
*/
|
*/
|
||||||
public class ConstantInvokeDynamic implements Member {
|
public class ConstantInvokeDynamic {
|
||||||
|
|
||||||
private final ConstantNameAndType nameAndType;
|
private final ConstantNameAndType nameAndType;
|
||||||
|
|
||||||
|
private final int bootstrapMethodIndex;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Invoke dynamic info in the constant pool.
|
* Invoke dynamic info in the constant pool.
|
||||||
* https://docs.oracle.com/javase/specs/jvms/se9/html/jvms-4.html#jvms-4.4.10
|
* https://docs.oracle.com/javase/specs/jvms/se9/html/jvms-4.html#jvms-4.4.10
|
||||||
@ -33,31 +35,35 @@ public class ConstantInvokeDynamic implements Member {
|
|||||||
* the name and type
|
* the name and type
|
||||||
*/
|
*/
|
||||||
ConstantInvokeDynamic( int bootstrapMethodAttrIndex, ConstantNameAndType nameAndType ) {
|
ConstantInvokeDynamic( int bootstrapMethodAttrIndex, ConstantNameAndType nameAndType ) {
|
||||||
|
this.bootstrapMethodIndex = bootstrapMethodAttrIndex;
|
||||||
this.nameAndType = nameAndType;
|
this.nameAndType = nameAndType;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@inheritDoc}
|
* The simple name of the generated method of the single function interface.
|
||||||
|
*
|
||||||
|
* @return the name
|
||||||
*/
|
*/
|
||||||
@Override
|
|
||||||
public String getName() {
|
public String getName() {
|
||||||
return nameAndType.getName();
|
return nameAndType.getName();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@inheritDoc}
|
* Get the signature of the factory method. For example "()Ljava.lang.Runnable;" for the lamba expression "Runnable
|
||||||
|
* run = () -> foo();"
|
||||||
|
*
|
||||||
|
* @return the type
|
||||||
*/
|
*/
|
||||||
@Override
|
|
||||||
public String getClassName() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the type of the method. For example "(Ljava.lang.String;)I"
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
public String getType() {
|
public String getType() {
|
||||||
return nameAndType.getType();
|
return nameAndType.getType();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
/**
|
||||||
|
* Get the index to the bootstrap methods.
|
||||||
|
*
|
||||||
|
* @return the index
|
||||||
|
*/
|
||||||
|
public int getBootstrapMethodIndex() {
|
||||||
|
return bootstrapMethodIndex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -19,9 +19,12 @@ import java.io.IOException;
|
|||||||
|
|
||||||
import javax.annotation.Nonnull;
|
import javax.annotation.Nonnull;
|
||||||
|
|
||||||
|
import de.inetsoftware.classparser.BootstrapMethod;
|
||||||
|
import de.inetsoftware.classparser.ClassFile;
|
||||||
import de.inetsoftware.classparser.Code;
|
import de.inetsoftware.classparser.Code;
|
||||||
import de.inetsoftware.classparser.CodeInputStream;
|
import de.inetsoftware.classparser.CodeInputStream;
|
||||||
import de.inetsoftware.classparser.ConstantClass;
|
import de.inetsoftware.classparser.ConstantClass;
|
||||||
|
import de.inetsoftware.classparser.ConstantInvokeDynamic;
|
||||||
import de.inetsoftware.classparser.ConstantPool;
|
import de.inetsoftware.classparser.ConstantPool;
|
||||||
import de.inetsoftware.classparser.ConstantRef;
|
import de.inetsoftware.classparser.ConstantRef;
|
||||||
import de.inetsoftware.classparser.MethodInfo;
|
import de.inetsoftware.classparser.MethodInfo;
|
||||||
@ -62,7 +65,7 @@ class JavaMethodWasmCodeBuilder extends WasmCodeBuilder {
|
|||||||
|
|
||||||
byteCode = code.getByteCode();
|
byteCode = code.getByteCode();
|
||||||
boolean hasReturn = !method.getType().endsWith( ")V" );
|
boolean hasReturn = !method.getType().endsWith( ")V" );
|
||||||
writeCode( byteCode, code.getConstantPool(), hasReturn );
|
writeCode( byteCode, code.getConstantPool(), method.getDeclaringClassFile(), hasReturn );
|
||||||
calculateVariables();
|
calculateVariables();
|
||||||
} catch( Exception ioex ) {
|
} catch( Exception ioex ) {
|
||||||
int lineNumber = byteCode == null ? -1 : byteCode.getLineNumber();
|
int lineNumber = byteCode == null ? -1 : byteCode.getLineNumber();
|
||||||
@ -77,12 +80,14 @@ class JavaMethodWasmCodeBuilder extends WasmCodeBuilder {
|
|||||||
* a stream of byte code
|
* a stream of byte code
|
||||||
* @param constantPool
|
* @param constantPool
|
||||||
* the constant pool of the the current class
|
* the constant pool of the the current class
|
||||||
|
* @param classFile
|
||||||
|
* the declaring class file
|
||||||
* @param hasReturn
|
* @param hasReturn
|
||||||
* if the method has a return value
|
* if the method has a return value
|
||||||
* @throws WasmException
|
* @throws WasmException
|
||||||
* if some Java code can't converted
|
* if some Java code can't converted
|
||||||
*/
|
*/
|
||||||
private void writeCode( CodeInputStream byteCode, ConstantPool constantPool, boolean hasReturn ) throws WasmException {
|
private void writeCode( CodeInputStream byteCode, ConstantPool constantPool, ClassFile classFile, boolean hasReturn ) throws WasmException {
|
||||||
int lineNumber = -1;
|
int lineNumber = -1;
|
||||||
try {
|
try {
|
||||||
boolean wide = false;
|
boolean wide = false;
|
||||||
@ -589,7 +594,14 @@ class JavaMethodWasmCodeBuilder extends WasmCodeBuilder {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
//TODO case 185: // invokeinterface
|
//TODO case 185: // invokeinterface
|
||||||
//TODO case 186: // invokedynamic
|
case 186: // invokedynamic
|
||||||
|
idx = byteCode.readUnsignedShort();
|
||||||
|
ConstantInvokeDynamic dynamic = (ConstantInvokeDynamic)constantPool.get( idx );
|
||||||
|
idx = byteCode.readUnsignedShort(); // ever zero
|
||||||
|
idx = dynamic.getBootstrapMethodIndex();
|
||||||
|
BootstrapMethod method = classFile.getBootstrapMethod( idx );
|
||||||
|
throw new WasmException( "InvokeDynamic/Lambda is not supported.", lineNumber );
|
||||||
|
//TODO break;
|
||||||
case 187: // new
|
case 187: // new
|
||||||
String name = ((ConstantClass)constantPool.get( byteCode.readUnsignedShort() )).getName();
|
String name = ((ConstantClass)constantPool.get( byteCode.readUnsignedShort() )).getName();
|
||||||
addStructInstruction( StructOperator.NEW_DEFAULT, name, null, codePos, lineNumber );
|
addStructInstruction( StructOperator.NEW_DEFAULT, name, null, codePos, lineNumber );
|
||||||
|
Loading…
x
Reference in New Issue
Block a user