8161708: javac, consider a different way to handle access code for operators
Reviewed-by: mcimadamore
This commit is contained in:
parent
e6f3a52942
commit
2bedc263dc
@ -55,7 +55,9 @@ import com.sun.tools.javac.comp.Attr;
|
|||||||
import com.sun.tools.javac.comp.AttrContext;
|
import com.sun.tools.javac.comp.AttrContext;
|
||||||
import com.sun.tools.javac.comp.Env;
|
import com.sun.tools.javac.comp.Env;
|
||||||
import com.sun.tools.javac.jvm.*;
|
import com.sun.tools.javac.jvm.*;
|
||||||
|
import com.sun.tools.javac.tree.JCTree.JCFieldAccess;
|
||||||
import com.sun.tools.javac.tree.JCTree.JCVariableDecl;
|
import com.sun.tools.javac.tree.JCTree.JCVariableDecl;
|
||||||
|
import com.sun.tools.javac.tree.JCTree.Tag;
|
||||||
import com.sun.tools.javac.util.*;
|
import com.sun.tools.javac.util.*;
|
||||||
import com.sun.tools.javac.util.DefinedBy.Api;
|
import com.sun.tools.javac.util.DefinedBy.Api;
|
||||||
import com.sun.tools.javac.util.Name;
|
import com.sun.tools.javac.util.Name;
|
||||||
@ -64,9 +66,15 @@ import static com.sun.tools.javac.code.Flags.*;
|
|||||||
import static com.sun.tools.javac.code.Kinds.*;
|
import static com.sun.tools.javac.code.Kinds.*;
|
||||||
import static com.sun.tools.javac.code.Kinds.Kind.*;
|
import static com.sun.tools.javac.code.Kinds.Kind.*;
|
||||||
import static com.sun.tools.javac.code.Scope.LookupKind.NON_RECURSIVE;
|
import static com.sun.tools.javac.code.Scope.LookupKind.NON_RECURSIVE;
|
||||||
|
import static com.sun.tools.javac.code.Symbol.OperatorSymbol.AccessCode.FIRSTASGOP;
|
||||||
import static com.sun.tools.javac.code.TypeTag.CLASS;
|
import static com.sun.tools.javac.code.TypeTag.CLASS;
|
||||||
import static com.sun.tools.javac.code.TypeTag.FORALL;
|
import static com.sun.tools.javac.code.TypeTag.FORALL;
|
||||||
import static com.sun.tools.javac.code.TypeTag.TYPEVAR;
|
import static com.sun.tools.javac.code.TypeTag.TYPEVAR;
|
||||||
|
import static com.sun.tools.javac.jvm.ByteCodes.iadd;
|
||||||
|
import static com.sun.tools.javac.jvm.ByteCodes.ishll;
|
||||||
|
import static com.sun.tools.javac.jvm.ByteCodes.lushrl;
|
||||||
|
import static com.sun.tools.javac.jvm.ByteCodes.lxor;
|
||||||
|
import static com.sun.tools.javac.jvm.ByteCodes.string_add;
|
||||||
|
|
||||||
/** Root class for Java symbols. It contains subclasses
|
/** Root class for Java symbols. It contains subclasses
|
||||||
* for specific sorts of symbols, such as variables, methods and operators,
|
* for specific sorts of symbols, such as variables, methods and operators,
|
||||||
@ -1950,15 +1958,90 @@ public abstract class Symbol extends AnnoConstruct implements Element {
|
|||||||
public static class OperatorSymbol extends MethodSymbol {
|
public static class OperatorSymbol extends MethodSymbol {
|
||||||
|
|
||||||
public int opcode;
|
public int opcode;
|
||||||
|
private int accessCode = Integer.MIN_VALUE;
|
||||||
|
|
||||||
public OperatorSymbol(Name name, Type type, int opcode, Symbol owner) {
|
public OperatorSymbol(Name name, Type type, int opcode, Symbol owner) {
|
||||||
super(PUBLIC | STATIC, name, type, owner);
|
super(PUBLIC | STATIC, name, type, owner);
|
||||||
this.opcode = opcode;
|
this.opcode = opcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public <R, P> R accept(Symbol.Visitor<R, P> v, P p) {
|
public <R, P> R accept(Symbol.Visitor<R, P> v, P p) {
|
||||||
return v.visitOperatorSymbol(this, p);
|
return v.visitOperatorSymbol(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getAccessCode(Tag tag) {
|
||||||
|
if (accessCode != Integer.MIN_VALUE && !tag.isIncOrDecUnaryOp()) {
|
||||||
|
return accessCode;
|
||||||
|
}
|
||||||
|
accessCode = AccessCode.from(tag, opcode);
|
||||||
|
return accessCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Access codes for dereferencing, assignment,
|
||||||
|
* and pre/post increment/decrement.
|
||||||
|
|
||||||
|
* All access codes for accesses to the current class are even.
|
||||||
|
* If a member of the superclass should be accessed instead (because
|
||||||
|
* access was via a qualified super), add one to the corresponding code
|
||||||
|
* for the current class, making the number odd.
|
||||||
|
* This numbering scheme is used by the backend to decide whether
|
||||||
|
* to issue an invokevirtual or invokespecial call.
|
||||||
|
*
|
||||||
|
* @see Gen#visitSelect(JCFieldAccess tree)
|
||||||
|
*/
|
||||||
|
public enum AccessCode {
|
||||||
|
UNKNOWN(-1, Tag.NO_TAG),
|
||||||
|
DEREF(0, Tag.NO_TAG),
|
||||||
|
ASSIGN(2, Tag.ASSIGN),
|
||||||
|
PREINC(4, Tag.PREINC),
|
||||||
|
PREDEC(6, Tag.PREDEC),
|
||||||
|
POSTINC(8, Tag.POSTINC),
|
||||||
|
POSTDEC(10, Tag.POSTDEC),
|
||||||
|
FIRSTASGOP(12, Tag.NO_TAG);
|
||||||
|
|
||||||
|
public final int code;
|
||||||
|
public final Tag tag;
|
||||||
|
public static final int numberOfAccessCodes = (lushrl - ishll + lxor + 2 - iadd) * 2 + FIRSTASGOP.code + 2;
|
||||||
|
|
||||||
|
AccessCode(int code, Tag tag) {
|
||||||
|
this.code = code;
|
||||||
|
this.tag = tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
static public AccessCode getFromCode(int code) {
|
||||||
|
for (AccessCode aCodes : AccessCode.values()) {
|
||||||
|
if (aCodes.code == code) {
|
||||||
|
return aCodes;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int from(Tag tag, int opcode) {
|
||||||
|
/** Map bytecode of binary operation to access code of corresponding
|
||||||
|
* assignment operation. This is always an even number.
|
||||||
|
*/
|
||||||
|
switch (tag) {
|
||||||
|
case PREINC:
|
||||||
|
return AccessCode.PREINC.code;
|
||||||
|
case PREDEC:
|
||||||
|
return AccessCode.PREDEC.code;
|
||||||
|
case POSTINC:
|
||||||
|
return AccessCode.POSTINC.code;
|
||||||
|
case POSTDEC:
|
||||||
|
return AccessCode.POSTDEC.code;
|
||||||
|
}
|
||||||
|
if (iadd <= opcode && opcode <= lxor) {
|
||||||
|
return (opcode - iadd) * 2 + FIRSTASGOP.code;
|
||||||
|
} else if (opcode == string_add) {
|
||||||
|
return (lxor + 1 - iadd) * 2 + FIRSTASGOP.code;
|
||||||
|
} else if (ishll <= opcode && opcode <= lushrl) {
|
||||||
|
return (opcode - ishll + lxor + 2 - iadd) * 2 + FIRSTASGOP.code;
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Symbol completer interface.
|
/** Symbol completer interface.
|
||||||
|
@ -3159,7 +3159,7 @@ public class Attr extends JCTree.Visitor {
|
|||||||
Type operand = attribExpr(tree.rhs, env);
|
Type operand = attribExpr(tree.rhs, env);
|
||||||
// Find operator.
|
// Find operator.
|
||||||
Symbol operator = tree.operator = operators.resolveBinary(tree, tree.getTag().noAssignOp(), owntype, operand);
|
Symbol operator = tree.operator = operators.resolveBinary(tree, tree.getTag().noAssignOp(), owntype, operand);
|
||||||
if (operator.kind == MTH &&
|
if (operator != operators.noOpSymbol &&
|
||||||
!owntype.isErroneous() &&
|
!owntype.isErroneous() &&
|
||||||
!operand.isErroneous()) {
|
!operand.isErroneous()) {
|
||||||
chk.checkDivZero(tree.rhs.pos(), operator, operand);
|
chk.checkDivZero(tree.rhs.pos(), operator, operand);
|
||||||
@ -3179,7 +3179,7 @@ public class Attr extends JCTree.Visitor {
|
|||||||
// Find operator.
|
// Find operator.
|
||||||
Symbol operator = tree.operator = operators.resolveUnary(tree, tree.getTag(), argtype);
|
Symbol operator = tree.operator = operators.resolveUnary(tree, tree.getTag(), argtype);
|
||||||
Type owntype = types.createErrorType(tree.type);
|
Type owntype = types.createErrorType(tree.type);
|
||||||
if (operator.kind == MTH &&
|
if (operator != operators.noOpSymbol &&
|
||||||
!argtype.isErroneous()) {
|
!argtype.isErroneous()) {
|
||||||
owntype = (tree.getTag().isIncOrDecUnaryOp())
|
owntype = (tree.getTag().isIncOrDecUnaryOp())
|
||||||
? tree.arg.type
|
? tree.arg.type
|
||||||
@ -3204,7 +3204,7 @@ public class Attr extends JCTree.Visitor {
|
|||||||
// Find operator.
|
// Find operator.
|
||||||
Symbol operator = tree.operator = operators.resolveBinary(tree, tree.getTag(), left, right);
|
Symbol operator = tree.operator = operators.resolveBinary(tree, tree.getTag(), left, right);
|
||||||
Type owntype = types.createErrorType(tree.type);
|
Type owntype = types.createErrorType(tree.type);
|
||||||
if (operator.kind == MTH &&
|
if (operator != operators.noOpSymbol &&
|
||||||
!left.isErroneous() &&
|
!left.isErroneous() &&
|
||||||
!right.isErroneous()) {
|
!right.isErroneous()) {
|
||||||
owntype = operator.type.getReturnType();
|
owntype = operator.type.getReturnType();
|
||||||
|
@ -38,6 +38,7 @@ import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition;
|
|||||||
import com.sun.tools.javac.util.List;
|
import com.sun.tools.javac.util.List;
|
||||||
|
|
||||||
import com.sun.tools.javac.code.Symbol.*;
|
import com.sun.tools.javac.code.Symbol.*;
|
||||||
|
import com.sun.tools.javac.code.Symbol.OperatorSymbol.AccessCode;
|
||||||
import com.sun.tools.javac.tree.JCTree.*;
|
import com.sun.tools.javac.tree.JCTree.*;
|
||||||
import com.sun.tools.javac.code.Type.*;
|
import com.sun.tools.javac.code.Type.*;
|
||||||
|
|
||||||
@ -49,6 +50,7 @@ import static com.sun.tools.javac.code.Flags.BLOCK;
|
|||||||
import static com.sun.tools.javac.code.Scope.LookupKind.NON_RECURSIVE;
|
import static com.sun.tools.javac.code.Scope.LookupKind.NON_RECURSIVE;
|
||||||
import static com.sun.tools.javac.code.TypeTag.*;
|
import static com.sun.tools.javac.code.TypeTag.*;
|
||||||
import static com.sun.tools.javac.code.Kinds.Kind.*;
|
import static com.sun.tools.javac.code.Kinds.Kind.*;
|
||||||
|
import static com.sun.tools.javac.code.Symbol.OperatorSymbol.AccessCode.DEREF;
|
||||||
import static com.sun.tools.javac.jvm.ByteCodes.*;
|
import static com.sun.tools.javac.jvm.ByteCodes.*;
|
||||||
import static com.sun.tools.javac.tree.JCTree.Tag.*;
|
import static com.sun.tools.javac.tree.JCTree.Tag.*;
|
||||||
|
|
||||||
@ -816,33 +818,6 @@ public class Lower extends TreeTranslator {
|
|||||||
* Access methods
|
* Access methods
|
||||||
*************************************************************************/
|
*************************************************************************/
|
||||||
|
|
||||||
/** Access codes for dereferencing, assignment,
|
|
||||||
* and pre/post increment/decrement.
|
|
||||||
* Access codes for assignment operations are determined by method accessCode
|
|
||||||
* below.
|
|
||||||
*
|
|
||||||
* All access codes for accesses to the current class are even.
|
|
||||||
* If a member of the superclass should be accessed instead (because
|
|
||||||
* access was via a qualified super), add one to the corresponding code
|
|
||||||
* for the current class, making the number odd.
|
|
||||||
* This numbering scheme is used by the backend to decide whether
|
|
||||||
* to issue an invokevirtual or invokespecial call.
|
|
||||||
*
|
|
||||||
* @see Gen#visitSelect(JCFieldAccess tree)
|
|
||||||
*/
|
|
||||||
private static final int
|
|
||||||
DEREFcode = 0,
|
|
||||||
ASSIGNcode = 2,
|
|
||||||
PREINCcode = 4,
|
|
||||||
PREDECcode = 6,
|
|
||||||
POSTINCcode = 8,
|
|
||||||
POSTDECcode = 10,
|
|
||||||
FIRSTASGOPcode = 12;
|
|
||||||
|
|
||||||
/** Number of access codes
|
|
||||||
*/
|
|
||||||
private static final int NCODES = accessCode(ByteCodes.lushrl) + 2;
|
|
||||||
|
|
||||||
/** A mapping from symbols to their access numbers.
|
/** A mapping from symbols to their access numbers.
|
||||||
*/
|
*/
|
||||||
private Map<Symbol,Integer> accessNums;
|
private Map<Symbol,Integer> accessNums;
|
||||||
@ -864,20 +839,6 @@ public class Lower extends TreeTranslator {
|
|||||||
*/
|
*/
|
||||||
private ListBuffer<Symbol> accessed;
|
private ListBuffer<Symbol> accessed;
|
||||||
|
|
||||||
/** Map bytecode of binary operation to access code of corresponding
|
|
||||||
* assignment operation. This is always an even number.
|
|
||||||
*/
|
|
||||||
private static int accessCode(int bytecode) {
|
|
||||||
if (ByteCodes.iadd <= bytecode && bytecode <= ByteCodes.lxor)
|
|
||||||
return (bytecode - iadd) * 2 + FIRSTASGOPcode;
|
|
||||||
else if (bytecode == ByteCodes.string_add)
|
|
||||||
return (ByteCodes.lxor + 1 - iadd) * 2 + FIRSTASGOPcode;
|
|
||||||
else if (ByteCodes.ishll <= bytecode && bytecode <= ByteCodes.lushrl)
|
|
||||||
return (bytecode - ishll + ByteCodes.lxor + 2 - iadd) * 2 + FIRSTASGOPcode;
|
|
||||||
else
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** return access code for identifier,
|
/** return access code for identifier,
|
||||||
* @param tree The tree representing the identifier use.
|
* @param tree The tree representing the identifier use.
|
||||||
* @param enclOp The closest enclosing operation node of tree,
|
* @param enclOp The closest enclosing operation node of tree,
|
||||||
@ -885,24 +846,24 @@ public class Lower extends TreeTranslator {
|
|||||||
*/
|
*/
|
||||||
private static int accessCode(JCTree tree, JCTree enclOp) {
|
private static int accessCode(JCTree tree, JCTree enclOp) {
|
||||||
if (enclOp == null)
|
if (enclOp == null)
|
||||||
return DEREFcode;
|
return AccessCode.DEREF.code;
|
||||||
else if (enclOp.hasTag(ASSIGN) &&
|
else if (enclOp.hasTag(ASSIGN) &&
|
||||||
tree == TreeInfo.skipParens(((JCAssign) enclOp).lhs))
|
tree == TreeInfo.skipParens(((JCAssign) enclOp).lhs))
|
||||||
return ASSIGNcode;
|
return AccessCode.ASSIGN.code;
|
||||||
else if (enclOp.getTag().isIncOrDecUnaryOp() &&
|
else if (enclOp.getTag().isIncOrDecUnaryOp() &&
|
||||||
tree == TreeInfo.skipParens(((JCUnary) enclOp).arg))
|
tree == TreeInfo.skipParens(((JCUnary) enclOp).arg))
|
||||||
return mapTagToUnaryOpCode(enclOp.getTag());
|
return (((JCUnary) enclOp).operator).getAccessCode(enclOp.getTag());
|
||||||
else if (enclOp.getTag().isAssignop() &&
|
else if (enclOp.getTag().isAssignop() &&
|
||||||
tree == TreeInfo.skipParens(((JCAssignOp) enclOp).lhs))
|
tree == TreeInfo.skipParens(((JCAssignOp) enclOp).lhs))
|
||||||
return accessCode(((OperatorSymbol) ((JCAssignOp) enclOp).operator).opcode);
|
return (((JCAssignOp) enclOp).operator).getAccessCode(enclOp.getTag());
|
||||||
else
|
else
|
||||||
return DEREFcode;
|
return AccessCode.DEREF.code;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Return binary operator that corresponds to given access code.
|
/** Return binary operator that corresponds to given access code.
|
||||||
*/
|
*/
|
||||||
private OperatorSymbol binaryAccessOperator(int acode) {
|
private OperatorSymbol binaryAccessOperator(int acode, Tag tag) {
|
||||||
return (OperatorSymbol)operators.lookupBinaryOp(sym -> accessCode(((OperatorSymbol)sym).opcode) == acode);
|
return operators.lookupBinaryOp(op -> op.getAccessCode(tag) == acode);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Return tree tag for assignment operation corresponding
|
/** Return tree tag for assignment operation corresponding
|
||||||
@ -984,7 +945,7 @@ public class Lower extends TreeTranslator {
|
|||||||
if (anum == null) {
|
if (anum == null) {
|
||||||
anum = accessed.length();
|
anum = accessed.length();
|
||||||
accessNums.put(vsym, anum);
|
accessNums.put(vsym, anum);
|
||||||
accessSyms.put(vsym, new MethodSymbol[NCODES]);
|
accessSyms.put(vsym, new MethodSymbol[AccessCode.numberOfAccessCodes]);
|
||||||
accessed.append(vsym);
|
accessed.append(vsym);
|
||||||
// System.out.println("accessing " + vsym + " in " + vsym.location());
|
// System.out.println("accessing " + vsym + " in " + vsym.location());
|
||||||
}
|
}
|
||||||
@ -996,13 +957,13 @@ public class Lower extends TreeTranslator {
|
|||||||
switch (vsym.kind) {
|
switch (vsym.kind) {
|
||||||
case VAR:
|
case VAR:
|
||||||
acode = accessCode(tree, enclOp);
|
acode = accessCode(tree, enclOp);
|
||||||
if (acode >= FIRSTASGOPcode) {
|
if (acode >= AccessCode.FIRSTASGOP.code) {
|
||||||
OperatorSymbol operator = binaryAccessOperator(acode);
|
OperatorSymbol operator = binaryAccessOperator(acode, enclOp.getTag());
|
||||||
if (operator.opcode == string_add)
|
if (operator.opcode == string_add)
|
||||||
argtypes = List.of(syms.objectType);
|
argtypes = List.of(syms.objectType);
|
||||||
else
|
else
|
||||||
argtypes = operator.type.getParameterTypes().tail;
|
argtypes = operator.type.getParameterTypes().tail;
|
||||||
} else if (acode == ASSIGNcode)
|
} else if (acode == AccessCode.ASSIGN.code)
|
||||||
argtypes = List.of(vsym.erasure(types));
|
argtypes = List.of(vsym.erasure(types));
|
||||||
else
|
else
|
||||||
argtypes = List.nil();
|
argtypes = List.nil();
|
||||||
@ -1010,7 +971,7 @@ public class Lower extends TreeTranslator {
|
|||||||
thrown = List.nil();
|
thrown = List.nil();
|
||||||
break;
|
break;
|
||||||
case MTH:
|
case MTH:
|
||||||
acode = DEREFcode;
|
acode = AccessCode.DEREF.code;
|
||||||
argtypes = vsym.erasure(types).getParameterTypes();
|
argtypes = vsym.erasure(types).getParameterTypes();
|
||||||
restype = vsym.erasure(types).getReturnType();
|
restype = vsym.erasure(types).getReturnType();
|
||||||
thrown = vsym.type.getThrownTypes();
|
thrown = vsym.type.getThrownTypes();
|
||||||
@ -1306,7 +1267,7 @@ public class Lower extends TreeTranslator {
|
|||||||
accessConstructorDef(cdef.pos, sym, accessConstrs.get(sym)));
|
accessConstructorDef(cdef.pos, sym, accessConstrs.get(sym)));
|
||||||
} else {
|
} else {
|
||||||
MethodSymbol[] accessors = accessSyms.get(sym);
|
MethodSymbol[] accessors = accessSyms.get(sym);
|
||||||
for (int i = 0; i < NCODES; i++) {
|
for (int i = 0; i < AccessCode.numberOfAccessCodes; i++) {
|
||||||
if (accessors[i] != null)
|
if (accessors[i] != null)
|
||||||
cdef.defs = cdef.defs.prepend(
|
cdef.defs = cdef.defs.prepend(
|
||||||
accessDef(cdef.pos, sym, accessors[i], i));
|
accessDef(cdef.pos, sym, accessors[i], i));
|
||||||
@ -1314,42 +1275,6 @@ public class Lower extends TreeTranslator {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Maps unary operator integer codes to JCTree.Tag objects
|
|
||||||
* @param unaryOpCode the unary operator code
|
|
||||||
*/
|
|
||||||
private static Tag mapUnaryOpCodeToTag(int unaryOpCode){
|
|
||||||
switch (unaryOpCode){
|
|
||||||
case PREINCcode:
|
|
||||||
return PREINC;
|
|
||||||
case PREDECcode:
|
|
||||||
return PREDEC;
|
|
||||||
case POSTINCcode:
|
|
||||||
return POSTINC;
|
|
||||||
case POSTDECcode:
|
|
||||||
return POSTDEC;
|
|
||||||
default:
|
|
||||||
return NO_TAG;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Maps JCTree.Tag objects to unary operator integer codes
|
|
||||||
* @param tag the JCTree.Tag
|
|
||||||
*/
|
|
||||||
private static int mapTagToUnaryOpCode(Tag tag){
|
|
||||||
switch (tag){
|
|
||||||
case PREINC:
|
|
||||||
return PREINCcode;
|
|
||||||
case PREDEC:
|
|
||||||
return PREDECcode;
|
|
||||||
case POSTINC:
|
|
||||||
return POSTINCcode;
|
|
||||||
case POSTDEC:
|
|
||||||
return POSTDECcode;
|
|
||||||
default:
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Construct definition of an access method.
|
/** Construct definition of an access method.
|
||||||
* @param pos The source code position of the definition.
|
* @param pos The source code position of the definition.
|
||||||
* @param vsym The private or protected symbol.
|
* @param vsym The private or protected symbol.
|
||||||
@ -1388,20 +1313,21 @@ public class Lower extends TreeTranslator {
|
|||||||
int acode1 = acode - (acode & 1);
|
int acode1 = acode - (acode & 1);
|
||||||
|
|
||||||
JCExpression expr; // The access method's return value.
|
JCExpression expr; // The access method's return value.
|
||||||
switch (acode1) {
|
AccessCode aCode = AccessCode.getFromCode(acode1);
|
||||||
case DEREFcode:
|
switch (aCode) {
|
||||||
|
case DEREF:
|
||||||
expr = ref;
|
expr = ref;
|
||||||
break;
|
break;
|
||||||
case ASSIGNcode:
|
case ASSIGN:
|
||||||
expr = make.Assign(ref, args.head);
|
expr = make.Assign(ref, args.head);
|
||||||
break;
|
break;
|
||||||
case PREINCcode: case POSTINCcode: case PREDECcode: case POSTDECcode:
|
case PREINC: case POSTINC: case PREDEC: case POSTDEC:
|
||||||
expr = makeUnary(mapUnaryOpCodeToTag(acode1), ref);
|
expr = makeUnary(aCode.tag, ref);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
expr = make.Assignop(
|
expr = make.Assignop(
|
||||||
treeTag(binaryAccessOperator(acode1)), ref, args.head);
|
treeTag(binaryAccessOperator(acode1, JCTree.Tag.NO_TAG)), ref, args.head);
|
||||||
((JCAssignOp) expr).operator = binaryAccessOperator(acode1);
|
((JCAssignOp) expr).operator = binaryAccessOperator(acode1, JCTree.Tag.NO_TAG);
|
||||||
}
|
}
|
||||||
stat = make.Return(expr.setType(sym.type));
|
stat = make.Return(expr.setType(sym.type));
|
||||||
} else {
|
} else {
|
||||||
@ -3275,7 +3201,7 @@ public class Lower extends TreeTranslator {
|
|||||||
// tree.lhs. However, we can still get the
|
// tree.lhs. However, we can still get the
|
||||||
// unerased type of tree.lhs as it is stored
|
// unerased type of tree.lhs as it is stored
|
||||||
// in tree.type in Attr.
|
// in tree.type in Attr.
|
||||||
Symbol newOperator = operators.resolveBinary(tree,
|
OperatorSymbol newOperator = operators.resolveBinary(tree,
|
||||||
newTag,
|
newTag,
|
||||||
tree.type,
|
tree.type,
|
||||||
tree.rhs.type);
|
tree.rhs.type);
|
||||||
@ -3304,7 +3230,7 @@ public class Lower extends TreeTranslator {
|
|||||||
JCMethodInvocation app = (JCMethodInvocation)tree.lhs;
|
JCMethodInvocation app = (JCMethodInvocation)tree.lhs;
|
||||||
// if operation is a += on strings,
|
// if operation is a += on strings,
|
||||||
// make sure to convert argument to string
|
// make sure to convert argument to string
|
||||||
JCExpression rhs = (((OperatorSymbol)tree.operator).opcode == string_add)
|
JCExpression rhs = tree.operator.opcode == string_add
|
||||||
? makeString(tree.rhs)
|
? makeString(tree.rhs)
|
||||||
: tree.rhs;
|
: tree.rhs;
|
||||||
app.args = List.of(rhs).prependList(app.args);
|
app.args = List.of(rhs).prependList(app.args);
|
||||||
|
@ -95,6 +95,7 @@ public class Operators {
|
|||||||
names = Names.instance(context);
|
names = Names.instance(context);
|
||||||
log = Log.instance(context);
|
log = Log.instance(context);
|
||||||
types = Types.instance(context);
|
types = Types.instance(context);
|
||||||
|
noOpSymbol = new OperatorSymbol(names.empty, Type.noType, -1, syms.noSymbol);
|
||||||
initOperatorNames();
|
initOperatorNames();
|
||||||
initUnaryOperators();
|
initUnaryOperators();
|
||||||
initBinaryOperators();
|
initBinaryOperators();
|
||||||
@ -145,7 +146,7 @@ public class Operators {
|
|||||||
/**
|
/**
|
||||||
* Entry point for resolving a unary operator given an operator tag and an argument type.
|
* Entry point for resolving a unary operator given an operator tag and an argument type.
|
||||||
*/
|
*/
|
||||||
Symbol resolveUnary(DiagnosticPosition pos, JCTree.Tag tag, Type op) {
|
OperatorSymbol resolveUnary(DiagnosticPosition pos, JCTree.Tag tag, Type op) {
|
||||||
return resolve(tag,
|
return resolve(tag,
|
||||||
unaryOperators,
|
unaryOperators,
|
||||||
unop -> unop.test(op),
|
unop -> unop.test(op),
|
||||||
@ -156,7 +157,7 @@ public class Operators {
|
|||||||
/**
|
/**
|
||||||
* Entry point for resolving a binary operator given an operator tag and a pair of argument types.
|
* Entry point for resolving a binary operator given an operator tag and a pair of argument types.
|
||||||
*/
|
*/
|
||||||
Symbol resolveBinary(DiagnosticPosition pos, JCTree.Tag tag, Type op1, Type op2) {
|
OperatorSymbol resolveBinary(DiagnosticPosition pos, JCTree.Tag tag, Type op1, Type op2) {
|
||||||
return resolve(tag,
|
return resolve(tag,
|
||||||
binaryOperators,
|
binaryOperators,
|
||||||
binop -> binop.test(op1, op2),
|
binop -> binop.test(op1, op2),
|
||||||
@ -169,8 +170,8 @@ public class Operators {
|
|||||||
* map. If there's a matching operator, its resolve routine is called and the result is returned;
|
* map. If there's a matching operator, its resolve routine is called and the result is returned;
|
||||||
* otherwise the result of a fallback function is returned.
|
* otherwise the result of a fallback function is returned.
|
||||||
*/
|
*/
|
||||||
private <O> Symbol resolve(Tag tag, Map<Name, List<O>> opMap, Predicate<O> opTestFunc,
|
private <O> OperatorSymbol resolve(Tag tag, Map<Name, List<O>> opMap, Predicate<O> opTestFunc,
|
||||||
Function<O, Symbol> resolveFunc, Supplier<Symbol> noResultFunc) {
|
Function<O, OperatorSymbol> resolveFunc, Supplier<OperatorSymbol> noResultFunc) {
|
||||||
return opMap.get(operatorName(tag)).stream()
|
return opMap.get(operatorName(tag)).stream()
|
||||||
.filter(opTestFunc)
|
.filter(opTestFunc)
|
||||||
.map(resolveFunc)
|
.map(resolveFunc)
|
||||||
@ -181,7 +182,7 @@ public class Operators {
|
|||||||
/**
|
/**
|
||||||
* Creates an operator symbol.
|
* Creates an operator symbol.
|
||||||
*/
|
*/
|
||||||
private Symbol makeOperator(Name name, List<OperatorType> formals, OperatorType res, int... opcodes) {
|
private OperatorSymbol makeOperator(Name name, List<OperatorType> formals, OperatorType res, int... opcodes) {
|
||||||
MethodType opType = new MethodType(
|
MethodType opType = new MethodType(
|
||||||
formals.stream()
|
formals.stream()
|
||||||
.map(o -> o.asType(syms))
|
.map(o -> o.asType(syms))
|
||||||
@ -201,10 +202,14 @@ public class Operators {
|
|||||||
((opcodes[0] << ByteCodes.preShift) | opcodes[1]);
|
((opcodes[0] << ByteCodes.preShift) | opcodes[1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** A symbol that stands for a missing operator.
|
||||||
|
*/
|
||||||
|
public final OperatorSymbol noOpSymbol;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Report an operator lookup error.
|
* Report an operator lookup error.
|
||||||
*/
|
*/
|
||||||
private Symbol reportErrorIfNeeded(DiagnosticPosition pos, Tag tag, Type... args) {
|
private OperatorSymbol reportErrorIfNeeded(DiagnosticPosition pos, Tag tag, Type... args) {
|
||||||
if (Stream.of(args).noneMatch(Type::isErroneous)) {
|
if (Stream.of(args).noneMatch(Type::isErroneous)) {
|
||||||
Name opName = operatorName(tag);
|
Name opName = operatorName(tag);
|
||||||
JCDiagnostic.Error opError = (args.length) == 1 ?
|
JCDiagnostic.Error opError = (args.length) == 1 ?
|
||||||
@ -212,7 +217,7 @@ public class Operators {
|
|||||||
Errors.OperatorCantBeApplied1(opName, args[0], args[1]);
|
Errors.OperatorCantBeApplied1(opName, args[0], args[1]);
|
||||||
log.error(pos, opError);
|
log.error(pos, opError);
|
||||||
}
|
}
|
||||||
return syms.noSymbol;
|
return noOpSymbol;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -263,10 +268,10 @@ public class Operators {
|
|||||||
final Name name;
|
final Name name;
|
||||||
|
|
||||||
/** The list of symbols associated with this operator (lazily populated). */
|
/** The list of symbols associated with this operator (lazily populated). */
|
||||||
Optional<Symbol[]> alternatives = Optional.empty();
|
Optional<OperatorSymbol[]> alternatives = Optional.empty();
|
||||||
|
|
||||||
/** An array of operator symbol suppliers (used to lazily populate the symbol list). */
|
/** An array of operator symbol suppliers (used to lazily populate the symbol list). */
|
||||||
List<Supplier<Symbol>> operatorSuppliers = List.nil();
|
List<Supplier<OperatorSymbol>> operatorSuppliers = List.nil();
|
||||||
|
|
||||||
@SuppressWarnings("varargs")
|
@SuppressWarnings("varargs")
|
||||||
OperatorHelper(Tag tag) {
|
OperatorHelper(Tag tag) {
|
||||||
@ -278,21 +283,21 @@ public class Operators {
|
|||||||
* using an applicability predicate; if the test suceeds that same operator is returned,
|
* using an applicability predicate; if the test suceeds that same operator is returned,
|
||||||
* otherwise a dummy symbol is returned.
|
* otherwise a dummy symbol is returned.
|
||||||
*/
|
*/
|
||||||
final Symbol doLookup(Predicate<Symbol> applicabilityTest) {
|
final OperatorSymbol doLookup(Predicate<OperatorSymbol> applicabilityTest) {
|
||||||
return Stream.of(alternatives.orElseGet(this::initOperators))
|
return Stream.of(alternatives.orElseGet(this::initOperators))
|
||||||
.filter(applicabilityTest)
|
.filter(applicabilityTest)
|
||||||
.findFirst()
|
.findFirst()
|
||||||
.orElse(syms.noSymbol);
|
.orElse(noOpSymbol);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This routine performs lazy instantiation of the operator symbols supported by this helper.
|
* This routine performs lazy instantiation of the operator symbols supported by this helper.
|
||||||
* After initialization is done, the suppliers are cleared, to free up memory.
|
* After initialization is done, the suppliers are cleared, to free up memory.
|
||||||
*/
|
*/
|
||||||
private Symbol[] initOperators() {
|
private OperatorSymbol[] initOperators() {
|
||||||
Symbol[] operators = operatorSuppliers.stream()
|
OperatorSymbol[] operators = operatorSuppliers.stream()
|
||||||
.map(op -> op.get())
|
.map(op -> op.get())
|
||||||
.toArray(Symbol[]::new);
|
.toArray(OperatorSymbol[]::new);
|
||||||
alternatives = Optional.of(operators);
|
alternatives = Optional.of(operators);
|
||||||
operatorSuppliers = null; //let GC do its work
|
operatorSuppliers = null; //let GC do its work
|
||||||
return operators;
|
return operators;
|
||||||
@ -311,10 +316,10 @@ public class Operators {
|
|||||||
/**
|
/**
|
||||||
* This routine implements the unary operator lookup process. It customizes the behavior
|
* This routine implements the unary operator lookup process. It customizes the behavior
|
||||||
* of the shared lookup routine in {@link OperatorHelper}, by using an unary applicability test
|
* of the shared lookup routine in {@link OperatorHelper}, by using an unary applicability test
|
||||||
* (see {@link UnaryOperatorHelper#isUnaryOperatorApplicable(OperatorSymbol, Type)}
|
* (see {@link UnaryOperatorHelper#isUnaryOperatorApplicable(OperatorOperatorSymbol, Type)}
|
||||||
*/
|
*/
|
||||||
final Symbol doLookup(Type t) {
|
final OperatorSymbol doLookup(Type t) {
|
||||||
return doLookup(op -> isUnaryOperatorApplicable((OperatorSymbol)op, t));
|
return doLookup(op -> isUnaryOperatorApplicable(op, t));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -336,7 +341,7 @@ public class Operators {
|
|||||||
* This method will be overridden by unary operator helpers to provide custom resolution
|
* This method will be overridden by unary operator helpers to provide custom resolution
|
||||||
* logic.
|
* logic.
|
||||||
*/
|
*/
|
||||||
abstract Symbol resolve(Type t);
|
abstract OperatorSymbol resolve(Type t);
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract class BinaryOperatorHelper extends OperatorHelper implements BiPredicate<Type, Type> {
|
abstract class BinaryOperatorHelper extends OperatorHelper implements BiPredicate<Type, Type> {
|
||||||
@ -350,8 +355,8 @@ public class Operators {
|
|||||||
* of the shared lookup routine in {@link OperatorHelper}, by using an unary applicability test
|
* of the shared lookup routine in {@link OperatorHelper}, by using an unary applicability test
|
||||||
* (see {@link BinaryOperatorHelper#isBinaryOperatorApplicable(OperatorSymbol, Type, Type)}
|
* (see {@link BinaryOperatorHelper#isBinaryOperatorApplicable(OperatorSymbol, Type, Type)}
|
||||||
*/
|
*/
|
||||||
final Symbol doLookup(Type t1, Type t2) {
|
final OperatorSymbol doLookup(Type t1, Type t2) {
|
||||||
return doLookup(op -> isBinaryOperatorApplicable((OperatorSymbol)op, t1, t2));
|
return doLookup(op -> isBinaryOperatorApplicable(op, t1, t2));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -375,7 +380,7 @@ public class Operators {
|
|||||||
* This method will be overridden by binary operator helpers to provide custom resolution
|
* This method will be overridden by binary operator helpers to provide custom resolution
|
||||||
* logic.
|
* logic.
|
||||||
*/
|
*/
|
||||||
abstract Symbol resolve(Type t1, Type t2);
|
abstract OperatorSymbol resolve(Type t1, Type t2);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -393,7 +398,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg) {
|
public OperatorSymbol resolve(Type arg) {
|
||||||
return doLookup(syms.objectType);
|
return doLookup(syms.objectType);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -421,7 +426,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg) {
|
public OperatorSymbol resolve(Type arg) {
|
||||||
return doLookup(unaryPromotion(arg));
|
return doLookup(unaryPromotion(arg));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -442,7 +447,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg) {
|
public OperatorSymbol resolve(Type arg) {
|
||||||
return doLookup(syms.booleanType);
|
return doLookup(syms.booleanType);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -458,7 +463,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg) {
|
public OperatorSymbol resolve(Type arg) {
|
||||||
return doLookup(types.unboxedTypeOrType(arg));
|
return doLookup(types.unboxedTypeOrType(arg));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -481,7 +486,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg1, Type arg2) {
|
public OperatorSymbol resolve(Type arg1, Type arg2) {
|
||||||
Type t = binaryPromotion(arg1, arg2);
|
Type t = binaryPromotion(arg1, arg2);
|
||||||
return doLookup(t, t);
|
return doLookup(t, t);
|
||||||
}
|
}
|
||||||
@ -504,7 +509,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg1, Type arg2) {
|
public OperatorSymbol resolve(Type arg1, Type arg2) {
|
||||||
return doLookup(syms.booleanType, syms.booleanType);
|
return doLookup(syms.booleanType, syms.booleanType);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -527,7 +532,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg1, Type arg2) {
|
public OperatorSymbol resolve(Type arg1, Type arg2) {
|
||||||
return doLookup(stringPromotion(arg1), stringPromotion(arg2));
|
return doLookup(stringPromotion(arg1), stringPromotion(arg2));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -570,7 +575,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type arg1, Type arg2) {
|
public OperatorSymbol resolve(Type arg1, Type arg2) {
|
||||||
return doLookup(unaryPromotion(arg1), unaryPromotion(arg2));
|
return doLookup(unaryPromotion(arg1), unaryPromotion(arg2));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -613,7 +618,7 @@ public class Operators {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Symbol resolve(Type t1, Type t2) {
|
public OperatorSymbol resolve(Type t1, Type t2) {
|
||||||
ComparisonKind kind = getKind(t1, t2);
|
ComparisonKind kind = getKind(t1, t2);
|
||||||
Type t = (kind == ComparisonKind.NUMERIC_OR_BOOLEAN) ?
|
Type t = (kind == ComparisonKind.NUMERIC_OR_BOOLEAN) ?
|
||||||
binaryPromotion(t1, t2) :
|
binaryPromotion(t1, t2) :
|
||||||
@ -798,12 +803,12 @@ public class Operators {
|
|||||||
.addBinaryOperator(BOOLEAN, BOOLEAN, BOOLEAN, bool_or));
|
.addBinaryOperator(BOOLEAN, BOOLEAN, BOOLEAN, bool_or));
|
||||||
}
|
}
|
||||||
|
|
||||||
Symbol lookupBinaryOp(Predicate<Symbol> applicabilityTest) {
|
OperatorSymbol lookupBinaryOp(Predicate<OperatorSymbol> applicabilityTest) {
|
||||||
return binaryOperators.values().stream()
|
return binaryOperators.values().stream()
|
||||||
.flatMap(List::stream)
|
.flatMap(List::stream)
|
||||||
.map(helper -> helper.doLookup(applicabilityTest))
|
.map(helper -> helper.doLookup(applicabilityTest))
|
||||||
.distinct()
|
.distinct()
|
||||||
.filter(sym -> sym != syms.noSymbol)
|
.filter(sym -> sym != noOpSymbol)
|
||||||
.findFirst().get();
|
.findFirst().get();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1795,7 +1795,7 @@ public class Gen extends JCTree.Visitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void visitAssignop(JCAssignOp tree) {
|
public void visitAssignop(JCAssignOp tree) {
|
||||||
OperatorSymbol operator = (OperatorSymbol) tree.operator;
|
OperatorSymbol operator = tree.operator;
|
||||||
Item l;
|
Item l;
|
||||||
if (operator.opcode == string_add) {
|
if (operator.opcode == string_add) {
|
||||||
l = concat.makeConcat(tree);
|
l = concat.makeConcat(tree);
|
||||||
@ -1827,7 +1827,7 @@ public class Gen extends JCTree.Visitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void visitUnary(JCUnary tree) {
|
public void visitUnary(JCUnary tree) {
|
||||||
OperatorSymbol operator = (OperatorSymbol)tree.operator;
|
OperatorSymbol operator = tree.operator;
|
||||||
if (tree.hasTag(NOT)) {
|
if (tree.hasTag(NOT)) {
|
||||||
CondItem od = genCond(tree.arg, false);
|
CondItem od = genCond(tree.arg, false);
|
||||||
result = od.negate();
|
result = od.negate();
|
||||||
@ -1909,7 +1909,7 @@ public class Gen extends JCTree.Visitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void visitBinary(JCBinary tree) {
|
public void visitBinary(JCBinary tree) {
|
||||||
OperatorSymbol operator = (OperatorSymbol)tree.operator;
|
OperatorSymbol operator = tree.operator;
|
||||||
if (operator.opcode == string_add) {
|
if (operator.opcode == string_add) {
|
||||||
result = concat.makeConcat(tree);
|
result = concat.makeConcat(tree);
|
||||||
} else if (tree.hasTag(AND)) {
|
} else if (tree.hasTag(AND)) {
|
||||||
|
@ -131,8 +131,7 @@ public abstract class StringConcat {
|
|||||||
tree = TreeInfo.skipParens(tree);
|
tree = TreeInfo.skipParens(tree);
|
||||||
if (tree.hasTag(PLUS) && tree.type.constValue() == null) {
|
if (tree.hasTag(PLUS) && tree.type.constValue() == null) {
|
||||||
JCTree.JCBinary op = (JCTree.JCBinary) tree;
|
JCTree.JCBinary op = (JCTree.JCBinary) tree;
|
||||||
if (op.operator.kind == MTH &&
|
if (op.operator.kind == MTH && op.operator.opcode == string_add) {
|
||||||
((Symbol.OperatorSymbol) op.operator).opcode == string_add) {
|
|
||||||
return res
|
return res
|
||||||
.appendList(collect(op.lhs, res))
|
.appendList(collect(op.lhs, res))
|
||||||
.appendList(collect(op.rhs, res));
|
.appendList(collect(op.rhs, res));
|
||||||
|
@ -1876,8 +1876,8 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
private Tag opcode;
|
private Tag opcode;
|
||||||
public JCExpression lhs;
|
public JCExpression lhs;
|
||||||
public JCExpression rhs;
|
public JCExpression rhs;
|
||||||
public Symbol operator;
|
public OperatorSymbol operator;
|
||||||
protected JCAssignOp(Tag opcode, JCTree lhs, JCTree rhs, Symbol operator) {
|
protected JCAssignOp(Tag opcode, JCTree lhs, JCTree rhs, OperatorSymbol operator) {
|
||||||
this.opcode = opcode;
|
this.opcode = opcode;
|
||||||
this.lhs = (JCExpression)lhs;
|
this.lhs = (JCExpression)lhs;
|
||||||
this.rhs = (JCExpression)rhs;
|
this.rhs = (JCExpression)rhs;
|
||||||
@ -1892,7 +1892,7 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
public JCExpression getVariable() { return lhs; }
|
public JCExpression getVariable() { return lhs; }
|
||||||
@DefinedBy(Api.COMPILER_TREE)
|
@DefinedBy(Api.COMPILER_TREE)
|
||||||
public JCExpression getExpression() { return rhs; }
|
public JCExpression getExpression() { return rhs; }
|
||||||
public Symbol getOperator() {
|
public OperatorSymbol getOperator() {
|
||||||
return operator;
|
return operator;
|
||||||
}
|
}
|
||||||
@Override @DefinedBy(Api.COMPILER_TREE)
|
@Override @DefinedBy(Api.COMPILER_TREE)
|
||||||
@ -1911,7 +1911,7 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
public static class JCUnary extends JCExpression implements UnaryTree {
|
public static class JCUnary extends JCExpression implements UnaryTree {
|
||||||
private Tag opcode;
|
private Tag opcode;
|
||||||
public JCExpression arg;
|
public JCExpression arg;
|
||||||
public Symbol operator;
|
public OperatorSymbol operator;
|
||||||
protected JCUnary(Tag opcode, JCExpression arg) {
|
protected JCUnary(Tag opcode, JCExpression arg) {
|
||||||
this.opcode = opcode;
|
this.opcode = opcode;
|
||||||
this.arg = arg;
|
this.arg = arg;
|
||||||
@ -1923,7 +1923,7 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
public Kind getKind() { return TreeInfo.tagToKind(getTag()); }
|
public Kind getKind() { return TreeInfo.tagToKind(getTag()); }
|
||||||
@DefinedBy(Api.COMPILER_TREE)
|
@DefinedBy(Api.COMPILER_TREE)
|
||||||
public JCExpression getExpression() { return arg; }
|
public JCExpression getExpression() { return arg; }
|
||||||
public Symbol getOperator() {
|
public OperatorSymbol getOperator() {
|
||||||
return operator;
|
return operator;
|
||||||
}
|
}
|
||||||
@Override @DefinedBy(Api.COMPILER_TREE)
|
@Override @DefinedBy(Api.COMPILER_TREE)
|
||||||
@ -1947,11 +1947,11 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
private Tag opcode;
|
private Tag opcode;
|
||||||
public JCExpression lhs;
|
public JCExpression lhs;
|
||||||
public JCExpression rhs;
|
public JCExpression rhs;
|
||||||
public Symbol operator;
|
public OperatorSymbol operator;
|
||||||
protected JCBinary(Tag opcode,
|
protected JCBinary(Tag opcode,
|
||||||
JCExpression lhs,
|
JCExpression lhs,
|
||||||
JCExpression rhs,
|
JCExpression rhs,
|
||||||
Symbol operator) {
|
OperatorSymbol operator) {
|
||||||
this.opcode = opcode;
|
this.opcode = opcode;
|
||||||
this.lhs = lhs;
|
this.lhs = lhs;
|
||||||
this.rhs = rhs;
|
this.rhs = rhs;
|
||||||
@ -1966,7 +1966,7 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
public JCExpression getLeftOperand() { return lhs; }
|
public JCExpression getLeftOperand() { return lhs; }
|
||||||
@DefinedBy(Api.COMPILER_TREE)
|
@DefinedBy(Api.COMPILER_TREE)
|
||||||
public JCExpression getRightOperand() { return rhs; }
|
public JCExpression getRightOperand() { return rhs; }
|
||||||
public Symbol getOperator() {
|
public OperatorSymbol getOperator() {
|
||||||
return operator;
|
return operator;
|
||||||
}
|
}
|
||||||
@Override @DefinedBy(Api.COMPILER_TREE)
|
@Override @DefinedBy(Api.COMPILER_TREE)
|
||||||
|
Loading…
Reference in New Issue
Block a user