mirror of
https://github.com/i-net-software/JWebAssembly.git
synced 2025-03-25 07:27:52 +01:00
duplicate THIS parameter also for interface calls
This commit is contained in:
parent
cb71374f57
commit
f2547bd0d2
@ -319,7 +319,7 @@ public class ModuleGenerator {
|
|||||||
for( WasmInstruction instruction : instructions ) {
|
for( WasmInstruction instruction : instructions ) {
|
||||||
switch( instruction.getType() ) {
|
switch( instruction.getType() ) {
|
||||||
case Call:
|
case Call:
|
||||||
case CallIndirect:
|
case CallVirtual:
|
||||||
((WasmCallInstruction)instruction).markAsNeeded( functions );
|
((WasmCallInstruction)instruction).markAsNeeded( functions );
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@ -548,7 +548,7 @@ public class ModuleGenerator {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case Call:
|
case Call:
|
||||||
case CallIndirect:
|
case CallVirtual:
|
||||||
((WasmCallInstruction)instruction).markAsNeeded( functions );
|
((WasmCallInstruction)instruction).markAsNeeded( functions );
|
||||||
break;
|
break;
|
||||||
case Struct:
|
case Struct:
|
||||||
|
@ -16,17 +16,7 @@
|
|||||||
*/
|
*/
|
||||||
package de.inetsoftware.jwebassembly.module;
|
package de.inetsoftware.jwebassembly.module;
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
|
|
||||||
import javax.annotation.Nonnull;
|
|
||||||
|
|
||||||
import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
|
import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
|
||||||
import de.inetsoftware.jwebassembly.wasm.MemoryOperator;
|
|
||||||
import de.inetsoftware.jwebassembly.wasm.NamedStorageType;
|
|
||||||
import de.inetsoftware.jwebassembly.wasm.StructOperator;
|
|
||||||
import de.inetsoftware.jwebassembly.wasm.ValueType;
|
|
||||||
import de.inetsoftware.jwebassembly.wasm.VariableOperator;
|
|
||||||
import de.inetsoftware.jwebassembly.wasm.WasmOptions;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WasmInstruction for a function call.
|
* WasmInstruction for a function call.
|
||||||
@ -34,16 +24,12 @@ import de.inetsoftware.jwebassembly.wasm.WasmOptions;
|
|||||||
* @author Volker Berlin
|
* @author Volker Berlin
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
class WasmCallIndirectInstruction extends WasmCallInstruction {
|
abstract class WasmCallIndirectInstruction extends WasmCallInstruction {
|
||||||
|
|
||||||
private int virtualFunctionIdx = -1;
|
|
||||||
|
|
||||||
private final StructType type;
|
private final StructType type;
|
||||||
|
|
||||||
private int tempVarIdx;
|
private int tempVarIdx;
|
||||||
|
|
||||||
private final WasmOptions options;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create an instance of a function call instruction
|
* Create an instance of a function call instruction
|
||||||
*
|
*
|
||||||
@ -55,21 +41,10 @@ class WasmCallIndirectInstruction extends WasmCallInstruction {
|
|||||||
* the line number in the Java source code
|
* the line number in the Java source code
|
||||||
* @param types
|
* @param types
|
||||||
* the type manager
|
* the type manager
|
||||||
* @param options
|
|
||||||
* compiler properties
|
|
||||||
*/
|
*/
|
||||||
WasmCallIndirectInstruction( FunctionName name, int javaCodePos, int lineNumber, TypeManager types, WasmOptions options ) {
|
WasmCallIndirectInstruction( FunctionName name, int javaCodePos, int lineNumber, TypeManager types ) {
|
||||||
super( name, javaCodePos, lineNumber, types, true );
|
super( name, javaCodePos, lineNumber, types, true );
|
||||||
this.type = types.valueOf( name.className );
|
this.type = types.valueOf( name.className );
|
||||||
this.options = options;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
Type getType() {
|
|
||||||
return Type.CallIndirect;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -90,12 +65,12 @@ class WasmCallIndirectInstruction extends WasmCallInstruction {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@inheritDoc}
|
* Get the variable index on which this can be found.
|
||||||
|
*
|
||||||
|
* @return the index of the variable
|
||||||
*/
|
*/
|
||||||
@Override
|
int getVariableIndexOfThis() {
|
||||||
void markAsNeeded( FunctionManager functions ) {
|
return tempVarIdx;
|
||||||
super.markAsNeeded( functions );
|
|
||||||
virtualFunctionIdx = functions.getFunctionIndex( getFunctionName() );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -103,29 +78,5 @@ class WasmCallIndirectInstruction extends WasmCallInstruction {
|
|||||||
*
|
*
|
||||||
* @return true, virtual call
|
* @return true, virtual call
|
||||||
*/
|
*/
|
||||||
boolean isVirtual() {
|
abstract boolean isVirtual();
|
||||||
return virtualFunctionIdx > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
public void writeTo( @Nonnull ModuleWriter writer ) throws IOException {
|
|
||||||
if( virtualFunctionIdx < 0 ) {
|
|
||||||
super.writeTo( writer );
|
|
||||||
} else {
|
|
||||||
// duplicate this on the stack
|
|
||||||
writer.writeLocal( VariableOperator.get, tempVarIdx );
|
|
||||||
|
|
||||||
if( options.useGC() ) {
|
|
||||||
writer.writeStructOperator( StructOperator.GET, type, new NamedStorageType( type, "", "vtable" ), 0 ); // vtable is ever on position 0
|
|
||||||
} else {
|
|
||||||
writer.writeConst( 0, ValueType.i32 ); // vtable is ever on position 0
|
|
||||||
writer.writeFunctionCall( WasmCodeBuilder.GET_I32 );
|
|
||||||
}
|
|
||||||
writer.writeMemoryOperator( MemoryOperator.load, ValueType.i32, virtualFunctionIdx * 4, 2 );
|
|
||||||
writer.writeVirtualFunctionCall( getFunctionName(), type );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,12 @@ import javax.annotation.Nonnull;
|
|||||||
|
|
||||||
import de.inetsoftware.jwebassembly.WasmException;
|
import de.inetsoftware.jwebassembly.WasmException;
|
||||||
import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
|
import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.MemoryOperator;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.NamedStorageType;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.StructOperator;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.ValueType;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.VariableOperator;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.WasmOptions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WasmInstruction for a function call.
|
* WasmInstruction for a function call.
|
||||||
@ -29,7 +35,9 @@ import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
|
|||||||
* @author Volker Berlin
|
* @author Volker Berlin
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
class WasmCallInterfaceInstruction extends WasmCallInstruction {
|
class WasmCallInterfaceInstruction extends WasmCallIndirectInstruction {
|
||||||
|
|
||||||
|
private final WasmOptions options;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create an instance of a function call instruction
|
* Create an instance of a function call instruction
|
||||||
@ -42,9 +50,12 @@ class WasmCallInterfaceInstruction extends WasmCallInstruction {
|
|||||||
* the line number in the Java source code
|
* the line number in the Java source code
|
||||||
* @param types
|
* @param types
|
||||||
* the type manager
|
* the type manager
|
||||||
|
* @param options
|
||||||
|
* compiler properties
|
||||||
*/
|
*/
|
||||||
WasmCallInterfaceInstruction( FunctionName name, int javaCodePos, int lineNumber, TypeManager types ) {
|
WasmCallInterfaceInstruction( FunctionName name, int javaCodePos, int lineNumber, TypeManager types, WasmOptions options ) {
|
||||||
super( name, javaCodePos, lineNumber, types, true );
|
super( name, javaCodePos, lineNumber, types );
|
||||||
|
this.options = options;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -55,12 +66,28 @@ class WasmCallInterfaceInstruction extends WasmCallInstruction {
|
|||||||
return Type.CallInterface;
|
return Type.CallInterface;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean isVirtual() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@inheritDoc}
|
* {@inheritDoc}
|
||||||
*/
|
*/
|
||||||
public void writeTo( @Nonnull ModuleWriter writer ) throws IOException {
|
public void writeTo( @Nonnull ModuleWriter writer ) throws IOException {
|
||||||
StructType type = getTypeManager().valueOf( getFunctionName().className );
|
FunctionName name = getFunctionName();
|
||||||
|
StructType type = getThisType();
|
||||||
int classIndex = type.getClassIndex();
|
int classIndex = type.getClassIndex();
|
||||||
|
|
||||||
|
//writer.writeLocal( VariableOperator.get, tempVarIdx ); // duplicate this on the stack
|
||||||
|
writer.writeConst( classIndex, ValueType.i32 );
|
||||||
|
//writer.writeConst( functionIndex, ValueType.i32 );
|
||||||
|
//writer.writeFunctionCall( callInterface ); // parameters: this, classIndex, functionIndex
|
||||||
|
|
||||||
|
//writer.writeVirtualFunctionCall( name, type );
|
||||||
throw new WasmException( "Interface calls are not supported.", getLineNumber() );
|
throw new WasmException( "Interface calls are not supported.", getLineNumber() );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,110 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2019 - 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.jwebassembly.module;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
|
||||||
|
import de.inetsoftware.jwebassembly.module.TypeManager.StructType;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.MemoryOperator;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.NamedStorageType;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.StructOperator;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.ValueType;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.VariableOperator;
|
||||||
|
import de.inetsoftware.jwebassembly.wasm.WasmOptions;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* WasmInstruction for a function call.
|
||||||
|
*
|
||||||
|
* @author Volker Berlin
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
class WasmCallVirtualInstruction extends WasmCallIndirectInstruction {
|
||||||
|
|
||||||
|
private int virtualFunctionIdx = -1;
|
||||||
|
|
||||||
|
private final WasmOptions options;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an instance of a function call instruction
|
||||||
|
*
|
||||||
|
* @param name
|
||||||
|
* the function name that should be called
|
||||||
|
* @param javaCodePos
|
||||||
|
* the code position/offset in the Java method
|
||||||
|
* @param lineNumber
|
||||||
|
* the line number in the Java source code
|
||||||
|
* @param types
|
||||||
|
* the type manager
|
||||||
|
* @param options
|
||||||
|
* compiler properties
|
||||||
|
*/
|
||||||
|
WasmCallVirtualInstruction( FunctionName name, int javaCodePos, int lineNumber, TypeManager types, WasmOptions options ) {
|
||||||
|
super( name, javaCodePos, lineNumber, types );
|
||||||
|
this.options = options;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
Type getType() {
|
||||||
|
return Type.CallVirtual;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
void markAsNeeded( FunctionManager functions ) {
|
||||||
|
super.markAsNeeded( functions );
|
||||||
|
virtualFunctionIdx = functions.getFunctionIndex( getFunctionName() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* if this call is executed virtual or if is was optimized.
|
||||||
|
*
|
||||||
|
* @return true, virtual call
|
||||||
|
*/
|
||||||
|
boolean isVirtual() {
|
||||||
|
return virtualFunctionIdx > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void writeTo( @Nonnull ModuleWriter writer ) throws IOException {
|
||||||
|
if( virtualFunctionIdx < 0 ) {
|
||||||
|
super.writeTo( writer );
|
||||||
|
} else {
|
||||||
|
// duplicate this on the stack
|
||||||
|
writer.writeLocal( VariableOperator.get, getVariableIndexOfThis() );
|
||||||
|
|
||||||
|
StructType type = getThisType();
|
||||||
|
if( options.useGC() ) {
|
||||||
|
writer.writeStructOperator( StructOperator.GET, type, new NamedStorageType( type, "", "vtable" ), 0 ); // vtable is ever on position 0
|
||||||
|
} else {
|
||||||
|
writer.writeConst( 0, ValueType.i32 ); // vtable is ever on position 0
|
||||||
|
writer.writeFunctionCall( WasmCodeBuilder.GET_I32 );
|
||||||
|
}
|
||||||
|
writer.writeMemoryOperator( MemoryOperator.load, ValueType.i32, virtualFunctionIdx * 4, 2 );
|
||||||
|
writer.writeVirtualFunctionCall( getFunctionName(), type );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -472,6 +472,43 @@ public abstract class WasmCodeBuilder {
|
|||||||
instructions.add( instruction );
|
instructions.add( instruction );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add indirect call to the instruction.
|
||||||
|
*
|
||||||
|
* @param indirectCall
|
||||||
|
* the instruction
|
||||||
|
*/
|
||||||
|
private void addCallIndirectInstruction( WasmCallIndirectInstruction indirectCall ) {
|
||||||
|
// For access to the vtable the THIS parameter must be duplicated on stack before the function parameters
|
||||||
|
|
||||||
|
// find the instruction that this push on stack
|
||||||
|
int count = indirectCall.getPopCount();
|
||||||
|
int idx = findPushInstruction( count, false );
|
||||||
|
WasmInstruction instr = instructions.get( idx );
|
||||||
|
int varIndex = -1;
|
||||||
|
// if it is a GET to a local variable then we can use it
|
||||||
|
if( instr.getType() == Type.Local ) {
|
||||||
|
WasmLocalInstruction local1 = (WasmLocalInstruction)instr;
|
||||||
|
if( local1.getOperator() == VariableOperator.get ) {
|
||||||
|
varIndex = local1.getIndex();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//alternate we need to create a new locale variable
|
||||||
|
if( varIndex < 0 ) {
|
||||||
|
int javaCodePos = indirectCall.getCodePosition();
|
||||||
|
varIndex = getTempVariable( indirectCall.getThisType(), instr.getCodePosition(), javaCodePos + 1 );
|
||||||
|
idx = count == 1 ? instructions.size() : findPushInstruction( count - 1, false );
|
||||||
|
instructions.add( idx, new DupThis( indirectCall, varIndex, javaCodePos ) );
|
||||||
|
}
|
||||||
|
indirectCall.setVariableIndexOfThis( varIndex );
|
||||||
|
instructions.add( indirectCall );
|
||||||
|
if( !options.useGC() ) {
|
||||||
|
// for later access of the vtable
|
||||||
|
functions.markAsNeeded( GET_I32 );
|
||||||
|
functions.markAsImport( GET_I32, GET_I32.getAnnotation() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add a virtual/method function call.
|
* Add a virtual/method function call.
|
||||||
*
|
*
|
||||||
@ -483,28 +520,7 @@ public abstract class WasmCodeBuilder {
|
|||||||
* the line number in the Java source code
|
* the line number in the Java source code
|
||||||
*/
|
*/
|
||||||
protected void addCallVirtualInstruction( FunctionName name, int javaCodePos, int lineNumber ) {
|
protected void addCallVirtualInstruction( FunctionName name, int javaCodePos, int lineNumber ) {
|
||||||
WasmCallIndirectInstruction virtualCall = new WasmCallIndirectInstruction( name, javaCodePos, lineNumber, types, options );
|
addCallIndirectInstruction( new WasmCallVirtualInstruction( name, javaCodePos, lineNumber, types, options ) );
|
||||||
int count = virtualCall.getPopCount();
|
|
||||||
int idx = findPushInstruction( count, false );
|
|
||||||
WasmInstruction instr = instructions.get( idx );
|
|
||||||
int varIndex = -1;
|
|
||||||
if( instr.getType() == Type.Local ) {
|
|
||||||
WasmLocalInstruction local1 = (WasmLocalInstruction)instr;
|
|
||||||
if( local1.getOperator() == VariableOperator.get ) {
|
|
||||||
varIndex = local1.getIndex();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if( varIndex < 0 ) {
|
|
||||||
varIndex = getTempVariable( virtualCall.getThisType(), instr.getCodePosition(), javaCodePos + 1 );
|
|
||||||
idx = count == 1 ? instructions.size() : findPushInstruction( count - 1, false );
|
|
||||||
instructions.add( idx, new DupThis( virtualCall, varIndex, javaCodePos ) );
|
|
||||||
}
|
|
||||||
virtualCall.setVariableIndexOfThis( varIndex );
|
|
||||||
instructions.add( virtualCall );
|
|
||||||
if( !options.useGC() ) {
|
|
||||||
functions.markAsNeeded( GET_I32 );
|
|
||||||
functions.markAsImport( GET_I32, GET_I32.getAnnotation() );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -517,9 +533,7 @@ public abstract class WasmCodeBuilder {
|
|||||||
* the line number in the Java source code
|
* the line number in the Java source code
|
||||||
*/
|
*/
|
||||||
protected void addCallInterfaceInstruction( FunctionName name, int javaCodePos, int lineNumber ) {
|
protected void addCallInterfaceInstruction( FunctionName name, int javaCodePos, int lineNumber ) {
|
||||||
WasmCallInterfaceInstruction interfaceCall = new WasmCallInterfaceInstruction( name, javaCodePos, lineNumber, types );
|
addCallIndirectInstruction( new WasmCallInterfaceInstruction( name, javaCodePos, lineNumber, types, options ) );
|
||||||
|
|
||||||
instructions.add( interfaceCall );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -35,7 +35,7 @@ abstract class WasmInstruction {
|
|||||||
* Type of instruction to faster differ as with instanceof.
|
* Type of instruction to faster differ as with instanceof.
|
||||||
*/
|
*/
|
||||||
static enum Type {
|
static enum Type {
|
||||||
Const, Convert, Local, Global, Table, Memory, Block, Numeric, Nop, Call, CallIndirect, CallInterface, Array, Struct, Dup, DupThis;
|
Const, Convert, Local, Global, Table, Memory, Block, Numeric, Nop, Call, CallVirtual, CallInterface, Array, Struct, Dup, DupThis;
|
||||||
}
|
}
|
||||||
|
|
||||||
private int javaCodePos;
|
private int javaCodePos;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user