8033004: Make base TypeAnnotationPosition data immutable, create better methods for creating base TypeAnnotationPositions
First of a series of major fixes to type annotations; cleans up interface with TypeAnnotationPosition Reviewed-by: jjg
This commit is contained in:
parent
4724c454cd
commit
3bd9bdd993
@ -425,11 +425,15 @@ public class SymbolMetadata {
|
|||||||
private final List<T> placeholderFor;
|
private final List<T> placeholderFor;
|
||||||
private final Symbol on;
|
private final Symbol on;
|
||||||
|
|
||||||
public Placeholder(Annotate.AnnotateRepeatedContext<T> ctx, List<T> placeholderFor, Symbol on) {
|
public Placeholder(Annotate.AnnotateRepeatedContext<T> ctx,
|
||||||
|
List<T> placeholderFor, Symbol on) {
|
||||||
super(on.type, List.<Pair<Symbol.MethodSymbol, Attribute>>nil(),
|
super(on.type, List.<Pair<Symbol.MethodSymbol, Attribute>>nil(),
|
||||||
ctx.isTypeCompound ?
|
ctx.isTypeCompound ?
|
||||||
((Attribute.TypeCompound)placeholderFor.head).position :
|
((Attribute.TypeCompound)placeholderFor.head).position :
|
||||||
new TypeAnnotationPosition());
|
// TODO: Eventually, we will need to get rid of this
|
||||||
|
// use of unknown, either by using null, or by
|
||||||
|
// throwing an assertion failure here.
|
||||||
|
TypeAnnotationPosition.unknown);
|
||||||
this.ctx = ctx;
|
this.ctx = ctx;
|
||||||
this.placeholderFor = placeholderFor;
|
this.placeholderFor = placeholderFor;
|
||||||
this.on = on;
|
this.on = on;
|
||||||
|
@ -119,13 +119,20 @@ public class TypeAnnotationPosition {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public TargetType type = TargetType.UNKNOWN;
|
public static final List<TypePathEntry> emptyPath = List.nil();
|
||||||
|
|
||||||
|
// NOTE: All of these will be converted to final fields eventually.
|
||||||
|
|
||||||
|
public final TargetType type;
|
||||||
|
|
||||||
// For generic/array types.
|
// For generic/array types.
|
||||||
public List<TypePathEntry> location = List.nil();
|
|
||||||
|
// This field is in the process of being made final. Do not
|
||||||
|
// introduce new mutations.
|
||||||
|
public List<TypePathEntry> location;
|
||||||
|
|
||||||
// Tree position.
|
// Tree position.
|
||||||
public int pos = -1;
|
public final int pos;
|
||||||
|
|
||||||
// For type casts, type tests, new, locals (as start_pc),
|
// For type casts, type tests, new, locals (as start_pc),
|
||||||
// and method and constructor reference type arguments.
|
// and method and constructor reference type arguments.
|
||||||
@ -138,13 +145,17 @@ public class TypeAnnotationPosition {
|
|||||||
public int[] lvarIndex = null;
|
public int[] lvarIndex = null;
|
||||||
|
|
||||||
// For type parameter bound
|
// For type parameter bound
|
||||||
public int bound_index = Integer.MIN_VALUE;
|
public final int bound_index;
|
||||||
|
|
||||||
// For type parameter and method parameter
|
// For type parameter and method parameter
|
||||||
public int parameter_index = Integer.MIN_VALUE;
|
public final int parameter_index;
|
||||||
|
|
||||||
// For class extends, implements, and throws clauses
|
// For class extends, implements, and throws clauses
|
||||||
public int type_index = Integer.MIN_VALUE;
|
|
||||||
|
// This field is effectively final. However, it needs to be
|
||||||
|
// modified by Gen for the time being. Do not introduce new
|
||||||
|
// mutations.
|
||||||
|
public int type_index;
|
||||||
|
|
||||||
// For exception parameters, index into exception table.
|
// For exception parameters, index into exception table.
|
||||||
// In com.sun.tools.javac.jvm.Gen.genCatch we first set the type_index
|
// In com.sun.tools.javac.jvm.Gen.genCatch we first set the type_index
|
||||||
@ -156,9 +167,10 @@ public class TypeAnnotationPosition {
|
|||||||
// If this type annotation is within a lambda expression,
|
// If this type annotation is within a lambda expression,
|
||||||
// store a pointer to the lambda expression tree in order
|
// store a pointer to the lambda expression tree in order
|
||||||
// to allow a later translation to the right method.
|
// to allow a later translation to the right method.
|
||||||
public JCLambda onLambda = null;
|
public final JCLambda onLambda;
|
||||||
|
|
||||||
public TypeAnnotationPosition() {}
|
// NOTE: This constructor will eventually go away, and be replaced
|
||||||
|
// by static builder methods.
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
@ -323,4 +335,847 @@ public class TypeAnnotationPosition {
|
|||||||
}
|
}
|
||||||
return loc.toList();
|
return loc.toList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// These methods are the new preferred way to create
|
||||||
|
// TypeAnnotationPositions
|
||||||
|
|
||||||
|
// Never make this a public constructor without creating a builder.
|
||||||
|
private TypeAnnotationPosition(final TargetType ttype,
|
||||||
|
final int pos,
|
||||||
|
final int parameter_index,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int bound_index,
|
||||||
|
final List<TypePathEntry> location) {
|
||||||
|
Assert.checkNonNull(location);
|
||||||
|
this.type = ttype;
|
||||||
|
this.pos = pos;
|
||||||
|
this.parameter_index = parameter_index;
|
||||||
|
this.onLambda = onLambda;
|
||||||
|
this.type_index = type_index;
|
||||||
|
this.bound_index = bound_index;
|
||||||
|
this.location = location;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method return.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodReturn(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_RETURN, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method return.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodReturn(final List<TypePathEntry> location) {
|
||||||
|
return methodReturn(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method return.
|
||||||
|
*
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition methodReturn(final int pos) {
|
||||||
|
return methodReturn(emptyPath, null, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method receiver.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodReceiver(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_RECEIVER, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method receiver.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodReceiver(final List<TypePathEntry> location) {
|
||||||
|
return methodReceiver(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method receiver.
|
||||||
|
*
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition methodReceiver(final int pos) {
|
||||||
|
return methodReceiver(emptyPath, null, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method formal parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this parameter.
|
||||||
|
* @param index The index of the parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodParameter(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int parameter_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_FORMAL_PARAMETER,
|
||||||
|
pos, parameter_index, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method formal parameter.
|
||||||
|
*
|
||||||
|
* @param onLambda The lambda for this parameter.
|
||||||
|
* @param index The index of the parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodParameter(final JCLambda onLambda,
|
||||||
|
final int parameter_index,
|
||||||
|
final int pos) {
|
||||||
|
return methodParameter(emptyPath, onLambda,
|
||||||
|
parameter_index, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method formal parameter.
|
||||||
|
*
|
||||||
|
* @param index The index of the parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodParameter(final int parameter_index,
|
||||||
|
final int pos) {
|
||||||
|
return methodParameter(null, parameter_index, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method formal parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param index The index of the parameter.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodParameter(final List<TypePathEntry> location,
|
||||||
|
final int parameter_index) {
|
||||||
|
return methodParameter(location, null, parameter_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method reference.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this method reference.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodRef(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_REFERENCE, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method reference.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this method reference.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodRef(final List<TypePathEntry> location) {
|
||||||
|
return methodRef(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a constructor reference.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this constructor reference.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
constructorRef(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CONSTRUCTOR_REFERENCE, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a constructor reference.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this constructor reference.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
constructorRef(final List<TypePathEntry> location) {
|
||||||
|
return constructorRef(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a field.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
field(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.FIELD, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a field.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
field(final List<TypePathEntry> location) {
|
||||||
|
return field(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a field.
|
||||||
|
*
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition field(final int pos) {
|
||||||
|
return field(emptyPath, null, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a local variable.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
localVariable(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.LOCAL_VARIABLE, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a local variable.
|
||||||
|
*
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
localVariable(final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return localVariable(emptyPath, onLambda, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a local variable.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
localVariable(final List<TypePathEntry> location) {
|
||||||
|
return localVariable(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for an exception parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
exceptionParameter(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.EXCEPTION_PARAMETER, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for an exception parameter.
|
||||||
|
*
|
||||||
|
* @param onLambda The lambda for this parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
exceptionParameter(final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return exceptionParameter(emptyPath, onLambda, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for an exception parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
exceptionParameter(final List<TypePathEntry> location) {
|
||||||
|
return exceptionParameter(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a resource variable.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
resourceVariable(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.RESOURCE_VARIABLE, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a resource variable.
|
||||||
|
*
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
resourceVariable(final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return resourceVariable(emptyPath, onLambda, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a resource variable.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
resourceVariable(final List<TypePathEntry> location) {
|
||||||
|
return resourceVariable(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a new.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
newObj(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.NEW, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a new.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition newObj(final int pos) {
|
||||||
|
return newObj(emptyPath, null, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a new.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
newObj(final List<TypePathEntry> location) {
|
||||||
|
return newObj(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a class extension.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index of the interface.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
classExtends(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CLASS_EXTENDS, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a class extension.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index of the interface.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
classExtends(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return classExtends(location, onLambda, -1, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a class extension.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param type_index The index of the interface.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
classExtends(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return classExtends(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a class extension.
|
||||||
|
*
|
||||||
|
* @param type_index The index of the interface.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition classExtends(final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return classExtends(emptyPath, null, type_index, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a class extension.
|
||||||
|
*
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition classExtends(final int pos) {
|
||||||
|
return classExtends(-1, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for an instanceof.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
instanceOf(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.INSTANCEOF, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for an instanceof.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
instanceOf(final List<TypePathEntry> location) {
|
||||||
|
return instanceOf(location, null, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a type cast.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index into an intersection type.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
typeCast(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CAST, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a type cast.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index into an intersection type.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
typeCast(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return typeCast(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method
|
||||||
|
* invocation type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index of the type argument.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodInvocationTypeArg(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_INVOCATION_TYPE_ARGUMENT,
|
||||||
|
pos, Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method
|
||||||
|
* invocation type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param type_index The index of the type argument.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodInvocationTypeArg(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return methodInvocationTypeArg(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a constructor
|
||||||
|
* invocation type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index of the type argument.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
constructorInvocationTypeArg(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CONSTRUCTOR_INVOCATION_TYPE_ARGUMENT,
|
||||||
|
pos, Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a constructor
|
||||||
|
* invocation type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param type_index The index of the type argument.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
constructorInvocationTypeArg(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return constructorInvocationTypeArg(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a type parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
typeParameter(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int parameter_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CLASS_TYPE_PARAMETER, pos,
|
||||||
|
parameter_index, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a type parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
typeParameter(final List<TypePathEntry> location,
|
||||||
|
final int parameter_index) {
|
||||||
|
return typeParameter(location, null, parameter_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method type parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodTypeParameter(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int parameter_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_TYPE_PARAMETER,
|
||||||
|
pos, parameter_index, onLambda,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method type parameter.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodTypeParameter(final List<TypePathEntry> location,
|
||||||
|
final int parameter_index) {
|
||||||
|
return methodTypeParameter(location, null, parameter_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a throws clause.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param type_index The index of the exception.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodThrows(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.THROWS, pos,
|
||||||
|
Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a throws clause.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param type_index The index of the exception.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodThrows(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return methodThrows(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method reference
|
||||||
|
* type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type argument.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodRefTypeArg(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_REFERENCE_TYPE_ARGUMENT,
|
||||||
|
pos, Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method reference
|
||||||
|
* type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type argument.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodRefTypeArg(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return methodRefTypeArg(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a constructor reference
|
||||||
|
* type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type argument.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
constructorRefTypeArg(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int type_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT,
|
||||||
|
pos, Integer.MIN_VALUE, onLambda,
|
||||||
|
type_index, Integer.MIN_VALUE,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a constructor reference
|
||||||
|
* type argument.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param parameter_index The index of the type argument.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
constructorRefTypeArg(final List<TypePathEntry> location,
|
||||||
|
final int type_index) {
|
||||||
|
return constructorRefTypeArg(location, null, type_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a type parameter bound.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param bound_index The index of the type parameter bound.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
typeParameterBound(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int parameter_index,
|
||||||
|
final int bound_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.CLASS_TYPE_PARAMETER_BOUND,
|
||||||
|
pos, parameter_index, onLambda,
|
||||||
|
Integer.MIN_VALUE, bound_index,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a type parameter bound.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param bound_index The index of the type parameter bound.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
typeParameterBound(final List<TypePathEntry> location,
|
||||||
|
final int parameter_index,
|
||||||
|
final int bound_index) {
|
||||||
|
return typeParameterBound(location, null, parameter_index,
|
||||||
|
bound_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method type
|
||||||
|
* parameter bound.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param onLambda The lambda for this variable.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param bound_index The index of the type parameter bound.
|
||||||
|
* @param pos The position from the associated tree node.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodTypeParameterBound(final List<TypePathEntry> location,
|
||||||
|
final JCLambda onLambda,
|
||||||
|
final int parameter_index,
|
||||||
|
final int bound_index,
|
||||||
|
final int pos) {
|
||||||
|
return new TypeAnnotationPosition(TargetType.METHOD_TYPE_PARAMETER_BOUND,
|
||||||
|
pos, parameter_index, onLambda,
|
||||||
|
Integer.MIN_VALUE, bound_index,
|
||||||
|
location);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a {@code TypeAnnotationPosition} for a method type
|
||||||
|
* parameter bound.
|
||||||
|
*
|
||||||
|
* @param location The type path.
|
||||||
|
* @param parameter_index The index of the type parameter.
|
||||||
|
* @param bound_index The index of the type parameter bound.
|
||||||
|
*/
|
||||||
|
public static TypeAnnotationPosition
|
||||||
|
methodTypeParameterBound(final List<TypePathEntry> location,
|
||||||
|
final int parameter_index,
|
||||||
|
final int bound_index) {
|
||||||
|
return methodTypeParameterBound(location, null, parameter_index,
|
||||||
|
bound_index, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Consider this deprecated on arrival. We eventually want to get
|
||||||
|
// rid of this value altogether. Do not use it for anything new.
|
||||||
|
public static final TypeAnnotationPosition unknown =
|
||||||
|
new TypeAnnotationPosition(TargetType.UNKNOWN, -1,
|
||||||
|
Integer.MIN_VALUE, null,
|
||||||
|
Integer.MIN_VALUE, Integer.MIN_VALUE,
|
||||||
|
emptyPath);
|
||||||
}
|
}
|
||||||
|
@ -663,8 +663,15 @@ public class TypeAnnotations {
|
|||||||
* type annotations: determine the type annotation positions.
|
* type annotations: determine the type annotation positions.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
private void resolveFrame(JCTree tree, JCTree frame,
|
// This method is considered deprecated, and will be removed
|
||||||
List<JCTree> path, TypeAnnotationPosition p) {
|
// in the near future. Don't use it for anything new.
|
||||||
|
private TypeAnnotationPosition
|
||||||
|
resolveFrame(JCTree tree,
|
||||||
|
JCTree frame,
|
||||||
|
List<JCTree> path,
|
||||||
|
JCLambda currentLambda,
|
||||||
|
int outer_type_index,
|
||||||
|
ListBuffer<TypePathEntry> location) {
|
||||||
/*
|
/*
|
||||||
System.out.println("Resolving tree: " + tree + " kind: " + tree.getKind());
|
System.out.println("Resolving tree: " + tree + " kind: " + tree.getKind());
|
||||||
System.out.println(" Framing tree: " + frame + " kind: " + frame.getKind());
|
System.out.println(" Framing tree: " + frame + " kind: " + frame.getKind());
|
||||||
@ -675,87 +682,101 @@ public class TypeAnnotations {
|
|||||||
|
|
||||||
switch (frame.getKind()) {
|
switch (frame.getKind()) {
|
||||||
case TYPE_CAST:
|
case TYPE_CAST:
|
||||||
JCTypeCast frameTC = (JCTypeCast) frame;
|
return TypeAnnotationPosition.typeCast(location.toList(),
|
||||||
p.type = TargetType.CAST;
|
currentLambda,
|
||||||
if (frameTC.clazz.hasTag(Tag.TYPEINTERSECTION)) {
|
outer_type_index,
|
||||||
// This case was already handled by INTERSECTION_TYPE
|
frame.pos);
|
||||||
} else {
|
|
||||||
p.type_index = 0;
|
|
||||||
}
|
|
||||||
p.pos = frame.pos;
|
|
||||||
return;
|
|
||||||
|
|
||||||
case INSTANCE_OF:
|
case INSTANCE_OF:
|
||||||
p.type = TargetType.INSTANCEOF;
|
return TypeAnnotationPosition.instanceOf(location.toList(),
|
||||||
p.pos = frame.pos;
|
currentLambda,
|
||||||
return;
|
frame.pos);
|
||||||
|
|
||||||
case NEW_CLASS:
|
case NEW_CLASS:
|
||||||
JCNewClass frameNewClass = (JCNewClass) frame;
|
final JCNewClass frameNewClass = (JCNewClass) frame;
|
||||||
if (frameNewClass.def != null) {
|
if (frameNewClass.def != null) {
|
||||||
// Special handling for anonymous class instantiations
|
// Special handling for anonymous class instantiations
|
||||||
JCClassDecl frameClassDecl = frameNewClass.def;
|
final JCClassDecl frameClassDecl = frameNewClass.def;
|
||||||
if (frameClassDecl.extending == tree) {
|
if (frameClassDecl.extending == tree) {
|
||||||
p.type = TargetType.CLASS_EXTENDS;
|
return TypeAnnotationPosition
|
||||||
p.type_index = -1;
|
.classExtends(location.toList(), currentLambda,
|
||||||
|
frame.pos);
|
||||||
} else if (frameClassDecl.implementing.contains(tree)) {
|
} else if (frameClassDecl.implementing.contains(tree)) {
|
||||||
p.type = TargetType.CLASS_EXTENDS;
|
final int type_index =
|
||||||
p.type_index = frameClassDecl.implementing.indexOf(tree);
|
frameClassDecl.implementing.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.classExtends(location.toList(), currentLambda,
|
||||||
|
type_index, frame.pos);
|
||||||
} else {
|
} else {
|
||||||
// In contrast to CLASS below, typarams cannot occur here.
|
// In contrast to CLASS below, typarams cannot occur here.
|
||||||
Assert.error("Could not determine position of tree " + tree +
|
throw new AssertionError("Could not determine position of tree " + tree +
|
||||||
" within frame " + frame);
|
" within frame " + frame);
|
||||||
}
|
}
|
||||||
} else if (frameNewClass.typeargs.contains(tree)) {
|
} else if (frameNewClass.typeargs.contains(tree)) {
|
||||||
p.type = TargetType.CONSTRUCTOR_INVOCATION_TYPE_ARGUMENT;
|
final int type_index =
|
||||||
p.type_index = frameNewClass.typeargs.indexOf(tree);
|
frameNewClass.typeargs.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.constructorInvocationTypeArg(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
type_index,
|
||||||
|
frame.pos);
|
||||||
} else {
|
} else {
|
||||||
p.type = TargetType.NEW;
|
return TypeAnnotationPosition
|
||||||
|
.newObj(location.toList(), currentLambda,
|
||||||
|
frame.pos);
|
||||||
}
|
}
|
||||||
p.pos = frame.pos;
|
|
||||||
return;
|
|
||||||
|
|
||||||
case NEW_ARRAY:
|
case NEW_ARRAY:
|
||||||
p.type = TargetType.NEW;
|
return TypeAnnotationPosition
|
||||||
p.pos = frame.pos;
|
.newObj(location.toList(), currentLambda, frame.pos);
|
||||||
return;
|
|
||||||
|
|
||||||
case ANNOTATION_TYPE:
|
case ANNOTATION_TYPE:
|
||||||
case CLASS:
|
case CLASS:
|
||||||
case ENUM:
|
case ENUM:
|
||||||
case INTERFACE:
|
case INTERFACE:
|
||||||
p.pos = frame.pos;
|
|
||||||
if (((JCClassDecl)frame).extending == tree) {
|
if (((JCClassDecl)frame).extending == tree) {
|
||||||
p.type = TargetType.CLASS_EXTENDS;
|
return TypeAnnotationPosition
|
||||||
p.type_index = -1;
|
.classExtends(location.toList(), currentLambda,
|
||||||
|
frame.pos);
|
||||||
} else if (((JCClassDecl)frame).implementing.contains(tree)) {
|
} else if (((JCClassDecl)frame).implementing.contains(tree)) {
|
||||||
p.type = TargetType.CLASS_EXTENDS;
|
final int type_index =
|
||||||
p.type_index = ((JCClassDecl)frame).implementing.indexOf(tree);
|
((JCClassDecl)frame).implementing.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.classExtends(location.toList(), currentLambda,
|
||||||
|
type_index, frame.pos);
|
||||||
} else if (((JCClassDecl)frame).typarams.contains(tree)) {
|
} else if (((JCClassDecl)frame).typarams.contains(tree)) {
|
||||||
p.type = TargetType.CLASS_TYPE_PARAMETER;
|
final int parameter_index =
|
||||||
p.parameter_index = ((JCClassDecl)frame).typarams.indexOf(tree);
|
((JCClassDecl)frame).typarams.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.typeParameter(location.toList(), currentLambda,
|
||||||
|
parameter_index, frame.pos);
|
||||||
} else {
|
} else {
|
||||||
Assert.error("Could not determine position of tree " + tree +
|
throw new AssertionError("Could not determine position of tree " +
|
||||||
" within frame " + frame);
|
tree + " within frame " + frame);
|
||||||
}
|
}
|
||||||
return;
|
|
||||||
|
|
||||||
case METHOD: {
|
case METHOD: {
|
||||||
JCMethodDecl frameMethod = (JCMethodDecl) frame;
|
final JCMethodDecl frameMethod = (JCMethodDecl) frame;
|
||||||
p.pos = frame.pos;
|
|
||||||
if (frameMethod.thrown.contains(tree)) {
|
if (frameMethod.thrown.contains(tree)) {
|
||||||
p.type = TargetType.THROWS;
|
final int type_index = frameMethod.thrown.indexOf(tree);
|
||||||
p.type_index = frameMethod.thrown.indexOf(tree);
|
return TypeAnnotationPosition
|
||||||
|
.methodThrows(location.toList(), currentLambda,
|
||||||
|
type_index, frame.pos);
|
||||||
} else if (frameMethod.restype == tree) {
|
} else if (frameMethod.restype == tree) {
|
||||||
p.type = TargetType.METHOD_RETURN;
|
return TypeAnnotationPosition
|
||||||
|
.methodReturn(location.toList(), currentLambda,
|
||||||
|
frame.pos);
|
||||||
} else if (frameMethod.typarams.contains(tree)) {
|
} else if (frameMethod.typarams.contains(tree)) {
|
||||||
p.type = TargetType.METHOD_TYPE_PARAMETER;
|
final int parameter_index =
|
||||||
p.parameter_index = frameMethod.typarams.indexOf(tree);
|
frameMethod.typarams.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.methodTypeParameter(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
parameter_index, frame.pos);
|
||||||
} else {
|
} else {
|
||||||
Assert.error("Could not determine position of tree " + tree +
|
throw new AssertionError("Could not determine position of tree " + tree +
|
||||||
" within frame " + frame);
|
" within frame " + frame);
|
||||||
}
|
}
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case PARAMETERIZED_TYPE: {
|
case PARAMETERIZED_TYPE: {
|
||||||
@ -766,25 +787,30 @@ public class TypeAnnotations {
|
|||||||
} else if (((JCTypeApply)frame).arguments.contains(tree)) {
|
} else if (((JCTypeApply)frame).arguments.contains(tree)) {
|
||||||
JCTypeApply taframe = (JCTypeApply) frame;
|
JCTypeApply taframe = (JCTypeApply) frame;
|
||||||
int arg = taframe.arguments.indexOf(tree);
|
int arg = taframe.arguments.indexOf(tree);
|
||||||
p.location = p.location.prepend(new TypePathEntry(TypePathEntryKind.TYPE_ARGUMENT, arg));
|
location = location.prepend(
|
||||||
|
new TypePathEntry(TypePathEntryKind.TYPE_ARGUMENT,
|
||||||
|
arg));
|
||||||
|
|
||||||
Type typeToUse;
|
Type typeToUse;
|
||||||
if (newPath.tail != null && newPath.tail.head.hasTag(Tag.NEWCLASS)) {
|
if (newPath.tail != null &&
|
||||||
// If we are within an anonymous class instantiation, use its type,
|
newPath.tail.head.hasTag(Tag.NEWCLASS)) {
|
||||||
// because it contains a correctly nested type.
|
// If we are within an anonymous class
|
||||||
|
// instantiation, use its type, because it
|
||||||
|
// contains a correctly nested type.
|
||||||
typeToUse = newPath.tail.head.type;
|
typeToUse = newPath.tail.head.type;
|
||||||
} else {
|
} else {
|
||||||
typeToUse = taframe.type;
|
typeToUse = taframe.type;
|
||||||
}
|
}
|
||||||
|
|
||||||
locateNestedTypes(typeToUse, p);
|
location = locateNestedTypes(typeToUse, location);
|
||||||
} else {
|
} else {
|
||||||
Assert.error("Could not determine type argument position of tree " + tree +
|
throw new AssertionError("Could not determine type argument position of tree " + tree +
|
||||||
" within frame " + frame);
|
" within frame " + frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
return;
|
newPath, currentLambda,
|
||||||
|
outer_type_index, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
case MEMBER_REFERENCE: {
|
case MEMBER_REFERENCE: {
|
||||||
@ -793,117 +819,140 @@ public class TypeAnnotations {
|
|||||||
if (mrframe.expr == tree) {
|
if (mrframe.expr == tree) {
|
||||||
switch (mrframe.mode) {
|
switch (mrframe.mode) {
|
||||||
case INVOKE:
|
case INVOKE:
|
||||||
p.type = TargetType.METHOD_REFERENCE;
|
return TypeAnnotationPosition
|
||||||
break;
|
.methodRef(location.toList(), currentLambda,
|
||||||
|
frame.pos);
|
||||||
case NEW:
|
case NEW:
|
||||||
p.type = TargetType.CONSTRUCTOR_REFERENCE;
|
return TypeAnnotationPosition
|
||||||
break;
|
.constructorRef(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
frame.pos);
|
||||||
default:
|
default:
|
||||||
Assert.error("Unknown method reference mode " + mrframe.mode +
|
throw new AssertionError("Unknown method reference mode " + mrframe.mode +
|
||||||
" for tree " + tree + " within frame " + frame);
|
" for tree " + tree + " within frame " + frame);
|
||||||
}
|
}
|
||||||
p.pos = frame.pos;
|
|
||||||
} else if (mrframe.typeargs != null &&
|
} else if (mrframe.typeargs != null &&
|
||||||
mrframe.typeargs.contains(tree)) {
|
mrframe.typeargs.contains(tree)) {
|
||||||
int arg = mrframe.typeargs.indexOf(tree);
|
final int type_index = mrframe.typeargs.indexOf(tree);
|
||||||
p.type_index = arg;
|
|
||||||
switch (mrframe.mode) {
|
switch (mrframe.mode) {
|
||||||
case INVOKE:
|
case INVOKE:
|
||||||
p.type = TargetType.METHOD_REFERENCE_TYPE_ARGUMENT;
|
return TypeAnnotationPosition
|
||||||
break;
|
.methodRefTypeArg(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
type_index, frame.pos);
|
||||||
case NEW:
|
case NEW:
|
||||||
p.type = TargetType.CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT;
|
return TypeAnnotationPosition
|
||||||
break;
|
.constructorRefTypeArg(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
type_index, frame.pos);
|
||||||
default:
|
default:
|
||||||
Assert.error("Unknown method reference mode " + mrframe.mode +
|
throw new AssertionError("Unknown method reference mode " + mrframe.mode +
|
||||||
" for tree " + tree + " within frame " + frame);
|
" for tree " + tree + " within frame " + frame);
|
||||||
}
|
}
|
||||||
p.pos = frame.pos;
|
|
||||||
} else {
|
} else {
|
||||||
Assert.error("Could not determine type argument position of tree " + tree +
|
throw new AssertionError("Could not determine type argument position of tree " + tree +
|
||||||
" within frame " + frame);
|
" within frame " + frame);
|
||||||
}
|
}
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case ARRAY_TYPE: {
|
case ARRAY_TYPE: {
|
||||||
ListBuffer<TypePathEntry> index = new ListBuffer<>();
|
location = location.prepend(TypePathEntry.ARRAY);
|
||||||
index = index.append(TypePathEntry.ARRAY);
|
|
||||||
List<JCTree> newPath = path.tail;
|
List<JCTree> newPath = path.tail;
|
||||||
while (true) {
|
while (true) {
|
||||||
JCTree npHead = newPath.tail.head;
|
JCTree npHead = newPath.tail.head;
|
||||||
if (npHead.hasTag(JCTree.Tag.TYPEARRAY)) {
|
if (npHead.hasTag(JCTree.Tag.TYPEARRAY)) {
|
||||||
newPath = newPath.tail;
|
newPath = newPath.tail;
|
||||||
index = index.append(TypePathEntry.ARRAY);
|
location = location.prepend(TypePathEntry.ARRAY);
|
||||||
} else if (npHead.hasTag(JCTree.Tag.ANNOTATED_TYPE)) {
|
} else if (npHead.hasTag(JCTree.Tag.ANNOTATED_TYPE)) {
|
||||||
newPath = newPath.tail;
|
newPath = newPath.tail;
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
p.location = p.location.prependList(index.toList());
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
newPath, currentLambda,
|
||||||
return;
|
outer_type_index, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
case TYPE_PARAMETER:
|
case TYPE_PARAMETER:
|
||||||
if (path.tail.tail.head.hasTag(JCTree.Tag.CLASSDEF)) {
|
if (path.tail.tail.head.hasTag(JCTree.Tag.CLASSDEF)) {
|
||||||
JCClassDecl clazz = (JCClassDecl)path.tail.tail.head;
|
final JCClassDecl clazz =
|
||||||
p.type = TargetType.CLASS_TYPE_PARAMETER_BOUND;
|
(JCClassDecl)path.tail.tail.head;
|
||||||
p.parameter_index = clazz.typarams.indexOf(path.tail.head);
|
final int parameter_index =
|
||||||
p.bound_index = ((JCTypeParameter)frame).bounds.indexOf(tree);
|
clazz.typarams.indexOf(path.tail.head);
|
||||||
if (((JCTypeParameter)frame).bounds.get(0).type.isInterface()) {
|
final int bound_index =
|
||||||
// Account for an implicit Object as bound 0
|
((JCTypeParameter)frame).bounds.get(0)
|
||||||
p.bound_index += 1;
|
.type.isInterface() ?
|
||||||
}
|
((JCTypeParameter)frame).bounds.indexOf(tree) + 1:
|
||||||
|
((JCTypeParameter)frame).bounds.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.typeParameterBound(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
parameter_index, bound_index,
|
||||||
|
frame.pos);
|
||||||
} else if (path.tail.tail.head.hasTag(JCTree.Tag.METHODDEF)) {
|
} else if (path.tail.tail.head.hasTag(JCTree.Tag.METHODDEF)) {
|
||||||
JCMethodDecl method = (JCMethodDecl)path.tail.tail.head;
|
final JCMethodDecl method =
|
||||||
p.type = TargetType.METHOD_TYPE_PARAMETER_BOUND;
|
(JCMethodDecl)path.tail.tail.head;
|
||||||
p.parameter_index = method.typarams.indexOf(path.tail.head);
|
final int parameter_index =
|
||||||
p.bound_index = ((JCTypeParameter)frame).bounds.indexOf(tree);
|
method.typarams.indexOf(path.tail.head);
|
||||||
if (((JCTypeParameter)frame).bounds.get(0).type.isInterface()) {
|
final int bound_index =
|
||||||
// Account for an implicit Object as bound 0
|
((JCTypeParameter)frame).bounds.get(0)
|
||||||
p.bound_index += 1;
|
.type.isInterface() ?
|
||||||
}
|
((JCTypeParameter)frame).bounds.indexOf(tree) + 1:
|
||||||
|
((JCTypeParameter)frame).bounds.indexOf(tree);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.methodTypeParameterBound(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
parameter_index,
|
||||||
|
bound_index,
|
||||||
|
frame.pos);
|
||||||
} else {
|
} else {
|
||||||
Assert.error("Could not determine position of tree " + tree +
|
throw new AssertionError("Could not determine position of tree " + tree +
|
||||||
" within frame " + frame);
|
" within frame " + frame);
|
||||||
}
|
}
|
||||||
p.pos = frame.pos;
|
|
||||||
return;
|
|
||||||
|
|
||||||
case VARIABLE:
|
case VARIABLE:
|
||||||
VarSymbol v = ((JCVariableDecl)frame).sym;
|
VarSymbol v = ((JCVariableDecl)frame).sym;
|
||||||
p.pos = frame.pos;
|
|
||||||
switch (v.getKind()) {
|
|
||||||
case LOCAL_VARIABLE:
|
|
||||||
p.type = TargetType.LOCAL_VARIABLE;
|
|
||||||
break;
|
|
||||||
case FIELD:
|
|
||||||
p.type = TargetType.FIELD;
|
|
||||||
break;
|
|
||||||
case PARAMETER:
|
|
||||||
if (v.getQualifiedName().equals(names._this)) {
|
|
||||||
// TODO: Intro a separate ElementKind?
|
|
||||||
p.type = TargetType.METHOD_RECEIVER;
|
|
||||||
} else {
|
|
||||||
p.type = TargetType.METHOD_FORMAL_PARAMETER;
|
|
||||||
p.parameter_index = methodParamIndex(path, frame);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case EXCEPTION_PARAMETER:
|
|
||||||
p.type = TargetType.EXCEPTION_PARAMETER;
|
|
||||||
break;
|
|
||||||
case RESOURCE_VARIABLE:
|
|
||||||
p.type = TargetType.RESOURCE_VARIABLE;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
Assert.error("Found unexpected type annotation for variable: " + v + " with kind: " + v.getKind());
|
|
||||||
}
|
|
||||||
if (v.getKind() != ElementKind.FIELD) {
|
if (v.getKind() != ElementKind.FIELD) {
|
||||||
v.owner.appendUniqueTypeAttributes(v.getRawTypeAttributes());
|
v.owner.appendUniqueTypeAttributes(v.getRawTypeAttributes());
|
||||||
}
|
}
|
||||||
return;
|
switch (v.getKind()) {
|
||||||
|
case LOCAL_VARIABLE:
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.localVariable(location.toList(), currentLambda,
|
||||||
|
frame.pos);
|
||||||
|
case FIELD:
|
||||||
|
return TypeAnnotationPosition.field(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
frame.pos);
|
||||||
|
case PARAMETER:
|
||||||
|
if (v.getQualifiedName().equals(names._this)) {
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.methodReceiver(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
frame.pos);
|
||||||
|
} else {
|
||||||
|
final int parameter_index =
|
||||||
|
methodParamIndex(path, frame);
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.methodParameter(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
parameter_index,
|
||||||
|
frame.pos);
|
||||||
|
}
|
||||||
|
case EXCEPTION_PARAMETER:
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.exceptionParameter(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
frame.pos);
|
||||||
|
case RESOURCE_VARIABLE:
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.resourceVariable(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
frame.pos);
|
||||||
|
default:
|
||||||
|
throw new AssertionError("Found unexpected type annotation for variable: " + v + " with kind: " + v.getKind());
|
||||||
|
}
|
||||||
|
|
||||||
case ANNOTATED_TYPE: {
|
case ANNOTATED_TYPE: {
|
||||||
if (frame == tree) {
|
if (frame == tree) {
|
||||||
@ -921,83 +970,89 @@ public class TypeAnnotations {
|
|||||||
// class/method as enclosing elements.
|
// class/method as enclosing elements.
|
||||||
// There is actually nothing to do for them.
|
// There is actually nothing to do for them.
|
||||||
} else {
|
} else {
|
||||||
locateNestedTypes(utype, p);
|
location = locateNestedTypes(utype, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
List<JCTree> newPath = path.tail;
|
List<JCTree> newPath = path.tail;
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
return;
|
newPath, currentLambda,
|
||||||
|
outer_type_index, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
case UNION_TYPE: {
|
case UNION_TYPE: {
|
||||||
List<JCTree> newPath = path.tail;
|
List<JCTree> newPath = path.tail;
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
return;
|
newPath, currentLambda,
|
||||||
|
outer_type_index, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
case INTERSECTION_TYPE: {
|
case INTERSECTION_TYPE: {
|
||||||
JCTypeIntersection isect = (JCTypeIntersection)frame;
|
JCTypeIntersection isect = (JCTypeIntersection)frame;
|
||||||
p.type_index = isect.bounds.indexOf(tree);
|
final List<JCTree> newPath = path.tail;
|
||||||
List<JCTree> newPath = path.tail;
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
newPath, currentLambda,
|
||||||
return;
|
isect.bounds.indexOf(tree), location);
|
||||||
}
|
}
|
||||||
|
|
||||||
case METHOD_INVOCATION: {
|
case METHOD_INVOCATION: {
|
||||||
JCMethodInvocation invocation = (JCMethodInvocation)frame;
|
JCMethodInvocation invocation = (JCMethodInvocation)frame;
|
||||||
if (!invocation.typeargs.contains(tree)) {
|
if (!invocation.typeargs.contains(tree)) {
|
||||||
Assert.error("{" + tree + "} is not an argument in the invocation: " + invocation);
|
throw new AssertionError("{" + tree + "} is not an argument in the invocation: " + invocation);
|
||||||
}
|
}
|
||||||
MethodSymbol exsym = (MethodSymbol) TreeInfo.symbol(invocation.getMethodSelect());
|
MethodSymbol exsym = (MethodSymbol) TreeInfo.symbol(invocation.getMethodSelect());
|
||||||
|
final int type_index = invocation.typeargs.indexOf(tree);
|
||||||
if (exsym == null) {
|
if (exsym == null) {
|
||||||
Assert.error("could not determine symbol for {" + invocation + "}");
|
throw new AssertionError("could not determine symbol for {" + invocation + "}");
|
||||||
} else if (exsym.isConstructor()) {
|
} else if (exsym.isConstructor()) {
|
||||||
p.type = TargetType.CONSTRUCTOR_INVOCATION_TYPE_ARGUMENT;
|
return TypeAnnotationPosition
|
||||||
|
.constructorInvocationTypeArg(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
type_index,
|
||||||
|
invocation.pos);
|
||||||
} else {
|
} else {
|
||||||
p.type = TargetType.METHOD_INVOCATION_TYPE_ARGUMENT;
|
return TypeAnnotationPosition
|
||||||
|
.methodInvocationTypeArg(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
type_index,
|
||||||
|
invocation.pos);
|
||||||
}
|
}
|
||||||
p.pos = invocation.pos;
|
|
||||||
p.type_index = invocation.typeargs.indexOf(tree);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case EXTENDS_WILDCARD:
|
case EXTENDS_WILDCARD:
|
||||||
case SUPER_WILDCARD: {
|
case SUPER_WILDCARD: {
|
||||||
// Annotations in wildcard bounds
|
// Annotations in wildcard bounds
|
||||||
p.location = p.location.prepend(TypePathEntry.WILDCARD);
|
final List<JCTree> newPath = path.tail;
|
||||||
List<JCTree> newPath = path.tail;
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
newPath, currentLambda,
|
||||||
return;
|
outer_type_index,
|
||||||
|
location.prepend(TypePathEntry.WILDCARD));
|
||||||
}
|
}
|
||||||
|
|
||||||
case MEMBER_SELECT: {
|
case MEMBER_SELECT: {
|
||||||
List<JCTree> newPath = path.tail;
|
final List<JCTree> newPath = path.tail;
|
||||||
resolveFrame(newPath.head, newPath.tail.head, newPath, p);
|
return resolveFrame(newPath.head, newPath.tail.head,
|
||||||
return;
|
newPath, currentLambda,
|
||||||
|
outer_type_index, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
Assert.error("Unresolved frame: " + frame + " of kind: " + frame.getKind() +
|
throw new AssertionError("Unresolved frame: " + frame +
|
||||||
|
" of kind: " + frame.getKind() +
|
||||||
"\n Looking for tree: " + tree);
|
"\n Looking for tree: " + tree);
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void locateNestedTypes(Type type, TypeAnnotationPosition p) {
|
private ListBuffer<TypePathEntry>
|
||||||
// The number of "steps" to get from the full type to the
|
locateNestedTypes(Type type,
|
||||||
// left-most outer type.
|
ListBuffer<TypePathEntry> depth) {
|
||||||
ListBuffer<TypePathEntry> depth = new ListBuffer<>();
|
|
||||||
|
|
||||||
Type encl = type.getEnclosingType();
|
Type encl = type.getEnclosingType();
|
||||||
while (encl != null &&
|
while (encl != null &&
|
||||||
encl.getKind() != TypeKind.NONE &&
|
encl.getKind() != TypeKind.NONE &&
|
||||||
encl.getKind() != TypeKind.ERROR) {
|
encl.getKind() != TypeKind.ERROR) {
|
||||||
depth = depth.append(TypePathEntry.INNER_TYPE);
|
depth = depth.prepend(TypePathEntry.INNER_TYPE);
|
||||||
encl = encl.getEnclosingType();
|
encl = encl.getEnclosingType();
|
||||||
}
|
}
|
||||||
if (depth.nonEmpty()) {
|
return depth;
|
||||||
p.location = p.location.prependList(depth.toList());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private int methodParamIndex(List<JCTree> path, JCTree param) {
|
private int methodParamIndex(List<JCTree> path, JCTree param) {
|
||||||
@ -1048,17 +1103,17 @@ public class TypeAnnotations {
|
|||||||
}
|
}
|
||||||
if (sigOnly) {
|
if (sigOnly) {
|
||||||
if (!tree.mods.annotations.isEmpty()) {
|
if (!tree.mods.annotations.isEmpty()) {
|
||||||
// Nothing to do for separateAnnotationsKinds if
|
|
||||||
// there are no annotations of either kind.
|
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
|
||||||
pos.type = TargetType.METHOD_RETURN;
|
|
||||||
if (tree.sym.isConstructor()) {
|
if (tree.sym.isConstructor()) {
|
||||||
pos.pos = tree.pos;
|
final TypeAnnotationPosition pos =
|
||||||
// Use null to mark that the annotations go with the symbol.
|
TypeAnnotationPosition.methodReturn(tree.pos);
|
||||||
|
// Use null to mark that the annotations go
|
||||||
|
// with the symbol.
|
||||||
separateAnnotationsKinds(tree, null, tree.sym, pos);
|
separateAnnotationsKinds(tree, null, tree.sym, pos);
|
||||||
} else {
|
} else {
|
||||||
pos.pos = tree.restype.pos;
|
final TypeAnnotationPosition pos =
|
||||||
separateAnnotationsKinds(tree.restype, tree.sym.type.getReturnType(),
|
TypeAnnotationPosition.methodReturn(tree.restype.pos);
|
||||||
|
separateAnnotationsKinds(tree.restype,
|
||||||
|
tree.sym.type.getReturnType(),
|
||||||
tree.sym, pos);
|
tree.sym, pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1067,10 +1122,10 @@ public class TypeAnnotations {
|
|||||||
// Nothing to do for separateAnnotationsKinds if
|
// Nothing to do for separateAnnotationsKinds if
|
||||||
// there are no annotations of either kind.
|
// there are no annotations of either kind.
|
||||||
// TODO: make sure there are no declaration annotations.
|
// TODO: make sure there are no declaration annotations.
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.METHOD_RECEIVER;
|
TypeAnnotationPosition.methodReceiver(tree.recvparam.vartype.pos);
|
||||||
pos.pos = tree.recvparam.vartype.pos;
|
separateAnnotationsKinds(tree.recvparam.vartype,
|
||||||
separateAnnotationsKinds(tree.recvparam.vartype, tree.recvparam.sym.type,
|
tree.recvparam.sym.type,
|
||||||
tree.recvparam.sym, pos);
|
tree.recvparam.sym, pos);
|
||||||
}
|
}
|
||||||
int i = 0;
|
int i = 0;
|
||||||
@ -1078,11 +1133,11 @@ public class TypeAnnotations {
|
|||||||
if (!param.mods.annotations.isEmpty()) {
|
if (!param.mods.annotations.isEmpty()) {
|
||||||
// Nothing to do for separateAnnotationsKinds if
|
// Nothing to do for separateAnnotationsKinds if
|
||||||
// there are no annotations of either kind.
|
// there are no annotations of either kind.
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.METHOD_FORMAL_PARAMETER;
|
TypeAnnotationPosition.methodParameter(i, param.vartype.pos);
|
||||||
pos.parameter_index = i;
|
separateAnnotationsKinds(param.vartype,
|
||||||
pos.pos = param.vartype.pos;
|
param.sym.type,
|
||||||
separateAnnotationsKinds(param.vartype, param.sym.type, param.sym, pos);
|
param.sym, pos);
|
||||||
}
|
}
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
@ -1119,11 +1174,9 @@ public class TypeAnnotations {
|
|||||||
if (!param.mods.annotations.isEmpty()) {
|
if (!param.mods.annotations.isEmpty()) {
|
||||||
// Nothing to do for separateAnnotationsKinds if
|
// Nothing to do for separateAnnotationsKinds if
|
||||||
// there are no annotations of either kind.
|
// there are no annotations of either kind.
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.METHOD_FORMAL_PARAMETER;
|
TypeAnnotationPosition.methodParameter(tree, i,
|
||||||
pos.parameter_index = i;
|
param.vartype.pos);
|
||||||
pos.pos = param.vartype.pos;
|
|
||||||
pos.onLambda = tree;
|
|
||||||
separateAnnotationsKinds(param.vartype, param.sym.type, param.sym, pos);
|
separateAnnotationsKinds(param.vartype, param.sym.type, param.sym, pos);
|
||||||
}
|
}
|
||||||
++i;
|
++i;
|
||||||
@ -1153,28 +1206,24 @@ public class TypeAnnotations {
|
|||||||
// Parameters are handled in visitMethodDef or visitLambda.
|
// Parameters are handled in visitMethodDef or visitLambda.
|
||||||
} else if (tree.sym.getKind() == ElementKind.FIELD) {
|
} else if (tree.sym.getKind() == ElementKind.FIELD) {
|
||||||
if (sigOnly) {
|
if (sigOnly) {
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.FIELD;
|
TypeAnnotationPosition.field(tree.pos);
|
||||||
pos.pos = tree.pos;
|
|
||||||
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
||||||
}
|
}
|
||||||
} else if (tree.sym.getKind() == ElementKind.LOCAL_VARIABLE) {
|
} else if (tree.sym.getKind() == ElementKind.LOCAL_VARIABLE) {
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.LOCAL_VARIABLE;
|
TypeAnnotationPosition.localVariable(currentLambda,
|
||||||
pos.pos = tree.pos;
|
tree.pos);
|
||||||
pos.onLambda = currentLambda;
|
|
||||||
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
||||||
} else if (tree.sym.getKind() == ElementKind.EXCEPTION_PARAMETER) {
|
} else if (tree.sym.getKind() == ElementKind.EXCEPTION_PARAMETER) {
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.EXCEPTION_PARAMETER;
|
TypeAnnotationPosition.exceptionParameter(currentLambda,
|
||||||
pos.pos = tree.pos;
|
tree.pos);
|
||||||
pos.onLambda = currentLambda;
|
|
||||||
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
||||||
} else if (tree.sym.getKind() == ElementKind.RESOURCE_VARIABLE) {
|
} else if (tree.sym.getKind() == ElementKind.RESOURCE_VARIABLE) {
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
pos.type = TargetType.RESOURCE_VARIABLE;
|
TypeAnnotationPosition.resourceVariable(currentLambda,
|
||||||
pos.pos = tree.pos;
|
tree.pos);
|
||||||
pos.onLambda = currentLambda;
|
|
||||||
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos);
|
||||||
} else if (tree.sym.getKind() == ElementKind.ENUM_CONSTANT) {
|
} else if (tree.sym.getKind() == ElementKind.ENUM_CONSTANT) {
|
||||||
// No type annotations can occur here.
|
// No type annotations can occur here.
|
||||||
@ -1218,7 +1267,8 @@ public class TypeAnnotations {
|
|||||||
|
|
||||||
private void copyNewClassAnnotationsToOwner(JCNewClass tree) {
|
private void copyNewClassAnnotationsToOwner(JCNewClass tree) {
|
||||||
Symbol sym = tree.def.sym;
|
Symbol sym = tree.def.sym;
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
final TypeAnnotationPosition pos =
|
||||||
|
TypeAnnotationPosition.newObj(tree.pos);
|
||||||
ListBuffer<Attribute.TypeCompound> newattrs = new ListBuffer<>();
|
ListBuffer<Attribute.TypeCompound> newattrs = new ListBuffer<>();
|
||||||
|
|
||||||
for (Attribute.TypeCompound old : sym.getRawTypeAttributes()) {
|
for (Attribute.TypeCompound old : sym.getRawTypeAttributes()) {
|
||||||
@ -1226,8 +1276,6 @@ public class TypeAnnotations {
|
|||||||
pos));
|
pos));
|
||||||
}
|
}
|
||||||
|
|
||||||
pos.type = TargetType.NEW;
|
|
||||||
pos.pos = tree.pos;
|
|
||||||
sym.owner.appendUniqueTypeAttributes(newattrs.toList());
|
sym.owner.appendUniqueTypeAttributes(newattrs.toList());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1236,16 +1284,16 @@ public class TypeAnnotations {
|
|||||||
if (tree.def != null &&
|
if (tree.def != null &&
|
||||||
!tree.def.mods.annotations.isEmpty()) {
|
!tree.def.mods.annotations.isEmpty()) {
|
||||||
JCClassDecl classdecl = tree.def;
|
JCClassDecl classdecl = tree.def;
|
||||||
TypeAnnotationPosition pos = new TypeAnnotationPosition();
|
TypeAnnotationPosition pos;
|
||||||
pos.type = TargetType.CLASS_EXTENDS;
|
|
||||||
pos.pos = tree.pos;
|
|
||||||
if (classdecl.extending == tree.clazz) {
|
if (classdecl.extending == tree.clazz) {
|
||||||
pos.type_index = -1;
|
pos = TypeAnnotationPosition.classExtends(tree.pos);
|
||||||
} else if (classdecl.implementing.contains(tree.clazz)) {
|
} else if (classdecl.implementing.contains(tree.clazz)) {
|
||||||
pos.type_index = classdecl.implementing.indexOf(tree.clazz);
|
final int index = classdecl.implementing.indexOf(tree.clazz);
|
||||||
|
pos = TypeAnnotationPosition.classExtends(index, tree.pos);
|
||||||
} else {
|
} else {
|
||||||
// In contrast to CLASS elsewhere, typarams cannot occur here.
|
// In contrast to CLASS elsewhere, typarams cannot occur here.
|
||||||
Assert.error("Could not determine position of tree " + tree);
|
throw new AssertionError("Could not determine position of tree " + tree);
|
||||||
}
|
}
|
||||||
Type before = classdecl.sym.type;
|
Type before = classdecl.sym.type;
|
||||||
separateAnnotationsKinds(classdecl, tree.clazz.type, classdecl.sym, pos);
|
separateAnnotationsKinds(classdecl, tree.clazz.type, classdecl.sym, pos);
|
||||||
@ -1273,14 +1321,16 @@ public class TypeAnnotations {
|
|||||||
|
|
||||||
// handle annotations associated with dimensions
|
// handle annotations associated with dimensions
|
||||||
for (int i = 0; i < dimAnnosCount; ++i) {
|
for (int i = 0; i < dimAnnosCount; ++i) {
|
||||||
TypeAnnotationPosition p = new TypeAnnotationPosition();
|
ListBuffer<TypePathEntry> location =
|
||||||
p.pos = tree.pos;
|
new ListBuffer<TypePathEntry>();
|
||||||
p.onLambda = currentLambda;
|
|
||||||
p.type = TargetType.NEW;
|
|
||||||
if (i != 0) {
|
if (i != 0) {
|
||||||
depth = depth.append(TypePathEntry.ARRAY);
|
depth = depth.append(TypePathEntry.ARRAY);
|
||||||
p.location = p.location.appendList(depth.toList());
|
location = location.appendList(depth.toList());
|
||||||
}
|
}
|
||||||
|
final TypeAnnotationPosition p =
|
||||||
|
TypeAnnotationPosition.newObj(location.toList(),
|
||||||
|
currentLambda,
|
||||||
|
tree.pos);
|
||||||
|
|
||||||
setTypeAnnotationPos(tree.dimAnnotations.get(i), p);
|
setTypeAnnotationPos(tree.dimAnnotations.get(i), p);
|
||||||
}
|
}
|
||||||
@ -1293,12 +1343,14 @@ public class TypeAnnotations {
|
|||||||
while (elemType != null) {
|
while (elemType != null) {
|
||||||
if (elemType.hasTag(JCTree.Tag.ANNOTATED_TYPE)) {
|
if (elemType.hasTag(JCTree.Tag.ANNOTATED_TYPE)) {
|
||||||
JCAnnotatedType at = (JCAnnotatedType)elemType;
|
JCAnnotatedType at = (JCAnnotatedType)elemType;
|
||||||
TypeAnnotationPosition p = new TypeAnnotationPosition();
|
final ListBuffer<TypePathEntry> locationbuf =
|
||||||
p.type = TargetType.NEW;
|
locateNestedTypes(elemType.type,
|
||||||
p.pos = tree.pos;
|
new ListBuffer<TypePathEntry>());
|
||||||
p.onLambda = currentLambda;
|
final List<TypePathEntry> location =
|
||||||
locateNestedTypes(elemType.type, p);
|
locationbuf.toList().prependList(depth.toList());
|
||||||
p.location = p.location.prependList(depth.toList());
|
final TypeAnnotationPosition p =
|
||||||
|
TypeAnnotationPosition.newObj(location, currentLambda,
|
||||||
|
tree.pos);
|
||||||
setTypeAnnotationPos(at.annotations, p);
|
setTypeAnnotationPos(at.annotations, p);
|
||||||
elemType = at.underlyingType;
|
elemType = at.underlyingType;
|
||||||
} else if (elemType.hasTag(JCTree.Tag.TYPEARRAY)) {
|
} else if (elemType.hasTag(JCTree.Tag.TYPEARRAY)) {
|
||||||
@ -1320,9 +1372,9 @@ public class TypeAnnotations {
|
|||||||
System.err.println(" tree: " + tree + " kind: " + tree.getKind());
|
System.err.println(" tree: " + tree + " kind: " + tree.getKind());
|
||||||
System.err.println(" frame: " + frame + " kind: " + frame.getKind());
|
System.err.println(" frame: " + frame + " kind: " + frame.getKind());
|
||||||
*/
|
*/
|
||||||
TypeAnnotationPosition p = new TypeAnnotationPosition();
|
final TypeAnnotationPosition p =
|
||||||
p.onLambda = currentLambda;
|
resolveFrame(tree, frame, frames.toList(), currentLambda, 0,
|
||||||
resolveFrame(tree, frame, frames.toList(), p);
|
new ListBuffer<TypePathEntry>());
|
||||||
setTypeAnnotationPos(annotations, p);
|
setTypeAnnotationPos(annotations, p);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -299,7 +299,13 @@ public class Annotate {
|
|||||||
if (typeAnnotation) {
|
if (typeAnnotation) {
|
||||||
if (a.attribute == null || !(a.attribute instanceof Attribute.TypeCompound)) {
|
if (a.attribute == null || !(a.attribute instanceof Attribute.TypeCompound)) {
|
||||||
// Create a new TypeCompound
|
// Create a new TypeCompound
|
||||||
Attribute.TypeCompound tc = new Attribute.TypeCompound(a.type, buf.toList(), new TypeAnnotationPosition());
|
|
||||||
|
Attribute.TypeCompound tc =
|
||||||
|
new Attribute.TypeCompound(a.type, buf.toList(),
|
||||||
|
// TODO: Eventually, we will get rid of this use of
|
||||||
|
// unknown, because we'll get a position from
|
||||||
|
// MemberEnter (task 8027262).
|
||||||
|
TypeAnnotationPosition.unknown);
|
||||||
a.attribute = tc;
|
a.attribute = tc;
|
||||||
return tc;
|
return tc;
|
||||||
} else {
|
} else {
|
||||||
|
@ -1469,26 +1469,46 @@ public class ClassReader {
|
|||||||
if (!TargetType.isValidTargetTypeValue(tag))
|
if (!TargetType.isValidTargetTypeValue(tag))
|
||||||
throw this.badClassFile("bad.type.annotation.value", String.format("0x%02X", tag));
|
throw this.badClassFile("bad.type.annotation.value", String.format("0x%02X", tag));
|
||||||
|
|
||||||
TypeAnnotationPosition position = new TypeAnnotationPosition();
|
|
||||||
TargetType type = TargetType.fromTargetTypeValue(tag);
|
TargetType type = TargetType.fromTargetTypeValue(tag);
|
||||||
|
|
||||||
position.type = type;
|
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
// instanceof
|
// instanceof
|
||||||
case INSTANCEOF:
|
case INSTANCEOF: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final TypeAnnotationPosition position =
|
||||||
|
TypeAnnotationPosition.instanceOf(readTypePath());
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// new expression
|
// new expression
|
||||||
case NEW:
|
case NEW: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final TypeAnnotationPosition position =
|
||||||
|
TypeAnnotationPosition.newObj(readTypePath());
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// constructor/method reference receiver
|
// constructor/method reference receiver
|
||||||
case CONSTRUCTOR_REFERENCE:
|
case CONSTRUCTOR_REFERENCE: {
|
||||||
case METHOD_REFERENCE:
|
final int offset = nextChar();
|
||||||
position.offset = nextChar();
|
final TypeAnnotationPosition position =
|
||||||
break;
|
TypeAnnotationPosition.constructorRef(readTypePath());
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
|
case METHOD_REFERENCE: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final TypeAnnotationPosition position =
|
||||||
|
TypeAnnotationPosition.methodRef(readTypePath());
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// local variable
|
// local variable
|
||||||
case LOCAL_VARIABLE:
|
case LOCAL_VARIABLE: {
|
||||||
// resource variable
|
final int table_length = nextChar();
|
||||||
case RESOURCE_VARIABLE:
|
final TypeAnnotationPosition position =
|
||||||
int table_length = nextChar();
|
TypeAnnotationPosition.localVariable(readTypePath());
|
||||||
|
|
||||||
position.lvarOffset = new int[table_length];
|
position.lvarOffset = new int[table_length];
|
||||||
position.lvarLength = new int[table_length];
|
position.lvarLength = new int[table_length];
|
||||||
position.lvarIndex = new int[table_length];
|
position.lvarIndex = new int[table_length];
|
||||||
@ -1498,67 +1518,142 @@ public class ClassReader {
|
|||||||
position.lvarLength[i] = nextChar();
|
position.lvarLength[i] = nextChar();
|
||||||
position.lvarIndex[i] = nextChar();
|
position.lvarIndex[i] = nextChar();
|
||||||
}
|
}
|
||||||
break;
|
return position;
|
||||||
|
}
|
||||||
|
// resource variable
|
||||||
|
case RESOURCE_VARIABLE: {
|
||||||
|
final int table_length = nextChar();
|
||||||
|
final TypeAnnotationPosition position =
|
||||||
|
TypeAnnotationPosition.resourceVariable(readTypePath());
|
||||||
|
|
||||||
|
position.lvarOffset = new int[table_length];
|
||||||
|
position.lvarLength = new int[table_length];
|
||||||
|
position.lvarIndex = new int[table_length];
|
||||||
|
|
||||||
|
for (int i = 0; i < table_length; ++i) {
|
||||||
|
position.lvarOffset[i] = nextChar();
|
||||||
|
position.lvarLength[i] = nextChar();
|
||||||
|
position.lvarIndex[i] = nextChar();
|
||||||
|
}
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// exception parameter
|
// exception parameter
|
||||||
case EXCEPTION_PARAMETER:
|
case EXCEPTION_PARAMETER: {
|
||||||
position.exception_index = nextChar();
|
final int exception_index = nextChar();
|
||||||
break;
|
final TypeAnnotationPosition position =
|
||||||
|
TypeAnnotationPosition.exceptionParameter(readTypePath());
|
||||||
|
position.exception_index = exception_index;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// method receiver
|
// method receiver
|
||||||
case METHOD_RECEIVER:
|
case METHOD_RECEIVER:
|
||||||
// Do nothing
|
return TypeAnnotationPosition.methodReceiver(readTypePath());
|
||||||
break;
|
|
||||||
// type parameter
|
// type parameter
|
||||||
case CLASS_TYPE_PARAMETER:
|
case CLASS_TYPE_PARAMETER: {
|
||||||
case METHOD_TYPE_PARAMETER:
|
final int parameter_index = nextByte();
|
||||||
position.parameter_index = nextByte();
|
return TypeAnnotationPosition
|
||||||
break;
|
.typeParameter(readTypePath(), parameter_index);
|
||||||
|
}
|
||||||
|
case METHOD_TYPE_PARAMETER: {
|
||||||
|
final int parameter_index = nextByte();
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.methodTypeParameter(readTypePath(), parameter_index);
|
||||||
|
}
|
||||||
// type parameter bound
|
// type parameter bound
|
||||||
case CLASS_TYPE_PARAMETER_BOUND:
|
case CLASS_TYPE_PARAMETER_BOUND: {
|
||||||
case METHOD_TYPE_PARAMETER_BOUND:
|
final int parameter_index = nextByte();
|
||||||
position.parameter_index = nextByte();
|
final int bound_index = nextByte();
|
||||||
position.bound_index = nextByte();
|
return TypeAnnotationPosition
|
||||||
break;
|
.typeParameterBound(readTypePath(), parameter_index,
|
||||||
|
bound_index);
|
||||||
|
}
|
||||||
|
case METHOD_TYPE_PARAMETER_BOUND: {
|
||||||
|
final int parameter_index = nextByte();
|
||||||
|
final int bound_index = nextByte();
|
||||||
|
return TypeAnnotationPosition
|
||||||
|
.methodTypeParameterBound(readTypePath(), parameter_index,
|
||||||
|
bound_index);
|
||||||
|
}
|
||||||
// class extends or implements clause
|
// class extends or implements clause
|
||||||
case CLASS_EXTENDS:
|
case CLASS_EXTENDS: {
|
||||||
position.type_index = nextChar();
|
final int type_index = nextChar();
|
||||||
break;
|
return TypeAnnotationPosition.classExtends(readTypePath(),
|
||||||
|
type_index);
|
||||||
|
}
|
||||||
// throws
|
// throws
|
||||||
case THROWS:
|
case THROWS: {
|
||||||
position.type_index = nextChar();
|
final int type_index = nextChar();
|
||||||
break;
|
return TypeAnnotationPosition.methodThrows(readTypePath(),
|
||||||
|
type_index);
|
||||||
|
}
|
||||||
// method parameter
|
// method parameter
|
||||||
case METHOD_FORMAL_PARAMETER:
|
case METHOD_FORMAL_PARAMETER: {
|
||||||
position.parameter_index = nextByte();
|
final int parameter_index = nextByte();
|
||||||
break;
|
return TypeAnnotationPosition.methodParameter(readTypePath(),
|
||||||
|
parameter_index);
|
||||||
|
}
|
||||||
// type cast
|
// type cast
|
||||||
case CAST:
|
case CAST: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final int type_index = nextByte();
|
||||||
|
final TypeAnnotationPosition position =
|
||||||
|
TypeAnnotationPosition.typeCast(readTypePath(), type_index);
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// method/constructor/reference type argument
|
// method/constructor/reference type argument
|
||||||
case CONSTRUCTOR_INVOCATION_TYPE_ARGUMENT:
|
case CONSTRUCTOR_INVOCATION_TYPE_ARGUMENT: {
|
||||||
case METHOD_INVOCATION_TYPE_ARGUMENT:
|
final int offset = nextChar();
|
||||||
case CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT:
|
final int type_index = nextByte();
|
||||||
case METHOD_REFERENCE_TYPE_ARGUMENT:
|
final TypeAnnotationPosition position = TypeAnnotationPosition
|
||||||
position.offset = nextChar();
|
.constructorInvocationTypeArg(readTypePath(), type_index);
|
||||||
position.type_index = nextByte();
|
position.offset = offset;
|
||||||
break;
|
return position;
|
||||||
|
}
|
||||||
|
case METHOD_INVOCATION_TYPE_ARGUMENT: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final int type_index = nextByte();
|
||||||
|
final TypeAnnotationPosition position = TypeAnnotationPosition
|
||||||
|
.methodInvocationTypeArg(readTypePath(), type_index);
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
|
case CONSTRUCTOR_REFERENCE_TYPE_ARGUMENT: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final int type_index = nextByte();
|
||||||
|
final TypeAnnotationPosition position = TypeAnnotationPosition
|
||||||
|
.constructorRefTypeArg(readTypePath(), type_index);
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
|
case METHOD_REFERENCE_TYPE_ARGUMENT: {
|
||||||
|
final int offset = nextChar();
|
||||||
|
final int type_index = nextByte();
|
||||||
|
final TypeAnnotationPosition position = TypeAnnotationPosition
|
||||||
|
.methodRefTypeArg(readTypePath(), type_index);
|
||||||
|
position.offset = offset;
|
||||||
|
return position;
|
||||||
|
}
|
||||||
// We don't need to worry about these
|
// We don't need to worry about these
|
||||||
case METHOD_RETURN:
|
case METHOD_RETURN:
|
||||||
|
return TypeAnnotationPosition.methodReturn(readTypePath());
|
||||||
case FIELD:
|
case FIELD:
|
||||||
break;
|
return TypeAnnotationPosition.field(readTypePath());
|
||||||
case UNKNOWN:
|
case UNKNOWN:
|
||||||
throw new AssertionError("jvm.ClassReader: UNKNOWN target type should never occur!");
|
throw new AssertionError("jvm.ClassReader: UNKNOWN target type should never occur!");
|
||||||
default:
|
default:
|
||||||
throw new AssertionError("jvm.ClassReader: Unknown target type for position: " + position);
|
throw new AssertionError("jvm.ClassReader: Unknown target type for position: " + type);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
{ // See whether there is location info and read it
|
List<TypeAnnotationPosition.TypePathEntry> readTypePath() {
|
||||||
int len = nextByte();
|
int len = nextByte();
|
||||||
ListBuffer<Integer> loc = new ListBuffer<>();
|
ListBuffer<Integer> loc = new ListBuffer<>();
|
||||||
for (int i = 0; i < len * TypeAnnotationPosition.TypePathEntry.bytesPerEntry; ++i)
|
for (int i = 0; i < len * TypeAnnotationPosition.TypePathEntry.bytesPerEntry; ++i)
|
||||||
loc = loc.append(nextByte());
|
loc = loc.append(nextByte());
|
||||||
position.location = TypeAnnotationPosition.getTypePathFromBinary(loc.toList());
|
|
||||||
}
|
|
||||||
|
|
||||||
return position;
|
return TypeAnnotationPosition.getTypePathFromBinary(loc.toList());
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Attribute readAttributeValue() {
|
Attribute readAttributeValue() {
|
||||||
|
@ -2299,6 +2299,9 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
|
|||||||
|
|
||||||
// Attribute.Compound if tag is ANNOTATION
|
// Attribute.Compound if tag is ANNOTATION
|
||||||
// Attribute.TypeCompound if tag is TYPE_ANNOTATION
|
// Attribute.TypeCompound if tag is TYPE_ANNOTATION
|
||||||
|
//
|
||||||
|
// NOTE: This field is slated for removal in the future. Do
|
||||||
|
// not use it for anything new.
|
||||||
public Attribute.Compound attribute;
|
public Attribute.Compound attribute;
|
||||||
|
|
||||||
protected JCAnnotation(Tag tag, JCTree annotationType, List<JCExpression> args) {
|
protected JCAnnotation(Tag tag, JCTree annotationType, List<JCExpression> args) {
|
||||||
|
Loading…
Reference in New Issue
Block a user