Add IR parser of invoke instructions

This commit is contained in:
Alexey Andreev 2016-12-03 19:45:59 +03:00
parent c16de76b23
commit 825acfc85a
6 changed files with 176 additions and 34 deletions

View File

@ -21,10 +21,6 @@ import org.teavm.model.Instruction;
import org.teavm.model.MethodReference; import org.teavm.model.MethodReference;
import org.teavm.model.Variable; import org.teavm.model.Variable;
/**
*
* @author Alexey Andreev
*/
public class InvokeInstruction extends Instruction { public class InvokeInstruction extends Instruction {
private InvocationType type; private InvocationType type;
private MethodReference method; private MethodReference method;

View File

@ -17,6 +17,7 @@ package org.teavm.model.text;
import java.util.Arrays; import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.Locale;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import org.teavm.model.*; import org.teavm.model.*;
import org.teavm.model.instructions.*; import org.teavm.model.instructions.*;
@ -45,7 +46,8 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void classConstant(VariableReader receiver, ValueType cst) { public void classConstant(VariableReader receiver, ValueType cst) {
sb.append("@").append(receiver.getIndex()).append(" := classOf ").append(cst); sb.append("@").append(receiver.getIndex()).append(" := classOf ");
escapeIdentifierIfNeeded(cst.toString(), sb);
} }
@Override @Override
@ -112,6 +114,27 @@ class InstructionStringifier implements InstructionReader {
} }
} }
private static void escapeIdentifierIfNeeded(String s, StringBuilder sb) {
boolean needsEscaping = false;
if (s.isEmpty()) {
needsEscaping = true;
} else if (!ListingLexer.isIdentifierStart(s.charAt(0))) {
needsEscaping = true;
} else {
for (int i = 1; i < s.length(); ++i) {
if (!ListingLexer.isIdentifierPart(s.charAt(i))) {
needsEscaping = true;
break;
}
}
}
if (needsEscaping) {
sb.append('`').append(s).append('`');
} else {
sb.append(s);
}
}
@Override @Override
public void binary(BinaryOperation op, VariableReader receiver, VariableReader first, VariableReader second, public void binary(BinaryOperation op, VariableReader receiver, VariableReader first, VariableReader second,
NumericOperandType type) { NumericOperandType type) {
@ -170,7 +193,8 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void cast(VariableReader receiver, VariableReader value, ValueType targetType) { public void cast(VariableReader receiver, VariableReader value, ValueType targetType) {
sb.append("@").append(receiver.getIndex()).append(" := cast @").append(value.getIndex()) sb.append("@").append(receiver.getIndex()).append(" := cast @").append(value.getIndex())
.append(" to ").append(targetType); .append(" to ");
escapeIdentifierIfNeeded(targetType.toString(), sb);
} }
@Override @Override
@ -186,10 +210,10 @@ class InstructionStringifier implements InstructionReader {
sb.append("@").append(receiver.getIndex()).append(" := cast @").append(value.getIndex()); sb.append("@").append(receiver.getIndex()).append(" := cast @").append(value.getIndex());
switch (direction) { switch (direction) {
case FROM_INTEGER: case FROM_INTEGER:
sb.append(" from INT to ").append(type); sb.append(" from int to ").append(type.name().toLowerCase(Locale.ROOT));
break; break;
case TO_INTEGER: case TO_INTEGER:
sb.append(" from ").append(type).append(" to INT"); sb.append(" from ").append(type.name().toLowerCase(Locale.ROOT)).append(" to int");
break; break;
} }
} }
@ -263,9 +287,9 @@ class InstructionStringifier implements InstructionReader {
sb.append("; "); sb.append("; ");
} }
SwitchTableEntryReader entry = table.get(i); SwitchTableEntryReader entry = table.get(i);
sb.append("case ").append(entry.getCondition()).append(": goto $").append(entry.getTarget().getIndex()); sb.append("if ").append(entry.getCondition()).append(" goto $").append(entry.getTarget().getIndex());
} }
sb.append(", default: goto $").append(defaultTarget.getIndex()); sb.append(" else goto $").append(defaultTarget.getIndex());
} }
@Override @Override
@ -283,13 +307,17 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void createArray(VariableReader receiver, ValueType itemType, VariableReader size) { public void createArray(VariableReader receiver, ValueType itemType, VariableReader size) {
sb.append("@").append(receiver.getIndex()).append(" := new ").append(itemType).append("[@") sb.append("@").append(receiver.getIndex()).append(" := new ");
.append(size.getIndex()).append(']'); escapeIdentifierIfNeeded(itemType.toString(), sb);
sb.append("[@").append(size.getIndex()).append(']');
} }
@Override @Override
public void createArray(VariableReader receiver, ValueType itemType, List<? extends VariableReader> dimensions) { public void createArray(VariableReader receiver, ValueType itemType, List<? extends VariableReader> dimensions) {
sb.append("@").append(receiver.getIndex()).append(" := new ").append(itemType).append("["); sb.append("@").append(receiver.getIndex()).append(" := new ");
escapeIdentifierIfNeeded(itemType.toString(), sb);
sb.append("[");
for (int i = 0; i < dimensions.size(); ++i) { for (int i = 0; i < dimensions.size(); ++i) {
if (i > 0) { if (i > 0) {
sb.append(", "); sb.append(", ");
@ -301,12 +329,15 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void create(VariableReader receiver, String type) { public void create(VariableReader receiver, String type) {
sb.append("@").append(receiver.getIndex()).append(" := new ").append(type).append(""); sb.append("@").append(receiver.getIndex()).append(" := new ");
escapeIdentifierIfNeeded(type, sb);
sb.append("");
} }
@Override @Override
public void getField(VariableReader receiver, VariableReader instance, FieldReference field, ValueType fieldType) { public void getField(VariableReader receiver, VariableReader instance, FieldReference field, ValueType fieldType) {
sb.append("@").append(receiver.getIndex()).append(" := field " + field); sb.append("@").append(receiver.getIndex()).append(" := field ");
escapeIdentifierIfNeeded(field.toString(), sb);
sb.append(field); sb.append(field);
if (instance != null) { if (instance != null) {
sb.append(" @").append(instance.getIndex()); sb.append(" @").append(instance.getIndex());
@ -315,7 +346,8 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void putField(VariableReader instance, FieldReference field, VariableReader value, ValueType fieldType) { public void putField(VariableReader instance, FieldReference field, VariableReader value, ValueType fieldType) {
sb.append("field " + field); sb.append("field ");
escapeIdentifierIfNeeded(field.toString(), sb);
if (instance != null) { if (instance != null) {
sb.append(" @").append(instance.getIndex()); sb.append(" @").append(instance.getIndex());
} }
@ -329,19 +361,20 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void cloneArray(VariableReader receiver, VariableReader array) { public void cloneArray(VariableReader receiver, VariableReader array) {
sb.append("@").append(receiver.getIndex()).append(" := clone @").append(array.getIndex()).append(""); sb.append("@").append(receiver.getIndex()).append(" := clone @").append(array.getIndex());
} }
@Override @Override
public void unwrapArray(VariableReader receiver, VariableReader array, ArrayElementType elementType) { public void unwrapArray(VariableReader receiver, VariableReader array, ArrayElementType elementType) {
sb.append("@").append(receiver.getIndex()).append(" := data @").append(array.getIndex()).append(""); sb.append("@").append(receiver.getIndex()).append(" := data @").append(array.getIndex()).append(" as ")
.append(elementType.name().toLowerCase(Locale.ROOT));
} }
@Override @Override
public void getElement(VariableReader receiver, VariableReader array, VariableReader index, public void getElement(VariableReader receiver, VariableReader array, VariableReader index,
ArrayElementType type) { ArrayElementType type) {
sb.append("@").append(receiver.getIndex()).append(" := @").append(array.getIndex()).append("[@") sb.append("@").append(receiver.getIndex()).append(" := @").append(array.getIndex()).append("[@")
.append(index.getIndex()).append("]").append(" as " + type.name().toLowerCase()); .append(index.getIndex()).append("]").append(" as " + type.name().toLowerCase(Locale.ROOT));
} }
@Override @Override
@ -356,6 +389,9 @@ class InstructionStringifier implements InstructionReader {
if (receiver != null) { if (receiver != null) {
sb.append("@").append(receiver.getIndex()).append(" := "); sb.append("@").append(receiver.getIndex()).append(" := ");
} }
if (instance == null) {
sb.append("invokeStatic ");
} else {
switch (type) { switch (type) {
case SPECIAL: case SPECIAL:
sb.append("invoke "); sb.append("invoke ");
@ -364,7 +400,9 @@ class InstructionStringifier implements InstructionReader {
sb.append("invokeVirtual "); sb.append("invokeVirtual ");
break; break;
} }
}
escapeIdentifierIfNeeded(method.toString(), sb);
if (instance != null) { if (instance != null) {
sb.append("@").append(instance.getIndex()); sb.append("@").append(instance.getIndex());
} }
@ -447,7 +485,8 @@ class InstructionStringifier implements InstructionReader {
@Override @Override
public void isInstance(VariableReader receiver, VariableReader value, ValueType type) { public void isInstance(VariableReader receiver, VariableReader value, ValueType type) {
sb.append("@").append(receiver.getIndex()).append(" := @").append(value.getIndex()) sb.append("@").append(receiver.getIndex()).append(" := @").append(value.getIndex())
.append(" instanceOf ").append(type); .append(" instanceOf ");
escapeIdentifierIfNeeded(type.toString(), sb);
} }
@Override @Override

View File

@ -187,7 +187,7 @@ class ListingLexer {
readEscapedIdentifier(); readEscapedIdentifier();
break; break;
default: default:
if (isIdentifierStart()) { if (isIdentifierStart(c)) {
readIdentifier(); readIdentifier();
} else if (c >= '0' && c <= '9') { } else if (c >= '0' && c <= '9') {
readNumber(); readNumber();
@ -212,7 +212,7 @@ class ListingLexer {
private void readIdentifierLike() throws IOException { private void readIdentifierLike() throws IOException {
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
while (isIdentifierPart()) { while (isIdentifierPart(c)) {
sb.append((char) c); sb.append((char) c);
nextChar(); nextChar();
} }
@ -224,7 +224,7 @@ class ListingLexer {
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
sb.append((char) c); sb.append((char) c);
nextChar(); nextChar();
while (isIdentifierPart()) { while (isIdentifierPart(c)) {
sb.append((char) c); sb.append((char) c);
nextChar(); nextChar();
} }
@ -244,7 +244,7 @@ class ListingLexer {
tokenValue = sb.toString(); tokenValue = sb.toString();
} }
private boolean isIdentifierStart() { static boolean isIdentifierStart(int c) {
if (c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z') { if (c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z') {
return true; return true;
} }
@ -256,8 +256,8 @@ class ListingLexer {
} }
} }
private boolean isIdentifierPart() { static boolean isIdentifierPart(int c) {
if (isIdentifierStart() || c >= '0' && c <= '9') { if (isIdentifierStart(c) || c >= '0' && c <= '9') {
return true; return true;
} }
switch (c) { switch (c) {
@ -330,6 +330,11 @@ class ListingLexer {
token = ListingToken.INTEGER; token = ListingToken.INTEGER;
if (c == '-') {
sb.append(c);
nextChar();
}
while (c >= '0' && c <= '9') { while (c >= '0' && c <= '9') {
sb.append((char) c); sb.append((char) c);
nextChar(); nextChar();
@ -374,7 +379,7 @@ class ListingLexer {
} else if (c == 'l' || c == 'L') { } else if (c == 'l' || c == 'L') {
nextChar(); nextChar();
token = ListingToken.LONG; token = ListingToken.LONG;
} else if (isIdentifierStart()) { } else if (isIdentifierStart(c)) {
throw new ListingParseException("Wrong number", index); throw new ListingParseException("Wrong number", index);
} }

View File

@ -17,13 +17,16 @@ package org.teavm.model.text;
import java.io.IOException; import java.io.IOException;
import java.io.Reader; import java.io.Reader;
import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
import org.teavm.model.BasicBlock; import org.teavm.model.BasicBlock;
import org.teavm.model.Incoming; import org.teavm.model.Incoming;
import org.teavm.model.MethodReference;
import org.teavm.model.Phi; import org.teavm.model.Phi;
import org.teavm.model.Program; import org.teavm.model.Program;
import org.teavm.model.TextLocation; import org.teavm.model.TextLocation;
@ -43,6 +46,8 @@ import org.teavm.model.instructions.EmptyInstruction;
import org.teavm.model.instructions.ExitInstruction; import org.teavm.model.instructions.ExitInstruction;
import org.teavm.model.instructions.FloatConstantInstruction; import org.teavm.model.instructions.FloatConstantInstruction;
import org.teavm.model.instructions.IntegerConstantInstruction; import org.teavm.model.instructions.IntegerConstantInstruction;
import org.teavm.model.instructions.InvocationType;
import org.teavm.model.instructions.InvokeInstruction;
import org.teavm.model.instructions.JumpInstruction; import org.teavm.model.instructions.JumpInstruction;
import org.teavm.model.instructions.LongConstantInstruction; import org.teavm.model.instructions.LongConstantInstruction;
import org.teavm.model.instructions.NumericOperandType; import org.teavm.model.instructions.NumericOperandType;
@ -194,6 +199,12 @@ public class ListingParser {
parseIf(block); parseIf(block);
break; break;
} }
case "invoke":
case "invokeStatic":
case "invokeVirtual": {
parseInvoke(block, null);
break;
}
default: default:
unexpected(); unexpected();
break; break;
@ -274,6 +285,11 @@ public class ListingParser {
lexer.nextToken(); lexer.nextToken();
parseClassLiteral(block, receiver); parseClassLiteral(block, receiver);
break; break;
case "invoke":
case "invokeVirtual":
case "invokeStatic":
parseInvoke(block, receiver);
break;
default: default:
unexpected(); unexpected();
break; break;
@ -485,6 +501,57 @@ public class ListingParser {
lexer.nextToken(); lexer.nextToken();
} }
private void parseInvoke(BasicBlock block, Variable receiver) throws IOException, ListingParseException {
InvokeInstruction insn = new InvokeInstruction();
insn.setReceiver(receiver);
boolean hasInstance = true;
switch ((String) lexer.getTokenValue()) {
case "invoke":
insn.setType(InvocationType.SPECIAL);
break;
case "invokeStatic":
insn.setType(InvocationType.SPECIAL);
hasInstance = false;
break;
case "invokeVirtual":
insn.setType(InvocationType.VIRTUAL);
break;
}
lexer.nextToken();
expect(ListingToken.IDENTIFIER);
MethodReference method = MethodReference.parseIfPossible((String) lexer.getTokenValue());
if (method == null) {
throw new ListingParseException("Unparseable method", lexer.getIndex());
}
insn.setMethod(method);
lexer.nextToken();
List<Variable> arguments = new ArrayList<>();
if (lexer.getToken() == ListingToken.VARIABLE) {
arguments.add(expectVariable());
while (lexer.getToken() == ListingToken.COMMA) {
lexer.nextToken();
arguments.add(expectVariable());
}
}
if (hasInstance) {
if (arguments.isEmpty()) {
throw new ListingParseException("This kind of invocation requires at least one argument",
lexer.getIndex());
}
insn.setInstance(arguments.get(0));
insn.getArguments().addAll(arguments.subList(1, arguments.size()));
} else {
insn.getArguments().addAll(arguments);
}
block.getInstructions().add(insn);
}
private void parseIf(BasicBlock block) throws IOException, ListingParseException { private void parseIf(BasicBlock block) throws IOException, ListingParseException {
Variable first = expectVariable(); Variable first = expectVariable();

View File

@ -16,6 +16,8 @@
package org.teavm.model.text; package org.teavm.model.text;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
@ -28,6 +30,8 @@ import org.teavm.model.instructions.ClassConstantInstruction;
import org.teavm.model.instructions.DoubleConstantInstruction; import org.teavm.model.instructions.DoubleConstantInstruction;
import org.teavm.model.instructions.FloatConstantInstruction; import org.teavm.model.instructions.FloatConstantInstruction;
import org.teavm.model.instructions.IntegerConstantInstruction; import org.teavm.model.instructions.IntegerConstantInstruction;
import org.teavm.model.instructions.InvocationType;
import org.teavm.model.instructions.InvokeInstruction;
import org.teavm.model.instructions.LongConstantInstruction; import org.teavm.model.instructions.LongConstantInstruction;
import org.teavm.model.instructions.StringConstantInstruction; import org.teavm.model.instructions.StringConstantInstruction;
@ -72,6 +76,32 @@ public class ParserTest {
assertTrue("ClassConstant", block.getInstructions().get(5) instanceof ClassConstantInstruction); assertTrue("ClassConstant", block.getInstructions().get(5) instanceof ClassConstantInstruction);
} }
@Test
public void invocation() throws Exception {
Program program = runTest("invocation");
assertEquals(1, program.basicBlockCount());
BasicBlock block = program.basicBlockAt(0);
assertTrue(block.getInstructions().get(0) instanceof InvokeInstruction);
assertTrue(block.getInstructions().get(1) instanceof InvokeInstruction);
assertTrue(block.getInstructions().get(2) instanceof InvokeInstruction);
InvokeInstruction invoke = (InvokeInstruction) block.getInstructions().get(0);
assertEquals(InvocationType.VIRTUAL, invoke.getType());
assertEquals(0, invoke.getArguments().size());
assertNotNull(invoke.getInstance());
invoke = (InvokeInstruction) block.getInstructions().get(1);
assertEquals(InvocationType.SPECIAL, invoke.getType());
assertEquals(1, invoke.getArguments().size());
assertNull(invoke.getInstance());
invoke = (InvokeInstruction) block.getInstructions().get(2);
assertEquals(InvocationType.SPECIAL, invoke.getType());
assertEquals(1, invoke.getArguments().size());
assertNotNull(invoke.getInstance());
}
private Program runTest(String name) throws IOException { private Program runTest(String name) throws IOException {
ClassLoader classLoader = ParserTest.class.getClassLoader(); ClassLoader classLoader = ParserTest.class.getClassLoader();
String path = "model/text/" + name + ".txt"; String path = "model/text/" + name + ".txt";

View File

@ -0,0 +1,5 @@
$block
@a := invokeVirtual `java.lang.String.toString()Ljava/lang/String;` @o
@b := invokeStatic `java.lang.Integer.parseInt(Ljava/lang/String;)I` @a
invoke `java.lang.String.charAt(I)C` @a, @b
return @b