aboutsummaryrefslogtreecommitdiff
path: root/src/lombok/javac/JavacAST.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/lombok/javac/JavacAST.java')
-rw-r--r--src/lombok/javac/JavacAST.java347
1 files changed, 0 insertions, 347 deletions
diff --git a/src/lombok/javac/JavacAST.java b/src/lombok/javac/JavacAST.java
deleted file mode 100644
index f2c83fb8..00000000
--- a/src/lombok/javac/JavacAST.java
+++ /dev/null
@@ -1,347 +0,0 @@
-/*
- * Copyright © 2009 Reinier Zwitserloot and Roel Spilker.
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * THE SOFTWARE.
- */
-package lombok.javac;
-
-import java.lang.reflect.Field;
-import java.util.ArrayList;
-import java.util.Collection;
-import java.util.List;
-
-import javax.annotation.processing.Messager;
-import javax.tools.Diagnostic;
-import javax.tools.JavaFileObject;
-
-import lombok.core.AST;
-
-import com.sun.source.util.Trees;
-import com.sun.tools.javac.code.Symtab;
-import com.sun.tools.javac.tree.JCTree;
-import com.sun.tools.javac.tree.TreeMaker;
-import com.sun.tools.javac.tree.JCTree.JCAnnotation;
-import com.sun.tools.javac.tree.JCTree.JCBlock;
-import com.sun.tools.javac.tree.JCTree.JCClassDecl;
-import com.sun.tools.javac.tree.JCTree.JCCompilationUnit;
-import com.sun.tools.javac.tree.JCTree.JCExpression;
-import com.sun.tools.javac.tree.JCTree.JCFieldAccess;
-import com.sun.tools.javac.tree.JCTree.JCImport;
-import com.sun.tools.javac.tree.JCTree.JCMethodDecl;
-import com.sun.tools.javac.tree.JCTree.JCStatement;
-import com.sun.tools.javac.tree.JCTree.JCVariableDecl;
-import com.sun.tools.javac.util.Context;
-import com.sun.tools.javac.util.Log;
-import com.sun.tools.javac.util.Name;
-import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition;
-
-/**
- * Wraps around javac's internal AST view to add useful features as well as the ability to visit parents from children,
- * something javac's own AST system does not offer.
- */
-public class JavacAST extends AST<JavacAST, JavacNode, JCTree> {
- private final Messager messager;
- private final Name.Table nameTable;
- private final TreeMaker treeMaker;
- private final Symtab symtab;
- private final Log log;
- private final Context context;
-
- /**
- * Creates a new JavacAST of the provided Compilation Unit.
- *
- * @param trees The trees instance to use to inspect the compilation unit. Generate via:
- * {@code Trees.getInstance(env)}
- * @param messager A Messager for warning and error reporting.
- * @param context A Context object for interfacing with the compiler.
- * @param top The compilation unit, which serves as the top level node in the tree to be built.
- */
- public JavacAST(Trees trees, Messager messager, Context context, JCCompilationUnit top) {
- super(top.sourcefile == null ? null : top.sourcefile.toString());
- setTop(buildCompilationUnit(top));
- this.context = context;
- this.messager = messager;
- this.log = Log.instance(context);
- this.nameTable = Name.Table.instance(context);
- this.treeMaker = TreeMaker.instance(context);
- this.symtab = Symtab.instance(context);
- }
-
- public Context getContext() {
- return context;
- }
-
- /** {@inheritDoc} */
- @Override public String getPackageDeclaration() {
- JCCompilationUnit unit = (JCCompilationUnit)top().get();
- return unit.pid instanceof JCFieldAccess ? unit.pid.toString() : null;
- }
-
- /** {@inheritDoc} */
- @Override public Collection<String> getImportStatements() {
- List<String> imports = new ArrayList<String>();
- JCCompilationUnit unit = (JCCompilationUnit)top().get();
- for (JCTree def : unit.defs) {
- if (def instanceof JCImport) {
- imports.add(((JCImport)def).qualid.toString());
- }
- }
-
- return imports;
- }
-
- /**
- * Runs through the entire AST, starting at the compilation unit, calling the provided visitor's visit methods
- * for each node, depth first.
- */
- public void traverse(JavacASTVisitor visitor) {
- top().traverse(visitor);
- }
-
- void traverseChildren(JavacASTVisitor visitor, JavacNode node) {
- for (JavacNode child : new ArrayList<JavacNode>(node.down())) {
- child.traverse(visitor);
- }
- }
-
- /** @return A Name object generated for the proper name table belonging to this AST. */
- public Name toName(String name) {
- return nameTable.fromString(name);
- }
-
- /** @return A TreeMaker instance that you can use to create new AST nodes. */
- public TreeMaker getTreeMaker() {
- return treeMaker;
- }
-
- /** @return The symbol table used by this AST for symbols. */
- public Symtab getSymbolTable() {
- return symtab;
- }
-
- /** {@inheritDoc} */
- @Override protected JavacNode buildTree(JCTree node, Kind kind) {
- switch (kind) {
- case COMPILATION_UNIT:
- return buildCompilationUnit((JCCompilationUnit) node);
- case TYPE:
- return buildType((JCClassDecl) node);
- case FIELD:
- return buildField((JCVariableDecl) node);
- case INITIALIZER:
- return buildInitializer((JCBlock) node);
- case METHOD:
- return buildMethod((JCMethodDecl) node);
- case ARGUMENT:
- return buildLocalVar((JCVariableDecl) node, kind);
- case LOCAL:
- return buildLocalVar((JCVariableDecl) node, kind);
- case STATEMENT:
- return buildStatementOrExpression(node);
- case ANNOTATION:
- return buildAnnotation((JCAnnotation) node);
- default:
- throw new AssertionError("Did not expect: " + kind);
- }
- }
-
- private JavacNode buildCompilationUnit(JCCompilationUnit top) {
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
- for (JCTree s : top.defs) {
- if (s instanceof JCClassDecl) {
- addIfNotNull(childNodes, buildType((JCClassDecl)s));
- } // else they are import statements, which we don't care about. Or Skip objects, whatever those are.
- }
-
- return new JavacNode(this, top, childNodes, Kind.COMPILATION_UNIT);
- }
-
- private JavacNode buildType(JCClassDecl type) {
- if (setAndGetAsHandled(type)) return null;
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
-
- for (JCTree def : type.defs) {
- for (JCAnnotation annotation : type.mods.annotations) addIfNotNull(childNodes, buildAnnotation(annotation));
- /* A def can be:
- * JCClassDecl for inner types
- * JCMethodDecl for constructors and methods
- * JCVariableDecl for fields
- * JCBlock for (static) initializers
- */
- if (def instanceof JCMethodDecl) addIfNotNull(childNodes, buildMethod((JCMethodDecl)def));
- else if (def instanceof JCClassDecl) addIfNotNull(childNodes, buildType((JCClassDecl)def));
- else if (def instanceof JCVariableDecl) addIfNotNull(childNodes, buildField((JCVariableDecl)def));
- else if (def instanceof JCBlock) addIfNotNull(childNodes, buildInitializer((JCBlock)def));
- }
-
- return putInMap(new JavacNode(this, type, childNodes, Kind.TYPE));
- }
-
- private JavacNode buildField(JCVariableDecl field) {
- if (setAndGetAsHandled(field)) return null;
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
- for (JCAnnotation annotation : field.mods.annotations) addIfNotNull(childNodes, buildAnnotation(annotation));
- addIfNotNull(childNodes, buildExpression(field.init));
- return putInMap(new JavacNode(this, field, childNodes, Kind.FIELD));
- }
-
- private JavacNode buildLocalVar(JCVariableDecl local, Kind kind) {
- if (setAndGetAsHandled(local)) return null;
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
- for (JCAnnotation annotation : local.mods.annotations) addIfNotNull(childNodes, buildAnnotation(annotation));
- addIfNotNull(childNodes, buildExpression(local.init));
- return putInMap(new JavacNode(this, local, childNodes, kind));
- }
-
- private JavacNode buildInitializer(JCBlock initializer) {
- if (setAndGetAsHandled(initializer)) return null;
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
- for (JCStatement statement: initializer.stats) addIfNotNull(childNodes, buildStatement(statement));
- return putInMap(new JavacNode(this, initializer, childNodes, Kind.INITIALIZER));
- }
-
- private JavacNode buildMethod(JCMethodDecl method) {
- if (setAndGetAsHandled(method)) return null;
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
- for (JCAnnotation annotation : method.mods.annotations) addIfNotNull(childNodes, buildAnnotation(annotation));
- for (JCVariableDecl param : method.params) addIfNotNull(childNodes, buildLocalVar(param, Kind.ARGUMENT));
- if (method.body != null && method.body.stats != null) {
- for (JCStatement statement : method.body.stats) addIfNotNull(childNodes, buildStatement(statement));
- }
- return putInMap(new JavacNode(this, method, childNodes, Kind.METHOD));
- }
-
- private JavacNode buildAnnotation(JCAnnotation annotation) {
- if (setAndGetAsHandled(annotation)) return null;
- return putInMap(new JavacNode(this, annotation, null, Kind.ANNOTATION));
- }
-
- private JavacNode buildExpression(JCExpression expression) {
- return buildStatementOrExpression(expression);
- }
-
- private JavacNode buildStatement(JCStatement statement) {
- return buildStatementOrExpression(statement);
- }
-
- private JavacNode buildStatementOrExpression(JCTree statement) {
- if (statement == null) return null;
- if (statement instanceof JCAnnotation) return null;
- if (statement instanceof JCClassDecl) return buildType((JCClassDecl)statement);
- if (statement instanceof JCVariableDecl) return buildLocalVar((JCVariableDecl)statement, Kind.LOCAL);
-
- if (setAndGetAsHandled(statement)) return null;
-
- return drill(statement);
- }
-
- private JavacNode drill(JCTree statement) {
- List<JavacNode> childNodes = new ArrayList<JavacNode>();
- for (FieldAccess fa : fieldsOf(statement.getClass())) childNodes.addAll(buildWithField(JavacNode.class, statement, fa));
- return putInMap(new JavacNode(this, statement, childNodes, Kind.STATEMENT));
- }
-
- /** For javac, both JCExpression and JCStatement are considered as valid children types. */
- @Override
- protected Collection<Class<? extends JCTree>> getStatementTypes() {
- Collection<Class<? extends JCTree>> collection = new ArrayList<Class<? extends JCTree>>(2);
- collection.add(JCStatement.class);
- collection.add(JCExpression.class);
- return collection;
- }
-
- private static void addIfNotNull(Collection<JavacNode> nodes, JavacNode node) {
- if (node != null) nodes.add(node);
- }
-
- /** Supply either a position or a node (in that case, position of the node is used) */
- void printMessage(Diagnostic.Kind kind, String message, JavacNode node, DiagnosticPosition pos) {
- JavaFileObject oldSource = null;
- JavaFileObject newSource = null;
- JCTree astObject = node == null ? null : node.get();
- JCCompilationUnit top = (JCCompilationUnit) top().get();
- newSource = top.sourcefile;
- if (newSource != null) {
- oldSource = log.useSource(newSource);
- if (pos == null) pos = astObject.pos();
- }
- try {
- switch (kind) {
- case ERROR:
- increaseErrorCount(messager);
- boolean prev = log.multipleErrors;
- log.multipleErrors = true;
- try {
- log.error(pos, "proc.messager", message);
- } finally {
- log.multipleErrors = prev;
- }
- break;
- default:
- case WARNING:
- log.warning(pos, "proc.messager", message);
- break;
- }
- } finally {
- if (oldSource != null) log.useSource(oldSource);
- }
- }
-
- /** {@inheritDoc} */
- @Override protected void setElementInASTCollection(Field field, Object refField, List<Collection<?>> chain, Collection<?> collection, int idx, JCTree newN) throws IllegalAccessException {
- com.sun.tools.javac.util.List<?> list = setElementInConsList(chain, collection, ((List<?>)collection).get(idx), newN);
- field.set(refField, list);
- }
-
- private com.sun.tools.javac.util.List<?> setElementInConsList(List<Collection<?>> chain, Collection<?> current, Object oldO, Object newO) {
- com.sun.tools.javac.util.List<?> oldL = (com.sun.tools.javac.util.List<?>) current;
- com.sun.tools.javac.util.List<?> newL = replaceInConsList(oldL, oldO, newO);
- if (chain.isEmpty()) return newL;
- List<Collection<?>> reducedChain = new ArrayList<Collection<?>>(chain);
- Collection<?> newCurrent = reducedChain.remove(reducedChain.size() -1);
- return setElementInConsList(reducedChain, newCurrent, oldL, newL);
- }
-
- private com.sun.tools.javac.util.List<?> replaceInConsList(com.sun.tools.javac.util.List<?> oldL, Object oldO, Object newO) {
- boolean repl = false;
- Object[] a = oldL.toArray();
- for (int i = 0; i < a.length; i++) {
- if (a[i] == oldO) {
- a[i] = newO;
- repl = true;
- }
- }
-
- if (repl) return com.sun.tools.javac.util.List.<Object>from(a);
- return oldL;
- }
-
- private void increaseErrorCount(Messager m) {
- try {
- Field f = m.getClass().getDeclaredField("errorCount");
- f.setAccessible(true);
- if (f.getType() == int.class) {
- int val = ((Number)f.get(m)).intValue();
- f.set(m, val +1);
- }
- } catch (Throwable t) {
- //Very unfortunate, but in most cases it still works fine, so we'll silently swallow it.
- }
- }
-}