From ff77d06f17b21b8eaef8be17677d79aa578d00ce Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Wed, 15 Jan 2020 10:45:03 -0500 Subject: [PATCH] 8236597: issues inferring type annotations on records Reviewed-by: mcimadamore --- .../com/sun/tools/javac/code/Symbol.java | 4 +- .../sun/tools/javac/code/TypeAnnotations.java | 5 +- .../com/sun/tools/javac/comp/TypeEnter.java | 19 +- .../sun/tools/javac/parser/JavacParser.java | 3 +- .../TypeAnnotationsPositionsOnRecords.java | 240 ++++++++++++++++++ 5 files changed, 261 insertions(+), 10 deletions(-) create mode 100644 test/langtools/tools/javac/annotations/typeAnnotations/TypeAnnotationsPositionsOnRecords.java diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java index 1e5012f365c..37fb6c83203 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symbol.java @@ -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. * * 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.jvm.*; 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.JCVariableDecl; import com.sun.tools.javac.tree.JCTree.Tag; @@ -1733,6 +1734,7 @@ public abstract class Symbol extends AnnoConstruct implements PoolConstant, Elem @SuppressWarnings("preview") public static class RecordComponent extends VarSymbol implements RecordComponentElement { public MethodSymbol accessor; + public JCTree.JCMethodDecl accessorMeth; /** * Construct a record component, given its flags, name, type and owner. diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java index 35565055339..dd7662fdb9a 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java @@ -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. * * 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.defs); + if (tree.sym.isRecord()) { + tree.sym.getRecordComponents().stream().forEach(rc -> scan(rc.accessorMeth)); + } } /** diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/TypeEnter.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/TypeEnter.java index 8ce3b7b3486..63d3f9898e7 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/TypeEnter.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/TypeEnter.java @@ -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 * 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, - 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(), // thrown @@ -1050,6 +1057,7 @@ public class TypeEnter implements Completer { null); memberEnter.memberEnter(getter, env); rec.accessor = getter.sym; + rec.accessorMeth = getter; } else if (implSym != null) { rec.accessor = implSym; } @@ -1155,11 +1163,8 @@ public class TypeEnter implements Completer { field.sym.flags_field &= ~Flags.VARARGS; } // now lets add the accessors - tree.defs.stream() - .filter(t -> t.hasTag(VARDEF)) - .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) + recordFields.stream() + .filter(vd -> (lookupMethod(syms.objectType.tsym, vd.name, List.nil()) == null)) .forEach(vd -> addAccessor(vd, env)); } } diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java index e0e90788e0e..3a063c2c589 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java @@ -3746,7 +3746,8 @@ public class JavacParser implements Parser { ListBuffer tmpParams = new ListBuffer<>(); for (JCVariableDecl param : headerFields) { 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)); } methDef.params = tmpParams.toList(); diff --git a/test/langtools/tools/javac/annotations/typeAnnotations/TypeAnnotationsPositionsOnRecords.java b/test/langtools/tools/javac/annotations/typeAnnotations/TypeAnnotationsPositionsOnRecords.java new file mode 100644 index 00000000000..17b94730e54 --- /dev/null +++ b/test/langtools/tools/javac/annotations/typeAnnotations/TypeAnnotationsPositionsOnRecords.java @@ -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("")) { + 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 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 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 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 annos) { + findAnnotations(cf, m, Attribute.RuntimeVisibleTypeAnnotations, annos); + findAnnotations(cf, m, Attribute.RuntimeInvisibleTypeAnnotations, annos); + } + + void findAnnotations(ClassFile cf, Field m, List annos) { + findAnnotations(cf, m, Attribute.RuntimeVisibleTypeAnnotations, annos); + findAnnotations(cf, m, Attribute.RuntimeInvisibleTypeAnnotations, annos); + } + + void findAnnotations(ClassFile cf, Method m, String name, List 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 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)); + } + } +}