mirror of
https://github.com/konsoletyper/teavm.git
synced 2025-01-06 10:15:18 +08:00
Developing JavaScript AST writer for JSBody
This commit is contained in:
parent
d4b7348394
commit
3e88314fbb
599
teavm-jso-impl/src/main/java/org/teavm/jso/plugin/AstWriter.java
Normal file
599
teavm-jso-impl/src/main/java/org/teavm/jso/plugin/AstWriter.java
Normal file
@ -0,0 +1,599 @@
|
||||
/*
|
||||
* Copyright 2015 Alexey Andreev.
|
||||
*
|
||||
* 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 org.teavm.jso.plugin;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.List;
|
||||
import org.mozilla.javascript.Node;
|
||||
import org.mozilla.javascript.ScriptRuntime;
|
||||
import org.mozilla.javascript.Token;
|
||||
import org.mozilla.javascript.ast.ArrayComprehension;
|
||||
import org.mozilla.javascript.ast.ArrayComprehensionLoop;
|
||||
import org.mozilla.javascript.ast.ArrayLiteral;
|
||||
import org.mozilla.javascript.ast.AstNode;
|
||||
import org.mozilla.javascript.ast.Block;
|
||||
import org.mozilla.javascript.ast.BreakStatement;
|
||||
import org.mozilla.javascript.ast.ConditionalExpression;
|
||||
import org.mozilla.javascript.ast.ContinueStatement;
|
||||
import org.mozilla.javascript.ast.DoLoop;
|
||||
import org.mozilla.javascript.ast.ElementGet;
|
||||
import org.mozilla.javascript.ast.EmptyStatement;
|
||||
import org.mozilla.javascript.ast.ExpressionStatement;
|
||||
import org.mozilla.javascript.ast.ForInLoop;
|
||||
import org.mozilla.javascript.ast.ForLoop;
|
||||
import org.mozilla.javascript.ast.FunctionCall;
|
||||
import org.mozilla.javascript.ast.FunctionNode;
|
||||
import org.mozilla.javascript.ast.GeneratorExpression;
|
||||
import org.mozilla.javascript.ast.GeneratorExpressionLoop;
|
||||
import org.mozilla.javascript.ast.IfStatement;
|
||||
import org.mozilla.javascript.ast.InfixExpression;
|
||||
import org.mozilla.javascript.ast.Label;
|
||||
import org.mozilla.javascript.ast.LabeledStatement;
|
||||
import org.mozilla.javascript.ast.LetNode;
|
||||
import org.mozilla.javascript.ast.NewExpression;
|
||||
import org.mozilla.javascript.ast.NumberLiteral;
|
||||
import org.mozilla.javascript.ast.ObjectLiteral;
|
||||
import org.mozilla.javascript.ast.ObjectProperty;
|
||||
import org.mozilla.javascript.ast.ParenthesizedExpression;
|
||||
import org.mozilla.javascript.ast.PropertyGet;
|
||||
import org.mozilla.javascript.ast.RegExpLiteral;
|
||||
import org.mozilla.javascript.ast.ReturnStatement;
|
||||
import org.mozilla.javascript.ast.StringLiteral;
|
||||
import org.teavm.codegen.SourceWriter;
|
||||
|
||||
/**
|
||||
*
|
||||
* @author Alexey Andreev
|
||||
*/
|
||||
public class AstWriter {
|
||||
private static final int PRECEDENCE_PRIMARY = 1;
|
||||
private static final int PRECEDENCE_MEMBER = 2;
|
||||
private static final int PRECEDENCE_FUNCTION = 3;
|
||||
private static final int PRECEDENCE_POSTFIX = 4;
|
||||
private static final int PRECEDENCE_PREFIX = 5;
|
||||
private static final int PRECEDENCE_MUL = 6;
|
||||
private static final int PRECEDENCE_ADD = 7;
|
||||
private static final int PRECEDENCE_SHIFT = 8;
|
||||
private static final int PRECEDENCE_RELATION = 9;
|
||||
private static final int PRECEDENCE_EQUALITY = 10;
|
||||
private static final int PRECEDENCE_BITWISE_AND = 11;
|
||||
private static final int PRECEDENCE_BITWISE_XOR = 12;
|
||||
private static final int PRECEDENCE_BITWISE_OR = 13;
|
||||
private static final int PRECEDENCE_AND = 14;
|
||||
private static final int PRECEDENCE_OR = 15;
|
||||
private static final int PRECEDENCE_COND = 16;
|
||||
private static final int PRECEDENCE_ASSIGN = 17;
|
||||
private static final int PRECEDENCE_COMMA = 18;
|
||||
private SourceWriter writer;
|
||||
|
||||
public AstWriter(SourceWriter writer) {
|
||||
this.writer = writer;
|
||||
}
|
||||
|
||||
public void print(AstNode node) throws IOException {
|
||||
print(node, PRECEDENCE_COMMA);
|
||||
}
|
||||
|
||||
private void print(AstNode node, int precedence) throws IOException {
|
||||
switch (node.getType()) {
|
||||
case Token.CALL:
|
||||
case Token.NEW:
|
||||
print((FunctionCall) node, precedence);
|
||||
break;
|
||||
case Token.FUNCTION:
|
||||
print((FunctionNode) node);
|
||||
break;
|
||||
case Token.ARRAYCOMP:
|
||||
print((ArrayComprehension) node);
|
||||
break;
|
||||
case Token.GETPROP:
|
||||
print((PropertyGet) node, precedence);
|
||||
break;
|
||||
case Token.GENEXPR:
|
||||
print((GeneratorExpression) node);
|
||||
break;
|
||||
case Token.NUMBER:
|
||||
print((NumberLiteral) node);
|
||||
break;
|
||||
case Token.STRING:
|
||||
print((StringLiteral) node);
|
||||
break;
|
||||
case Token.REGEXP:
|
||||
print((RegExpLiteral) node);
|
||||
break;
|
||||
case Token.OBJECTLIT:
|
||||
print((ObjectLiteral) node);
|
||||
break;
|
||||
case Token.ARRAYLIT:
|
||||
print((ArrayLiteral) node);
|
||||
break;
|
||||
case Token.BLOCK:
|
||||
print((Block) node);
|
||||
break;
|
||||
case Token.HOOK:
|
||||
print((ConditionalExpression) node, precedence);
|
||||
break;
|
||||
case Token.GETELEM:
|
||||
print((ElementGet) node, precedence);
|
||||
break;
|
||||
case Token.LETEXPR:
|
||||
print((LetNode) node);
|
||||
break;
|
||||
case Token.LP:
|
||||
print((ParenthesizedExpression) node, precedence);
|
||||
break;
|
||||
case Token.EMPTY:
|
||||
if (node instanceof EmptyStatement) {
|
||||
writer.append(';');
|
||||
}
|
||||
break;
|
||||
case Token.EXPR_VOID:
|
||||
case Token.EXPR_RESULT:
|
||||
if (node instanceof ExpressionStatement) {
|
||||
print((ExpressionStatement) node);
|
||||
} else if (node instanceof LabeledStatement) {
|
||||
print((LabeledStatement) node);
|
||||
}
|
||||
break;
|
||||
case Token.BREAK:
|
||||
print((BreakStatement) node);
|
||||
break;
|
||||
case Token.CONTINUE:
|
||||
print((ContinueStatement) node);
|
||||
break;
|
||||
case Token.RETURN:
|
||||
print((ReturnStatement) node);
|
||||
break;
|
||||
case Token.DO:
|
||||
print((DoLoop) node);
|
||||
break;
|
||||
case Token.FOR:
|
||||
if (node instanceof ForInLoop) {
|
||||
print((ForInLoop) node);
|
||||
} else if (node instanceof ForLoop) {
|
||||
print((ForLoop) node);
|
||||
}
|
||||
break;
|
||||
case Token.IF:
|
||||
print((IfStatement) node);
|
||||
break;
|
||||
default:
|
||||
if (node instanceof InfixExpression) {
|
||||
printInfix((InfixExpression) node, precedence);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private void print(Block node) throws IOException {
|
||||
writer.append('{').softNewLine().indent();
|
||||
for (Node child = node.getFirstChild(); child != null; child = child.getNext()) {
|
||||
print((AstNode) child);
|
||||
writer.softNewLine();
|
||||
}
|
||||
writer.outdent().append('}');
|
||||
}
|
||||
|
||||
private void print(LabeledStatement node) throws IOException {
|
||||
for (Label label : node.getLabels()) {
|
||||
writer.append(label.getName()).append(':').ws();
|
||||
}
|
||||
print(node.getStatement());
|
||||
}
|
||||
|
||||
private void print(BreakStatement node) throws IOException {
|
||||
writer.append("break");
|
||||
if (node.getBreakLabel() != null) {
|
||||
writer.append(' ').append(node.getBreakLabel().getString());
|
||||
}
|
||||
writer.append(';');
|
||||
}
|
||||
|
||||
private void print(ContinueStatement node) throws IOException {
|
||||
writer.append("continue");
|
||||
if (node.getLabel() != null) {
|
||||
writer.append(' ').append(node.getLabel().getString());
|
||||
}
|
||||
writer.append(';');
|
||||
}
|
||||
|
||||
private void print(ReturnStatement node) throws IOException {
|
||||
writer.append("return");
|
||||
if (node.getReturnValue() != null) {
|
||||
writer.append(' ');
|
||||
print(node.getReturnValue());
|
||||
}
|
||||
writer.append(';');
|
||||
}
|
||||
|
||||
private void print(DoLoop node) throws IOException {
|
||||
writer.append("do");
|
||||
if (node.getBody() instanceof Block) {
|
||||
writer.ws();
|
||||
} else {
|
||||
writer.append(' ');
|
||||
}
|
||||
print(node.getBody());
|
||||
writer.append("while").ws().append('(');
|
||||
writer.append(");");
|
||||
}
|
||||
|
||||
private void print(ForInLoop node) throws IOException {
|
||||
writer.append("for");
|
||||
if (node.isForEach()) {
|
||||
writer.append(" each");
|
||||
}
|
||||
writer.ws().append("(");
|
||||
print(node.getIterator());
|
||||
writer.append(" in ");
|
||||
print(node.getIteratedObject());
|
||||
writer.append(')').ws();
|
||||
print(node.getBody());
|
||||
}
|
||||
|
||||
private void print(ForLoop node) throws IOException {
|
||||
writer.append("for").ws().append('(');
|
||||
print(node.getInitializer());
|
||||
writer.append(';');
|
||||
print(node.getCondition());
|
||||
writer.append(';');
|
||||
print(node.getIncrement());
|
||||
writer.append(';');
|
||||
writer.append(')').ws();
|
||||
print(node.getBody());
|
||||
}
|
||||
|
||||
private void print(IfStatement node) throws IOException {
|
||||
writer.append("if").ws().append('(');
|
||||
print(node.getCondition());
|
||||
writer.append(')').ws();
|
||||
print(node.getThenPart());
|
||||
if (node.getElsePart() != null) {
|
||||
writer.ws().append("else ");
|
||||
print(node.getElsePart());
|
||||
}
|
||||
}
|
||||
|
||||
private void print(ExpressionStatement node) throws IOException {
|
||||
print(node.getExpression());
|
||||
writer.append(';');
|
||||
}
|
||||
|
||||
private void print(ElementGet node, int precedence) throws IOException {
|
||||
if (precedence < PRECEDENCE_MEMBER) {
|
||||
writer.append('(');
|
||||
}
|
||||
print(node.getTarget(), PRECEDENCE_MEMBER);
|
||||
if (precedence < PRECEDENCE_MEMBER) {
|
||||
writer.append(')');
|
||||
}
|
||||
writer.append('[');
|
||||
print(node.getElement());
|
||||
writer.append(']');
|
||||
}
|
||||
|
||||
private void print(PropertyGet node, int precedence) throws IOException {
|
||||
if (precedence < PRECEDENCE_MEMBER) {
|
||||
writer.append('(');
|
||||
}
|
||||
print(node.getLeft());
|
||||
if (precedence < PRECEDENCE_MEMBER) {
|
||||
writer.append(')');
|
||||
}
|
||||
writer.ws().append('.').ws();
|
||||
print(node.getRight());
|
||||
}
|
||||
|
||||
private void print(FunctionCall node, int precedence) throws IOException {
|
||||
if (node instanceof NewExpression) {
|
||||
writer.append("new ");
|
||||
}
|
||||
if (precedence < PRECEDENCE_FUNCTION) {
|
||||
writer.append('(');
|
||||
}
|
||||
print(node.getTarget(), PRECEDENCE_FUNCTION);
|
||||
if (precedence < PRECEDENCE_FUNCTION) {
|
||||
writer.append(')');
|
||||
}
|
||||
writer.append('(');
|
||||
printList(node.getArguments());
|
||||
writer.append(')');
|
||||
if (node instanceof NewExpression) {
|
||||
writer.ws();
|
||||
print(((NewExpression) node).getInitializer());
|
||||
}
|
||||
}
|
||||
|
||||
private void print(ConditionalExpression node, int precedence) throws IOException {
|
||||
if (precedence < PRECEDENCE_COND) {
|
||||
writer.append('(');
|
||||
}
|
||||
print(node.getTestExpression(), PRECEDENCE_COND - 1);
|
||||
writer.ws().append('?').ws();
|
||||
print(node.getTrueExpression(), PRECEDENCE_COND - 1);
|
||||
writer.ws().append(':').ws();
|
||||
print(node.getFalseExpression(), PRECEDENCE_COND);
|
||||
if (precedence < PRECEDENCE_COND) {
|
||||
writer.append(')');
|
||||
}
|
||||
print(node.getTestExpression(), precedence);
|
||||
}
|
||||
|
||||
private void printList(List<? extends AstNode> nodes) throws IOException {
|
||||
if (nodes == null || nodes.isEmpty()) {
|
||||
return;
|
||||
}
|
||||
print(nodes.get(0));
|
||||
for (int i = 1; i < nodes.size(); ++i) {
|
||||
writer.append(',').ws();
|
||||
print(nodes.get(i));
|
||||
}
|
||||
}
|
||||
|
||||
private void print(ArrayComprehension node) throws IOException {
|
||||
writer.append("[");
|
||||
for (ArrayComprehensionLoop loop : node.getLoops()) {
|
||||
writer.append("for").ws().append("(");
|
||||
print(loop.getIterator());
|
||||
writer.append(" of ");
|
||||
print(loop.getIteratedObject());
|
||||
writer.append(')');
|
||||
}
|
||||
if (node.getFilter() != null) {
|
||||
writer.append("if").ws().append("(");
|
||||
print(node.getFilter());
|
||||
writer.append(")");
|
||||
}
|
||||
print(node.getResult());
|
||||
writer.append(']');
|
||||
}
|
||||
|
||||
private void print(GeneratorExpression node) throws IOException {
|
||||
writer.append("(");
|
||||
for (GeneratorExpressionLoop loop : node.getLoops()) {
|
||||
writer.append("for").ws().append("(");
|
||||
print(loop.getIterator());
|
||||
writer.append(" of ");
|
||||
print(loop.getIteratedObject());
|
||||
writer.append(')');
|
||||
}
|
||||
if (node.getFilter() != null) {
|
||||
writer.append("if").ws().append("(");
|
||||
print(node.getFilter());
|
||||
writer.append(")");
|
||||
}
|
||||
print(node.getResult());
|
||||
writer.append(')');
|
||||
}
|
||||
|
||||
private void print(NumberLiteral node) throws IOException {
|
||||
writer.append(node.getNumber());
|
||||
}
|
||||
|
||||
private void print(StringLiteral node) throws IOException {
|
||||
writer.append(node.getQuoteCharacter());
|
||||
writer.append(ScriptRuntime.escapeString(node.getString(), node.getQuoteCharacter()));
|
||||
writer.append(node.getQuoteCharacter());
|
||||
}
|
||||
|
||||
private void print(RegExpLiteral node) throws IOException {
|
||||
writer.append('/').append(node.getValue()).append('/').append(node.getFlags());
|
||||
}
|
||||
|
||||
private void print(ArrayLiteral node) throws IOException {
|
||||
writer.append('[');
|
||||
printList(node.getElements());
|
||||
writer.append(']');
|
||||
}
|
||||
|
||||
private void print(ObjectLiteral node) throws IOException {
|
||||
writer.append('{');
|
||||
if (node.getElements() != null && !node.getElements().isEmpty()) {
|
||||
print(node.getElements().get(0));
|
||||
for (int i = 1; i < node.getElements().size(); ++i) {
|
||||
writer.append(',').ws();
|
||||
print(node.getElements().get(i));
|
||||
}
|
||||
}
|
||||
printList(node.getElements());
|
||||
writer.append('}');
|
||||
}
|
||||
|
||||
private void print(ObjectProperty node) throws IOException {
|
||||
StringBuilder sb = new StringBuilder();
|
||||
if (node.isGetterMethod()) {
|
||||
sb.append("get ");
|
||||
} else if (node.isSetterMethod()) {
|
||||
sb.append("set ");
|
||||
}
|
||||
print(node.getLeft());
|
||||
writer.ws().append(':').ws();
|
||||
print(node.getRight());
|
||||
}
|
||||
|
||||
private void print(FunctionNode node) throws IOException {
|
||||
if (!node.isMethod()) {
|
||||
writer.append("function");
|
||||
}
|
||||
if (node.getFunctionName() != null) {
|
||||
writer.append(' ').append(node.getFunctionName());
|
||||
}
|
||||
writer.append('(');
|
||||
printList(node.getParams());
|
||||
writer.append(')');
|
||||
|
||||
if (node.isExpressionClosure()) {
|
||||
if (node.getBody().getLastChild() instanceof ReturnStatement) {
|
||||
print(((ReturnStatement) node.getBody().getLastChild()).getReturnValue());
|
||||
if (node.getFunctionType() == FunctionNode.FUNCTION_STATEMENT) {
|
||||
writer.append(";");
|
||||
}
|
||||
}
|
||||
} else {
|
||||
print(node.getBody());
|
||||
writer.softNewLine();
|
||||
}
|
||||
if (node.getFunctionType() == FunctionNode.FUNCTION_STATEMENT || node.isMethod()) {
|
||||
writer.softNewLine();
|
||||
}
|
||||
}
|
||||
|
||||
private void print(LetNode node) throws IOException {
|
||||
writer.append("let").ws().append('(');
|
||||
printList(node.getVariables().getVariables());
|
||||
writer.append(')');
|
||||
writer.append(node.getBody());
|
||||
}
|
||||
|
||||
private void print(ParenthesizedExpression node, int precedence) throws IOException {
|
||||
print(node.getExpression(), precedence);
|
||||
}
|
||||
|
||||
private void printInfix(InfixExpression node, int precedence) throws IOException {
|
||||
int innerPrecedence = getPrecedence(node.getType());
|
||||
|
||||
if (innerPrecedence > precedence) {
|
||||
writer.append('(');
|
||||
}
|
||||
|
||||
int leftPrecedence;
|
||||
switch (node.getType()) {
|
||||
case Token.ASSIGN:
|
||||
case Token.ASSIGN_ADD:
|
||||
case Token.ASSIGN_SUB:
|
||||
case Token.ASSIGN_MUL:
|
||||
case Token.ASSIGN_DIV:
|
||||
case Token.ASSIGN_MOD:
|
||||
case Token.ASSIGN_BITAND:
|
||||
case Token.ASSIGN_BITXOR:
|
||||
case Token.ASSIGN_BITOR:
|
||||
case Token.ASSIGN_LSH:
|
||||
case Token.ASSIGN_RSH:
|
||||
case Token.ASSIGN_URSH:
|
||||
leftPrecedence = innerPrecedence - 1;
|
||||
break;
|
||||
default:
|
||||
leftPrecedence = innerPrecedence;
|
||||
}
|
||||
print(node.getLeft(), leftPrecedence);
|
||||
|
||||
boolean ws = requiresWhitespaces(node.getType());
|
||||
if (ws) {
|
||||
writer.append(' ');
|
||||
} else {
|
||||
writer.ws();
|
||||
}
|
||||
writer.append(AstNode.operatorToString(node.getType()));
|
||||
if (ws) {
|
||||
writer.append(' ');
|
||||
} else {
|
||||
writer.ws();
|
||||
}
|
||||
|
||||
int rightPrecedence;
|
||||
switch (node.getType()) {
|
||||
case Token.DIV:
|
||||
case Token.MOD:
|
||||
case Token.SUB:
|
||||
case Token.LSH:
|
||||
case Token.RSH:
|
||||
case Token.URSH:
|
||||
case Token.IN:
|
||||
case Token.INSTANCEOF:
|
||||
case Token.EQ:
|
||||
case Token.NE:
|
||||
case Token.SHEQ:
|
||||
case Token.SHNE:
|
||||
case Token.GT:
|
||||
case Token.GE:
|
||||
case Token.LT:
|
||||
case Token.LE:
|
||||
rightPrecedence = innerPrecedence - 1;
|
||||
break;
|
||||
default:
|
||||
rightPrecedence = innerPrecedence;
|
||||
}
|
||||
print(node.getRight(), rightPrecedence);
|
||||
|
||||
if (innerPrecedence > precedence) {
|
||||
writer.append('(');
|
||||
}
|
||||
}
|
||||
|
||||
private int getPrecedence(int token) {
|
||||
switch (token) {
|
||||
case Token.MUL:
|
||||
case Token.DIV:
|
||||
case Token.MOD:
|
||||
return PRECEDENCE_MUL;
|
||||
case Token.ADD:
|
||||
case Token.SUB:
|
||||
return PRECEDENCE_ADD;
|
||||
case Token.LSH:
|
||||
case Token.RSH:
|
||||
case Token.URSH:
|
||||
return PRECEDENCE_SHIFT;
|
||||
case Token.LT:
|
||||
case Token.LE:
|
||||
case Token.GT:
|
||||
case Token.GE:
|
||||
case Token.IN:
|
||||
case Token.INSTANCEOF:
|
||||
return PRECEDENCE_RELATION;
|
||||
case Token.EQ:
|
||||
case Token.NE:
|
||||
case Token.SHEQ:
|
||||
case Token.SHNE:
|
||||
return PRECEDENCE_EQUALITY;
|
||||
case Token.BITAND:
|
||||
return PRECEDENCE_BITWISE_AND;
|
||||
case Token.BITXOR:
|
||||
return PRECEDENCE_BITWISE_XOR;
|
||||
case Token.BITOR:
|
||||
return PRECEDENCE_BITWISE_OR;
|
||||
case Token.AND:
|
||||
return PRECEDENCE_AND;
|
||||
case Token.OR:
|
||||
return PRECEDENCE_OR;
|
||||
case Token.ASSIGN:
|
||||
case Token.ASSIGN_ADD:
|
||||
case Token.ASSIGN_SUB:
|
||||
case Token.ASSIGN_MUL:
|
||||
case Token.ASSIGN_DIV:
|
||||
case Token.ASSIGN_MOD:
|
||||
case Token.ASSIGN_BITAND:
|
||||
case Token.ASSIGN_BITXOR:
|
||||
case Token.ASSIGN_BITOR:
|
||||
case Token.ASSIGN_LSH:
|
||||
case Token.ASSIGN_RSH:
|
||||
case Token.ASSIGN_URSH:
|
||||
return PRECEDENCE_ASSIGN;
|
||||
default:
|
||||
return PRECEDENCE_COMMA;
|
||||
}
|
||||
}
|
||||
|
||||
private boolean requiresWhitespaces(int token) {
|
||||
switch (token) {
|
||||
case Token.IN:
|
||||
case Token.TYPEOF:
|
||||
case Token.INSTANCEOF:
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
@ -68,4 +68,6 @@ public class JSBodyGenerator implements Generator {
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user