提交 323df2e0 编写于 作者: J jlahoda

8038788: javac behaves incorrectly for annotations after method type parameters in some cases

Reviewed-by: jjg, emc
上级 77b42f7e
......@@ -3406,16 +3406,28 @@ public class JavacParser implements Parser {
* | ModifiersOpt
* ( Type Ident
* ( VariableDeclaratorsRest ";" | MethodDeclaratorRest )
* | VOID Ident MethodDeclaratorRest
* | TypeParameters (Type | VOID) Ident MethodDeclaratorRest
* | VOID Ident VoidMethodDeclaratorRest
* | TypeParameters [Annotations]
* ( Type Ident MethodDeclaratorRest
* | VOID Ident VoidMethodDeclaratorRest
* )
* | Ident ConstructorDeclaratorRest
* | TypeParameters Ident ConstructorDeclaratorRest
* | ClassOrInterfaceOrEnumDeclaration
* )
* InterfaceBodyDeclaration =
* ";"
* | ModifiersOpt Type Ident
* ( ConstantDeclaratorsRest | InterfaceMethodDeclaratorRest ";" )
* | ModifiersOpt
* ( Type Ident
* ( ConstantDeclaratorsRest ";" | MethodDeclaratorRest )
* | VOID Ident MethodDeclaratorRest
* | TypeParameters [Annotations]
* ( Type Ident MethodDeclaratorRest
* | VOID Ident VoidMethodDeclaratorRest
* )
* | ClassOrInterfaceOrEnumDeclaration
* )
*
*/
protected List<JCTree> classOrInterfaceBodyDeclaration(Name className, boolean isInterface) {
if (token.kind == SEMI) {
......@@ -3444,28 +3456,29 @@ public class JavacParser implements Parser {
}
List<JCAnnotation> annosAfterParams = annotationsOpt(Tag.ANNOTATION);
if (annosAfterParams.nonEmpty()) {
checkAnnotationsAfterTypeParams(annosAfterParams.head.pos);
mods.annotations = mods.annotations.appendList(annosAfterParams);
if (mods.pos == Position.NOPOS)
mods.pos = mods.annotations.head.pos;
}
Token tk = token;
pos = token.pos;
JCExpression type;
boolean isVoid = token.kind == VOID;
if (isVoid) {
if (annosAfterParams.nonEmpty())
illegal(annosAfterParams.head.pos);
type = to(F.at(pos).TypeIdent(TypeTag.VOID));
nextToken();
} else {
if (annosAfterParams.nonEmpty()) {
checkAnnotationsAfterTypeParams(annosAfterParams.head.pos);
mods.annotations = mods.annotations.appendList(annosAfterParams);
if (mods.pos == Position.NOPOS)
mods.pos = mods.annotations.head.pos;
}
// method returns types are un-annotated types
type = unannotatedType();
}
if (token.kind == LPAREN && !isInterface && type.hasTag(IDENT)) {
if (isInterface || tk.name() != className)
error(pos, "invalid.meth.decl.ret.type.req");
else if (annosAfterParams.nonEmpty())
illegal(annosAfterParams.head.pos);
return List.of(methodDeclaratorRest(
pos, mods, null, names.init, typarams,
isInterface, true, dc));
......@@ -3497,13 +3510,9 @@ public class JavacParser implements Parser {
}
/** MethodDeclaratorRest =
* FormalParameters BracketsOpt [Throws TypeList] ( MethodBody | [DEFAULT AnnotationValue] ";")
* FormalParameters BracketsOpt [THROWS TypeList] ( MethodBody | [DEFAULT AnnotationValue] ";")
* VoidMethodDeclaratorRest =
* FormalParameters [Throws TypeList] ( MethodBody | ";")
* InterfaceMethodDeclaratorRest =
* FormalParameters BracketsOpt [THROWS TypeList] ";"
* VoidInterfaceMethodDeclaratorRest =
* FormalParameters [THROWS TypeList] ";"
* FormalParameters [THROWS TypeList] ( MethodBody | ";")
* ConstructorDeclaratorRest =
* "(" FormalParameterListOpt ")" [THROWS TypeList] MethodBody
*/
......
/*
* Copyright (c) 2014, 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 8038788
* @summary Verify proper handling of annotations after method's type parameters.
* @build AfterMethodTypeParams
* @run main AfterMethodTypeParams
*/
import java.io.IOException;
import java.io.StringWriter;
import java.net.URI;
import java.util.*;
import javax.lang.model.element.Name;
import javax.tools.*;
import com.sun.source.tree.*;
import com.sun.source.util.*;
public class AfterMethodTypeParams {
public static void main(String... args) throws IOException {
new AfterMethodTypeParams().run();
}
void run() throws IOException {
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
for (TestCase tc : testCases) {
String test = TEMPLATE.replace("CONTENT", tc.snippet);
List<JavaFileObject> files = Arrays.asList(new MyFileObject(test));
StringWriter out = new StringWriter();
List<String> options = Arrays.asList("-XDrawDiagnostics", "-XDshouldStopPolicy=FLOW");
JavacTask task = (JavacTask) compiler.getTask(out, null, null, options, null, files);
new TreePathScanner<Void, Void>() {
boolean seenAnnotation;
@Override
public Void visitAnnotation(AnnotationTree node, Void p) {
Name name = ((IdentifierTree) node.getAnnotationType()).getName();
seenAnnotation |= name.contentEquals("TA") || name.contentEquals("DA");
return null;
}
@Override
public Void visitCompilationUnit(CompilationUnitTree node, Void p) {
super.visitCompilationUnit(node, p);
if (!seenAnnotation)
error(test, "Annotation was missing");
return null;
}
}.scan(task.parse(), null);
task.analyze();
if (!tc.error.equals(out.toString().trim())) {
error(test, "Incorrect errors: " + out.toString());
}
}
if (errors > 0) {
throw new IllegalStateException("Errors found");
}
}
int errors;
void error(String code, String error) {
System.out.println("Error detected: " + error);
System.out.println("Code:");
System.out.println(code);
errors++;
}
static String TEMPLATE =
"import java.lang.annotation.*;\n" +
"public class Test {\n" +
" CONTENT\n" +
"}\n" +
"@Target({ElementType.METHOD, ElementType.CONSTRUCTOR})\n" +
"@interface DA { }\n" +
"@Target(ElementType.TYPE_USE)\n" +
"@interface TA { }\n";
static class MyFileObject extends SimpleJavaFileObject {
final String text;
public MyFileObject(String text) {
super(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE);
this.text = text;
}
@Override
public CharSequence getCharContent(boolean ignoreEncodingErrors) {
return text;
}
}
static TestCase[] testCases = new TestCase[] {
new TestCase("<T> @DA int foo1() { return 0;}", ""),
new TestCase("<T> @DA void foo2() { }", ""),
new TestCase("<T> @TA int foo3() { return 0;}", ""),
new TestCase("<T> @TA void foo4() { }",
"Test.java:3:9: compiler.err.annotation.type.not.applicable"),
new TestCase("<T> @DA Test() { }", "Test.java:3:9: compiler.err.illegal.start.of.type"),
new TestCase("<T> @TA Test() { }", "Test.java:3:9: compiler.err.illegal.start.of.type"),
};
static class TestCase {
final String snippet;
final String error;
public TestCase(String snippet, String error) {
this.snippet = snippet;
this.error = error;
}
}
}
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册