diff --git a/src/java.compiler/share/classes/javax/annotation/processing/AbstractProcessor.java b/src/java.compiler/share/classes/javax/annotation/processing/AbstractProcessor.java index 1d934a27ef5..b1fe372ea01 100644 --- a/src/java.compiler/share/classes/javax/annotation/processing/AbstractProcessor.java +++ b/src/java.compiler/share/classes/javax/annotation/processing/AbstractProcessor.java @@ -25,6 +25,7 @@ package javax.annotation.processing; +import java.util.List; import java.util.Set; import java.util.HashSet; import java.util.Collections; @@ -80,10 +81,9 @@ public abstract class AbstractProcessor implements Processor { */ public Set getSupportedOptions() { SupportedOptions so = this.getClass().getAnnotation(SupportedOptions.class); - if (so == null) - return Collections.emptySet(); - else - return arrayToSet(so.value(), false); + return (so == null) ? + Set.of() : + arrayToSet(so.value(), false, "option value", "@SupportedOptions"); } /** @@ -110,12 +110,13 @@ public abstract class AbstractProcessor implements Processor { "No SupportedAnnotationTypes annotation " + "found on " + this.getClass().getName() + ", returning an empty set."); - return Collections.emptySet(); + return Set.of(); } else { boolean stripModulePrefixes = initialized && processingEnv.getSourceVersion().compareTo(SourceVersion.RELEASE_8) <= 0; - return arrayToSet(sat.value(), stripModulePrefixes); + return arrayToSet(sat.value(), stripModulePrefixes, + "annotation type", "@SupportedAnnotationTypes"); } } @@ -181,7 +182,7 @@ public abstract class AbstractProcessor implements Processor { AnnotationMirror annotation, ExecutableElement member, String userText) { - return Collections.emptyList(); + return List.of(); } /** @@ -195,17 +196,33 @@ public abstract class AbstractProcessor implements Processor { return initialized; } - private static Set arrayToSet(String[] array, - boolean stripModulePrefixes) { + private Set arrayToSet(String[] array, + boolean stripModulePrefixes, + String contentType, + String annotationName) { assert array != null; - Set set = new HashSet<>(array.length); + Set set = new HashSet<>(); for (String s : array) { + boolean stripped = false; if (stripModulePrefixes) { int index = s.indexOf('/'); - if (index != -1) + if (index != -1) { s = s.substring(index + 1); + stripped = true; + } + } + boolean added = set.add(s); + // Don't issue a duplicate warning when the module name is + // stripped off to avoid spurious warnings in a case like + // "foo/a.B", "bar/a.B". + if (!added && !stripped && isInitialized() ) { + processingEnv.getMessager().printMessage(Diagnostic.Kind.WARNING, + "Duplicate " + contentType + + " ``" + s + "'' for processor " + + this.getClass().getName() + + " in its " + annotationName + + "annotation."); } - set.add(s); } return Collections.unmodifiableSet(set); } diff --git a/src/java.compiler/share/classes/javax/annotation/processing/Processor.java b/src/java.compiler/share/classes/javax/annotation/processing/Processor.java index 86f2696a7ce..9e34f914510 100644 --- a/src/java.compiler/share/classes/javax/annotation/processing/Processor.java +++ b/src/java.compiler/share/classes/javax/annotation/processing/Processor.java @@ -252,7 +252,7 @@ public interface Processor { * "/"} character. For example, if a processor supports {@code * "a.B"}, this can include multiple annotation types named {@code * a.B} which reside in different modules. To only support {@code - * a.B} in the {@code Foo} module, instead use {@code "Foo/a.B"}. + * a.B} in the {@code foo} module, instead use {@code "foo/a.B"}. * * If a module name is included, only an annotation in that module * is matched. In particular, if a module name is given in an diff --git a/test/langtools/tools/javac/processing/warnings/TestRepeatedSupportedItems.java b/test/langtools/tools/javac/processing/warnings/TestRepeatedSupportedItems.java new file mode 100644 index 00000000000..76fb603bee8 --- /dev/null +++ b/test/langtools/tools/javac/processing/warnings/TestRepeatedSupportedItems.java @@ -0,0 +1,66 @@ +/* + * Copyright (c) 2006, 2019, 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 + * @bug 8146726 + * @summary Test that warnings about repeated supported options and annotation types output as expected. + * @compile TestRepeatedSupportedItems.java + * @compile/ref=au_8.out -XDrawDiagnostics -processor TestRepeatedSupportedItems -proc:only -source 8 -Xlint:-options TestRepeatedSupportedItems.java + * @compile/ref=au_current.out -XDrawDiagnostics -processor TestRepeatedSupportedItems -proc:only -Xlint:-options TestRepeatedSupportedItems.java + */ + +import java.lang.annotation.*; +import java.util.Set; +import java.util.HashSet; +import java.util.Arrays; +import javax.annotation.processing.*; +import javax.lang.model.SourceVersion; +import javax.lang.model.element.TypeElement; + +/** + * A warning should be issued by the logic in + * javax.annotation.processing.AbstractProcessor for the repeated + * information. The "Foo" option warnings occur regardless of source + * level. The number of times the Baz annotation type is repeated + * depends on whether or not the source level supports modules. + */ +@SupportedAnnotationTypes({"foo/Baz", "foo/Baz", "bar/Baz", "Baz", "Baz"}) +@SupportedOptions({"Foo", "Foo"}) +@Baz +public class TestRepeatedSupportedItems extends AbstractProcessor { + + @Override + public SourceVersion getSupportedSourceVersion() { + return SourceVersion.latest(); + } + + public boolean process(Set annotations, + RoundEnvironment roundEnvironment) { + return true; + } +} + +@Retention(RetentionPolicy.RUNTIME) +@interface Baz { +} diff --git a/test/langtools/tools/javac/processing/warnings/au_8.out b/test/langtools/tools/javac/processing/warnings/au_8.out new file mode 100644 index 00000000000..2b0a04434a6 --- /dev/null +++ b/test/langtools/tools/javac/processing/warnings/au_8.out @@ -0,0 +1,4 @@ +- compiler.warn.proc.messager: Duplicate annotation type ``Baz'' for processor TestRepeatedSupportedItems in its @SupportedAnnotationTypesannotation. +- compiler.warn.proc.messager: Duplicate annotation type ``Baz'' for processor TestRepeatedSupportedItems in its @SupportedAnnotationTypesannotation. +- compiler.warn.proc.messager: Duplicate option value ``Foo'' for processor TestRepeatedSupportedItems in its @SupportedOptionsannotation. +3 warnings diff --git a/test/langtools/tools/javac/processing/warnings/au_current.out b/test/langtools/tools/javac/processing/warnings/au_current.out new file mode 100644 index 00000000000..1c19c9a148d --- /dev/null +++ b/test/langtools/tools/javac/processing/warnings/au_current.out @@ -0,0 +1,4 @@ +- compiler.warn.proc.messager: Duplicate annotation type ``foo/Baz'' for processor TestRepeatedSupportedItems in its @SupportedAnnotationTypesannotation. +- compiler.warn.proc.messager: Duplicate annotation type ``Baz'' for processor TestRepeatedSupportedItems in its @SupportedAnnotationTypesannotation. +- compiler.warn.proc.messager: Duplicate option value ``Foo'' for processor TestRepeatedSupportedItems in its @SupportedOptionsannotation. +3 warnings