8236597: issues inferring type annotations on records
Reviewed-by: mcimadamore
This commit is contained in:
parent
c0dce25756
commit
ff77d06f17
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1999, 2019, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1999, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -57,6 +57,7 @@ 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.jvm.PoolConstant;
|
import com.sun.tools.javac.jvm.PoolConstant;
|
||||||
|
import com.sun.tools.javac.tree.JCTree;
|
||||||
import com.sun.tools.javac.tree.JCTree.JCFieldAccess;
|
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.tree.JCTree.Tag;
|
||||||
@ -1733,6 +1734,7 @@ public abstract class Symbol extends AnnoConstruct implements PoolConstant, Elem
|
|||||||
@SuppressWarnings("preview")
|
@SuppressWarnings("preview")
|
||||||
public static class RecordComponent extends VarSymbol implements RecordComponentElement {
|
public static class RecordComponent extends VarSymbol implements RecordComponentElement {
|
||||||
public MethodSymbol accessor;
|
public MethodSymbol accessor;
|
||||||
|
public JCTree.JCMethodDecl accessorMeth;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a record component, given its flags, name, type and owner.
|
* Construct a record component, given its flags, name, type and owner.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2009, 2019, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2009, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -1129,6 +1129,9 @@ public class TypeAnnotations {
|
|||||||
scan(tree.implementing);
|
scan(tree.implementing);
|
||||||
}
|
}
|
||||||
scan(tree.defs);
|
scan(tree.defs);
|
||||||
|
if (tree.sym.isRecord()) {
|
||||||
|
tree.sym.getRecordComponents().stream().forEach(rc -> scan(rc.accessorMeth));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1040,9 +1040,16 @@ public class TypeEnter implements Completer {
|
|||||||
* it could be that some of those annotations are not applicable to the accessor, they will be striped
|
* it could be that some of those annotations are not applicable to the accessor, they will be striped
|
||||||
* away later at Check::validateAnnotation
|
* away later at Check::validateAnnotation
|
||||||
*/
|
*/
|
||||||
JCMethodDecl getter = make.at(tree.pos).MethodDef(make.Modifiers(Flags.PUBLIC | Flags.GENERATED_MEMBER, tree.mods.annotations),
|
JCMethodDecl getter = make.at(tree.pos).
|
||||||
|
MethodDef(
|
||||||
|
make.Modifiers(Flags.PUBLIC | Flags.GENERATED_MEMBER, tree.mods.annotations),
|
||||||
tree.sym.name,
|
tree.sym.name,
|
||||||
make.Type(tree.sym.type),
|
/* we need to special case for the case when the user declared the type as an ident
|
||||||
|
* if we don't do that then we can have issues if type annotations are applied to the
|
||||||
|
* return type: javac issues an error if a type annotation is applied to java.lang.String
|
||||||
|
* but applying a type annotation to String is kosher
|
||||||
|
*/
|
||||||
|
tree.vartype.hasTag(IDENT) ? make.Ident(tree.vartype.type.tsym) : make.Type(tree.sym.type),
|
||||||
List.nil(),
|
List.nil(),
|
||||||
List.nil(),
|
List.nil(),
|
||||||
List.nil(), // thrown
|
List.nil(), // thrown
|
||||||
@ -1050,6 +1057,7 @@ public class TypeEnter implements Completer {
|
|||||||
null);
|
null);
|
||||||
memberEnter.memberEnter(getter, env);
|
memberEnter.memberEnter(getter, env);
|
||||||
rec.accessor = getter.sym;
|
rec.accessor = getter.sym;
|
||||||
|
rec.accessorMeth = getter;
|
||||||
} else if (implSym != null) {
|
} else if (implSym != null) {
|
||||||
rec.accessor = implSym;
|
rec.accessor = implSym;
|
||||||
}
|
}
|
||||||
@ -1155,11 +1163,8 @@ public class TypeEnter implements Completer {
|
|||||||
field.sym.flags_field &= ~Flags.VARARGS;
|
field.sym.flags_field &= ~Flags.VARARGS;
|
||||||
}
|
}
|
||||||
// now lets add the accessors
|
// now lets add the accessors
|
||||||
tree.defs.stream()
|
recordFields.stream()
|
||||||
.filter(t -> t.hasTag(VARDEF))
|
.filter(vd -> (lookupMethod(syms.objectType.tsym, vd.name, List.nil()) == null))
|
||||||
.map(t -> (JCVariableDecl) t)
|
|
||||||
// lets stay clear of adding a forbidden name, javac will fail later anyway
|
|
||||||
.filter(vd -> (vd.sym.flags_field & RECORD) != 0 && lookupMethod(syms.objectType.tsym, vd.name, List.nil()) == null)
|
|
||||||
.forEach(vd -> addAccessor(vd, env));
|
.forEach(vd -> addAccessor(vd, env));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3746,7 +3746,8 @@ public class JavacParser implements Parser {
|
|||||||
ListBuffer<JCVariableDecl> tmpParams = new ListBuffer<>();
|
ListBuffer<JCVariableDecl> tmpParams = new ListBuffer<>();
|
||||||
for (JCVariableDecl param : headerFields) {
|
for (JCVariableDecl param : headerFields) {
|
||||||
tmpParams.add(F.at(param)
|
tmpParams.add(F.at(param)
|
||||||
.VarDef(F.Modifiers(Flags.PARAMETER | param.mods.flags & Flags.VARARGS),
|
// we will get flags plus annotations from the record component
|
||||||
|
.VarDef(F.Modifiers(Flags.PARAMETER | param.mods.flags & Flags.VARARGS | param.mods.flags & Flags.FINAL, param.mods.annotations),
|
||||||
param.name, param.vartype, null));
|
param.name, param.vartype, null));
|
||||||
}
|
}
|
||||||
methDef.params = tmpParams.toList();
|
methDef.params = tmpParams.toList();
|
||||||
|
@ -0,0 +1,240 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @summary Verify location of type annotations on records
|
||||||
|
* @library /tools/lib
|
||||||
|
* @modules
|
||||||
|
* jdk.jdeps/com.sun.tools.classfile
|
||||||
|
* jdk.compiler/com.sun.tools.javac.api
|
||||||
|
* jdk.compiler/com.sun.tools.javac.main
|
||||||
|
* jdk.compiler/com.sun.tools.javac.code
|
||||||
|
* jdk.compiler/com.sun.tools.javac.util
|
||||||
|
* @build toolbox.ToolBox toolbox.JavacTask
|
||||||
|
* @compile --enable-preview -source ${jdk.version} TypeAnnotationsPositionsOnRecords.java
|
||||||
|
* @run main/othervm --enable-preview TypeAnnotationsPositionsOnRecords
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
|
||||||
|
import java.lang.annotation.*;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
import com.sun.tools.classfile.*;
|
||||||
|
import com.sun.tools.javac.util.Assert;
|
||||||
|
|
||||||
|
import toolbox.JavacTask;
|
||||||
|
import toolbox.ToolBox;
|
||||||
|
|
||||||
|
public class TypeAnnotationsPositionsOnRecords {
|
||||||
|
|
||||||
|
final String src =
|
||||||
|
"""
|
||||||
|
import java.lang.annotation.*;
|
||||||
|
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target({ ElementType.TYPE_USE })
|
||||||
|
@interface Nullable {}
|
||||||
|
|
||||||
|
record Record1(@Nullable String t) {}
|
||||||
|
|
||||||
|
record Record2(@Nullable String t) {
|
||||||
|
public Record2 {}
|
||||||
|
}
|
||||||
|
|
||||||
|
record Record3(@Nullable String t1, @Nullable String t2) {}
|
||||||
|
|
||||||
|
record Record4(@Nullable String t1, @Nullable String t2) {
|
||||||
|
public Record4 {}
|
||||||
|
}
|
||||||
|
|
||||||
|
record Record5(String t1, @Nullable String t2) {}
|
||||||
|
|
||||||
|
record Record6(String t1, @Nullable String t2) {
|
||||||
|
public Record6 {}
|
||||||
|
}
|
||||||
|
""";
|
||||||
|
|
||||||
|
public static void main(String... args) throws Exception {
|
||||||
|
new TypeAnnotationsPositionsOnRecords().run();
|
||||||
|
}
|
||||||
|
|
||||||
|
ToolBox tb = new ToolBox();
|
||||||
|
|
||||||
|
void run() throws Exception {
|
||||||
|
compileTestClass();
|
||||||
|
checkClassFile(new File(Paths.get(System.getProperty("user.dir"),
|
||||||
|
"Record1.class").toUri()), 0);
|
||||||
|
checkClassFile(new File(Paths.get(System.getProperty("user.dir"),
|
||||||
|
"Record2.class").toUri()), 0);
|
||||||
|
checkClassFile(new File(Paths.get(System.getProperty("user.dir"),
|
||||||
|
"Record3.class").toUri()), 0, 1);
|
||||||
|
checkClassFile(new File(Paths.get(System.getProperty("user.dir"),
|
||||||
|
"Record4.class").toUri()), 0, 1);
|
||||||
|
checkClassFile(new File(Paths.get(System.getProperty("user.dir"),
|
||||||
|
"Record5.class").toUri()), 1);
|
||||||
|
checkClassFile(new File(Paths.get(System.getProperty("user.dir"),
|
||||||
|
"Record6.class").toUri()), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void compileTestClass() throws Exception {
|
||||||
|
new JavacTask(tb)
|
||||||
|
.sources(src)
|
||||||
|
.options("--enable-preview", "-source", Integer.toString(Runtime.version().feature()))
|
||||||
|
.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
void checkClassFile(final File cfile, int... taPositions) throws Exception {
|
||||||
|
ClassFile classFile = ClassFile.read(cfile);
|
||||||
|
int accessorPos = 0;
|
||||||
|
int checkedAccessors = 0;
|
||||||
|
for (Method method : classFile.methods) {
|
||||||
|
String methodName = method.getName(classFile.constant_pool);
|
||||||
|
if (methodName.equals("toString") || methodName.equals("hashCode") || methodName.equals("equals")) {
|
||||||
|
// ignore
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (methodName.equals("<init>")) {
|
||||||
|
checkConstructor(classFile, method, taPositions);
|
||||||
|
} else {
|
||||||
|
for (int taPos : taPositions) {
|
||||||
|
if (taPos == accessorPos) {
|
||||||
|
checkAccessor(classFile, method);
|
||||||
|
checkedAccessors++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
accessorPos++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
checkFields(classFile, taPositions);
|
||||||
|
Assert.check(checkedAccessors == taPositions.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* there can be several parameters annotated we have to check that the ones annotated are the
|
||||||
|
* expected ones
|
||||||
|
*/
|
||||||
|
void checkConstructor(ClassFile classFile, Method method, int... positions) throws Exception {
|
||||||
|
List<TypeAnnotation> annos = new ArrayList<>();
|
||||||
|
findAnnotations(classFile, method, annos);
|
||||||
|
Assert.check(annos.size() == positions.length);
|
||||||
|
int i = 0;
|
||||||
|
for (int pos : positions) {
|
||||||
|
TypeAnnotation ta = annos.get(i);
|
||||||
|
Assert.check(ta.position.type.toString().equals("METHOD_FORMAL_PARAMETER"));
|
||||||
|
Assert.check(ta.position.parameter_index == pos);
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* this case is simpler as there can only be one annotation at the accessor and it has to be applied
|
||||||
|
* at the return type
|
||||||
|
*/
|
||||||
|
void checkAccessor(ClassFile classFile, Method method) {
|
||||||
|
List<TypeAnnotation> annos = new ArrayList<>();
|
||||||
|
findAnnotations(classFile, method, annos);
|
||||||
|
Assert.check(annos.size() == 1);
|
||||||
|
TypeAnnotation ta = annos.get(0);
|
||||||
|
Assert.check(ta.position.type.toString().equals("METHOD_RETURN"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* here we have to check that only the fields for which its position matches with the one of the
|
||||||
|
* original annotated record component are annotated
|
||||||
|
*/
|
||||||
|
void checkFields(ClassFile classFile, int... positions) {
|
||||||
|
if (positions != null && positions.length > 0) {
|
||||||
|
int fieldPos = 0;
|
||||||
|
int annotationPos = 0;
|
||||||
|
int currentAnnoPosition = positions[annotationPos];
|
||||||
|
int annotatedFields = 0;
|
||||||
|
for (Field field : classFile.fields) {
|
||||||
|
List<TypeAnnotation> annos = new ArrayList<>();
|
||||||
|
findAnnotations(classFile, field, annos);
|
||||||
|
if (fieldPos != currentAnnoPosition) {
|
||||||
|
Assert.check(annos.size() == 0);
|
||||||
|
} else {
|
||||||
|
Assert.check(annos.size() == 1);
|
||||||
|
TypeAnnotation ta = annos.get(0);
|
||||||
|
Assert.check(ta.position.type.toString().equals("FIELD"));
|
||||||
|
annotationPos++;
|
||||||
|
currentAnnoPosition = annotationPos < positions.length ? positions[annotationPos] : -1;
|
||||||
|
annotatedFields++;
|
||||||
|
}
|
||||||
|
fieldPos++;
|
||||||
|
}
|
||||||
|
Assert.check(annotatedFields == positions.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// utility methods
|
||||||
|
void findAnnotations(ClassFile cf, Method m, List<TypeAnnotation> annos) {
|
||||||
|
findAnnotations(cf, m, Attribute.RuntimeVisibleTypeAnnotations, annos);
|
||||||
|
findAnnotations(cf, m, Attribute.RuntimeInvisibleTypeAnnotations, annos);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findAnnotations(ClassFile cf, Field m, List<TypeAnnotation> annos) {
|
||||||
|
findAnnotations(cf, m, Attribute.RuntimeVisibleTypeAnnotations, annos);
|
||||||
|
findAnnotations(cf, m, Attribute.RuntimeInvisibleTypeAnnotations, annos);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findAnnotations(ClassFile cf, Method m, String name, List<TypeAnnotation> annos) {
|
||||||
|
int index = m.attributes.getIndex(cf.constant_pool, name);
|
||||||
|
if (index != -1) {
|
||||||
|
Attribute attr = m.attributes.get(index);
|
||||||
|
assert attr instanceof RuntimeTypeAnnotations_attribute;
|
||||||
|
RuntimeTypeAnnotations_attribute tAttr = (RuntimeTypeAnnotations_attribute)attr;
|
||||||
|
annos.addAll(Arrays.asList(tAttr.annotations));
|
||||||
|
}
|
||||||
|
|
||||||
|
int cindex = m.attributes.getIndex(cf.constant_pool, Attribute.Code);
|
||||||
|
if (cindex != -1) {
|
||||||
|
Attribute cattr = m.attributes.get(cindex);
|
||||||
|
assert cattr instanceof Code_attribute;
|
||||||
|
Code_attribute cAttr = (Code_attribute)cattr;
|
||||||
|
index = cAttr.attributes.getIndex(cf.constant_pool, name);
|
||||||
|
if (index != -1) {
|
||||||
|
Attribute attr = cAttr.attributes.get(index);
|
||||||
|
assert attr instanceof RuntimeTypeAnnotations_attribute;
|
||||||
|
RuntimeTypeAnnotations_attribute tAttr = (RuntimeTypeAnnotations_attribute)attr;
|
||||||
|
annos.addAll(Arrays.asList(tAttr.annotations));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void findAnnotations(ClassFile cf, Field m, String name, List<TypeAnnotation> annos) {
|
||||||
|
int index = m.attributes.getIndex(cf.constant_pool, name);
|
||||||
|
if (index != -1) {
|
||||||
|
Attribute attr = m.attributes.get(index);
|
||||||
|
assert attr instanceof RuntimeTypeAnnotations_attribute;
|
||||||
|
RuntimeTypeAnnotations_attribute tAttr = (RuntimeTypeAnnotations_attribute)attr;
|
||||||
|
annos.addAll(Arrays.asList(tAttr.annotations));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user