/* * Copyright (C) 2009-2012 The Project Lombok Authors. * * 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.Method; import java.util.Objects; import java.util.regex.Pattern; import com.sun.tools.javac.main.JavaCompiler; import com.sun.tools.javac.tree.JCTree; import com.sun.tools.javac.tree.JCTree.JCBinary; import com.sun.tools.javac.tree.JCTree.JCExpression; import com.sun.tools.javac.tree.JCTree.JCFieldAccess; import com.sun.tools.javac.tree.JCTree.JCIdent; import com.sun.tools.javac.tree.JCTree.JCLiteral; import com.sun.tools.javac.tree.JCTree.JCPrimitiveTypeTree; import com.sun.tools.javac.tree.JCTree.JCUnary; import com.sun.tools.javac.tree.TreeMaker; /** * Container for static utility methods relevant to lombok's operation on javac. */ public class Javac { private Javac() { // prevent instantiation } /** Matches any of the 8 primitive names, such as {@code boolean}. */ private static final Pattern PRIMITIVE_TYPE_NAME_PATTERN = Pattern.compile("^(boolean|byte|short|int|long|float|double|char)$"); /** * Checks if the given expression (that really ought to refer to a type * expression) represents a primitive type. */ public static boolean isPrimitive(JCExpression ref) { String typeName = ref.toString(); return PRIMITIVE_TYPE_NAME_PATTERN.matcher(typeName).matches(); } /** * Turns an expression into a guessed intended literal. Only works for * literals, as you can imagine. * * Will for example turn a TrueLiteral into 'Boolean.valueOf(true)'. */ public static Object calculateGuess(JCExpression expr) { if (expr instanceof JCLiteral) { JCLiteral lit = (JCLiteral) expr; if (lit.getKind() == com.sun.source.tree.Tree.Kind.BOOLEAN_LITERAL) { return ((Number) lit.value).intValue() == 0 ? false : true; } return lit.value; } else if (expr instanceof JCIdent || expr instanceof JCFieldAccess) { String x = expr.toString(); if (x.endsWith(".class")) x = x.substring(0, x.length() - 6); else { int idx = x.lastIndexOf('.'); if (idx > -1) x = x.substring(idx + 1); } return x; } else return null; } public static final Object CTC_BOOLEAN = getTypeTag("BOOLEAN"); public static final Object CTC_INT = getTypeTag("INT"); public static final Object CTC_DOUBLE = getTypeTag("DOUBLE"); public static final Object CTC_FLOAT = getTypeTag("FLOAT"); public static final Object CTC_SHORT = getTypeTag("SHORT"); public static final Object CTC_BYTE = getTypeTag("BYTE"); public static final Object CTC_LONG = getTypeTag("LONG"); public static final Object CTC_CHAR = getTypeTag("CHAR"); public static final Object CTC_VOID = getTypeTag("VOID"); public static final Object CTC_NONE = getTypeTag("NONE"); public static final Object CTC_BOT = getTypeTag("BOT"); public static final Object CTC_CLASS = getTypeTag("CLASS"); public static final Object CTC_NOT_EQUAL = getTreeTag("NE"); public static final Object CTC_NOT = getTreeTag("NOT"); public static final Object CTC_BITXOR = getTreeTag("BITXOR"); public static final Object CTC_UNSIGNED_SHIFT_RIGHT = getTreeTag("USR"); public static final Object CTC_MUL = getTreeTag("MUL"); public static final Object CTC_PLUS = getTreeTag("PLUS"); public static final Object CTC_EQUAL = getTreeTag("EQ"); // /** // * Retrieves the current ordinal position of a type tag. // * // * For JDK 8 this is the ordinal position within the // * com.sun.tools.javac.code.TypeTag enum for JDK 7 and lower, // * this is the value of the constant within // * com.sun.tools.javac.code.TypeTags // * // * Solves the problem of compile time constant inlining, resulting in lombok // * having the wrong value (javac compiler changes private api constants from // * time to time). // * // * @param identifier // * @return the ordinal value of the typetag constant // */ // public static int getTypeTag(String identifier) { // try { // if (JavaCompiler.version().startsWith("1.8")) { // Object enumInstance = Class.forName("com.sun.tools.javac.code.TypeTag").getField(identifier).get(null); // return (int) Class.forName("com.sun.tools.javac.code.TypeTag").getField("order").get(enumInstance); // // } else { // return (int) Class.forName("com.sun.tools.javac.code.TypeTags").getField(identifier).get(null); // } // } catch (NoSuchFieldException e) { // throw new RuntimeException(e); // } catch (IllegalAccessException e) { // throw new RuntimeException(e); // } catch (Exception e) { // if (e instanceof RuntimeException) throw (RuntimeException) e; // throw new RuntimeException(e); // } // } public static boolean compareCTC(Object ctc1, Object ctc2) { return Objects.equals(ctc1, ctc2); } /** * Retrieves the current type tag. The actual type object differs depending on the Compiler version * * For JDK 8 this is an enum value of type com.sun.tools.javac.code.TypeTag * for JDK 7 and lower, this is the value of the constant within com.sun.tools.javac.code.TypeTags * * Solves the problem of compile time constant inlining, resulting in lombok * having the wrong value (javac compiler changes private api constants from * time to time). * * @param identifier * @return the ordinal value of the typetag constant */ public static Object getTypeTag(String identifier) { try { if (JavaCompiler.version().startsWith("1.8")) { return Class.forName("com.sun.tools.javac.code.TypeTag").getField(identifier).get(null); } else { return Class.forName("com.sun.tools.javac.code.TypeTags").getField(identifier).get(null); } } catch (NoSuchFieldException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (Exception e) { if (e instanceof RuntimeException) throw (RuntimeException) e; throw new RuntimeException(e); } } public static Object getTreeTag(String identifier) { try { if (JavaCompiler.version().startsWith("1.8")) { return Class.forName("com.sun.tools.javac.tree.JCTree$Tag").getField(identifier).get(null); } else { return Class.forName("com.sun.tools.javac.tree.JCTree").getField(identifier).get(null); } } catch (NoSuchFieldException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (Exception e) { if (e instanceof RuntimeException) throw (RuntimeException) e; throw new RuntimeException(e); } } public static Object getTreeTypeTag(JCPrimitiveTypeTree tree) { return tree.typetag; } public static Object getTreeTypeTag(JCLiteral tree) { return tree.typetag; } public static JCExpression makeTypeIdent(TreeMaker maker, Object ctc) { try { Method createIdent; if (JavaCompiler.version().startsWith("1.8")) { createIdent = TreeMaker.class.getMethod("TypeIdent", Class.forName("com.sun.tools.javac.code.TypeTag")); } else { createIdent = TreeMaker.class.getMethod("TypeIdent", Integer.TYPE); } return (JCExpression) createIdent.invoke(maker, ctc); } catch (NoSuchMethodException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (Exception e) { if (e instanceof RuntimeException) throw (RuntimeException) e; throw new RuntimeException(e); } } public static JCLiteral makeLiteral(TreeMaker maker, Object ctc, Object argument) { try { Method createLiteral; if (JavaCompiler.version().startsWith("1.8")) { createLiteral = TreeMaker.class.getMethod("Literal", Class.forName("com.sun.tools.javac.code.TypeTag"), Object.class); } else { createLiteral = TreeMaker.class.getMethod("Literal", Integer.TYPE, Object.class); } return (JCLiteral) createLiteral.invoke(maker, ctc, argument); } catch (NoSuchMethodException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (Exception e) { if (e instanceof RuntimeException) throw (RuntimeException) e; throw new RuntimeException(e); } } public static JCUnary makeUnary(TreeMaker maker, Object ctc, JCExpression argument) { try { Method createUnary; if (JavaCompiler.version().startsWith("1.8")) { createUnary = TreeMaker.class.getMethod("Unary", Class.forName("com.sun.tools.javac.code.TypeTag"), JCExpression.class); } else { createUnary = TreeMaker.class.getMethod("Unary", Integer.TYPE, JCExpression.class); } return (JCUnary) createUnary.invoke(maker, ctc, argument); } catch (NoSuchMethodException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (Exception e) { if (e instanceof RuntimeException) throw (RuntimeException) e; throw new RuntimeException(e); } } public static JCBinary makeBinary(TreeMaker maker, Object ctc, JCExpression rhsArgument, JCExpression lhsArgument) { try { Method createUnary; if (JavaCompiler.version().startsWith("1.8")) { createUnary = TreeMaker.class.getMethod("Binary", Class.forName("com.sun.tools.javac.code.TypeTag"), JCExpression.class, JCExpression.class); } else { createUnary = TreeMaker.class.getMethod("Binary", Integer.TYPE, JCExpression.class, JCExpression.class); } return (JCBinary) createUnary.invoke(maker, ctc, rhsArgument, lhsArgument); } catch (NoSuchMethodException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (Exception e) { if (e instanceof RuntimeException) throw (RuntimeException) e; throw new RuntimeException(e); } } }