/* * Copyright (C) 2007 JĂșlio Vilmar Gesser. * Copyright (C) 2012 Mozilla Foundation * * This file is part of Java 1.5 parser and Abstract Syntax Tree. * * Java 1.5 parser and Abstract Syntax Tree is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Java 1.5 parser and Abstract Syntax Tree is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with Java 1.5 parser and Abstract Syntax Tree. If not, see <http://www.gnu.org/licenses/>. */ /* * Created on 05/10/2006 */ package nu.validator.htmlparser.rusttranslate; import japa.parser.ast.BlockComment; import japa.parser.ast.CompilationUnit; import japa.parser.ast.LineComment; import japa.parser.ast.TypeParameter; import japa.parser.ast.body.BodyDeclaration; import japa.parser.ast.body.ClassOrInterfaceDeclaration; import japa.parser.ast.body.ConstructorDeclaration; import japa.parser.ast.body.EmptyMemberDeclaration; import japa.parser.ast.body.EmptyTypeDeclaration; import japa.parser.ast.body.EnumConstantDeclaration; import japa.parser.ast.body.EnumDeclaration; import japa.parser.ast.body.FieldDeclaration; import japa.parser.ast.body.InitializerDeclaration; import japa.parser.ast.body.JavadocComment; import japa.parser.ast.body.MethodDeclaration; import japa.parser.ast.body.ModifierSet; import japa.parser.ast.body.Parameter; import japa.parser.ast.body.TypeDeclaration; import japa.parser.ast.body.VariableDeclarator; import japa.parser.ast.body.VariableDeclaratorId; import japa.parser.ast.expr.AnnotationExpr; import japa.parser.ast.expr.ArrayAccessExpr; import japa.parser.ast.expr.ArrayCreationExpr; import japa.parser.ast.expr.ArrayInitializerExpr; import japa.parser.ast.expr.AssignExpr; import japa.parser.ast.expr.BinaryExpr; import japa.parser.ast.expr.BooleanLiteralExpr; import japa.parser.ast.expr.CastExpr; import japa.parser.ast.expr.CharLiteralExpr; import japa.parser.ast.expr.ClassExpr; import japa.parser.ast.expr.ConditionalExpr; import japa.parser.ast.expr.DoubleLiteralExpr; import japa.parser.ast.expr.EnclosedExpr; import japa.parser.ast.expr.Expression; import japa.parser.ast.expr.FieldAccessExpr; import japa.parser.ast.expr.InstanceOfExpr; import japa.parser.ast.expr.IntegerLiteralExpr; import japa.parser.ast.expr.IntegerLiteralMinValueExpr; import japa.parser.ast.expr.LongLiteralExpr; import japa.parser.ast.expr.LongLiteralMinValueExpr; import japa.parser.ast.expr.MemberValuePair; import japa.parser.ast.expr.MethodCallExpr; import japa.parser.ast.expr.NameExpr; import japa.parser.ast.expr.NullLiteralExpr; import japa.parser.ast.expr.ObjectCreationExpr; import japa.parser.ast.expr.QualifiedNameExpr; import japa.parser.ast.expr.StringLiteralExpr; import japa.parser.ast.expr.SuperExpr; import japa.parser.ast.expr.ThisExpr; import japa.parser.ast.expr.UnaryExpr; import japa.parser.ast.expr.UnaryExpr.Operator; import japa.parser.ast.expr.VariableDeclarationExpr; import japa.parser.ast.stmt.AssertStmt; import japa.parser.ast.stmt.BlockStmt; import japa.parser.ast.stmt.BreakStmt; import japa.parser.ast.stmt.CatchClause; import japa.parser.ast.stmt.ContinueStmt; import japa.parser.ast.stmt.DoStmt; import japa.parser.ast.stmt.EmptyStmt; import japa.parser.ast.stmt.ExplicitConstructorInvocationStmt; import japa.parser.ast.stmt.ExpressionStmt; import japa.parser.ast.stmt.ForStmt; import japa.parser.ast.stmt.ForeachStmt; import japa.parser.ast.stmt.IfStmt; import japa.parser.ast.stmt.LabeledStmt; import japa.parser.ast.stmt.ReturnStmt; import japa.parser.ast.stmt.Statement; import japa.parser.ast.stmt.SwitchEntryStmt; import japa.parser.ast.stmt.SwitchStmt; import japa.parser.ast.stmt.SynchronizedStmt; import japa.parser.ast.stmt.ThrowStmt; import japa.parser.ast.stmt.TryStmt; import japa.parser.ast.stmt.TypeDeclarationStmt; import japa.parser.ast.stmt.WhileStmt; import japa.parser.ast.type.ClassOrInterfaceType; import japa.parser.ast.type.PrimitiveType; import japa.parser.ast.type.ReferenceType; import japa.parser.ast.type.Type; import japa.parser.ast.type.VoidType; import japa.parser.ast.type.WildcardType; import japa.parser.ast.visitor.VoidVisitorAdapter; import java.util.HashSet; import java.util.Iterator; import java.util.LinkedList; import java.util.List; import java.util.Set; import nu.validator.htmlparser.cpptranslate.TranslatorUtils; /** * @author Julio Vilmar Gesser * @author Henri Sivonen */ public final class RustVisitor extends VoidVisitorAdapter<Object> { private static final String[] MODS = { "Tokenizer", "TreeBuilder", "MetaScanner", "AttributeName", "ElementName", "HtmlAttributes", "StackNode", "UTF16Buffer", "StateSnapshot", }; private boolean inMethodSignature = false; private Set<String> fields = new HashSet<String>(); private Set<String> constants = new HashSet<String>(); private Expression loopUpdate = null; private static class SourcePrinter { private int level = 0; private boolean indented = false; private final StringBuilder buf = new StringBuilder(); public void indent() { level++; } public void unindent() { level--; } private void makeIndent() { for (int i = 0; i < level; i++) { buf.append(" "); } } public void print(String arg) { if (!indented) { makeIndent(); indented = true; } buf.append(arg); } public void printLn(String arg) { print(arg); printLn(); } public void printLn() { buf.append("\n"); indented = false; } public String getSource() { return buf.toString(); } @Override public String toString() { return getSource(); } } private final SourcePrinter printer = new SourcePrinter(); public String getSource() { return printer.getSource(); } private void printModifiers(int modifiers) { if (ModifierSet.isPrivate(modifiers)) { printer.print("private "); } if (ModifierSet.isProtected(modifiers)) { printer.print("protected "); } if (ModifierSet.isPublic(modifiers)) { printer.print("public "); } if (ModifierSet.isAbstract(modifiers)) { printer.print("abstract "); } if (ModifierSet.isStatic(modifiers)) { printer.print("static "); } if (ModifierSet.isFinal(modifiers)) { printer.print("final "); } if (ModifierSet.isNative(modifiers)) { printer.print("native "); } if (ModifierSet.isStrictfp(modifiers)) { printer.print("strictfp "); } if (ModifierSet.isSynchronized(modifiers)) { printer.print("synchronized "); } if (ModifierSet.isTransient(modifiers)) { printer.print("transient "); } if (ModifierSet.isVolatile(modifiers)) { printer.print("volatile "); } } private void printMethods(List<BodyDeclaration> members, Object arg) { for (BodyDeclaration member : members) { if (member instanceof MethodDeclaration) { MethodDeclaration meth = (MethodDeclaration) member; if (meth.getName().startsWith("fatal") || meth.getName().startsWith("err") || meth.getName().startsWith("warn") || meth.getName().startsWith("maybeErr") || meth.getName().startsWith("maybeWarn") || meth.getName().startsWith("note") || "releaseArray".equals(meth.getName()) || "deleteArray".equals(meth.getName()) || "delete".equals(meth.getName())) { continue; } printer.printLn(); member.accept(this, arg); printer.printLn(); } } } private void printFields(List<BodyDeclaration> members, Object arg) { for (BodyDeclaration member : members) { if (member instanceof FieldDeclaration) { FieldDeclaration field = (FieldDeclaration) member; int mods = field.getModifiers(); if (ModifierSet.isStatic(mods) && ModifierSet.isFinal(mods)) { continue; } fields.add(field.getVariables().get(0).getId().getName()); printer.printLn(); member.accept(this, arg); printer.printLn(); } } } private void printConstants(List<BodyDeclaration> members, Object arg) { for (BodyDeclaration member : members) { if (member instanceof FieldDeclaration) { FieldDeclaration field = (FieldDeclaration) member; int mods = field.getModifiers(); if (!(ModifierSet.isStatic(mods) && ModifierSet.isFinal(mods))) { continue; } constants.add(field.getVariables().get(0).getId().getName()); printer.printLn(); member.accept(this, arg); printer.printLn(); } } } private void printMemberAnnotations(List<AnnotationExpr> annotations, Object arg) { if (annotations != null) { for (AnnotationExpr a : annotations) { a.accept(this, arg); printer.printLn(); } } } private void printArguments(List<Expression> args, Object arg) { printer.print("("); if (args != null) { for (Iterator<Expression> i = args.iterator(); i.hasNext();) { Expression e = i.next(); e.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } } printer.print(")"); } private void printJavadoc(JavadocComment javadoc, Object arg) { if (javadoc != null) { javadoc.accept(this, arg); } } public void visit(CompilationUnit n, Object arg) { if (n.getTypes() != null) { for (Iterator<TypeDeclaration> i = n.getTypes().iterator(); i.hasNext();) { i.next().accept(this, arg); printer.printLn(); if (i.hasNext()) { printer.printLn(); } } } } public void visit(NameExpr n, Object arg) { if (fields.contains(n.getName())) { printer.print("self."); } printer.print(n.getName()); } public void visit(QualifiedNameExpr n, Object arg) { n.getQualifier().accept(this, arg); printer.print("."); printer.print(n.getName()); } public void visit(ClassOrInterfaceDeclaration n, Object arg) { for (int i = 0; i < MODS.length; i++) { String mod = MODS[i]; if (!mod.equals(n.getName())) { printer.print("mod "); printer.print(mod); printer.printLn(";"); } } printJavadoc(n.getJavaDoc(), arg); if (n.getMembers() != null) { printConstants(n.getMembers(), arg); } printer.printLn(); printer.printLn(); printer.print("struct "); printer.print(n.getName()); printer.printLn(" {"); printer.indent(); if (n.getMembers() != null) { printFields(n.getMembers(), arg); } printer.unindent(); printer.print("}"); printer.printLn(); printer.printLn(); printer.print("impl "); printer.print(n.getName()); printer.printLn(" {"); printer.indent(); if (n.getMembers() != null) { printMethods(n.getMembers(), arg); } printer.unindent(); printer.print("}"); } public void visit(EmptyTypeDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); printer.print(";"); } public void visit(JavadocComment n, Object arg) { printer.print("/**"); printer.print(n.getContent()); printer.printLn("*/"); } public void visit(ClassOrInterfaceType n, Object arg) { if (n.getScope() != null) { n.getScope().accept(this, arg); printer.print("."); } printer.print(n.getName()); } public void visit(TypeParameter n, Object arg) { printer.print(n.getName()); if (n.getTypeBound() != null) { printer.print(" extends "); for (Iterator<ClassOrInterfaceType> i = n.getTypeBound().iterator(); i.hasNext();) { ClassOrInterfaceType c = i.next(); c.accept(this, arg); if (i.hasNext()) { printer.print(" & "); } } } } public void visit(PrimitiveType n, Object arg) { switch (n.getType()) { case Boolean: printer.print("bool"); break; case Byte: printer.print("i8"); break; case Char: printer.print("u16"); break; case Double: printer.print("f64"); break; case Float: printer.print("f32"); break; case Int: printer.print("i32"); break; case Long: printer.print("i64"); break; case Short: printer.print("i16"); break; } } public void visit(ReferenceType n, Object arg) { // if (inMethodSignature) { // printer.print("&"); // } else { // printer.print("~"); // } printer.print("@"); for (int i = 0; i < n.getArrayCount(); i++) { printer.print("["); } n.getType().accept(this, arg); for (int i = 0; i < n.getArrayCount(); i++) { printer.print("]"); } } public void visit(WildcardType n, Object arg) { printer.print("?"); if (n.getExtends() != null) { printer.print(" extends "); n.getExtends().accept(this, arg); } if (n.getSuper() != null) { printer.print(" super "); n.getSuper().accept(this, arg); } } public void visit(FieldDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); // printMemberAnnotations(n.getAnnotations(), arg); boolean field = true; int mods = n.getModifiers(); if (ModifierSet.isStatic(mods) && ModifierSet.isFinal(mods)) { if (!ModifierSet.isPrivate(mods)) { printer.print("pub "); } printer.print("const "); field = false; } else if (!ModifierSet.isFinal(mods)) { printer.print("mut "); } List<VariableDeclarator> vars = n.getVariables(); printVariableDeclarator(n.getType(), vars, arg, field); printer.print(field ? "," : ";"); } private void printVariableDeclarator(Type type, List<VariableDeclarator> vars, Object arg, boolean field) { if (vars.size() != 1) { throw new RuntimeException(); } VariableDeclarator decl = vars.get(0); VariableDeclaratorId id = decl.getId(); printer.print(id.getName()); printer.print(": "); for (int i = 0; i < id.getArrayCount(); i++) { printer.print("["); } type.accept(this, arg); for (int i = 0; i < id.getArrayCount(); i++) { printer.print("]"); } Expression init = decl.getInit(); if (init != null && !field) { printer.print(" = "); init.accept(this, arg); } } public void visit(ArrayInitializerExpr n, Object arg) { printer.print("["); if (n.getValues() != null) { printer.print(" "); for (Iterator<Expression> i = n.getValues().iterator(); i.hasNext();) { Expression expr = i.next(); expr.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } printer.print(" "); } printer.print("]"); } public void visit(VoidType n, Object arg) { printer.print("void"); } public void visit(ArrayAccessExpr n, Object arg) { n.getName().accept(this, arg); printer.print("["); n.getIndex().accept(this, arg); printer.print("]"); } public void visit(ArrayCreationExpr n, Object arg) { printer.print("new "); n.getType().accept(this, arg); if (n.getDimensions() != null) { for (Expression dim : n.getDimensions()) { printer.print("["); dim.accept(this, arg); printer.print("]"); } for (int i = 0; i < n.getArrayCount(); i++) { printer.print("[]"); } } else { for (int i = 0; i < n.getArrayCount(); i++) { printer.print("[]"); } printer.print(" "); n.getInitializer().accept(this, arg); } } public void visit(AssignExpr n, Object arg) { n.getTarget().accept(this, arg); printer.print(" "); switch (n.getOperator()) { case assign: printer.print("="); break; case and: printer.print("&="); break; case or: printer.print("|="); break; case xor: printer.print("^="); break; case plus: printer.print("+="); break; case minus: printer.print("-="); break; case rem: printer.print("%="); break; case slash: printer.print("/="); break; case star: printer.print("*="); break; case lShift: printer.print("<<="); break; case rSignedShift: printer.print(">>="); break; case rUnsignedShift: printer.print(">>>="); break; } printer.print(" "); n.getValue().accept(this, arg); } public void visit(BinaryExpr n, Object arg) { n.getLeft().accept(this, arg); printer.print(" "); switch (n.getOperator()) { case or: printer.print("||"); break; case and: printer.print("&&"); break; case binOr: printer.print("|"); break; case binAnd: printer.print("&"); break; case xor: printer.print("^"); break; case equals: printer.print("=="); break; case notEquals: printer.print("!="); break; case less: printer.print("<"); break; case greater: printer.print(">"); break; case lessEquals: printer.print("<="); break; case greaterEquals: printer.print(">="); break; case lShift: printer.print("<<"); break; case rSignedShift: printer.print(">>"); break; case rUnsignedShift: printer.print(">>>"); break; case plus: printer.print("+"); break; case minus: printer.print("-"); break; case times: printer.print("*"); break; case divide: printer.print("/"); break; case remainder: printer.print("%"); break; } printer.print(" "); n.getRight().accept(this, arg); } public void visit(CastExpr n, Object arg) { printer.print("("); n.getType().accept(this, arg); printer.print(") "); n.getExpr().accept(this, arg); } public void visit(ClassExpr n, Object arg) { n.getType().accept(this, arg); printer.print(".class"); } public void visit(ConditionalExpr n, Object arg) { n.getCondition().accept(this, arg); printer.print(" ? "); n.getThenExpr().accept(this, arg); printer.print(" : "); n.getElseExpr().accept(this, arg); } public void visit(EnclosedExpr n, Object arg) { printer.print("("); n.getInner().accept(this, arg); printer.print(")"); } public void visit(FieldAccessExpr n, Object arg) { String scope = n.getScope().toString(); printer.print(scope); boolean mod = false; for (int i = 0; i < MODS.length; i++) { if (MODS[i].equals(scope)) { mod = true; break; } } printer.print(mod ? "::" : "."); if ("length".equals(n.getField())) { printer.print("len() as i32"); } else { printer.print(n.getField()); } } public void visit(InstanceOfExpr n, Object arg) { n.getExpr().accept(this, arg); printer.print(" instanceof "); n.getType().accept(this, arg); } public void visit(CharLiteralExpr n, Object arg) { // printer.print("'"); // char c = n.getValue().charAt(0); // switch (c) { // case '\b': // printer.print("\\b"); // break; // case '\t': // printer.print("\\t"); // break; // case '\n': // printer.print("\\n"); // break; // case '\f': // printer.print("\\f"); // break; // case '\r': // printer.print("\\r"); // break; // case '\'': // printer.print("\\'"); // break; // case '\\': // printer.print(n.getValue()); // break; // default: // if (c < ' ' || c > '~') { // String hex = Integer.toHexString(c); // switch (hex.length()) { // case 1: // printer.print("\\u000"+hex); // break; // case 2: // printer.print("\\u00"+hex); // break; // case 3: // printer.print("\\u0"+hex); // break; // case 4: // printer.print("\\u"+hex); // break; // } // } else { // printer.print(""+c); // } // break; // } // printer.print("'"); String str = n.getValue(); if (str.length() == 1) { String hex = Integer.toHexString(str.charAt(0)); switch (hex.length()) { case 1: printer.print("0x0"+hex); break; case 2: printer.print("0x"+hex); break; case 3: printer.print("0x0"+hex); break; case 4: printer.print("0x"+hex); break; } } else if ("\\n".equals(str)) { printer.print("0x0A"); } else if ("\\r".equals(str)) { printer.print("0x0D"); } else if ("\\t".equals(str)) { printer.print("0x09"); } else if ("\\\"".equals(str)) { printer.print("0x22"); } else if ("\\'".equals(str)) { printer.print("0x27"); } else { throw new RuntimeException(str); } } public void visit(DoubleLiteralExpr n, Object arg) { printer.print(n.getValue()); } public void visit(IntegerLiteralExpr n, Object arg) { printer.print(n.getValue()); } public void visit(LongLiteralExpr n, Object arg) { printer.print(n.getValue()); } public void visit(IntegerLiteralMinValueExpr n, Object arg) { printer.print(n.getValue()); } public void visit(LongLiteralMinValueExpr n, Object arg) { printer.print(n.getValue()); } public void visit(StringLiteralExpr n, Object arg) { printer.print("\""); printer.print(n.getValue()); printer.print("\""); } public void visit(BooleanLiteralExpr n, Object arg) { printer.print(String.valueOf(n.getValue())); } public void visit(NullLiteralExpr n, Object arg) { printer.print("null"); } public void visit(ThisExpr n, Object arg) { if (n.getClassExpr() != null) { n.getClassExpr().accept(this, arg); printer.print("."); } printer.print("self"); } public void visit(SuperExpr n, Object arg) { if (n.getClassExpr() != null) { n.getClassExpr().accept(this, arg); printer.print("."); } printer.print("super"); } public void visit(MethodCallExpr n, Object arg) { if (n.getScope() != null) { n.getScope().accept(this, arg); printer.print("."); } printer.print(n.getName()); printArguments(n.getArgs(), arg); } public void visit(ObjectCreationExpr n, Object arg) { if (n.getScope() != null) { n.getScope().accept(this, arg); printer.print("."); } printer.print("new "); n.getType().accept(this, arg); printArguments(n.getArgs(), arg); if (n.getAnonymousClassBody() != null) { printer.printLn(" {"); printer.indent(); printMethods(n.getAnonymousClassBody(), arg); printer.unindent(); printer.print("}"); } } public void visit(UnaryExpr n, Object arg) { Operator op = n.getOperator(); if (op == null) { n.getExpr().accept(this, arg); return; } switch (op) { case positive: printer.print("+"); n.getExpr().accept(this, arg); break; case negative: printer.print("-"); n.getExpr().accept(this, arg); break; case inverse: printer.print("i32::compl("); n.getExpr().accept(this, arg); printer.print(")"); break; case not: printer.print("!"); n.getExpr().accept(this, arg); break; case preIncrement: case posIncrement: n.getExpr().accept(this, arg); printer.print(" = "); n.getExpr().accept(this, arg); printer.print(" + 1"); break; case preDecrement: case posDecrement: n.getExpr().accept(this, arg); printer.print(" = "); n.getExpr().accept(this, arg); printer.print(" - 1"); break; } } public void visit(ConstructorDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); printMemberAnnotations(n.getAnnotations(), arg); printModifiers(n.getModifiers()); if (n.getTypeParameters() != null) { printer.print(" "); } printer.print(n.getName()); printer.print("("); if (n.getParameters() != null) { for (Iterator<Parameter> i = n.getParameters().iterator(); i.hasNext();) { Parameter p = i.next(); p.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } } printer.print(")"); if (n.getThrows() != null) { printer.print(" throws "); for (Iterator<NameExpr> i = n.getThrows().iterator(); i.hasNext();) { NameExpr name = i.next(); name.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } } printer.print(" "); n.getBlock().accept(this, arg); } public void visit(MethodDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); // printMemberAnnotations(n.getAnnotations(), arg); // printModifiers(n.getModifiers()); // printTypeParameters(n.getTypeParameters(), arg); // if (n.getTypeParameters() != null) { // printer.print(" "); // } printer.print("fn "); printer.print(n.getName()); printer.print("("); inMethodSignature = true; if (n.getParameters() != null) { for (Iterator<Parameter> i = n.getParameters().iterator(); i.hasNext();) { Parameter p = i.next(); p.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } } inMethodSignature = false; printer.print(")"); Type type = n.getType(); if (!(type instanceof VoidType)) { printer.print(" -> "); type.accept(this, arg); } // for (int i = 0; i < n.getArrayCount(); i++) { // printer.print("[]"); // } // if (n.getThrows() != null) { // printer.print(" throws "); // for (Iterator<NameExpr> i = n.getThrows().iterator(); i.hasNext();) { // NameExpr name = i.next(); // name.accept(this, arg); // if (i.hasNext()) { // printer.print(", "); // } // } // } if (n.getBody() == null) { printer.print(";"); } else { printer.print(" "); n.getBody().accept(this, arg); } } public void visit(Parameter n, Object arg) { // printAnnotations(n.getAnnotations(), arg); // printModifiers(n.getModifiers()); VariableDeclaratorId id = n.getId(); printer.print(id.getName()); // if (n.isVarArgs()) { // printer.print("..."); // } printer.print(": "); n.getType().accept(this, arg); } public void visit(ExplicitConstructorInvocationStmt n, Object arg) { if (n.isThis()) { printer.print("this"); } else { if (n.getExpr() != null) { n.getExpr().accept(this, arg); printer.print("."); } printer.print("super"); } printArguments(n.getArgs(), arg); printer.print(";"); } public void visit(VariableDeclarationExpr n, Object arg) { // printAnnotations(n.getAnnotations(), arg); printer.print("let "); if (!ModifierSet.isFinal(n.getModifiers())) { printer.print("mut "); } // printModifiers(n.getModifiers()); List<VariableDeclarator> vars = n.getVars(); printVariableDeclarator(n.getType(), vars, arg, false); } public void visit(TypeDeclarationStmt n, Object arg) { n.getTypeDeclaration().accept(this, arg); } public void visit(AssertStmt n, Object arg) { Expression check = n.getCheck(); if (check instanceof BooleanLiteralExpr) { BooleanLiteralExpr bool = (BooleanLiteralExpr) check; if (!bool.getValue()) { printer.print("fail;"); return; } } printer.print("assert "); check.accept(this, arg); printer.print(";"); } public void visit(BlockStmt n, Object arg) { printer.printLn("{"); if (n.getStmts() != null) { printer.indent(); for (Statement s : n.getStmts()) { s.accept(this, arg); printer.printLn(); } printer.unindent(); } printer.print("}"); } public void visit(LabeledStmt n, Object arg) { assert arg == null; n.getStmt().accept(this, n.getLabel()); } public void visit(EmptyStmt n, Object arg) { printer.print(";"); } public void visit(ExpressionStmt n, Object arg) { Expression plusplus = null; Expression ex = n.getExpression(); if (ex instanceof MethodCallExpr) { MethodCallExpr meth = (MethodCallExpr) ex; if (meth.getName().startsWith("fatal") || meth.getName().startsWith("err") || meth.getName().startsWith("warn") || meth.getName().startsWith("maybeErr") || meth.getName().startsWith("maybeWarn") || meth.getName().startsWith("note") || "releaseArray".equals(meth.getName()) || "deleteArray".equals(meth.getName()) || "delete".equals(meth.getName())) { return; } } if (ex instanceof AssignExpr) { AssignExpr ax = (AssignExpr) ex; Expression left = ax.getTarget(); if (left instanceof ArrayAccessExpr) { ArrayAccessExpr aae = (ArrayAccessExpr) left; Expression index = aae.getIndex(); if (index instanceof UnaryExpr) { UnaryExpr unex = (UnaryExpr) index; if (unex.getOperator() == Operator.posIncrement) { plusplus = unex.getExpr(); unex.setOperator(null); } } } } n.getExpression().accept(this, arg); printer.print(";"); if (plusplus != null) { printer.printLn(); plusplus.accept(this, arg); printer.print(" = "); plusplus.accept(this, arg); printer.print(" + 1;"); } } public void visit(SwitchStmt n, Object arg) { printer.print("match "); n.getSelector().accept(this, arg); printer.printLn(" {"); if (n.getEntries() != null) { printer.indent(); List<Expression> labels = new LinkedList<Expression>(); for (SwitchEntryStmt e : n.getEntries()) { labels.add(e.getLabel()); List<Statement> stmts = e.getStmts(); if (stmts != null) { if (stmts.get(stmts.size() - 1) instanceof BreakStmt) { BreakStmt brk = (BreakStmt)stmts.get(stmts.size() - 1); if (brk.getId() == null) { stmts.remove(stmts.size() - 1); } } if (!stmts.isEmpty()) { boolean first = true; for (Expression label : labels) { if (!first) { printer.print(" | "); } first = false; if (label == null) { printer.print("_"); } else { label.accept(this, arg); } } printer.printLn(" => {"); printer.indent(); for (Statement statement : stmts) { statement.accept(this, arg); printer.printLn(); } printer.unindent(); printer.printLn("}"); } labels.clear(); } } printer.unindent(); } printer.print("}"); } public void visit(SwitchEntryStmt n, Object arg) { throw new RuntimeException("Not supposed to come here."); } public void visit(BreakStmt n, Object arg) { printer.print("break"); if (n.getId() != null && !"charsetloop".equals(n.getId()) && !"charactersloop".equals(n.getId())) { printer.print(" "); printer.print(n.getId()); } printer.print(";"); } public void visit(ReturnStmt n, Object arg) { printer.print("return"); if (n.getExpr() != null) { printer.print(" "); n.getExpr().accept(this, arg); } printer.print(";"); } public void visit(EnumDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); printMemberAnnotations(n.getAnnotations(), arg); printModifiers(n.getModifiers()); printer.print("enum "); printer.print(n.getName()); if (n.getImplements() != null) { printer.print(" implements "); for (Iterator<ClassOrInterfaceType> i = n.getImplements().iterator(); i.hasNext();) { ClassOrInterfaceType c = i.next(); c.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } } printer.printLn(" {"); printer.indent(); if (n.getEntries() != null) { printer.printLn(); for (Iterator<EnumConstantDeclaration> i = n.getEntries().iterator(); i.hasNext();) { EnumConstantDeclaration e = i.next(); e.accept(this, arg); if (i.hasNext()) { printer.print(", "); } } } if (n.getMembers() != null) { printer.printLn(";"); printMethods(n.getMembers(), arg); } else { if (n.getEntries() != null) { printer.printLn(); } } printer.unindent(); printer.print("}"); throw new RuntimeException("Unsupported syntax."); } public void visit(EnumConstantDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); printMemberAnnotations(n.getAnnotations(), arg); printer.print(n.getName()); if (n.getArgs() != null) { printArguments(n.getArgs(), arg); } if (n.getClassBody() != null) { printer.printLn(" {"); printer.indent(); printMethods(n.getClassBody(), arg); printer.unindent(); printer.printLn("}"); } throw new RuntimeException("Unsupported syntax."); } public void visit(EmptyMemberDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); printer.print(";"); } public void visit(InitializerDeclaration n, Object arg) { printJavadoc(n.getJavaDoc(), arg); if (n.isStatic()) { printer.print("static "); } n.getBlock().accept(this, arg); } public void visit(IfStmt n, Object arg) { Expression cond = n.getCondition(); if (cond instanceof BinaryExpr) { BinaryExpr binex = (BinaryExpr) cond; Expression left = binex.getLeft(); if (left instanceof UnaryExpr) { UnaryExpr unex = (UnaryExpr) left; if (unex.getOperator() == Operator.preIncrement) { unex.getExpr().accept(this, arg); printer.print(" = "); unex.getExpr().accept(this, arg); printer.printLn(" + 1;"); unex.setOperator(null); } } } if (!TranslatorUtils.isErrorHandlerIf(n.getCondition(), false)) { if (TranslatorUtils.isErrorOnlyBlock(n.getThenStmt(), false)) { if (n.getElseStmt() != null && !TranslatorUtils.isErrorOnlyBlock(n.getElseStmt(), false)) { printer.print("if "); if (n.getCondition() instanceof BinaryExpr) { BinaryExpr binExpr = (BinaryExpr) n.getCondition(); switch (binExpr.getOperator()) { case equals: binExpr.getLeft().accept(this, arg); printer.print(" != "); binExpr.getRight().accept(this, arg); break; case notEquals: binExpr.getLeft().accept(this, arg); printer.print(" == "); binExpr.getRight().accept(this, arg); break; default: printer.print("!("); n.getCondition().accept(this, arg); printer.print(")"); break; } } else { printer.print("!("); n.getCondition().accept(this, arg); printer.print(")"); } printer.print(" "); n.getElseStmt().accept(this, arg); } } else { printer.print("if "); n.getCondition().accept(this, arg); printer.print(" "); n.getThenStmt().accept(this, arg); if (n.getElseStmt() != null && !TranslatorUtils.isErrorOnlyBlock(n.getElseStmt(), false)) { printer.print(" else "); n.getElseStmt().accept(this, arg); } } } } public void visit(WhileStmt n, Object arg) { printer.print("while "); n.getCondition().accept(this, arg); printer.print(" "); n.getBody().accept(this, arg); } public void visit(ContinueStmt n, Object arg) { if (loopUpdate != null) { loopUpdate.accept(this, arg); printer.printLn(";"); } printer.print("loop"); if (n.getId() != null) { printer.print(" "); printer.print(n.getId()); } printer.print(";"); } public void visit(DoStmt n, Object arg) { printer.print("do "); n.getBody().accept(this, arg); printer.print(" while ("); n.getCondition().accept(this, arg); printer.print(");"); throw new RuntimeException("Unsupported syntax."); } public void visit(ForeachStmt n, Object arg) { printer.print("for ("); n.getVariable().accept(this, arg); printer.print(" : "); n.getIterable().accept(this, arg); printer.print(") "); n.getBody().accept(this, arg); throw new RuntimeException("Unsupported syntax."); } public void visit(ForStmt n, Object arg) { String label = null; if (arg instanceof String) { label = (String) arg; arg = null; } if (n.getInit() == null && n.getCompare() == null && n.getUpdate() == null) { printer.print("loop "); if (label != null) { printer.print(label); printer.print(": "); } n.getBody().accept(this, arg); return; } assert label == null || "charsetloop".equals(label) || "charactersloop".equals(label); Expression oldLoopUpdate = loopUpdate; loopUpdate = n.getUpdate().get(0); if (n.getInit() != null) { n.getInit().get(0).accept(this, arg); printer.printLn(";"); } if (n.getCompare() == null) { printer.print("loop "); } else { printer.print("while "); n.getCompare().accept(this, arg); printer.print(" "); } Statement body = n.getBody(); if (body instanceof BlockStmt) { BlockStmt blockStmt = (BlockStmt) body; printer.printLn("{"); printer.indent(); if (blockStmt.getStmts() != null) { for (Statement s : blockStmt.getStmts()) { s.accept(this, arg); printer.printLn(); } } if (loopUpdate != null) { loopUpdate.accept(this, arg); printer.printLn(";"); } printer.unindent(); printer.print("}"); } else { throw new RuntimeException(); } loopUpdate = oldLoopUpdate; } public void visit(ThrowStmt n, Object arg) { printer.print("throw "); n.getExpr().accept(this, arg); printer.print(";"); } public void visit(SynchronizedStmt n, Object arg) { printer.print("synchronized ("); n.getExpr().accept(this, arg); printer.print(") "); n.getBlock().accept(this, arg); } public void visit(TryStmt n, Object arg) { printer.print("try "); n.getTryBlock().accept(this, arg); if (n.getCatchs() != null) { for (CatchClause c : n.getCatchs()) { c.accept(this, arg); } } if (n.getFinallyBlock() != null) { printer.print(" finally "); n.getFinallyBlock().accept(this, arg); } } public void visit(CatchClause n, Object arg) { printer.print(" catch ("); n.getExcept().accept(this, arg); printer.print(") "); n.getCatchBlock().accept(this, arg); } // public void visit(AnnotationDeclaration n, Object arg) { // printJavadoc(n.getJavaDoc(), arg); // printMemberAnnotations(n.getAnnotations(), arg); // printModifiers(n.getModifiers()); // // printer.print("@interface "); // printer.print(n.getName()); // printer.printLn(" {"); // printer.indent(); // if (n.getMembers() != null) { // printMembers(n.getMembers(), arg); // } // printer.unindent(); // printer.print("}"); // } // // public void visit(AnnotationMemberDeclaration n, Object arg) { // printJavadoc(n.getJavaDoc(), arg); // printMemberAnnotations(n.getAnnotations(), arg); // printModifiers(n.getModifiers()); // // n.getType().accept(this, arg); // printer.print(" "); // printer.print(n.getName()); // printer.print("()"); // if (n.getDefaultValue() != null) { // printer.print(" default "); // n.getDefaultValue().accept(this, arg); // } // printer.print(";"); // } // // public void visit(MarkerAnnotationExpr n, Object arg) { // printer.print("@"); // n.getName().accept(this, arg); // } // // public void visit(SingleMemberAnnotationExpr n, Object arg) { // printer.print("@"); // n.getName().accept(this, arg); // printer.print("("); // n.getMemberValue().accept(this, arg); // printer.print(")"); // } // // public void visit(NormalAnnotationExpr n, Object arg) { // printer.print("@"); // n.getName().accept(this, arg); // printer.print("("); // if (n.getPairs() != null) { // for (Iterator<MemberValuePair> i = n.getPairs().iterator(); i.hasNext();) { // MemberValuePair m = i.next(); // m.accept(this, arg); // if (i.hasNext()) { // printer.print(", "); // } // } // } // printer.print(")"); // } public void visit(MemberValuePair n, Object arg) { printer.print(n.getName()); printer.print(" = "); n.getValue().accept(this, arg); } public void visit(LineComment n, Object arg) { printer.print("//"); printer.printLn(n.getContent()); } public void visit(BlockComment n, Object arg) { printer.print("/*"); printer.print(n.getContent()); printer.printLn("*/"); } }