From 37880ba60d6b68bd9050a8ebc87a61e2ae8a07d2 Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Thu, 9 Jul 2020 17:37:53 -0400 Subject: [PATCH] 8242529: javac defines type annotations incorrectly for record members (constructor and property accessor) Reviewed-by: psandoz, jlahoda --- .../com/sun/tools/javac/comp/TypeEnter.java | 11 ++- .../javac/diags/examples/KindnameRecord.java | 5 +- ...otPreservingNestedTypeAnnotationsTest.java | 73 +++++++++++++++++++ 3 files changed, 83 insertions(+), 6 deletions(-) create mode 100644 test/langtools/tools/javac/processing/model/element/RecordNotPreservingNestedTypeAnnotationsTest.java 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 be8f866684e..20ad15bd7b0 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 @@ -1087,9 +1087,11 @@ 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 */ + TreeCopier tc = new TreeCopier(make.at(tree.pos)); List originalAnnos = rec.getOriginalAnnos().isEmpty() ? rec.getOriginalAnnos() : - new TreeCopier(make.at(tree.pos)).copy(rec.getOriginalAnnos()); + tc.copy(rec.getOriginalAnnos()); + JCVariableDecl recordField = TreeInfo.recordFields((JCClassDecl) env.tree).stream().filter(rf -> rf.name == tree.name).findAny().get(); JCMethodDecl getter = make.at(tree.pos). MethodDef( make.Modifiers(PUBLIC | Flags.GENERATED_MEMBER, originalAnnos), @@ -1099,7 +1101,7 @@ public class TypeEnter implements Completer { * 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), + tc.copy(recordField.vartype), List.nil(), List.nil(), List.nil(), // thrown @@ -1404,10 +1406,11 @@ public class TypeEnter implements Completer { * parameter in the constructor. */ RecordComponent rc = ((ClassSymbol) owner).getRecordComponent(arg.sym); + TreeCopier tc = new TreeCopier(make.at(arg.pos)); arg.mods.annotations = rc.getOriginalAnnos().isEmpty() ? List.nil() : - new TreeCopier(make.at(arg.pos)).copy(rc.getOriginalAnnos()); - arg.vartype = tmpRecordFieldDecls.head.vartype; + tc.copy(rc.getOriginalAnnos()); + arg.vartype = tc.copy(tmpRecordFieldDecls.head.vartype); tmpRecordFieldDecls = tmpRecordFieldDecls.tail; } return md; diff --git a/test/langtools/tools/javac/diags/examples/KindnameRecord.java b/test/langtools/tools/javac/diags/examples/KindnameRecord.java index dff2c5242f3..3ee2450e9ed 100644 --- a/test/langtools/tools/javac/diags/examples/KindnameRecord.java +++ b/test/langtools/tools/javac/diags/examples/KindnameRecord.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2010, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2010, 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 @@ -23,7 +23,8 @@ // key: compiler.misc.kindname.record.component // key: compiler.misc.kindname.record -// key: compiler.misc.count.error +// key: compiler.misc.count.error.plural +// key: compiler.misc.kindname.method // key: compiler.err.already.defined // key: compiler.err.error // key: compiler.note.preview.filename diff --git a/test/langtools/tools/javac/processing/model/element/RecordNotPreservingNestedTypeAnnotationsTest.java b/test/langtools/tools/javac/processing/model/element/RecordNotPreservingNestedTypeAnnotationsTest.java new file mode 100644 index 00000000000..425753b065b --- /dev/null +++ b/test/langtools/tools/javac/processing/model/element/RecordNotPreservingNestedTypeAnnotationsTest.java @@ -0,0 +1,73 @@ +/* + * 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 8242529 + * @summary javac defines type annotations incorrectly for record members (constructor and property accessor) + * @modules + * jdk.compiler/com.sun.tools.javac.util + * @compile --enable-preview -source ${jdk.version} RecordNotPreservingNestedTypeAnnotationsTest.java + * @run main/othervm --enable-preview RecordNotPreservingNestedTypeAnnotationsTest + */ + +import java.lang.annotation.Annotation; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; +import java.lang.reflect.*; +import java.util.concurrent.Callable; +import com.sun.tools.javac.util.Assert; + +public record RecordNotPreservingNestedTypeAnnotationsTest(@RegularAnnotation @TypeAnnotation Callable<@TypeAnnotation ?> foo) { + public static void main(String[] args) throws Exception { + RecordComponent recordComponent = RecordNotPreservingNestedTypeAnnotationsTest.class.getRecordComponents()[0]; + checkAnnotations(recordComponent.getAnnotations(), recordComponent.getAnnotatedType()); + + Method accessor = recordComponent.getAccessor(); + checkAnnotations(accessor.getAnnotations(), accessor.getAnnotatedReturnType()); + + Constructor constructor = RecordNotPreservingNestedTypeAnnotationsTest.class.getConstructor(Callable.class); + checkAnnotations(constructor.getParameterAnnotations()[0], constructor.getAnnotatedParameterTypes()[0]); + + Field field = RecordNotPreservingNestedTypeAnnotationsTest.class.getDeclaredField(recordComponent.getName()); + checkAnnotations(field.getAnnotations(), field.getAnnotatedType()); + } + + static void checkAnnotations(Annotation[] decAnnotations, AnnotatedType annoType) { + Assert.check(decAnnotations.length == 1); + Assert.check(decAnnotations[0] instanceof RegularAnnotation); + + Assert.check(annoType.getAnnotations()[0] instanceof TypeAnnotation); + var annoTypeArgs = ((AnnotatedParameterizedType) annoType).getAnnotatedActualTypeArguments(); + Assert.check(annoTypeArgs.length == 1); + Assert.check(annoTypeArgs[0].getAnnotations()[0] instanceof TypeAnnotation); + } + + @Retention(RetentionPolicy.RUNTIME) + @Target({ElementType.TYPE_USE}) + @interface TypeAnnotation {} + + @Retention(RetentionPolicy.RUNTIME) + @interface RegularAnnotation {} +}