From f8e08a996521d93e11b2f0e32945a2d477e2d67b Mon Sep 17 00:00:00 2001 From: Joe Darcy Date: Thu, 17 Oct 2013 19:10:19 -0700 Subject: [PATCH 01/28] 8026838: Fix new doclint issues in javax.annotation.processing Reviewed-by: jjg --- .../classes/javax/annotation/processing/Processor.java | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/langtools/src/share/classes/javax/annotation/processing/Processor.java b/langtools/src/share/classes/javax/annotation/processing/Processor.java index c898e74255f..637c5a9c73c 100644 --- a/langtools/src/share/classes/javax/annotation/processing/Processor.java +++ b/langtools/src/share/classes/javax/annotation/processing/Processor.java @@ -209,11 +209,11 @@ public interface Processor { *
*
SupportedOptionString: *
Identifiers - *

+ * *

Identifiers: *
Identifier *
Identifier {@code .} Identifiers - *

+ * *

Identifier: *
Syntactic identifier, including keywords and literals *
@@ -250,7 +250,7 @@ public interface Processor { *
SupportedAnnotationTypeString: *
TypeName DotStaropt *
* - *

+ * *

DotStar: *
. * * From cf30c203379008ebae37bf00f1839a69cd53ca26 Mon Sep 17 00:00:00 2001 From: Jonathan Gibbons Date: Fri, 18 Oct 2013 15:03:34 -0700 Subject: [PATCH 02/28] 8026749: Missing LV table in lambda bodies Reviewed-by: vromero, jlahoda --- .../com/sun/tools/javac/code/Flags.java | 8 +- .../com/sun/tools/javac/comp/Flow.java | 6 +- .../sun/tools/javac/comp/LambdaToMethod.java | 18 +- .../classes/com/sun/tools/javac/jvm/Gen.java | 4 +- .../javac/lambda/LocalVariableTable.java | 220 ++++++++++++++++++ 5 files changed, 248 insertions(+), 8 deletions(-) create mode 100644 langtools/test/tools/javac/lambda/LocalVariableTable.java diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/Flags.java b/langtools/src/share/classes/com/sun/tools/javac/code/Flags.java index 2f59cfab7fd..3b1c40e2c92 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/Flags.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/Flags.java @@ -270,6 +270,11 @@ public class Flags { */ public static final long POTENTIALLY_AMBIGUOUS = 1L<<48; + /** + * Flag that marks a synthetic method body for a lambda expression + */ + public static final long LAMBDA_METHOD = 1L<<49; + /** Modifier masks. */ public static final int @@ -378,7 +383,8 @@ public class Flags { NOT_IN_PROFILE(Flags.NOT_IN_PROFILE), BAD_OVERRIDE(Flags.BAD_OVERRIDE), SIGNATURE_POLYMORPHIC(Flags.SIGNATURE_POLYMORPHIC), - THROWS(Flags.THROWS); + THROWS(Flags.THROWS), + LAMBDA_METHOD(Flags.LAMBDA_METHOD); Flag(long flag) { this.value = flag; diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Flow.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Flow.java index 8c69c814965..0c1b2d29970 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Flow.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Flow.java @@ -1718,9 +1718,9 @@ public class Flow { if (tree.body == null) { return; } - /* MemberEnter can generate synthetic methods, ignore them + /* Ignore synthetic methods, except for translated lambda methods. */ - if ((tree.sym.flags() & SYNTHETIC) != 0) { + if ((tree.sym.flags() & (SYNTHETIC | LAMBDA_METHOD)) == SYNTHETIC) { return; } @@ -1795,7 +1795,7 @@ public class Flow { protected void initParam(JCVariableDecl def) { inits.incl(def.sym.adr); uninits.excl(def.sym.adr); - } + } public void visitVarDef(JCVariableDecl tree) { boolean track = trackable(tree.sym); diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index e56aa535706..73fb8913371 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -30,6 +30,7 @@ import com.sun.tools.javac.tree.JCTree.JCMemberReference.ReferenceKind; import com.sun.tools.javac.tree.TreeMaker; import com.sun.tools.javac.tree.TreeTranslator; import com.sun.tools.javac.code.Attribute; +import com.sun.tools.javac.code.Flags; import com.sun.tools.javac.code.Kinds; import com.sun.tools.javac.code.Scope; import com.sun.tools.javac.code.Symbol; @@ -1315,7 +1316,9 @@ public class LambdaToMethod extends TreeTranslator { ListBuffer paramBuff = new ListBuffer(); int i = 0; for (List l = ptypes; l.nonEmpty(); l = l.tail) { - paramBuff.append(make.Param(make.paramName(i++), l.head, owner)); + JCVariableDecl param = make.Param(make.paramName(i++), l.head, owner); + param.sym.pos = tree.pos; + paramBuff.append(param); } List params = paramBuff.toList(); @@ -1755,7 +1758,7 @@ public class LambdaToMethod extends TreeTranslator { ((VarSymbol)ret).pos = ((VarSymbol)sym).pos; break; case CAPTURED_VAR: - ret = new VarSymbol(SYNTHETIC | FINAL, name, types.erasure(sym.type), translatedSym) { + ret = new VarSymbol(SYNTHETIC | FINAL | PARAMETER, name, types.erasure(sym.type), translatedSym) { @Override public Symbol baseSymbol() { //keep mapping with original captured symbol @@ -1763,8 +1766,17 @@ public class LambdaToMethod extends TreeTranslator { } }; break; + case LOCAL_VAR: + ret = new VarSymbol(FINAL, name, types.erasure(sym.type), translatedSym); + ((VarSymbol) ret).pos = ((VarSymbol) sym).pos; + break; + case PARAM: + ret = new VarSymbol(FINAL | PARAMETER, name, types.erasure(sym.type), translatedSym); + ((VarSymbol) ret).pos = ((VarSymbol) sym).pos; + break; default: ret = makeSyntheticVar(FINAL, name, types.erasure(sym.type), translatedSym); + ((VarSymbol) ret).pos = ((VarSymbol) sym).pos; } if (ret != sym) { ret.setDeclarationAttributes(sym.getRawAttributes()); @@ -1845,7 +1857,7 @@ public class LambdaToMethod extends TreeTranslator { // If instance access isn't needed, make it static. // Interface instance methods must be default methods. // Lambda methods are private synthetic. - translatedSym.flags_field = SYNTHETIC | + translatedSym.flags_field = SYNTHETIC | LAMBDA_METHOD | PRIVATE | (thisReferenced? (inInterface? DEFAULT : 0) : STATIC); diff --git a/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java b/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java index b48b727090f..34e61a011ee 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java +++ b/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java @@ -2892,7 +2892,8 @@ public class Gen extends JCTree.Visitor { @Override public void visitMethodDef(JCMethodDecl tree) { - if ((tree.sym.flags() & (SYNTHETIC | GENERATEDCONSTR)) != 0) { + if ((tree.sym.flags() & (SYNTHETIC | GENERATEDCONSTR)) != 0 + && (tree.sym.flags() & LAMBDA_METHOD) == 0) { return; } if (tree.name.equals(names.clinit)) { @@ -2906,6 +2907,7 @@ public class Gen extends JCTree.Visitor { return; } currentMethod = tree.sym; + super.visitMethodDef(tree); } diff --git a/langtools/test/tools/javac/lambda/LocalVariableTable.java b/langtools/test/tools/javac/lambda/LocalVariableTable.java new file mode 100644 index 00000000000..8a4d04a061b --- /dev/null +++ b/langtools/test/tools/javac/lambda/LocalVariableTable.java @@ -0,0 +1,220 @@ +/* + * Copyright (c) 2013, 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 8025998 8026749 + * @summary Missing LV table in lambda bodies + * @compile -g LocalVariableTable.java + * @run main LocalVariableTable + */ + +import java.io.*; +import java.lang.annotation.*; +import java.util.*; +import com.sun.tools.classfile.*; + +/* + * The test checks that a LocalVariableTable attribute is generated for the + * method bodies representing lambda expressions, and checks that the expected + * set of entries is found in the attribute. + * + * Since the bug was about missing entries in the LVT, not malformed entries, + * the test is not intended to be a detailed test of the contents of each + * LocalVariableTable entry: it is assumed that if a entry is present, it + * will have the correct contents. + * + * The test looks for test cases represented by nested classes whose + * name begins with "Lambda". Each such class contains a lambda expression + * that will mapped into a lambda method, and because the test is compiled + * with -g, these methods should have a LocalVariableTable. The set of + * expected names in the LVT is provided in an annotation on the class for + * the test case. + */ +public class LocalVariableTable { + public static void main(String... args) throws Exception { + new LocalVariableTable().run(); + } + + void run() throws Exception { + // the declared classes are returned in an unspecified order, + // so for neatness, sort them by name before processing them + Class[] classes = getClass().getDeclaredClasses(); + Arrays.sort(classes, (c1, c2) -> c1.getName().compareTo(c2.getName())); + + for (Class c : classes) { + if (c.getSimpleName().startsWith("Lambda")) + check(c); + } + if (errors > 0) + throw new Exception(errors + " errors found"); + } + + /** Check an individual test case. */ + void check(Class c) throws Exception { + System.err.println("Checking " + c.getSimpleName()); + + Expect expect = c.getAnnotation(Expect.class); + if (expect == null) { + error("@Expect not found for class " + c.getSimpleName()); + return; + } + + ClassFile cf = ClassFile.read(getClass().getResource(c.getName() + ".class").openStream()); + Method m = getLambdaMethod(cf); + if (m == null) { + error("lambda method not found"); + return; + } + + Code_attribute code = (Code_attribute) m.attributes.get(Attribute.Code); + if (code == null) { + error("Code attribute not found"); + return; + } + + LocalVariableTable_attribute lvt = + (LocalVariableTable_attribute) code.attributes.get(Attribute.LocalVariableTable); + if (lvt == null) { + error("LocalVariableTable attribute not found"); + return; + } + + Set foundNames = new LinkedHashSet<>(); + for (LocalVariableTable_attribute.Entry e: lvt.local_variable_table) { + foundNames.add(cf.constant_pool.getUTF8Value(e.name_index)); + } + + Set expectNames = new LinkedHashSet<>(Arrays.asList(expect.value())); + if (!foundNames.equals(expectNames)) { + Set foundOnly = new LinkedHashSet<>(foundNames); + foundOnly.removeAll(expectNames); + for (String s: foundOnly) + error("Unexpected name found: " + s); + Set expectOnly = new LinkedHashSet<>(expectNames); + expectOnly.removeAll(foundNames); + for (String s: expectOnly) + error("Expected name not found: " + s); + } + } + + /** Get a method whose name begins "lambda$...". */ + Method getLambdaMethod(ClassFile cf) throws ConstantPoolException { + for (Method m: cf.methods) { + if (m.getName(cf.constant_pool).startsWith("lambda$")) + return m; + } + return null; + } + + /** Report an error. */ + void error(String msg) { + System.err.println("Error: " + msg); + errors++; + } + + int errors; + + /** + * Annotation used to provide the set of names expected in the LVT attribute. + */ + @Retention(RetentionPolicy.RUNTIME) + @interface Expect { + String[] value(); + } + + /** Functional interface with nullary method. */ + interface Run0 { + public void run(); + } + + /** Functional interface with 1-ary method. */ + interface Run1 { + public void run(int a0); + } + + /** Functional interface with 2-ary method. */ + interface Run2 { + public void run(int a0, int a1); + } + + /* + * ---------- Test cases --------------------------------------------------- + */ + + @Expect({ "x" }) + static class Lambda_Args0_Local1 { + Run0 r = () -> { int x = 0; }; + } + + @Expect({ "x", "this" }) + static class Lambda_Args0_Local1_this { + int v; + Run0 r = () -> { int x = v; }; + } + + @Expect({ "a" }) + static class Lambda_Args1_Local0 { + Run1 r = (a) -> { }; + } + + @Expect({ "a", "x" }) + static class Lambda_Args1_Local1 { + Run1 r = (a) -> { int x = a; }; + } + + @Expect({ "a", "x" }) + static class Lambda_Args1_Local1_Captured1 { + void m() { + int v = 0; + Run1 r = (a) -> { int x = a + v; }; + } + } + + @Expect({ "a1", "a2", "x1", "x2", "this" }) + static class Lambda_Args2_Local2_Captured2_this { + int v; + void m() { + int v1 = 0; + int v2 = 0; + Run2 r = (a1, a2) -> { + int x1 = a1 + v1 + v; + int x2 = a2 + v2 + v; + }; + } + } + + @Expect({ "e" }) + static class Lambda_Try_Catch { + private static Runnable asUncheckedRunnable(Closeable c) { + return () -> { + try { + c.close(); + } catch (IOException e) { + throw new UncheckedIOException(e); + } + }; + } + } +} + From cee70e2272f58a97c03aa8faf696b008d96a7769 Mon Sep 17 00:00:00 2001 From: Bhavesh Patel Date: Fri, 18 Oct 2013 16:34:42 -0700 Subject: [PATCH 03/28] 8026567: Use meaningful style names for strong and italic styles Reviewed-by: jjg --- .../html/AbstractExecutableMemberWriter.java | 4 +- .../formats/html/AbstractIndexWriter.java | 4 +- .../formats/html/AbstractMemberWriter.java | 2 +- .../html/AnnotationTypeFieldWriterImpl.java | 4 +- ...nnotationTypeRequiredMemberWriterImpl.java | 4 +- .../html/AnnotationTypeWriterImpl.java | 6 +- .../doclets/formats/html/ClassWriterImpl.java | 6 +- .../formats/html/EnumConstantWriterImpl.java | 4 +- .../doclets/formats/html/FieldWriterImpl.java | 12 +-- .../doclets/formats/html/HelpWriter.java | 2 +- .../formats/html/HtmlDocletWriter.java | 4 +- .../formats/html/MethodWriterImpl.java | 12 +-- .../formats/html/NestedClassWriterImpl.java | 4 +- .../formats/html/PackageFrameWriter.java | 2 +- .../formats/html/PackageTreeWriter.java | 2 +- .../formats/html/PackageWriterImpl.java | 2 +- .../html/ProfilePackageFrameWriter.java | 2 +- .../html/ProfilePackageWriterImpl.java | 2 +- .../formats/html/ProfileWriterImpl.java | 2 +- .../formats/html/PropertyWriterImpl.java | 12 +-- .../formats/html/SubWriterHolderWriter.java | 8 +- .../formats/html/TagletWriterImpl.java | 20 ++-- .../doclets/formats/html/TreeWriter.java | 2 +- .../formats/html/markup/HtmlDocWriter.java | 2 +- .../formats/html/markup/HtmlStyle.java | 18 +++- .../internal/toolkit/resources/stylesheet.css | 6 +- .../com/sun/javadoc/AuthorDD/AuthorDD.java | 6 +- .../TestAnnotationTypes.java | 4 +- .../TestClassCrossReferences.java | 4 +- .../javadoc/testClassTree/TestClassTree.java | 12 +-- .../TestConstructorIndent.java | 4 +- .../TestDeprecatedDocs.java | 12 +-- .../TestExternalOverridenMethod.java | 6 +- .../com/sun/javadoc/testHref/TestHref.java | 4 +- .../TestHtmlDefinitionListTag.java | 100 +++++++++--------- .../testHtmlStrongTag/TestHtmlStrongTag.java | 6 +- .../com/sun/javadoc/testIndex/TestIndex.java | 16 +-- .../javadoc/testInterface/TestInterface.java | 6 +- .../sun/javadoc/testJavaFX/TestJavaFX.java | 18 ++-- .../sun/javadoc/testLegacyTaglet/Check.java | 2 +- .../testLinkOption/TestLinkOption.java | 6 +- .../TestMemberInheritence.java | 4 +- .../testMemberSummary/TestMemberSummary.java | 4 +- .../testNavigation/TestNavigation.java | 14 +-- .../TestNewLanguageFeatures.java | 74 ++++++------- .../TestOverridenMethodDocCopy.java | 4 +- .../TestOverridenPrivateMethods.java | 12 +-- ...verridenPrivateMethodsWithPackageFlag.java | 14 +-- ...verridenPrivateMethodsWithPrivateFlag.java | 14 +-- .../TestPackageDeprecation.java | 6 +- .../testParamTaglet/TestParamTaglet.java | 6 +- .../TestPrivateClasses.java | 24 ++--- .../javadoc/testProfiles/TestProfiles.java | 10 +- .../TestProfilesConfiguration.java | 4 +- .../TestSerializedFormDeprecationInfo.java | 28 ++--- .../javadoc/testSimpleTag/TestSimpleTag.java | 12 +-- .../TestSimpleTagInherit.java | 6 +- .../javadoc/testSinceTag/TestSinceTag.java | 6 +- .../javadoc/testTagOutput/TestTagOutput.java | 12 +-- .../sun/javadoc/testTaglets/TestTaglets.java | 2 +- .../sun/javadoc/testTaglets/taglets/Foo.java | 2 +- .../testThrowsHead/TestThrowsHead.java | 6 +- .../TestTypeAnnotations.java | 14 +-- .../javadoc/testValueTag/TestValueTag.java | 4 +- 64 files changed, 326 insertions(+), 310 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractExecutableMemberWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractExecutableMemberWriter.java index 23f4f00cd7f..4d8cd2c8616 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractExecutableMemberWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractExecutableMemberWriter.java @@ -101,10 +101,10 @@ public abstract class AbstractExecutableMemberWriter extends AbstractMemberWrite Content tdSummary) { ExecutableMemberDoc emd = (ExecutableMemberDoc)member; String name = emd.name(); - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getDocLink(context, cd, (MemberDoc) emd, name, false)); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); addParameters(emd, false, code, name.length() - 1); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractIndexWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractIndexWriter.java index 4b7ca05e384..51992fbcb67 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractIndexWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractIndexWriter.java @@ -176,7 +176,7 @@ public class AbstractIndexWriter extends HtmlDocletWriter { String name = (member instanceof ExecutableMemberDoc)? member.name() + ((ExecutableMemberDoc)member).flatSignature() : member.name(); - Content span = HtmlTree.SPAN(HtmlStyle.strong, + Content span = HtmlTree.SPAN(HtmlStyle.memberNameLink, getDocLink(LinkInfoImpl.Kind.INDEX, member, name)); Content dt = HtmlTree.DT(span); dt.addContent(" - "); @@ -198,7 +198,7 @@ public class AbstractIndexWriter extends HtmlDocletWriter { */ protected void addComment(ProgramElementDoc element, Content contentTree) { Tag[] tags; - Content span = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); + Content span = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); HtmlTree div = new HtmlTree(HtmlTag.DIV); div.addStyle(HtmlStyle.block); if (Util.isDeprecated(element)) { diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractMemberWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractMemberWriter.java index e44e4a3509e..e7502f376ef 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractMemberWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AbstractMemberWriter.java @@ -480,7 +480,7 @@ public abstract class AbstractMemberWriter { if (cd != null && !(pgmdoc instanceof ConstructorDoc) && !(pgmdoc instanceof ClassDoc)) { HtmlTree name = new HtmlTree(HtmlTag.SPAN); - name.addStyle(HtmlStyle.strong); + name.addStyle(HtmlStyle.typeNameLabel); name.addContent(cd.name() + "."); tdLast.addContent(name); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeFieldWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeFieldWriterImpl.java index bb936d37324..55dcd84fc64 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeFieldWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeFieldWriterImpl.java @@ -237,9 +237,9 @@ public class AnnotationTypeFieldWriterImpl extends AbstractMemberWriter */ protected void addSummaryLink(LinkInfoImpl.Kind context, ClassDoc cd, ProgramElementDoc member, Content tdSummary) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getDocLink(context, (MemberDoc) member, member.name(), false)); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeRequiredMemberWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeRequiredMemberWriterImpl.java index bdbe8d547c2..a3392011ae2 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeRequiredMemberWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeRequiredMemberWriterImpl.java @@ -239,9 +239,9 @@ public class AnnotationTypeRequiredMemberWriterImpl extends AbstractMemberWriter */ protected void addSummaryLink(LinkInfoImpl.Kind context, ClassDoc cd, ProgramElementDoc member, Content tdSummary) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getDocLink(context, (MemberDoc) member, member.name(), false)); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeWriterImpl.java index c90ae08c27e..b8866fe1a1b 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/AnnotationTypeWriterImpl.java @@ -227,7 +227,7 @@ public class AnnotationTypeWriterImpl extends SubWriterHolderWriter addSrcLink(annotationType, annotationName, pre); pre.addContent(parameterLinks); } else { - Content span = HtmlTree.SPAN(HtmlStyle.strong, annotationName); + Content span = HtmlTree.SPAN(HtmlStyle.memberNameLabel, annotationName); span.addContent(parameterLinks); pre.addContent(span); } @@ -262,8 +262,8 @@ public class AnnotationTypeWriterImpl extends SubWriterHolderWriter annotationInfoTree.addContent(hr); Tag[] deprs = annotationType.tags("deprecated"); if (Util.isDeprecated(annotationType)) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); - Content div = HtmlTree.DIV(HtmlStyle.block, strong); + Content deprLabel = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); + Content div = HtmlTree.DIV(HtmlStyle.block, deprLabel); if (deprs.length > 0) { Tag[] commentTags = deprs[0].inlineTags(); if (commentTags.length > 0) { diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ClassWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ClassWriterImpl.java index 7899445dd4a..92b7224d505 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ClassWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ClassWriterImpl.java @@ -254,7 +254,7 @@ public class ClassWriterImpl extends SubWriterHolderWriter addSrcLink(classDoc, className, pre); pre.addContent(parameterLinks); } else { - Content span = HtmlTree.SPAN(HtmlStyle.strong, className); + Content span = HtmlTree.SPAN(HtmlStyle.typeNameLabel, className); span.addContent(parameterLinks); pre.addContent(span); } @@ -547,8 +547,8 @@ public class ClassWriterImpl extends SubWriterHolderWriter classInfoTree.addContent(hr); Tag[] deprs = classDoc.tags("deprecated"); if (Util.isDeprecated(classDoc)) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); - Content div = HtmlTree.DIV(HtmlStyle.block, strong); + Content deprLabel = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); + Content div = HtmlTree.DIV(HtmlStyle.block, deprLabel); if (deprs.length > 0) { Tag[] commentTags = deprs[0].inlineTags(); if (commentTags.length > 0) { diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/EnumConstantWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/EnumConstantWriterImpl.java index b971d07668d..34a3a5027aa 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/EnumConstantWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/EnumConstantWriterImpl.java @@ -224,9 +224,9 @@ public class EnumConstantWriterImpl extends AbstractMemberWriter */ protected void addSummaryLink(LinkInfoImpl.Kind context, ClassDoc cd, ProgramElementDoc member, Content tdSummary) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getDocLink(context, (MemberDoc) member, member.name(), false)); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/FieldWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/FieldWriterImpl.java index 104967129f7..5e5cebf07f7 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/FieldWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/FieldWriterImpl.java @@ -140,11 +140,11 @@ public class FieldWriterImpl extends AbstractMemberWriter holder.typeName() : holder.qualifiedTypeName(), false); Content codeLink = HtmlTree.CODE(link); - Content strong = HtmlTree.SPAN(HtmlStyle.strong, holder.isClass()? + Content descfrmLabel = HtmlTree.SPAN(HtmlStyle.descfrmTypeLabel, holder.isClass()? writer.descfrmClassLabel : writer.descfrmInterfaceLabel); - strong.addContent(writer.getSpace()); - strong.addContent(codeLink); - fieldDocTree.addContent(HtmlTree.DIV(HtmlStyle.block, strong)); + descfrmLabel.addContent(writer.getSpace()); + descfrmLabel.addContent(codeLink); + fieldDocTree.addContent(HtmlTree.DIV(HtmlStyle.block, descfrmLabel)); writer.addInlineComment(field, fieldDocTree); } } @@ -258,9 +258,9 @@ public class FieldWriterImpl extends AbstractMemberWriter */ protected void addSummaryLink(LinkInfoImpl.Kind context, ClassDoc cd, ProgramElementDoc member, Content tdSummary) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getDocLink(context, cd , (MemberDoc) member, member.name(), false)); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HelpWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HelpWriter.java index 5e477dcc8d2..e1111faf9e9 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HelpWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HelpWriter.java @@ -316,7 +316,7 @@ public class HelpWriter extends HtmlDocletWriter { liConst.addContent(constPara); ul.addContent(liConst); Content divContent = HtmlTree.DIV(HtmlStyle.contentContainer, ul); - Content line30 = HtmlTree.SPAN(HtmlStyle.italic, getResource("doclet.Help_line_30")); + Content line30 = HtmlTree.SPAN(HtmlStyle.emphasizedPhrase, getResource("doclet.Help_line_30")); divContent.addContent(line30); contentTree.addContent(divContent); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java index 19a578c85cb..b06dc156097 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java @@ -1051,7 +1051,7 @@ public class HtmlDocletWriter extends HtmlDocWriter { public Content italicsClassName(ClassDoc cd, boolean qual) { Content name = new StringContent((qual)? cd.qualifiedName(): cd.name()); - return (cd.isInterface())? HtmlTree.SPAN(HtmlStyle.italic, name): name; + return (cd.isInterface())? HtmlTree.SPAN(HtmlStyle.interfaceName, name): name; } /** @@ -1567,7 +1567,7 @@ public class HtmlDocletWriter extends HtmlDocWriter { Content div; Content result = commentTagsToContent(null, doc, tags, first); if (depr) { - Content italic = HtmlTree.SPAN(HtmlStyle.italic, result); + Content italic = HtmlTree.SPAN(HtmlStyle.deprecationComment, result); div = HtmlTree.DIV(HtmlStyle.block, italic); htmltree.addContent(div); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/MethodWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/MethodWriterImpl.java index 8b3712d3594..a6290e237cf 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/MethodWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/MethodWriterImpl.java @@ -160,11 +160,11 @@ public class MethodWriterImpl extends AbstractExecutableMemberWriter holder.typeName() : holder.qualifiedTypeName(), false); Content codelLink = HtmlTree.CODE(link); - Content strong = HtmlTree.SPAN(HtmlStyle.strong, holder.asClassDoc().isClass()? + Content descfrmLabel = HtmlTree.SPAN(HtmlStyle.descfrmTypeLabel, holder.asClassDoc().isClass()? writer.descfrmClassLabel : writer.descfrmInterfaceLabel); - strong.addContent(writer.getSpace()); - strong.addContent(codelLink); - methodDocTree.addContent(HtmlTree.DIV(HtmlStyle.block, strong)); + descfrmLabel.addContent(writer.getSpace()); + descfrmLabel.addContent(codelLink); + methodDocTree.addContent(HtmlTree.DIV(HtmlStyle.block, descfrmLabel)); writer.addInlineComment(method, methodDocTree); } } @@ -310,7 +310,7 @@ public class MethodWriterImpl extends AbstractExecutableMemberWriter label = writer.specifiedByLabel; context = LinkInfoImpl.Kind.METHOD_SPECIFIED_BY; } - Content dt = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, label)); + Content dt = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.overrideSpecifyLabel, label)); dl.addContent(dt); Content overriddenTypeLink = writer.getLink(new LinkInfoImpl(writer.configuration, context, overriddenType)); @@ -365,7 +365,7 @@ public class MethodWriterImpl extends AbstractExecutableMemberWriter Content intfaclink = writer.getLink(new LinkInfoImpl( writer.configuration, LinkInfoImpl.Kind.METHOD_SPECIFIED_BY, intfac)); Content codeIntfacLink = HtmlTree.CODE(intfaclink); - Content dt = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, writer.specifiedByLabel)); + Content dt = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.overrideSpecifyLabel, writer.specifiedByLabel)); dl.addContent(dt); Content methlink = writer.getDocLink( LinkInfoImpl.Kind.MEMBER, implementedMeth, diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/NestedClassWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/NestedClassWriterImpl.java index d0537af4a41..c21e0aef881 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/NestedClassWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/NestedClassWriterImpl.java @@ -166,9 +166,9 @@ public class NestedClassWriterImpl extends AbstractMemberWriter */ protected void addSummaryLink(LinkInfoImpl.Kind context, ClassDoc cd, ProgramElementDoc member, Content tdSummary) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getLink(new LinkInfoImpl(configuration, context, (ClassDoc)member))); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageFrameWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageFrameWriter.java index 3268a446062..48e09440afc 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageFrameWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageFrameWriter.java @@ -183,7 +183,7 @@ public class PackageFrameWriter extends HtmlDocletWriter { printedHeader = true; } Content arr_i_name = new StringContent(arr[i].name()); - if (arr[i].isInterface()) arr_i_name = HtmlTree.SPAN(HtmlStyle.italic, arr_i_name); + if (arr[i].isInterface()) arr_i_name = HtmlTree.SPAN(HtmlStyle.interfaceName, arr_i_name); Content link = getLink(new LinkInfoImpl(configuration, LinkInfoImpl.Kind.PACKAGE_FRAME, arr[i]).label(arr_i_name).target("classFrame")); Content li = HtmlTree.LI(link); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageTreeWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageTreeWriter.java index 655af8e4c03..8d3d05d9372 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageTreeWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageTreeWriter.java @@ -155,7 +155,7 @@ public class PackageTreeWriter extends AbstractTreeWriter { * @param div the content tree to which the link will be added */ protected void addLinkToMainTree(Content div) { - Content span = HtmlTree.SPAN(HtmlStyle.strong, + Content span = HtmlTree.SPAN(HtmlStyle.packageHierarchyLabel, getResource("doclet.Package_Hierarchies")); div.addContent(span); HtmlTree ul = new HtmlTree (HtmlTag.UL); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageWriterImpl.java index c19bf2d6549..ef5b148ed1f 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PackageWriterImpl.java @@ -141,7 +141,7 @@ public class PackageWriterImpl extends HtmlDocletWriter if (Util.isDeprecated(packageDoc)) { HtmlTree deprDiv = new HtmlTree(HtmlTag.DIV); deprDiv.addStyle(HtmlStyle.deprecatedContent); - Content deprPhrase = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); + Content deprPhrase = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); deprDiv.addContent(deprPhrase); if (deprs.length > 0) { Tag[] commentTags = deprs[0].inlineTags(); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageFrameWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageFrameWriter.java index c61b7f869c5..4e0552b48b9 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageFrameWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageFrameWriter.java @@ -174,7 +174,7 @@ public class ProfilePackageFrameWriter extends HtmlDocletWriter { printedHeader = true; } Content arr_i_name = new StringContent(arr[i].name()); - if (arr[i].isInterface()) arr_i_name = HtmlTree.SPAN(HtmlStyle.italic, arr_i_name); + if (arr[i].isInterface()) arr_i_name = HtmlTree.SPAN(HtmlStyle.interfaceName, arr_i_name); Content link = getLink(new LinkInfoImpl(configuration, LinkInfoImpl.Kind.PACKAGE_FRAME, arr[i]).label(arr_i_name).target("classFrame")); Content li = HtmlTree.LI(link); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java index 7f2c6e61111..dea7479619c 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java @@ -158,7 +158,7 @@ public class ProfilePackageWriterImpl extends HtmlDocletWriter if (Util.isDeprecated(packageDoc)) { HtmlTree deprDiv = new HtmlTree(HtmlTag.DIV); deprDiv.addStyle(HtmlStyle.deprecatedContent); - Content deprPhrase = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); + Content deprPhrase = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); deprDiv.addContent(deprPhrase); if (deprs.length > 0) { Tag[] commentTags = deprs[0].inlineTags(); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfileWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfileWriterImpl.java index ebe8b0d1e69..15838e0cf9d 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfileWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfileWriterImpl.java @@ -187,7 +187,7 @@ public class ProfileWriterImpl extends HtmlDocletWriter deprs = pkg.tags("deprecated"); HtmlTree deprDiv = new HtmlTree(HtmlTag.DIV); deprDiv.addStyle(HtmlStyle.deprecatedContent); - Content deprPhrase = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); + Content deprPhrase = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); deprDiv.addContent(deprPhrase); if (deprs.length > 0) { Tag[] commentTags = deprs[0].inlineTags(); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PropertyWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PropertyWriterImpl.java index 9bf4bf0ca26..d772c13be59 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PropertyWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/PropertyWriterImpl.java @@ -136,11 +136,11 @@ public class PropertyWriterImpl extends AbstractMemberWriter holder.typeName() : holder.qualifiedTypeName(), false); Content codeLink = HtmlTree.CODE(link); - Content strong = HtmlTree.SPAN(HtmlStyle.strong, holder.isClass()? + Content descfrmLabel = HtmlTree.SPAN(HtmlStyle.descfrmTypeLabel, holder.isClass()? writer.descfrmClassLabel : writer.descfrmInterfaceLabel); - strong.addContent(writer.getSpace()); - strong.addContent(codeLink); - propertyDocTree.addContent(HtmlTree.DIV(HtmlStyle.block, strong)); + descfrmLabel.addContent(writer.getSpace()); + descfrmLabel.addContent(codeLink); + propertyDocTree.addContent(HtmlTree.DIV(HtmlStyle.block, descfrmLabel)); writer.addInlineComment(property, propertyDocTree); } } @@ -255,14 +255,14 @@ public class PropertyWriterImpl extends AbstractMemberWriter */ protected void addSummaryLink(LinkInfoImpl.Kind context, ClassDoc cd, ProgramElementDoc member, Content tdSummary) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, + Content memberLink = HtmlTree.SPAN(HtmlStyle.memberNameLink, writer.getDocLink(context, cd, (MemberDoc) member, member.name().substring(0, member.name().lastIndexOf("Property")), false, true)); - Content code = HtmlTree.CODE(strong); + Content code = HtmlTree.CODE(memberLink); tdSummary.addContent(code); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/SubWriterHolderWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/SubWriterHolderWriter.java index 4e15a00c14a..f634638b874 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/SubWriterHolderWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/SubWriterHolderWriter.java @@ -175,8 +175,8 @@ public abstract class SubWriterHolderWriter extends HtmlDocletWriter { Tag[] deprs = member.tags("deprecated"); Content div; if (Util.isDeprecated((ProgramElementDoc) member)) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); - div = HtmlTree.DIV(HtmlStyle.block, strong); + Content deprLabel = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); + div = HtmlTree.DIV(HtmlStyle.block, deprLabel); div.addContent(getSpace()); if (deprs.length > 0) { addInlineDeprecatedComment(member, deprs[0], div); @@ -186,8 +186,8 @@ public abstract class SubWriterHolderWriter extends HtmlDocletWriter { } else { ClassDoc cd = ((ProgramElementDoc)member).containingClass(); if (cd != null && Util.isDeprecated(cd)) { - Content strong = HtmlTree.SPAN(HtmlStyle.strong, deprecatedPhrase); - div = HtmlTree.DIV(HtmlStyle.block, strong); + Content deprLabel = HtmlTree.SPAN(HtmlStyle.deprecatedLabel, deprecatedPhrase); + div = HtmlTree.DIV(HtmlStyle.block, deprLabel); div.addContent(getSpace()); tdSummary.addContent(div); } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TagletWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TagletWriterImpl.java index 7ff2e2efc45..c48805eeea1 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TagletWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TagletWriterImpl.java @@ -97,7 +97,7 @@ public class TagletWriterImpl extends TagletWriter { Tag[] deprs = doc.tags("deprecated"); if (doc instanceof ClassDoc) { if (Util.isDeprecated((ProgramElementDoc) doc)) { - result.addContent(HtmlTree.SPAN(HtmlStyle.strong, + result.addContent(HtmlTree.SPAN(HtmlStyle.deprecatedLabel, new StringContent(configuration.getText("doclet.Deprecated")))); result.addContent(RawHtml.nbsp); if (deprs.length > 0) { @@ -112,18 +112,18 @@ public class TagletWriterImpl extends TagletWriter { } else { MemberDoc member = (MemberDoc) doc; if (Util.isDeprecated((ProgramElementDoc) doc)) { - result.addContent(HtmlTree.SPAN(HtmlStyle.strong, + result.addContent(HtmlTree.SPAN(HtmlStyle.deprecatedLabel, new StringContent(configuration.getText("doclet.Deprecated")))); result.addContent(RawHtml.nbsp); if (deprs.length > 0) { Content body = commentTagsToOutput(null, doc, deprs[0].inlineTags(), false); if (!body.isEmpty()) - result.addContent(HtmlTree.SPAN(HtmlStyle.italic, body)); + result.addContent(HtmlTree.SPAN(HtmlStyle.deprecationComment, body)); } } else { if (Util.isDeprecated(member.containingClass())) { - result.addContent(HtmlTree.SPAN(HtmlStyle.strong, + result.addContent(HtmlTree.SPAN(HtmlStyle.deprecatedLabel, new StringContent(configuration.getText("doclet.Deprecated")))); result.addContent(RawHtml.nbsp); } @@ -151,7 +151,7 @@ public class TagletWriterImpl extends TagletWriter { * {@inheritDoc} */ public Content getParamHeader(String header) { - HtmlTree result = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, + HtmlTree result = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.paramLabel, new StringContent(header))); return result; } @@ -186,7 +186,7 @@ public class TagletWriterImpl extends TagletWriter { */ public Content returnTagOutput(Tag returnTag) { ContentBuilder result = new ContentBuilder(); - result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, + result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.returnLabel, new StringContent(configuration.getText("doclet.Returns"))))); result.addContent(HtmlTree.DD(htmlWriter.commentTagsToContent( returnTag, null, returnTag.inlineTags(), false))); @@ -231,7 +231,7 @@ public class TagletWriterImpl extends TagletWriter { return body; ContentBuilder result = new ContentBuilder(); - result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, + result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.seeLabel, new StringContent(configuration.getText("doclet.See_Also"))))); result.addContent(HtmlTree.DD(body)); return result; @@ -250,7 +250,7 @@ public class TagletWriterImpl extends TagletWriter { */ public Content simpleTagOutput(Tag[] simpleTags, String header) { ContentBuilder result = new ContentBuilder(); - result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, new RawHtml(header)))); + result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.simpleTagLabel, new RawHtml(header)))); ContentBuilder body = new ContentBuilder(); for (int i = 0; i < simpleTags.length; i++) { if (i > 0) { @@ -268,7 +268,7 @@ public class TagletWriterImpl extends TagletWriter { */ public Content simpleTagOutput(Tag simpleTag, String header) { ContentBuilder result = new ContentBuilder(); - result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, new RawHtml(header)))); + result.addContent(HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.simpleTagLabel, new RawHtml(header)))); Content body = htmlWriter.commentTagsToContent( simpleTag, null, simpleTag.inlineTags(), false); result.addContent(HtmlTree.DD(body)); @@ -279,7 +279,7 @@ public class TagletWriterImpl extends TagletWriter { * {@inheritDoc} */ public Content getThrowsHeader() { - HtmlTree result = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.strong, + HtmlTree result = HtmlTree.DT(HtmlTree.SPAN(HtmlStyle.throwsLabel, new StringContent(configuration.getText("doclet.Throws")))); return result; } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TreeWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TreeWriter.java index 988facacb58..7ccde03602b 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TreeWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/TreeWriter.java @@ -131,7 +131,7 @@ public class TreeWriter extends AbstractTreeWriter { return; } if (!classesonly) { - Content span = HtmlTree.SPAN(HtmlStyle.strong, + Content span = HtmlTree.SPAN(HtmlStyle.packageHierarchyLabel, getResource("doclet.Package_Hierarchies")); contentTree.addContent(span); HtmlTree ul = new HtmlTree(HtmlTag.UL); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlDocWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlDocWriter.java index ee8cc9cb12e..3c63c114556 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlDocWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlDocWriter.java @@ -236,7 +236,7 @@ public abstract class HtmlDocWriter extends HtmlWriter { String stylename, String title, String target) { Content body = label; if (strong) { - body = HtmlTree.SPAN(HtmlStyle.strong, body); + body = HtmlTree.SPAN(HtmlStyle.typeNameLink, body); } if (stylename != null && stylename.length() != 0) { HtmlTree t = new HtmlTree(HtmlTag.FONT, body); diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlStyle.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlStyle.java index fc49d7f1360..bc5dba83a9b 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlStyle.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlStyle.java @@ -52,37 +52,51 @@ public enum HtmlStyle { constantValuesContainer, contentContainer, deprecatedContent, + deprecatedLabel, deprecatedSummary, + deprecationComment, description, + descfrmTypeLabel, details, docSummary, + emphasizedPhrase, header, horizontal, footer, indexContainer, indexHeader, inheritance, - italic, + interfaceName, legalCopy, + memberNameLabel, + memberNameLink, memberSummary, nameValue, navBarCell1Rev, navList, + overrideSpecifyLabel, overviewSummary, + packageHierarchyLabel, + paramLabel, + returnLabel, rowColor, + seeLabel, serializedFormContainer, + simpleTagLabel, skipNav, sourceContainer, sourceLineNo, - strong, subNav, subNavList, subTitle, summary, tabEnd, tableTab, + throwsLabel, title, topNav, + typeNameLabel, + typeNameLink, typeSummary, useSummary; } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/resources/stylesheet.css b/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/resources/stylesheet.css index 140b86bbccd..97356f24fa2 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/resources/stylesheet.css +++ b/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/resources/stylesheet.css @@ -514,9 +514,11 @@ h1.hidden { display:block; margin:3px 0 0 0; } -.strong { +.deprecatedLabel, .descfrmTypeLabel, .memberNameLabel, .memberNameLink, +.overrideSpecifyLabel, .packageHierarchyLabel, .paramLabel, .returnLabel, +.seeLabel, .simpleTagLabel, .throwsLabel, .typeNameLabel, .typeNameLink { font-weight:bold; } -.italic { +.deprecationComment, .emphasizedPhrase, .interfaceName { font-style:italic; } diff --git a/langtools/test/com/sun/javadoc/AuthorDD/AuthorDD.java b/langtools/test/com/sun/javadoc/AuthorDD/AuthorDD.java index e65f81aaa88..126f1de6f8d 100644 --- a/langtools/test/com/sun/javadoc/AuthorDD/AuthorDD.java +++ b/langtools/test/com/sun/javadoc/AuthorDD/AuthorDD.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4651598 + * @bug 4651598 8026567 * @summary Javadoc wrongly inserts
tags when using multiple @author tags * @author dkramer * @run main AuthorDD @@ -86,12 +86,12 @@ public class AuthorDD // Test single @since tag: - { "
Since:
"+NL+"
JDK 1.0
", + { "
Since:
"+NL+"
JDK 1.0
", BUGID + FS + "p1" + FS + "C1.html" }, // Test multiple @author tags: - { "
Author:
"+NL+"
Doug Kramer, Jamie, Neal
", + { "
Author:
"+NL+"
Doug Kramer, Jamie, Neal
", BUGID + FS + "p1" + FS + "C1.html" }, }; diff --git a/langtools/test/com/sun/javadoc/testAnnotationTypes/TestAnnotationTypes.java b/langtools/test/com/sun/javadoc/testAnnotationTypes/TestAnnotationTypes.java index 02286c30245..990813148ce 100644 --- a/langtools/test/com/sun/javadoc/testAnnotationTypes/TestAnnotationTypes.java +++ b/langtools/test/com/sun/javadoc/testAnnotationTypes/TestAnnotationTypes.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4973609 8015249 8025633 + * @bug 4973609 8015249 8025633 8026567 * @summary Make sure that annotation types with 0 members does not have * extra HR tags. * @author jamieh @@ -55,7 +55,7 @@ public class TestAnnotationTypes extends JavadocTester { {BUG_ID + FS + "pkg" + FS + "AnnotationTypeField.html", "

Field Summary

"}, {BUG_ID + FS + "pkg" + FS + "AnnotationTypeField.html", - "DEFAULT_NAME" + " "}, {BUG_ID + FS + "pkg" + FS + "AnnotationTypeField.html", diff --git a/langtools/test/com/sun/javadoc/testClassCrossReferences/TestClassCrossReferences.java b/langtools/test/com/sun/javadoc/testClassCrossReferences/TestClassCrossReferences.java index 8b8d3c945c5..d9ed7941637 100644 --- a/langtools/test/com/sun/javadoc/testClassCrossReferences/TestClassCrossReferences.java +++ b/langtools/test/com/sun/javadoc/testClassCrossReferences/TestClassCrossReferences.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4652655 4857717 8025633 + * @bug 4652655 4857717 8025633 8026567 * @summary This test verifies that class cross references work properly. * @author jamieh * @library ../lib/ @@ -48,7 +48,7 @@ public class TestClassCrossReferences extends JavadocTester { "Link to external member gcd"}, {BUG_ID + FS + "C.html", - "
" + NL + "
Overrides:
" + NL + + "
" + NL + "
Overrides:
" + NL + "
toString in class java.lang.Object
" + NL + "
"} }; diff --git a/langtools/test/com/sun/javadoc/testClassTree/TestClassTree.java b/langtools/test/com/sun/javadoc/testClassTree/TestClassTree.java index 8fcb71397cb..a652fe023c4 100644 --- a/langtools/test/com/sun/javadoc/testClassTree/TestClassTree.java +++ b/langtools/test/com/sun/javadoc/testClassTree/TestClassTree.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2004, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2004, 2013, 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,7 @@ /* * @test - * @bug 4632553 4973607 + * @bug 4632553 4973607 8026567 * @summary No need to include type name (class, interface, etc.) before * every single type in class tree. * Make sure class tree includes heirarchy for enums and annotation @@ -49,12 +49,12 @@ public class TestClassTree extends JavadocTester { private static final String[][] TEST = { {BUG_ID + FS + "pkg" + FS + "package-tree.html", "
    " + NL + "
  • pkg.ParentClass"}, + "title=\"class in pkg\">ParentClass"}, {BUG_ID + FS + "pkg" + FS + "package-tree.html", "

    Annotation Type Hierarchy

    " + NL + "
      " + NL + "
    • pkg.AnnotationType " + + "title=\"annotation in pkg\">AnnotationType " + "(implements java.lang.annotation.Annotation)
    • " + NL + "
    "}, {BUG_ID + FS + "pkg" + FS + "package-tree.html", @@ -63,14 +63,14 @@ public class TestClassTree extends JavadocTester { "
  • java.lang.Enum<E> (implements java.lang." + "Comparable<T>, java.io.Serializable)" + NL + "
      " + NL + "
    • pkg.Coin
    • " + NL + + "title=\"enum in pkg\">Coin" + NL + "
    " + NL + "
  • " + NL + "
" + NL + "" + NL + "" }, }; private static final String[][] NEGATED_TEST = { {BUG_ID + FS + "pkg" + FS + "package-tree.html", "
  • class pkg.ParentClass
  • "} + "title=\"class in pkg\">ParentClass"} }; /** diff --git a/langtools/test/com/sun/javadoc/testConstructorIndent/TestConstructorIndent.java b/langtools/test/com/sun/javadoc/testConstructorIndent/TestConstructorIndent.java index 27422853b03..189e8270236 100644 --- a/langtools/test/com/sun/javadoc/testConstructorIndent/TestConstructorIndent.java +++ b/langtools/test/com/sun/javadoc/testConstructorIndent/TestConstructorIndent.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4904037 + * @bug 4904037 8026567 * @summary The constructor comments should be surrounded by *
    . Check for this in the output. * @author jamieh @@ -47,7 +47,7 @@ public class TestConstructorIndent extends JavadocTester { private static final String[][] TEST = { {BUG_ID + FS + "C.html", "
    " + "This is just a simple constructor.
    " + NL + - "
    " + NL + "
    Parameters:
    " + NL + + "
    " + NL + "
    Parameters:
    " + NL + "
    i - a param.
    " + NL +"
    " } }; diff --git a/langtools/test/com/sun/javadoc/testDeprecatedDocs/TestDeprecatedDocs.java b/langtools/test/com/sun/javadoc/testDeprecatedDocs/TestDeprecatedDocs.java index c7c946ae73c..0f185fbdfbd 100644 --- a/langtools/test/com/sun/javadoc/testDeprecatedDocs/TestDeprecatedDocs.java +++ b/langtools/test/com/sun/javadoc/testDeprecatedDocs/TestDeprecatedDocs.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2013, 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,7 @@ /* * @test - * @bug 4927552 + * @bug 4927552 8026567 * @summary * @author jamieh * @library ../lib/ @@ -76,20 +76,20 @@ public class TestDeprecatedDocs extends JavadocTester { {TARGET_FILE, "pkg.DeprecatedClassByAnnotation.field"}, {TARGET_FILE2, "
    @Deprecated" + NL +
    -                 "public class DeprecatedClassByAnnotation" + NL +
    +                 "public class DeprecatedClassByAnnotation" + NL +
                      "extends java.lang.Object
    "}, {TARGET_FILE2, "
    @Deprecated" + NL +
                      "public int field
    " + NL + - "
    Deprecated. 
    "}, + "
    Deprecated. 
    "}, {TARGET_FILE2, "
    @Deprecated" + NL +
                      "public DeprecatedClassByAnnotation()
    " + NL + - "
    Deprecated. 
    "}, + "
    Deprecated. 
    "}, {TARGET_FILE2, "
    @Deprecated" + NL +
                      "public void method()
    " + NL + - "
    Deprecated. 
    "}, + "
    Deprecated. 
    "}, }; private static final String[][] NEGATED_TEST = NO_TEST; diff --git a/langtools/test/com/sun/javadoc/testExternalOverridenMethod/TestExternalOverridenMethod.java b/langtools/test/com/sun/javadoc/testExternalOverridenMethod/TestExternalOverridenMethod.java index 4798cee22ea..41bf3e279ac 100644 --- a/langtools/test/com/sun/javadoc/testExternalOverridenMethod/TestExternalOverridenMethod.java +++ b/langtools/test/com/sun/javadoc/testExternalOverridenMethod/TestExternalOverridenMethod.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4857717 8025633 + * @bug 4857717 8025633 8026567 * @summary Test to make sure that externally overriden and implemented methods * are documented properly. The method should still include "implements" or * "overrides" documentation even though the method is external. @@ -38,13 +38,13 @@ public class TestExternalOverridenMethod extends JavadocTester { private static final String BUG_ID = "4857717"; private static final String[][] TEST = { {BUG_ID + FS + "pkg" + FS + "XReader.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    read in class " + "FilterReader
    "}, {BUG_ID + FS + "pkg" + FS + "XReader.html", - "
    Specified by:
    " + NL + + "
    Specified by:
    " + NL + "
    readInt in interface " + "C4<E extends C4<E>>" + "public abstract class C4<E extends C4<E>>" }, }; private static final String[][] NEGATED_TEST = diff --git a/langtools/test/com/sun/javadoc/testHtmlDefinitionListTag/TestHtmlDefinitionListTag.java b/langtools/test/com/sun/javadoc/testHtmlDefinitionListTag/TestHtmlDefinitionListTag.java index 7d4b28d887a..7e81b1e3004 100644 --- a/langtools/test/com/sun/javadoc/testHtmlDefinitionListTag/TestHtmlDefinitionListTag.java +++ b/langtools/test/com/sun/javadoc/testHtmlDefinitionListTag/TestHtmlDefinitionListTag.java @@ -25,7 +25,7 @@ /* * @test - * @bug 6786690 6820360 8025633 + * @bug 6786690 6820360 8025633 8026567 * @summary This test verifies the nesting of definition list tags. * @author Bhavesh Patel * @library ../lib/ @@ -43,7 +43,7 @@ public class TestHtmlDefinitionListTag extends JavadocTester { // for default value. private static final String[][] TEST_ALL = { {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    public class " +
    -                 "C1" + NL +
    +                 "C1" + NL +
                      "extends java.lang.Object" + NL + "implements java.io.Serializable
    "}, {BUG_ID + FS + "pkg1" + FS + "C4.html", "
    " + NL + "
    Default:
    " + NL + "
    true
    " + NL + @@ -54,64 +54,64 @@ public class TestHtmlDefinitionListTag extends JavadocTester { // enclosing comments, tags and deprecated information. private static final String[][] TEST_CMNT_DEPR = { {BUG_ID + FS + "pkg1" + FS + "package-summary.html", "
    " + NL + - "
    Since:
    " + NL + + "
    Since:
    " + NL + "
    JDK1.0
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Since:
    " + NL + - "
    JDK1.0
    " + NL + "
    See Also:
    " + NL + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Since:
    " + NL + + "
    JDK1.0
    " + NL + "
    See Also:
    " + NL + "
    " + "C2, " + NL + "" + "Serialized Form
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Since:
    " + NL + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Since:
    " + NL + "
    1.4
    " + NL + - "
    See Also:
    " + NL + "
    " + + "
    See Also:
    " + NL + "
    " + "" + "setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    "+ NL + "
    Parameters:
    " + NL + "
    title" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    "+ NL + "
    Parameters:
    " + NL + "
    title" + " - the title
    " + NL + "
    test - boolean value" + - "
    " + NL + "
    Throws:
    " + NL + + "
    " + NL + "
    Throws:
    " + NL + "
    java.lang.IllegalArgumentException - if the " + "owner's" + NL + " GraphicsConfiguration is not from a screen " + "device
    " + NL + "
    HeadlessException
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Parameters:
    " + NL + "
    undecorated" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Parameters:
    " + NL + "
    undecorated" + " - true if no decorations are" + NL + " to be enabled;" + NL + " false " + - "if decorations are to be enabled.
    " + NL + "
    Since:" + + "if decorations are to be enabled.
    " + NL + "
    Since:" + "
    " + NL + "
    1.4
    " + NL + - "
    See Also:
    " + NL + "
    " + + "
    See Also:
    " + NL + "
    " + "readObject()" + "
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Throws:
    " + NL + - "
    java.io.IOException
    " + NL + "
    See Also:" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Throws:
    " + NL + + "
    java.io.IOException
    " + NL + "
    See Also:" + "
    " + NL + "
    " + "setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C2.html", "
    " + NL + "
    Parameters:" + - "
    " + NL + "
    set - boolean
    " + NL + "
    " + + {BUG_ID + FS + "pkg1" + FS + "C2.html", "
    " + NL + "
    Parameters:" + + "
    " + NL + "
    set - boolean
    " + NL + "
    " + "Since:
    " + NL + "
    1.4
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "
    " + NL + "
    Throws:" + + {BUG_ID + FS + "serialized-form.html", "
    " + NL + "
    Throws:" + "
    " + NL + "
    " + - "java.io.IOException
    " + NL + "
    See Also:" + + "java.io.IOException
    " + NL + "
    See Also:" + "
    " + NL + "
    " + "C1.setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + - " As of JDK version 1.5, replaced by" + NL + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + + " As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean)." + NL + "
    This field indicates whether the C1 is " + - "undecorated.
    " + NL + " " + NL + "
    " + NL + "
    Since:
    " + NL + - "
    1.4
    " + NL + "
    See Also:" + + "undecorated." + NL + " " + NL + "
    " + NL + "
    Since:
    " + NL + + "
    1.4
    " + NL + "
    See Also:" + "
    " + NL + "
    " + "C1.setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + - " As of JDK version 1.5, replaced by" + NL + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + + " As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean)." + NL + "
    Reads the object stream.
    " + NL + - "
    " + NL + "
    Throws:" + + "
    " + NL + "
    Throws:" + "
    " + NL + "
    " + "IOException
    " + NL + "
    java.io.IOException
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + " " + NL + "
    The name for this class.
    "}}; @@ -122,55 +122,55 @@ public class TestHtmlDefinitionListTag extends JavadocTester { // and deprecated information. private static final String[][] TEST_NODEPR = { {BUG_ID + FS + "pkg1" + FS + "package-summary.html", "
    " + NL + - "
    Since:
    " + NL + + "
    Since:
    " + NL + "
    JDK1.0
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Since:" + - "
    " + NL + "
    JDK1.0
    " + NL + "
    See Also:" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Since:" + + "
    " + NL + "
    JDK1.0
    " + NL + "
    See Also:" + "
    " + NL + "
    " + "C2, " + NL + "" + "Serialized Form
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Parameters:" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Parameters:" + "
    " + NL + "
    title - the title
    " + NL + "
    " + - "test - boolean value
    " + NL + "
    Throws:" + + "test - boolean value" + NL + "
    Throws:" + "
    " + NL + "
    java.lang.IllegalArgumentException" + " - if the owner's" + NL + " GraphicsConfiguration" + " is not from a screen device
    " + NL + "
    " + "HeadlessException
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Parameters:" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Parameters:" + "
    " + NL + "
    undecorated - true" + " if no decorations are" + NL + " to be enabled;" + NL + " false if decorations are to be enabled." + - "
    " + NL + "
    Since:
    " + NL + "
    1.4
    " + NL + - "
    See Also:
    " + NL + "
    " + + "
    " + NL + "
    Since:
    " + NL + "
    1.4
    " + NL + + "
    See Also:
    " + NL + "
    " + "readObject()
    " + NL + "
    "}, - {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Throws:" + + {BUG_ID + FS + "pkg1" + FS + "C1.html", "
    " + NL + "
    Throws:" + "
    " + NL + "
    java.io.IOException
    " + NL + "
    " + - "See Also:
    " + NL + "
    " + + "See Also:
    " + NL + "
    " + "setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "
    " + NL + "
    Throws:" + + {BUG_ID + FS + "serialized-form.html", "
    " + NL + "
    Throws:" + "
    " + NL + "
    " + - "java.io.IOException
    " + NL + "
    See Also:" + + "java.io.IOException" + NL + "
    See Also:" + "
    " + NL + "
    " + "C1.setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + - " As of JDK version 1.5, replaced by" + NL + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + + " As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean)." + NL + "
    This field indicates whether the C1 is " + - "undecorated.
    " + NL + " " + NL + "
    " + NL + "
    Since:
    " + NL + - "
    1.4
    " + NL + "
    See Also:" + + "undecorated." + NL + " " + NL + "
    " + NL + "
    Since:
    " + NL + + "
    1.4
    " + NL + "
    See Also:" + "
    " + NL + "
    " + "C1.setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + - " As of JDK version 1.5, replaced by" + NL + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + + " As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean)." + NL + "
    Reads the object stream.
    " + NL + - "
    " + NL + "
    Throws:" + + "
    " + NL + "
    Throws:" + "
    " + NL + "
    " + "IOException
    " + NL + "
    java.io.IOException
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + " " + NL + "
    " + "The name for this class.
    "}}; @@ -186,12 +186,12 @@ public class TestHtmlDefinitionListTag extends JavadocTester { "title=\"enum in pkg1\">C1.ModalExclusionType " + "APPLICATION_EXCLUDE" + NL + ""}, {BUG_ID + FS + "serialized-form.html", "
    boolean " +
    -                 "undecorated
    " + NL + "
    " + - "Deprecated. As of JDK version 1.5, replaced by" + NL + + "undecorated" + NL + "
    " + + "Deprecated. As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean).
    " + NL + ""}, - {BUG_ID + FS + "serialized-form.html", "" + - "Deprecated. As of JDK version" + + {BUG_ID + FS + "serialized-form.html", "" + + "Deprecated. As of JDK version" + " 1.5, replaced by" + NL + " " + "setUndecorated(boolean).
    " + NL + ""}}; diff --git a/langtools/test/com/sun/javadoc/testHtmlStrongTag/TestHtmlStrongTag.java b/langtools/test/com/sun/javadoc/testHtmlStrongTag/TestHtmlStrongTag.java index eef768a445d..b80d105f7c6 100644 --- a/langtools/test/com/sun/javadoc/testHtmlStrongTag/TestHtmlStrongTag.java +++ b/langtools/test/com/sun/javadoc/testHtmlStrongTag/TestHtmlStrongTag.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2013, 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 @@ -25,7 +25,7 @@ /* * @test - * @bug 6786028 + * @bug 6786028 8026567 * @summary This test verifys the use of HTML tag instead of by Javadoc std doclet. * @author Bhavesh Patel * @library ../lib/ @@ -38,7 +38,7 @@ public class TestHtmlStrongTag extends JavadocTester { private static final String BUG_ID = "6786028"; private static final String[][] TEST1 = { - {BUG_ID + FS + "pkg1" + FS + "C1.html", "See Also:"}}; + {BUG_ID + FS + "pkg1" + FS + "C1.html", "See Also:"}}; private static final String[][] NEGATED_TEST1 = { {BUG_ID + FS + "pkg1" + FS + "C1.html", "Method Summary"}, {BUG_ID + FS + "pkg1" + FS + "C1.html", ""}, diff --git a/langtools/test/com/sun/javadoc/testIndex/TestIndex.java b/langtools/test/com/sun/javadoc/testIndex/TestIndex.java index ae916ba4040..45e08fe6120 100644 --- a/langtools/test/com/sun/javadoc/testIndex/TestIndex.java +++ b/langtools/test/com/sun/javadoc/testIndex/TestIndex.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2013, 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,7 @@ /* * @test - * @bug 4852280 4517115 4973608 4994589 + * @bug 4852280 4517115 4973608 4994589 8026567 * @summary Perform tests on index.html file. * Also test that index-all.html has the appropriate output. * Test for unnamed package in index. @@ -53,27 +53,27 @@ public class TestIndex extends JavadocTester { //Test index-all.html {BUG_ID + FS + "index-all.html", - "C" + + "C" + " - Class in pkg"}, {BUG_ID + FS + "index-all.html", "" + - "Interface - Interface in " + + "Interface - Interface in " + "pkg"}, {BUG_ID + FS + "index-all.html", "" + - "AnnotationType - Annotation Type in " + + "AnnotationType - Annotation Type in " + "pkg"}, {BUG_ID + FS + "index-all.html", "" + - "Coin - Enum in " + + "Coin - Enum in " + "pkg"}, {BUG_ID + FS + "index-all.html", "Class in <Unnamed>"}, {BUG_ID + FS + "index-all.html", - "
    " + NL + "
    " + + "
    " + NL + "
    " + "Java - Static variable in class pkg.C
    " + NL + "
     
    " + NL + - "
    JDK " + + "
    JDK " + "- Static variable in class pkg." + "C
    " + NL + "
     
    " + NL + "
    "}, }; diff --git a/langtools/test/com/sun/javadoc/testInterface/TestInterface.java b/langtools/test/com/sun/javadoc/testInterface/TestInterface.java index 39f8e212629..fba2b6a63af 100644 --- a/langtools/test/com/sun/javadoc/testInterface/TestInterface.java +++ b/langtools/test/com/sun/javadoc/testInterface/TestInterface.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4682448 4947464 5029946 8025633 + * @bug 4682448 4947464 5029946 8025633 8026567 * @summary Verify that the public modifier does not show up in the * documentation for public methods, as recommended by the JLS. * If A implements I and B extends A, B should be in the list of @@ -83,7 +83,7 @@ public class TestInterface extends JavadocTester { }, //Make sure "Specified By" has substituted type parameters. {BUG_ID + FS + "pkg" + FS + "Child.html", - "
    Specified by:
    " + NL + + "
    Specified by:
    " + NL + "
    method" + " in interface " + "" + @@ -92,7 +92,7 @@ public class TestInterface extends JavadocTester { }, //Make sure "Overrides" has substituted type parameters. {BUG_ID + FS + "pkg" + FS + "Child.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    method" + " in class Parent<See Also:
    " + NL + "
    getRate(), " + NL + + "
    See Also:
    " + NL + "
    getRate(), " + NL + "setRate(double)
    "}, {"./" + BUG_ID + "/C.html", "
    public final void setRate(double value)
    " + NL + "
    Sets the value of the property rate.
    " + NL + - "
    " + NL + "
    Property description:
    " }, + "
    " + NL + "
    Property description:
    " }, {"./" + BUG_ID + "/C.html", "
    public final double getRate()
    " + NL + "
    Gets the value of the property rate.
    " + NL + - "
    " + NL + "
    Property description:
    " }, + "
    " + NL + "
    Property description:
    " }, {"./" + BUG_ID + "/C.html", - "rate" + NL + + "rate" + NL + "
    Defines the direction/speed at which the Timeline is expected to"}, {"./" + BUG_ID + "/C.html", - "Default value:"}, + "Default value:"}, {"./" + BUG_ID + "/C.html", - "Since:
    " + NL + "
    JavaFX 8.0
    " }, + "Since:
    " + NL + "
    JavaFX 8.0
    " }, {"./" + BUG_ID + "/C.html", "

    Sets the value of the property Property"}, {"./" + BUG_ID + "/C.html", "

    Gets the value of the property Property"}, {"./" + BUG_ID + "/C.html", - "Property description:"}, + "Property description:"}, {"./" + BUG_ID + "/C.html", - "setTestMethodProperty() " }, + "setTestMethodProperty() " }, {"./" + BUG_ID + "/C.html", "

    isPaused

    " + NL + "
    public final double isPaused()
    " + NL + diff --git a/langtools/test/com/sun/javadoc/testLegacyTaglet/Check.java b/langtools/test/com/sun/javadoc/testLegacyTaglet/Check.java index a294603b99a..de511c13b3f 100644 --- a/langtools/test/com/sun/javadoc/testLegacyTaglet/Check.java +++ b/langtools/test/com/sun/javadoc/testLegacyTaglet/Check.java @@ -126,7 +126,7 @@ public class Check implements Taglet { * @param tag the tag representation of this custom tag. */ public String toString(Tag tag) { - return "
    " + TAG_HEADER + ":
    " + tag.text() + + return "
    " + TAG_HEADER + ":
    " + tag.text() + "
    \n"; } diff --git a/langtools/test/com/sun/javadoc/testLinkOption/TestLinkOption.java b/langtools/test/com/sun/javadoc/testLinkOption/TestLinkOption.java index 18a1d2b7e47..271ac18c956 100644 --- a/langtools/test/com/sun/javadoc/testLinkOption/TestLinkOption.java +++ b/langtools/test/com/sun/javadoc/testLinkOption/TestLinkOption.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2013, 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,7 @@ /* * @test - * @bug 4720957 5020118 + * @bug 4720957 5020118 8026567 * @summary Test to make sure that -link and -linkoffline link to * right files. * @author jamieh @@ -61,7 +61,7 @@ public class TestLinkOption extends JavadocTester { "Object p3)" }, {BUG_ID + "-1" + FS + "java" + FS + "lang" + FS + "StringBuilderChild.html", - "
    public abstract class StringBuilderChild" + NL +
    +                "
    public abstract class StringBuilderChild" + NL +
                     "extends Object
    " }, diff --git a/langtools/test/com/sun/javadoc/testMemberInheritence/TestMemberInheritence.java b/langtools/test/com/sun/javadoc/testMemberInheritence/TestMemberInheritence.java index 62b9235e952..d77db44d53f 100644 --- a/langtools/test/com/sun/javadoc/testMemberInheritence/TestMemberInheritence.java +++ b/langtools/test/com/sun/javadoc/testMemberInheritence/TestMemberInheritence.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4638588 4635809 6256068 6270645 8025633 + * @bug 4638588 4635809 6256068 6270645 8025633 8026567 * @summary Test to make sure that members are inherited properly in the Javadoc. * Verify that inheritence labels are correct. * @author jamieh @@ -72,7 +72,7 @@ public class TestMemberInheritence extends JavadocTester { // Test overriding/implementing methods with generic parameters. {BUG_ID + FS + "pkg" + FS + "BaseClass.html", - "
    " + NL + "
    Specified by:
    " + NL + + "
    " + NL + "
    Specified by:
    " + NL + "
    " + "getAnnotation in interface " + "" + diff --git a/langtools/test/com/sun/javadoc/testMemberSummary/TestMemberSummary.java b/langtools/test/com/sun/javadoc/testMemberSummary/TestMemberSummary.java index eff1de42046..306e8cc9de4 100644 --- a/langtools/test/com/sun/javadoc/testMemberSummary/TestMemberSummary.java +++ b/langtools/test/com/sun/javadoc/testMemberSummary/TestMemberSummary.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4951228 6290760 8025633 + * @bug 4951228 6290760 8025633 8026567 * @summary Test the case where the overriden method returns a different * type than the method in the child class. Make sure the * documentation is inherited but the return type isn't. @@ -49,7 +49,7 @@ public class TestMemberSummary extends JavadocTester { // Check return type in member summary. {BUG_ID + FS + "pkg" + FS + "PublicChild.html", "PublicChild" + NL + - "" + + "" + "returnTypeTest()" }, // Check return type in member detail. diff --git a/langtools/test/com/sun/javadoc/testNavigation/TestNavigation.java b/langtools/test/com/sun/javadoc/testNavigation/TestNavigation.java index d8187147bd2..bf9711eaa5c 100644 --- a/langtools/test/com/sun/javadoc/testNavigation/TestNavigation.java +++ b/langtools/test/com/sun/javadoc/testNavigation/TestNavigation.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4131628 4664607 7025314 8023700 7198273 8025633 + * @bug 4131628 4664607 7025314 8023700 7198273 8025633 8026567 * @summary Make sure the Next/Prev Class links iterate through all types. * Make sure the navagation is 2 columns, not 3. * @author jamieh @@ -46,17 +46,17 @@ public class TestNavigation extends JavadocTester { private static final String[][] TEST = { {BUG_ID + FS + "pkg" + FS + "A.html", "
  • Prev Class
  • "}, {BUG_ID + FS + "pkg" + FS + "A.html", - "Next Class"}, + "Next Class"}, {BUG_ID + FS + "pkg" + FS + "C.html", - "Prev Class"}, + "Prev Class"}, {BUG_ID + FS + "pkg" + FS + "C.html", - "Next Class"}, + "Next Class"}, {BUG_ID + FS + "pkg" + FS + "E.html", - "Prev Class"}, + "Prev Class"}, {BUG_ID + FS + "pkg" + FS + "E.html", - "Next Class"}, + "Next Class"}, {BUG_ID + FS + "pkg" + FS + "I.html", - "Prev Class"}, + "Prev Class"}, {BUG_ID + FS + "pkg" + FS + "I.html", "
  • Next Class
  • "}, // Test for 4664607 {BUG_ID + FS + "pkg" + FS + "I.html", diff --git a/langtools/test/com/sun/javadoc/testNewLanguageFeatures/TestNewLanguageFeatures.java b/langtools/test/com/sun/javadoc/testNewLanguageFeatures/TestNewLanguageFeatures.java index 65013120dc8..6f9337e3fdb 100644 --- a/langtools/test/com/sun/javadoc/testNewLanguageFeatures/TestNewLanguageFeatures.java +++ b/langtools/test/com/sun/javadoc/testNewLanguageFeatures/TestNewLanguageFeatures.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4789689 4905985 4927164 4827184 4993906 5004549 7025314 7010344 8025633 + * @bug 4789689 4905985 4927164 4827184 4993906 5004549 7025314 7010344 8025633 8026567 * @summary Run Javadoc on a set of source files that demonstrate new * language features. Check the output to ensure that the new * language features are properly documented. @@ -53,7 +53,7 @@ public class TestNewLanguageFeatures extends JavadocTester { {BUG_ID + FS + "pkg" + FS + "Coin.html", "Enum Coin"}, //Make sure enum signature is correct. {BUG_ID + FS + "pkg" + FS + "Coin.html", "
    public enum " +
    -                     "Coin" + NL +
    +                     "Coin" + NL +
                          "extends java.lang.Enum<Coin>
    " }, @@ -62,7 +62,7 @@ public class TestNewLanguageFeatures extends JavadocTester { " "}, //Detail for enum constant {BUG_ID + FS + "pkg" + FS + "Coin.html", - "Dime"}, + "Dime"}, //Automatically insert documentation for values() and valueOf(). {BUG_ID + FS + "pkg" + FS + "Coin.html", "Returns an array containing the constants of this enum type,"}, @@ -80,11 +80,11 @@ public class TestNewLanguageFeatures extends JavadocTester { "Class TypeParameters<E>"}, //Check class type parameters section. {BUG_ID + FS + "pkg" + FS + "TypeParameters.html", - "
    Type Parameters:
    " + NL + "
    E - " + + "
    Type Parameters:
    " + NL + "
    E - " + "the type parameter for this class."}, //Type parameters in @see/@link {BUG_ID + FS + "pkg" + FS + "TypeParameters.html", - "
    " + NL + "
    See Also:
    " + NL + "
    " + + "
    " + NL + "
    See Also:
    " + NL + "
    " + "" + "TypeParameters
    " + NL + "
    "}, //Method that uses class type parameter. @@ -93,7 +93,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "parameter in TypeParameters\">E param)"}, //Method type parameter section. {BUG_ID + FS + "pkg" + FS + "TypeParameters.html", - "Type Parameters:
    " + NL + "
    T - This is the first " + + "Type Parameters:
    " + NL + "
    T - This is the first " + "type parameter.
    " + NL + "
    V - This is the second type " + "parameter."}, //Signature of method with type parameters @@ -118,7 +118,7 @@ public class TestNewLanguageFeatures extends JavadocTester { //Signature of subclass that has type parameters. {BUG_ID + FS + "pkg" + FS + "TypeParameterSubClass.html", - "
    public class TypeParameterSubClass<T extends " +
    +                "
    public class TypeParameterSubClass<T extends " +
                     "java.lang.String>" + NL + "extends " +
                     "" +
                     "TypeParameterSuperClass<T>
    "}, @@ -168,7 +168,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "Annotation Type AnnotationType"}, //Make sure the signature is correct. {BUG_ID + FS + "pkg" + FS + "AnnotationType.html", - "public @interface AnnotationType"}, + "public @interface AnnotationType"}, //Make sure member summary headings are correct. {BUG_ID + FS + "pkg" + FS + "AnnotationType.html", "

    Required Element Summary

    "}, @@ -198,7 +198,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "optional" + "=\"Class Annotation\"," + NL + " " + - "required=1994)" + NL + "public class " + + "required=1994)" + NL + "public class " + "AnnotationTypeUsage" + NL + "extends java.lang.Object
    "}, //FIELD @@ -299,7 +299,7 @@ public class TestNewLanguageFeatures extends JavadocTester { {BUG_ID + FS + "pkg1" + FS + "B.html", "
    @A"},
                 {BUG_ID + FS + "pkg1" + FS + "B.html",
    -                "public interface B
    "}, + "public interface B"}, //============================================================== @@ -320,7 +320,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "Foo " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo.html", - "ClassUseTest1<T extends " + "Foo" + " & " + @@ -333,8 +333,8 @@ public class TestNewLanguageFeatures extends JavadocTester { "pkg2\">Foo " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo.html", - "ClassUseTest1." + - "ClassUseTest1." + + "method" + "(T t) " }, @@ -372,7 +372,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "" }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo2.html", - "ClassUseTest1<T extends " + "Foo" + " & " + @@ -386,8 +386,8 @@ public class TestNewLanguageFeatures extends JavadocTester { "" }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo2.html", - "" + - "ClassUseTest1." + + "ClassUseTest1.method" + "(T t) " }, @@ -401,7 +401,7 @@ public class TestNewLanguageFeatures extends JavadocTester { " " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "ParamTest.html", - "ClassUseTest2<T extends " + "" + "ParamTest<" + @@ -415,8 +415,8 @@ public class TestNewLanguageFeatures extends JavadocTester { " " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "ParamTest.html", - "ClassUseTest2." + - "ClassUseTest2." + + "method" + "(T t) " }, @@ -456,7 +456,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "Foo3 " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo3.html", - "ClassUseTest2<T extends " + "" + "ParamTest<" + @@ -470,8 +470,8 @@ public class TestNewLanguageFeatures extends JavadocTester { "" }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo3.html", - "ClassUseTest2." + - "ClassUseTest2." + + "method" + "(T t) " }, @@ -500,7 +500,7 @@ public class TestNewLanguageFeatures extends JavadocTester { " " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "ParamTest2.html", - "ClassUseTest3<T extends " + "" + "ParamTest2<java.util.List<? extends " + @@ -515,8 +515,8 @@ public class TestNewLanguageFeatures extends JavadocTester { " " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "ParamTest2.html", - "ClassUseTest3" + - ".ClassUseTest3" + + ".method(T t) " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "ParamTest2.html", @@ -538,7 +538,7 @@ public class TestNewLanguageFeatures extends JavadocTester { "" }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo4.html", - "ClassUseTest3<T extends " + "" + "ParamTest2<java.util.List<? extends " + @@ -552,8 +552,8 @@ public class TestNewLanguageFeatures extends JavadocTester { "pkg2\">Foo4 " }, {BUG_ID + FS + "pkg2" + FS + "class-use" + FS + "Foo4.html", - "ClassUseTest3." + - "ClassUseTest3." + + "method(T t)" + " " }, @@ -586,8 +586,8 @@ public class TestNewLanguageFeatures extends JavadocTester { "Method and Description" + NL + "" + NL + "" + NL + "" + NL + "void" + NL + - "ClassUseTest3." + - "ClassUseTest3." + + "method(java." + "util.Set<Foo4> p) " + NL + @@ -663,14 +663,14 @@ public class TestNewLanguageFeatures extends JavadocTester { // TYPE PARAMETER IN INDEX //================================= {BUG_ID + FS + "index-all.html", - "" + + "" + "method(Vector<Object>)" }, //================================= // TYPE PARAMETER IN INDEX //================================= {BUG_ID + FS + "index-all.html", - "" + + "" + "method(Vector<Object>)" }, }; @@ -679,7 +679,7 @@ public class TestNewLanguageFeatures extends JavadocTester { // ENUM TESTING //================================= //NO constructor section - {BUG_ID + FS + "pkg" + FS + "Coin.html", "Constructor Summary"}, + {BUG_ID + FS + "pkg" + FS + "Coin.html", "

    Constructor Summary

    "}, //================================= // TYPE PARAMETER TESTING //================================= @@ -698,25 +698,25 @@ public class TestNewLanguageFeatures extends JavadocTester { {BUG_ID + FS + "pkg" + FS + "AnnotationTypeUsage.html", "@AnnotationTypeUndocumented(optional=\"Class Annotation\"," + NL + " required=1994)" + NL + - "public class AnnotationTypeUsage
    extends java.lang.Object
    "}, + "public class AnnotationTypeUsage
    extends java.lang.Object
    "}, //FIELD {BUG_ID + FS + "pkg" + FS + "AnnotationTypeUsage.html", "@AnnotationTypeUndocumented(optional=\"Field Annotation\"," + NL + " required=1994)" + NL + - "public int field"}, + "public int field"}, //CONSTRUCTOR {BUG_ID + FS + "pkg" + FS + "AnnotationTypeUsage.html", "@AnnotationTypeUndocumented(optional=\"Constructor Annotation\"," + NL + " required=1994)" + NL + - "public AnnotationTypeUsage()"}, + "public AnnotationTypeUsage()"}, //METHOD {BUG_ID + FS + "pkg" + FS + "AnnotationTypeUsage.html", "@AnnotationTypeUndocumented(optional=\"Method Annotation\"," + NL + " required=1994)" + NL + - "public void method()"}, + "public void method()"}, //================================= // Make sure annotation types do not diff --git a/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenMethodDocCopy.java b/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenMethodDocCopy.java index 506fde14a71..92bdf84d62f 100644 --- a/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenMethodDocCopy.java +++ b/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenMethodDocCopy.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4368820 8025633 + * @bug 4368820 8025633 8026567 * @summary Inherited comment should link directly to member, not just * class * @author jamieh @@ -46,7 +46,7 @@ public class TestOverridenMethodDocCopy extends JavadocTester { //Input for string search tests. private static final String[][] TEST = { {BUG_ID + FS + "pkg1" + FS + "SubClass.html", - "Description copied from class: " + + "Description copied from class: " + "" + "BaseClass" } diff --git a/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethods.java b/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethods.java index 21a6760d308..4a0b54cac34 100644 --- a/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethods.java +++ b/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethods.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4634891 + * @bug 4634891 8026567 * @summary Determine if overriden methods are properly documented when * -protected (default) visibility flag is used. * @author jamieh @@ -40,12 +40,12 @@ public class TestOverridenPrivateMethods extends JavadocTester { private static final String[][] TEST = { //The public method should be overriden {BUG_ID + FS + "pkg1" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    " + "publicMethod in class " + "BaseClass
    "}, //The public method in different package should be overriden {BUG_ID + FS + "pkg2" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    " + "publicMethod in class " + "BaseClass
    "}, @@ -55,7 +55,7 @@ public class TestOverridenPrivateMethodsWithPackageFlag extends JavadocTester { //The package private method should be overriden since the base and sub class are in the same //package. {BUG_ID + FS + "pkg1" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    " + "packagePrivateMethod in class " + "BaseClass
    "} @@ -65,18 +65,18 @@ public class TestOverridenPrivateMethodsWithPackageFlag extends JavadocTester { //The private method in should not be overriden {BUG_ID + FS + "pkg1" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    "}, //The private method in different package should not be overriden {BUG_ID + FS + "pkg2" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    "}, //The package private method should not be overriden since the base and sub class are in //different packages. {BUG_ID + FS + "pkg2" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    "}, }; diff --git a/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethodsWithPrivateFlag.java b/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethodsWithPrivateFlag.java index fbcd30e8b37..d42cad7e151 100644 --- a/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethodsWithPrivateFlag.java +++ b/langtools/test/com/sun/javadoc/testOverridenMethods/TestOverridenPrivateMethodsWithPrivateFlag.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4634891 + * @bug 4634891 8026567 * @summary Determine if overriden methods are properly documented when * -protected (default) visibility flag is used. * @author jamieh @@ -40,18 +40,18 @@ public class TestOverridenPrivateMethodsWithPrivateFlag extends JavadocTester { private static final String[][] TEST = { //The public method should be overriden {BUG_ID + FS + "pkg1" + FS + "SubClass.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Overrides:" + NL + + "
    Overrides:
    " + NL + "
    Deprecated." + NL + - "
    This package is Deprecated." + + "
    Deprecated." + NL + + "
    This package is Deprecated." + "
    " }, {BUG_ID + "-1" + FS + "deprecated-list.html", diff --git a/langtools/test/com/sun/javadoc/testParamTaglet/TestParamTaglet.java b/langtools/test/com/sun/javadoc/testParamTaglet/TestParamTaglet.java index 22715395250..dd2e11b0381 100644 --- a/langtools/test/com/sun/javadoc/testParamTaglet/TestParamTaglet.java +++ b/langtools/test/com/sun/javadoc/testParamTaglet/TestParamTaglet.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4802275 4967243 + * @bug 4802275 4967243 8026567 * @summary Make sure param tags are still printed even though they do not * match up with a real parameters. * Make sure inheritDoc cannot be used in an invalid param tag. @@ -48,12 +48,12 @@ public class TestParamTaglet extends JavadocTester { private static final String[][] TEST = { //Regular param tags. {BUG_ID + FS + "pkg" + FS + "C.html", - "Parameters:" + NL + "
    param1 - testing 1 2 3.
    " + + "Parameters:" + NL + "
    param1 - testing 1 2 3.
    " + NL + "
    param2 - testing 1 2 3." }, //Param tags that don't match with any real parameters. {BUG_ID + FS + "pkg" + FS + "C.html", - "Parameters:" + NL + "
    p1 - testing 1 2 3.
    " + + "Parameters:" + NL + "
    p1 - testing 1 2 3.
    " + NL + "
    p2 - testing 1 2 3." }, //{@inherit} doc misuse does not cause doclet to throw exception. diff --git a/langtools/test/com/sun/javadoc/testPrivateClasses/TestPrivateClasses.java b/langtools/test/com/sun/javadoc/testPrivateClasses/TestPrivateClasses.java index 6da9e9c1df7..2e422bbf9af 100644 --- a/langtools/test/com/sun/javadoc/testPrivateClasses/TestPrivateClasses.java +++ b/langtools/test/com/sun/javadoc/testPrivateClasses/TestPrivateClasses.java @@ -23,7 +23,7 @@ /* * @test - * @bug 4780441 4874845 4978816 8014017 8016328 8025633 + * @bug 4780441 4874845 4978816 8014017 8016328 8025633 8026567 * @summary Make sure that when the -private flag is not used, members * inherited from package private class are documented in the child. * @@ -112,12 +112,12 @@ public class TestPrivateClasses extends JavadocTester { private static final String[][] NEGATED_TEST1 = { // Should not document that a method overrides method from private class. {BUG_ID + "-1" + FS + "pkg" + FS + "PublicChild.html", - "Overrides:"}, + "Overrides:"}, // Should not document that a method specified by private interface. {BUG_ID + "-1" + FS + "pkg" + FS + "PublicChild.html", - "Specified by:"}, + "Specified by:"}, {BUG_ID + "-1" + FS + "pkg" + FS + "PublicInterface.html", - "Specified by:"}, + "Specified by:"}, // Should not mention that any documentation was copied. {BUG_ID + "-1" + FS + "pkg" + FS + "PublicChild.html", "Description copied from"}, @@ -139,7 +139,7 @@ public class TestPrivateClasses extends JavadocTester { //Do not inherit private interface method with generic parameters. //This method has been implemented. {BUG_ID + "-1" + FS + "pkg2" + FS + "C.html", - "hello"}, + "hello"}, }; // Test output when -private flag is used. @@ -176,14 +176,14 @@ public class TestPrivateClasses extends JavadocTester { }, // Should document that a method overrides method from private class. {BUG_ID + "-2" + FS + "pkg" + FS + "PublicChild.html", - "
    Overrides:
    " + NL + + "
    Overrides:
    " + NL + "
    " + "methodOverridenFromParent in class " + "" + "PrivateParent
    "}, // Should document that a method is specified by private interface. {BUG_ID + "-2" + FS + "pkg" + FS + "PublicChild.html", - "
    Specified by:
    " + NL + + "
    Specified by:
    " + NL + "
    " + "methodInterface in interface " + "" + @@ -227,11 +227,11 @@ public class TestPrivateClasses extends JavadocTester { //Since private flag is used, we can document that private interface method //with generic parameters has been implemented. {BUG_ID + "-2" + FS + "pkg2" + FS + "C.html", - "Description copied from interface: " + + "Description copied from interface: " + "I"}, {BUG_ID + "-2" + FS + "pkg2" + FS + "C.html", - "
    Specified by:
    " + NL + + "
    Specified by:
    " + NL + "
    hello" + " in interface " + "I" + @@ -240,14 +240,14 @@ public class TestPrivateClasses extends JavadocTester { //Make sure when no modifier appear in the class signature, the //signature is displayed correctly without extra space at the beginning. {BUG_ID + "-2" + FS + "pkg" + FS + "PrivateParent.html", - "
    class PrivateParent"},
    +            "
    class PrivateParent"},
     
           {BUG_ID + "-2" + FS + "pkg" + FS + "PublicChild.html",
    -            "
    public class PublicChild"},
    +            "
    public class PublicChild"},
         };
         private static final String[][] NEGATED_TEST2 = {
             {BUG_ID + "-2" + FS + "pkg" + FS + "PrivateParent.html",
    -            "
     class PrivateParent"},
    +            "
     class PrivateParent"},
         };
     
         /**
    diff --git a/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java b/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java
    index 1d308246c3f..0dc858dddd6 100644
    --- a/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java
    +++ b/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java
    @@ -23,7 +23,7 @@
     
     /*
      * @test
    - * @bug      8006124 8009684 8016921 8023700 8024096 8008164
    + * @bug      8006124 8009684 8016921 8023700 8024096 8008164 8026567
      * @summary  Test javadoc support for profiles.
      * @author   Bhavesh Patel, Evgeniya Stepanova
      * @library  ../lib/
    @@ -122,17 +122,17 @@ public class TestProfiles extends JavadocTester {
             },
             {PROFILE_BUG_ID + FS + "deprecated-list.html",""
                 + "pkg2.Class1Pkg2"
    -            + NL +"
    Class1Pkg2. This class is deprecated
    " + + NL +"
    Class1Pkg2. This class is deprecated
    " }, //Test deprecated package in profile {PROFILE_BUG_ID + FS + "deprecated-list.html","" + "pkgDeprecated" - + NL +"
    This package is Deprecated." + + NL +"
    This package is Deprecated." + " Use pkg1.
    " }, {PROFILE_BUG_ID + FS + "pkgDeprecated" + FS + "package-summary.html", - "
    Deprecated." - + NL + "
    This package is Deprecated." + "
    Deprecated." + + NL + "
    This package is Deprecated." + " Use pkg1.
    " }, // need to add teststring when JDK-8015496 will be fixed diff --git a/langtools/test/com/sun/javadoc/testProfiles/TestProfilesConfiguration.java b/langtools/test/com/sun/javadoc/testProfiles/TestProfilesConfiguration.java index 287cca02952..f194bddff93 100644 --- a/langtools/test/com/sun/javadoc/testProfiles/TestProfilesConfiguration.java +++ b/langtools/test/com/sun/javadoc/testProfiles/TestProfilesConfiguration.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8006124 8009684 8015663 8015496 + * @bug 8006124 8009684 8015663 8015496 8026567 * @summary Test javadoc options support for profiles. * @author Evgeniya Stepanova * @library ../lib/ @@ -87,7 +87,7 @@ public class TestProfilesConfiguration extends JavadocTester { {PROFILE_CONFIGURATION_BUG_ID + FS + "compact1-summary.html", "

    pkgDeprecated

    " + NL + "
    " + - "Deprecated.
    " + "Deprecated.
    " } }; private static final String[][] PROFILES_CONFIGURATION_NEGATED_TEST = { diff --git a/langtools/test/com/sun/javadoc/testSerializedFormDeprecationInfo/TestSerializedFormDeprecationInfo.java b/langtools/test/com/sun/javadoc/testSerializedFormDeprecationInfo/TestSerializedFormDeprecationInfo.java index 06217b94a56..1f391d08184 100644 --- a/langtools/test/com/sun/javadoc/testSerializedFormDeprecationInfo/TestSerializedFormDeprecationInfo.java +++ b/langtools/test/com/sun/javadoc/testSerializedFormDeprecationInfo/TestSerializedFormDeprecationInfo.java @@ -25,7 +25,7 @@ /* * @test - * @bug 6802694 8025633 + * @bug 6802694 8025633 8026567 * @summary This test verifies deprecation info in serialized-form.html. * @author Bhavesh Patel * @library ../lib/ @@ -42,29 +42,29 @@ public class TestSerializedFormDeprecationInfo extends JavadocTester { // display the inline comments, tags and deprecation information if any. private static final String[][] TEST_CMNT_DEPR = { {BUG_ID + FS + "serialized-form.html", "
    " + NL + - "
    Throws:
    " + NL + "
    " + - "java.io.IOException
    "+ NL + "
    See Also:" + + "
    Throws:
    " + NL + "
    " + + "java.io.IOException
    "+ NL + "
    See Also:" + "
    " + NL + "
    " + "C1.setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + - " As of JDK version 1.5, replaced by" + NL + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + + " As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean).
    " + NL + "
    This field indicates whether the C1 " + "is undecorated.
    " + NL + " " + NL + - "
    " + NL + "
    Since:
    " + NL + - "
    1.4
    " + NL + "
    See Also:" + + "
    " + NL + "
    Since:
    " + NL + + "
    1.4
    " + NL + "
    See Also:" + "
    " + NL + "
    " + "C1.setUndecorated(boolean)
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + - " As of JDK version 1.5, replaced by" + NL + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + + " As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean).
    " + NL + "
    Reads the object stream.
    " + NL + - "
    " + NL + "
    Throws:
    " + NL + "
    " + + "
    " + NL + "
    Throws:
    " + NL + "
    " + "IOException
    " + NL + "
    java.io.IOException
    " + NL + "
    "}, - {BUG_ID + FS + "serialized-form.html", "Deprecated." + + {BUG_ID + FS + "serialized-form.html", "Deprecated." + " 
    " + NL + "
    " + "The name for this class.
    "}}; @@ -73,12 +73,12 @@ public class TestSerializedFormDeprecationInfo extends JavadocTester { // information if any. private static final String[][] TEST_NOCMNT = { {BUG_ID + FS + "serialized-form.html", "
    boolean undecorated
    " + NL + - "
    Deprecated. " + + "
    Deprecated. " + "As of JDK version 1.5, replaced by" + NL + " " + "setUndecorated(boolean).
    " + NL + ""}, - {BUG_ID + FS + "serialized-form.html", "" + - "Deprecated. As of JDK version" + + {BUG_ID + FS + "serialized-form.html", "" + + "Deprecated. As of JDK version" + " 1.5, replaced by" + NL + " " + "setUndecorated(boolean).
    " + NL + ""}}; diff --git a/langtools/test/com/sun/javadoc/testSimpleTag/TestSimpleTag.java b/langtools/test/com/sun/javadoc/testSimpleTag/TestSimpleTag.java index 94b3eb134ef..e48e805ae6e 100644 --- a/langtools/test/com/sun/javadoc/testSimpleTag/TestSimpleTag.java +++ b/langtools/test/com/sun/javadoc/testSimpleTag/TestSimpleTag.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2013, 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,7 @@ /* * @test - * @bug 4695326 4750173 4920381 + * @bug 4695326 4750173 4920381 8026567 * @summary Test the declarartion of simple tags using -tag. Verify that * "-tag name" is a shortcut for "-tag name:a:Name:". Also verity that * you can escape the ":" character with a back slash so that it is not @@ -42,13 +42,13 @@ public class TestSimpleTag extends JavadocTester { private static final String[][] TEST = new String[][] { {"./" + BUG_ID + "/C.html", - "Todo:"}, + "Todo:"}, {"./" + BUG_ID + "/C.html", - "EJB Beans:"}, + "EJB Beans:"}, {"./" + BUG_ID + "/C.html", - "Regular Tag:"}, + "Regular Tag:"}, {"./" + BUG_ID + "/C.html", - "Back-Slash-Tag:"}, + "Back-Slash-Tag:"}, }; private static final String[] ARGS = new String[] { diff --git a/langtools/test/com/sun/javadoc/testSimpleTagInherit/TestSimpleTagInherit.java b/langtools/test/com/sun/javadoc/testSimpleTagInherit/TestSimpleTagInherit.java index 4b5ac2310b9..83617bd93bb 100644 --- a/langtools/test/com/sun/javadoc/testSimpleTagInherit/TestSimpleTagInherit.java +++ b/langtools/test/com/sun/javadoc/testSimpleTagInherit/TestSimpleTagInherit.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8008768 + * @bug 8008768 8026567 * @summary Using {@inheritDoc} in simple tag defined via -tag fails * @library ../lib/ * @build JavadocTester TestSimpleTagInherit @@ -46,10 +46,10 @@ public class TestSimpleTagInherit extends JavadocTester { //Input for string search tests. private static final String[][] TEST = { { BUG_ID + FS + "p" + FS + "TestClass.html", - "
    Custom:
    " + NL + + "
    Custom:
    " + NL + "
    doc for BaseClass class
    " }, { BUG_ID + FS + "p" + FS + "TestClass.html", - "
    Custom:
    " + NL + + "
    Custom:
    " + NL + "
    doc for BaseClass method
    " } }; private static final String[][] NEGATED_TEST = NO_TEST; diff --git a/langtools/test/com/sun/javadoc/testSinceTag/TestSinceTag.java b/langtools/test/com/sun/javadoc/testSinceTag/TestSinceTag.java index 531047cc065..b3538994c81 100644 --- a/langtools/test/com/sun/javadoc/testSinceTag/TestSinceTag.java +++ b/langtools/test/com/sun/javadoc/testSinceTag/TestSinceTag.java @@ -23,7 +23,7 @@ /* * @test - * @bug 7180906 + * @bug 7180906 8026567 * @summary Test to make sure that the since tag works correctly * @author Bhavesh Patel * @library ../lib/ @@ -48,11 +48,11 @@ public class TestSinceTag extends JavadocTester { //Input for string search tests. private static final String[][] TEST = { {BUG_ID + FS + "pkg1" + FS + "C1.html", - "
    " + NL + "
    Since:
    " + NL + + "
    " + NL + "
    Since:
    " + NL + "
    JDK1.0
    " }, {BUG_ID + FS + "serialized-form.html", - "
    " + NL + "
    Since:
    " + NL + + "
    " + NL + "
    Since:
    " + NL + "
    1.4
    " } }; diff --git a/langtools/test/com/sun/javadoc/testTagOutput/TestTagOutput.java b/langtools/test/com/sun/javadoc/testTagOutput/TestTagOutput.java index c7d5f7a186e..aecf75154e8 100644 --- a/langtools/test/com/sun/javadoc/testTagOutput/TestTagOutput.java +++ b/langtools/test/com/sun/javadoc/testTagOutput/TestTagOutput.java @@ -25,7 +25,7 @@ /* * @test - * @bug 8026370 + * @bug 8026370 8026567 * @summary This test checks the generated tag output. * @author Bhavesh Patel * @library ../lib/ @@ -38,15 +38,15 @@ public class TestTagOutput extends JavadocTester { private static final String BUG_ID = "8026370"; private static final String[][] TEST = { {BUG_ID + FS + "pkg1" + FS + "DeprecatedTag.html", - "
    Deprecated. 
    "}, + "
    Deprecated. 
    "}, {BUG_ID + FS + "pkg1" + FS + "DeprecatedTag.html", - "
    Deprecated. " + - "Do not use this.
    "}}; + "
    Deprecated. " + + "Do not use this.
    "}}; private static final String[][] NEGATED_TEST = { {BUG_ID + FS + "pkg1" + FS + "DeprecatedTag.html", - "
    Deprecated." + - " 
    "}}; + "
    Deprecated." + + " 
    "}}; private static final String[] ARGS = new String[] { diff --git a/langtools/test/com/sun/javadoc/testTaglets/TestTaglets.java b/langtools/test/com/sun/javadoc/testTaglets/TestTaglets.java index e49e857e6fe..563b84a2e6b 100644 --- a/langtools/test/com/sun/javadoc/testTaglets/TestTaglets.java +++ b/langtools/test/com/sun/javadoc/testTaglets/TestTaglets.java @@ -55,7 +55,7 @@ public class TestTaglets extends JavadocTester { //Input for string search tests. private static final String[][] TEST_4654308 = new String[][] { - {"4654308" + FS + "C.html", "Foo:" + + {"4654308" + FS + "C.html", "Foo:" + "
    my only method is here" + "
    "} }; diff --git a/langtools/test/com/sun/javadoc/testTaglets/taglets/Foo.java b/langtools/test/com/sun/javadoc/testTaglets/taglets/Foo.java index b0b7e475681..d3183718ec8 100644 --- a/langtools/test/com/sun/javadoc/testTaglets/taglets/Foo.java +++ b/langtools/test/com/sun/javadoc/testTaglets/taglets/Foo.java @@ -50,7 +50,7 @@ public class Foo extends BaseTaglet { */ public Content getTagletOutput(Tag tag, TagletWriter writer) { ArrayList inlineTags = new ArrayList(); - inlineTags.add(new TextTag(tag.holder(), "
    Foo:
    ")); + inlineTags.add(new TextTag(tag.holder(), "
    Foo:
    ")); inlineTags.addAll(Arrays.asList(tag.inlineTags())); inlineTags.add(new TextTag(tag.holder(), "
    ")); return writer.commentTagsToOutput(tag, diff --git a/langtools/test/com/sun/javadoc/testThrowsHead/TestThrowsHead.java b/langtools/test/com/sun/javadoc/testThrowsHead/TestThrowsHead.java index 49408e4ed01..67a55e226af 100644 --- a/langtools/test/com/sun/javadoc/testThrowsHead/TestThrowsHead.java +++ b/langtools/test/com/sun/javadoc/testThrowsHead/TestThrowsHead.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2013, 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,7 @@ /** * @test - * @bug 4530727 + * @bug 4530727 8026567 * @summary When an exception is declared in the method signature but * not documented with a throws tag, we generate a link to it in the * throws section. Make sure that the link is below a Throws heading. @@ -38,7 +38,7 @@ public class TestThrowsHead extends JavadocTester { private static final String BUG_ID = "4530727"; private static final String[][] TEST = { - {BUG_ID + FS + "C.html", "
    Throws:"} + {BUG_ID + FS + "C.html", "
    Throws:"} }; private static final String[][] NEGATED_TEST = NO_TEST; private static final String[] ARGS = new String[] { diff --git a/langtools/test/com/sun/javadoc/testTypeAnnotations/TestTypeAnnotations.java b/langtools/test/com/sun/javadoc/testTypeAnnotations/TestTypeAnnotations.java index 0627c07fccd..ad94e1d50b4 100644 --- a/langtools/test/com/sun/javadoc/testTypeAnnotations/TestTypeAnnotations.java +++ b/langtools/test/com/sun/javadoc/testTypeAnnotations/TestTypeAnnotations.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8005091 8009686 8025633 + * @bug 8005091 8009686 8025633 8026567 * @summary Make sure that type annotations are displayed correctly * @author Bhavesh Patel * @library ../lib/ @@ -74,12 +74,12 @@ public class TestTypeAnnotations extends JavadocTester { // Test for type annotations on Class Parameters (ClassParameters.java). {BUG_ID + FS + "typeannos" + FS + "ExtendsBound.html", - "class ExtendsBound<K extends ExtendsBound<K extends @ClassParamA java.lang.String>" }, {BUG_ID + FS + "typeannos" + FS + "ExtendsGeneric.html", - "
    class ExtendsGeneric<K extends " +
    +            "
    class ExtendsGeneric<K extends " +
                 "@ClassParamA Unannotated< java.lang.String>>"
             },
             {BUG_ID + FS + "typeannos" + FS + "TwoBounds.html",
    -            "
    class TwoBounds<K extends class TwoBounds<K extends " +
                 "@ClassParamA java.lang.String,V extends @ClassParamB" +
                 " java.lang.String>"
             },
             {BUG_ID + FS + "typeannos" + FS + "Complex1.html",
    -            "class Complex1<K extends Complex1<K extends " +
                 "@ClassParamA java.lang.String & java.lang.Runnable>"
             },
             {BUG_ID + FS + "typeannos" + FS + "Complex2.html",
    -            "class Complex2<K extends java.lang." +
    +            "class Complex2<K extends java.lang." +
                 "String & @ClassParamB java.lang.Runnable>"
             },
             {BUG_ID + FS + "typeannos" + FS + "ComplexBoth.html",
    -            "class ComplexBoth<K extends ComplexBoth<K extends @ClassParamA java.lang.String & @ClassParamA" +
    diff --git a/langtools/test/com/sun/javadoc/testValueTag/TestValueTag.java b/langtools/test/com/sun/javadoc/testValueTag/TestValueTag.java
    index 1d24a52798a..f035444b510 100644
    --- a/langtools/test/com/sun/javadoc/testValueTag/TestValueTag.java
    +++ b/langtools/test/com/sun/javadoc/testValueTag/TestValueTag.java
    @@ -23,7 +23,7 @@
     
     /*
      * @test
    - * @bug      4764045 8004825
    + * @bug      4764045 8004825 8026567
      * @summary  This test ensures that the value tag works in all
      * use cases. The explainations for each test case are written below.
      * @author   jamieh
    @@ -94,7 +94,7 @@ public class TestValueTag extends JavadocTester {
                 "Result: \"Test 17 passes\""},
             //Test @value tag used with custom tag.
             {BUG_ID + FS + "pkg1" + FS + "CustomTagUsage.html",
    -            "
    Todo:
    " + NL + + "
    Todo:
    " + NL + "
    the value of this constant is 55.
    "}, //Test @value errors printed dues to invalid use or when used with //non-constant or with bad references. From 027b935c1b780731322112ba7d458235e9fd273a Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Sat, 19 Oct 2013 17:43:09 +0100 Subject: [PATCH 04/28] 8024809: javac, some lambda programs are rejected by flow analysis Reviewed-by: jjg, dlsmith --- .../com/sun/tools/javac/comp/Attr.java | 3 - .../tools/javac/lambda/8016081/T8016081.java | 2 +- .../test/tools/javac/lambda/LambdaExpr13.java | 7 +- .../SelfInitializerInLambdaTesta.java | 65 +++++++++++++++++++ .../T8024809/SelfInitializerInLambdaTesta.out | 7 ++ .../SelfInitializerInLambdaTestb.java | 40 ++++++++++++ .../T8024809/SelfInitializerInLambdaTestb.out | 3 + .../test/tools/javac/lambda/TestSelfRef.java | 16 +++-- 8 files changed, 131 insertions(+), 12 deletions(-) create mode 100644 langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.java create mode 100644 langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.out create mode 100644 langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.java create mode 100644 langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.out diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java index 458c16caacd..046a5087f1a 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java @@ -314,9 +314,6 @@ public class Attr extends JCTree.Visitor { case CLASSDEF: //class def is always an owner return ((JCClassDecl)env.tree).sym; - case LAMBDA: - //a lambda is an owner - return a fresh synthetic method symbol - return new MethodSymbol(0, names.empty, null, syms.methodClass); case BLOCK: //static/instance init blocks are owner Symbol blockSym = env.info.scope.owner; diff --git a/langtools/test/tools/javac/lambda/8016081/T8016081.java b/langtools/test/tools/javac/lambda/8016081/T8016081.java index 648955f83cc..055712747a7 100644 --- a/langtools/test/tools/javac/lambda/8016081/T8016081.java +++ b/langtools/test/tools/javac/lambda/8016081/T8016081.java @@ -32,7 +32,7 @@ class T8016081 { interface fint { int get(); } @interface atype { - fint fld = ()->( fld == null ?0 : 1); + fint fld = ()->1; } @atype class T {} diff --git a/langtools/test/tools/javac/lambda/LambdaExpr13.java b/langtools/test/tools/javac/lambda/LambdaExpr13.java index 7a5181e54fe..2b2d75e6e32 100644 --- a/langtools/test/tools/javac/lambda/LambdaExpr13.java +++ b/langtools/test/tools/javac/lambda/LambdaExpr13.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2013, 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 @@ -26,13 +26,14 @@ * @bug 8003280 * @summary Add lambda tests * check that recursive lambda (through field ref) is accepted in all contexts + * but field initialization * @compile LambdaExpr13.java */ class LambdaExpr13 { - Runnable ir = () -> { ir.run(); };; - static Runnable sr = () -> { sr.run(); }; + Runnable ir; + static Runnable sr; { ir = () -> { ir.run(); }; } static { sr = () -> { sr.run(); }; } diff --git a/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.java b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.java new file mode 100644 index 00000000000..408f7b8f228 --- /dev/null +++ b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.java @@ -0,0 +1,65 @@ +/* + * Copyright (c) 2013, 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 8024809 + * @summary javac, some lambda programs are rejected by flow analysis + * @compile/fail/ref=SelfInitializerInLambdaTesta.out -XDrawDiagnostics SelfInitializerInLambdaTesta.java + */ + +public class SelfInitializerInLambdaTesta { + + final Runnable r1 = ()->System.out.println(r1); + + final Object lock = new Object(); + + final Runnable r2 = ()->{ + System.out.println(r2); + synchronized (lock){} + }; + + final Runnable r3 = ()->{ + synchronized (lock){ + System.out.println(r3); + } + }; + + final Runnable r4 = ()->{ + System.out.println(r4); + }; + + interface SAM { + int m(String s); + } + + final SAM s1 = (String s)->{ + System.out.println(s + s1.toString()); + return 0; + }; + + final SAM s2 = (s)->{ + System.out.println(s + s2.toString()); + return 0; + }; +} diff --git a/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.out b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.out new file mode 100644 index 00000000000..fcf50e891a9 --- /dev/null +++ b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTesta.out @@ -0,0 +1,7 @@ +SelfInitializerInLambdaTesta.java:33:48: compiler.err.illegal.self.ref +SelfInitializerInLambdaTesta.java:38:28: compiler.err.illegal.self.ref +SelfInitializerInLambdaTesta.java:44:32: compiler.err.illegal.self.ref +SelfInitializerInLambdaTesta.java:49:28: compiler.err.illegal.self.ref +SelfInitializerInLambdaTesta.java:57:32: compiler.err.illegal.self.ref +SelfInitializerInLambdaTesta.java:62:32: compiler.err.illegal.self.ref +6 errors diff --git a/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.java b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.java new file mode 100644 index 00000000000..87c254f7387 --- /dev/null +++ b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.java @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2013, 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 8024809 + * @summary javac, some lambda programs are rejected by flow analysis + * @compile/fail/ref=SelfInitializerInLambdaTestb.out -XDrawDiagnostics SelfInitializerInLambdaTestb.java + */ + +public class SelfInitializerInLambdaTestb { + + final Runnable r1; + + final Runnable r2 = ()-> System.out.println(r1); + + SelfInitializerInLambdaTestb() { + r1 = ()->System.out.println(r1); + } +} diff --git a/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.out b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.out new file mode 100644 index 00000000000..ebbdcab396c --- /dev/null +++ b/langtools/test/tools/javac/lambda/T8024809/SelfInitializerInLambdaTestb.out @@ -0,0 +1,3 @@ +SelfInitializerInLambdaTestb.java:35:49: compiler.err.var.might.not.have.been.initialized: r1 +SelfInitializerInLambdaTestb.java:38:37: compiler.err.var.might.not.have.been.initialized: r1 +2 errors diff --git a/langtools/test/tools/javac/lambda/TestSelfRef.java b/langtools/test/tools/javac/lambda/TestSelfRef.java index 967673e9fc2..58ab1426103 100644 --- a/langtools/test/tools/javac/lambda/TestSelfRef.java +++ b/langtools/test/tools/javac/lambda/TestSelfRef.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2013, 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 @@ -29,7 +29,6 @@ * consistently w.r.t. local inner classes */ -import com.sun.source.util.JavacTask; import java.net.URI; import java.util.Arrays; import javax.tools.Diagnostic; @@ -38,6 +37,7 @@ import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import javax.tools.StandardJavaFileManager; import javax.tools.ToolProvider; +import com.sun.source.util.JavacTask; public class TestSelfRef { @@ -176,10 +176,16 @@ public class TestSelfRef { check(); } - void check() { + boolean isErrorExpected() { //illegal forward ref - boolean errorExpected = ik.inMethodContext(sk) && - (rk.selfRef || rk.forwardRef); + boolean result = ik.inMethodContext(sk) && (rk.selfRef || rk.forwardRef); + result |= (rk == RefKind.SELF_LAMBDA || rk == RefKind.FORWARD_LAMBDA); + return result; + } + + void check() { + checkCount++; + boolean errorExpected = isErrorExpected(); if (diagChecker.errorFound != errorExpected) { throw new Error("invalid diagnostics for source:\n" + source.getCharContent(true) + From 84cadf2e45a66c7eeb40edefea68afa69a361dea Mon Sep 17 00:00:00 2001 From: Werner Dietl Date: Sun, 20 Oct 2013 12:01:43 -0700 Subject: [PATCH 05/28] 8025109: Better encapsulation for AnnotatedType Reviewed-by: jjg --- .../com/sun/tools/javac/code/Symbol.java | 80 +++++++++---------- .../sun/tools/javac/code/TypeAnnotations.java | 4 +- .../com/sun/tools/javac/comp/Attr.java | 2 - 3 files changed, 42 insertions(+), 44 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/Symbol.java b/langtools/src/share/classes/com/sun/tools/javac/code/Symbol.java index 5efd2b66b7c..cb3637c8557 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/Symbol.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/Symbol.java @@ -100,7 +100,7 @@ public abstract class Symbol extends AnnoConstruct implements Element { /** The attributes of this symbol are contained in this * SymbolMetadata. The SymbolMetadata instance is NOT immutable. */ - protected SymbolMetadata annotations; + protected SymbolMetadata metadata; /** An accessor method for the attributes of this symbol. @@ -108,9 +108,9 @@ public abstract class Symbol extends AnnoConstruct implements Element { * method to make sure that the class symbol is loaded. */ public List getRawAttributes() { - return (annotations == null) + return (metadata == null) ? List.nil() - : annotations.getDeclarationAttributes(); + : metadata.getDeclarationAttributes(); } /** An accessor method for the type attributes of this symbol. @@ -118,9 +118,9 @@ public abstract class Symbol extends AnnoConstruct implements Element { * method to make sure that the class symbol is loaded. */ public List getRawTypeAttributes() { - return (annotations == null) + return (metadata == null) ? List.nil() - : annotations.getTypeAttributes(); + : metadata.getTypeAttributes(); } /** Fetch a particular annotation from a symbol. */ @@ -132,106 +132,106 @@ public abstract class Symbol extends AnnoConstruct implements Element { } public boolean annotationsPendingCompletion() { - return annotations == null ? false : annotations.pendingCompletion(); + return metadata == null ? false : metadata.pendingCompletion(); } public void appendAttributes(List l) { if (l.nonEmpty()) { - initedAnnos().append(l); + initedMetadata().append(l); } } public void appendClassInitTypeAttributes(List l) { if (l.nonEmpty()) { - initedAnnos().appendClassInitTypeAttributes(l); + initedMetadata().appendClassInitTypeAttributes(l); } } public void appendInitTypeAttributes(List l) { if (l.nonEmpty()) { - initedAnnos().appendInitTypeAttributes(l); + initedMetadata().appendInitTypeAttributes(l); } } public void appendTypeAttributesWithCompletion(final Annotate.AnnotateRepeatedContext ctx) { - initedAnnos().appendTypeAttributesWithCompletion(ctx); + initedMetadata().appendTypeAttributesWithCompletion(ctx); } public void appendUniqueTypeAttributes(List l) { if (l.nonEmpty()) { - initedAnnos().appendUniqueTypes(l); + initedMetadata().appendUniqueTypes(l); } } public List getClassInitTypeAttributes() { - return (annotations == null) + return (metadata == null) ? List.nil() - : annotations.getClassInitTypeAttributes(); + : metadata.getClassInitTypeAttributes(); } public List getInitTypeAttributes() { - return (annotations == null) + return (metadata == null) ? List.nil() - : annotations.getInitTypeAttributes(); + : metadata.getInitTypeAttributes(); } public List getDeclarationAttributes() { - return (annotations == null) + return (metadata == null) ? List.nil() - : annotations.getDeclarationAttributes(); + : metadata.getDeclarationAttributes(); } public boolean hasAnnotations() { - return (annotations != null && !annotations.isEmpty()); + return (metadata != null && !metadata.isEmpty()); } public boolean hasTypeAnnotations() { - return (annotations != null && !annotations.isTypesEmpty()); + return (metadata != null && !metadata.isTypesEmpty()); } public void prependAttributes(List l) { if (l.nonEmpty()) { - initedAnnos().prepend(l); + initedMetadata().prepend(l); } } public void resetAnnotations() { - initedAnnos().reset(); + initedMetadata().reset(); } public void setAttributes(Symbol other) { - if (annotations != null || other.annotations != null) { - initedAnnos().setAttributes(other.annotations); + if (metadata != null || other.metadata != null) { + initedMetadata().setAttributes(other.metadata); } } public void setDeclarationAttributes(List a) { - if (annotations != null || a.nonEmpty()) { - initedAnnos().setDeclarationAttributes(a); + if (metadata != null || a.nonEmpty()) { + initedMetadata().setDeclarationAttributes(a); } } public void setDeclarationAttributesWithCompletion(final Annotate.AnnotateRepeatedContext ctx) { - initedAnnos().setDeclarationAttributesWithCompletion(ctx); + initedMetadata().setDeclarationAttributesWithCompletion(ctx); } public void setTypeAttributes(List a) { - if (annotations != null || a.nonEmpty()) { - if (annotations == null) - annotations = new SymbolMetadata(this); - annotations.setTypeAttributes(a); + if (metadata != null || a.nonEmpty()) { + if (metadata == null) + metadata = new SymbolMetadata(this); + metadata.setTypeAttributes(a); } } - private SymbolMetadata initedAnnos() { - if (annotations == null) - annotations = new SymbolMetadata(this); - return annotations; + private SymbolMetadata initedMetadata() { + if (metadata == null) + metadata = new SymbolMetadata(this); + return metadata; } /** This method is intended for debugging only. */ - public SymbolMetadata getAnnotations() { - return annotations; + public SymbolMetadata getMetadata() { + return metadata; } // @@ -862,10 +862,10 @@ public abstract class Symbol extends AnnoConstruct implements Element { } private void mergeAttributes() { - if (annotations == null && - package_info.annotations != null) { - annotations = new SymbolMetadata(this); - annotations.setAttributes(package_info.annotations); + if (metadata == null && + package_info.metadata != null) { + metadata = new SymbolMetadata(this); + metadata.setAttributes(package_info.metadata); } } diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java b/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java index 36da3077f05..a9c29756455 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java @@ -404,11 +404,11 @@ public class TypeAnnotations { depth = depth.append(TypePathEntry.ARRAY); while (arType.elemtype.hasTag(TypeTag.ARRAY)) { if (arType.elemtype.isAnnotated()) { - Type.AnnotatedType aelemtype = (Type.AnnotatedType) arType.elemtype; + Type aelemtype = arType.elemtype; arType = (Type.ArrayType) aelemtype.unannotatedType(); ArrayType prevToMod = tomodify; tomodify = new Type.ArrayType(null, arType.tsym); - prevToMod.elemtype = (Type.AnnotatedType) tomodify.annotatedType(arType.elemtype.getAnnotationMirrors()); + prevToMod.elemtype = tomodify.annotatedType(arType.elemtype.getAnnotationMirrors()); } else { arType = (Type.ArrayType) arType.elemtype; tomodify.elemtype = new Type.ArrayType(null, arType.tsym); diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java index 046a5087f1a..16c5e5f6dfb 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java @@ -4062,8 +4062,6 @@ public class Attr extends JCTree.Visitor { * Apply the annotations to the particular type. */ public void annotateType(final JCTree tree, final List annotations) { - // Callers ensure this. - // Assert.check(annotations != null && annotations.nonEmpty()); annotate.typeAnnotation(new Annotate.Worker() { @Override public String toString() { From 29582c0ec0229a963e9d05fd5a4435f6f5cd8975 Mon Sep 17 00:00:00 2001 From: Werner Dietl Date: Sun, 20 Oct 2013 12:46:12 -0700 Subject: [PATCH 06/28] 8026791: wrong type_path encoded for method_return on an inner class constructor Reviewed-by: jjg --- .../sun/tools/javac/code/TypeAnnotations.java | 11 ++++++++++- .../referenceinfos/Constructors.java | 18 ++++++++++-------- 2 files changed, 20 insertions(+), 9 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java b/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java index a9c29756455..a9030cac1f1 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java @@ -323,7 +323,16 @@ public class TypeAnnotations { if (type == null) { // When type is null, put the type annotations to the symbol. // This is used for constructor return annotations, for which - // no appropriate type exists. + // we use the type of the enclosing class. + type = sym.getEnclosingElement().asType(); + + // Declaration annotations are always allowed on constructor returns. + // Therefore, use typeAnnotations instead of onlyTypeAnnos. + type = typeWithAnnotations(typetree, type, typeAnnotations, typeAnnotations); + // Note that we don't use the result, the call to + // typeWithAnnotations side-effects the type annotation positions. + // This is important for constructors of nested classes. + sym.appendUniqueTypeAttributes(typeAnnotations); return; } diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/Constructors.java b/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/Constructors.java index a8d1311eb47..3106341ddc5 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/Constructors.java +++ b/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/Constructors.java @@ -21,14 +21,16 @@ * questions. */ -import static com.sun.tools.classfile.TypeAnnotation.TargetType.*; - /* * @test + * @bug 8026791 * @summary Test population of reference info for constructor results * @compile -g Driver.java ReferenceInfoUtil.java Constructors.java * @run main Driver Constructors */ + +import static com.sun.tools.classfile.TypeAnnotation.TargetType.*; + public class Constructors { @TADescriptions({ @@ -42,8 +44,8 @@ public class Constructors { } @TADescriptions({ - @TADescription(annotation = "TA", type = METHOD_RETURN), - @TADescription(annotation = "TB", type = METHOD_RETURN), + @TADescription(annotation = "TA", type = METHOD_RETURN, genericLocation = {1, 0}), + @TADescription(annotation = "TB", type = METHOD_RETURN, genericLocation = {1, 0}), @TADescription(annotation = "TC", type = METHOD_FORMAL_PARAMETER, paramIndex = 0) }) @TestClass("Test$Inner") @@ -56,9 +58,9 @@ public class Constructors { @TADescriptions({ @TADescription(annotation = "TA", type = METHOD_RECEIVER), - @TADescription(annotation = "TB", type = METHOD_RETURN), + @TADescription(annotation = "TB", type = METHOD_RETURN, genericLocation = {1, 0}), @TADescription(annotation = "TC", type = METHOD_RECEIVER), - @TADescription(annotation = "TD", type = METHOD_RETURN), + @TADescription(annotation = "TD", type = METHOD_RETURN, genericLocation = {1, 0}), @TADescription(annotation = "TE", type = METHOD_FORMAL_PARAMETER, paramIndex = 0) }) @TestClass("Test$Inner") @@ -72,9 +74,9 @@ public class Constructors { @TADescriptions({ @TADescription(annotation = "TA", type = METHOD_RECEIVER), @TADescription(annotation = "TB", type = METHOD_RECEIVER, genericLocation = {1, 0}), - @TADescription(annotation = "TC", type = METHOD_RETURN), + @TADescription(annotation = "TC", type = METHOD_RETURN, genericLocation = {1, 0, 1, 0}), @TADescription(annotation = "TD", type = METHOD_RECEIVER, genericLocation = {1, 0}), - @TADescription(annotation = "TE", type = METHOD_RETURN), + @TADescription(annotation = "TE", type = METHOD_RETURN, genericLocation = {1, 0, 1, 0}), @TADescription(annotation = "TF", type = METHOD_FORMAL_PARAMETER, paramIndex = 0) }) @TestClass("Outer$Middle$Inner") From 24394acaead810e36d93f6d0926fa91449349eff Mon Sep 17 00:00:00 2001 From: Kumar Srinivasan Date: Sun, 20 Oct 2013 12:54:17 -0700 Subject: [PATCH 07/28] 8026931: MethodParameters tests failing on Windows Reviewed-by: jjg, vromero --- langtools/test/tools/javac/MethodParameters/Tester.java | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/langtools/test/tools/javac/MethodParameters/Tester.java b/langtools/test/tools/javac/MethodParameters/Tester.java index 4b3bb04580f..1833985412c 100644 --- a/langtools/test/tools/javac/MethodParameters/Tester.java +++ b/langtools/test/tools/javac/MethodParameters/Tester.java @@ -27,7 +27,6 @@ import java.nio.charset.StandardCharsets; import java.nio.file.Files; import java.util.ArrayList; import java.util.Arrays; -import java.util.Collections; import java.util.List; /** @@ -152,14 +151,14 @@ public class Tester { throws FileNotFoundException, IOException { List refFileList = Files.readAllLines(refFile.toPath(), StandardCharsets.UTF_8); - List sbList = Arrays.asList(sb.split(System.getProperty("line.separator"))); + List sbList = Arrays.asList(sb.split("[\r\n]+")); // Check if test output contains unexpected lines or is missing expected lines. - List sbOnly = new ArrayList(sbList); + List sbOnly = new ArrayList<>(sbList); sbOnly.removeAll(refFileList); for (String line: sbOnly) error("unexpected line found: " + line); - List refOnly = new ArrayList(refFileList); + List refOnly = new ArrayList<>(refFileList); refOnly.removeAll(sbList); for (String line: refOnly) error("expected line not found: " + line); From 43754fccbd32f9125c125e31b83fb9d466a8f6ed Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Mon, 21 Oct 2013 15:55:02 +0100 Subject: [PATCH 08/28] 8026956: test tools/javac/lambda/TargetType58.java is failing after a libs change Reviewed-by: jfranck --- langtools/test/tools/javac/lambda/TargetType58.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/langtools/test/tools/javac/lambda/TargetType58.java b/langtools/test/tools/javac/lambda/TargetType58.java index cb3ca9027e4..d3829ac3f96 100644 --- a/langtools/test/tools/javac/lambda/TargetType58.java +++ b/langtools/test/tools/javac/lambda/TargetType58.java @@ -35,7 +35,7 @@ import java.util.stream.*; class TargetType58 { void test(List li) { - g(li, s -> s.substream(200), Collections.emptyList()); + g(li, s -> s.skip(200), Collections.emptyList()); } , From 7b509ee9873ea2e78fc308ddba14a84183b306ba Mon Sep 17 00:00:00 2001 From: Joe Darcy Date: Mon, 21 Oct 2013 15:37:11 -0700 Subject: [PATCH 09/28] 8026984: Clarity intended use of jdk.Exported Reviewed-by: psandoz, mr, alanb --- langtools/src/share/classes/jdk/Exported.java | 25 ++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/langtools/src/share/classes/jdk/Exported.java b/langtools/src/share/classes/jdk/Exported.java index dc9a9392db7..a768c8dae1f 100644 --- a/langtools/src/share/classes/jdk/Exported.java +++ b/langtools/src/share/classes/jdk/Exported.java @@ -38,7 +38,30 @@ import java.lang.annotation.*; * com.sun.*} are official parts of the JDK meant to be generally * usable while other portions of {@code com.sun.*} are not. This * annotation type allows those portions to be easily and - * programmaticly distinguished. + * programmatically distinguished. + * + *

    If in one release a type or package is + * @Exported(true), in a subsequent major release such a + * type or package can transition to @Exported(false). + * + *

    If a type or package is @Exported(false) in a + * release, it may be removed in a subsequent major release. + * + *

    If a top-level type has an @Exported annotation, + * any nested member types with the top-level type should have an + * @Exported annotation with the same value. + * + * (In exceptional cases, if a nested type is going to be removed + * before its enclosing type, the nested type's could be + * @Exported(false) while its enclosing type was + * @Exported(true).) + * + * Likewise, if a package has an @Exported annotation, + * top-level types within that package should also have an + * @Exported annotation. + * + * Sometimes a top-level type may have a different + * @Exported value than its package. * * @since 1.8 */ From 4a79e1cec8c0820d3a0be20fc9313039475c85ea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Joel=20Borggr=C3=A9n-Franck?= Date: Tue, 22 Oct 2013 03:36:44 +0200 Subject: [PATCH 10/28] 8026855: AnnoConstruct.getAnnotationsByType includes inherited indirectly present annotations even when containee type is not inheritable In AnnoConstruct.getAnnotationByType() check that the annotation sought after is inherited before looking on supertypes. Reviewed-by: jjg --- .../sun/tools/javac/code/AnnoConstruct.java | 4 +- .../model/element/TestNonInherited.java | 78 +++++++++++++++++++ 2 files changed, 81 insertions(+), 1 deletion(-) create mode 100644 langtools/test/tools/javac/processing/model/element/TestNonInherited.java diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java b/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java index 052336bbf2b..4d298f9b0c8 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java @@ -25,6 +25,7 @@ package com.sun.tools.javac.code; import java.lang.annotation.Annotation; +import java.lang.annotation.Inherited; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; @@ -112,7 +113,8 @@ public abstract class AnnoConstruct implements AnnotatedConstruct { } // Deal with inherited annotations - if (direct == null && container == null) + if (direct == null && container == null && + annoType.isAnnotationPresent(Inherited.class)) return getInheritedAnnotations(annoType); // Pack them in an array diff --git a/langtools/test/tools/javac/processing/model/element/TestNonInherited.java b/langtools/test/tools/javac/processing/model/element/TestNonInherited.java new file mode 100644 index 00000000000..cd5ec225ce7 --- /dev/null +++ b/langtools/test/tools/javac/processing/model/element/TestNonInherited.java @@ -0,0 +1,78 @@ +/* + * Copyright (c) 2013, 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 8026855 + * @summary Javac should only look on supertypes for repeatable annotations if + * both container and containee are inherited. + * @library /tools/javac/lib + * @build JavacTestingAbstractProcessor TestNonInherited + * @compile -processor TestNonInherited -proc:only TestNonInherited.java + */ + +import com.sun.tools.javac.util.Assert; + +import java.lang.annotation.*; +import java.util.Arrays; +import java.util.Set; +import javax.annotation.processing.*; +import javax.lang.model.element.*; +import javax.lang.model.util.*; + +import static javax.lang.model.util.ElementFilter.*; + +@TestNonInherited.Foo(1) +public class TestNonInherited extends JavacTestingAbstractProcessor { + public boolean process(Set annotations, + RoundEnvironment roundEnv) { + if (!roundEnv.processingOver()) { + boolean hasRun = false; + for (Element element : roundEnv.getRootElements()) + for (TypeElement te : typesIn(element.getEnclosedElements())) + if (te.getQualifiedName().contentEquals("TestNonInherited.T2")) { + hasRun = true; + Foo[] foos = te.getAnnotationsByType(Foo.class); + System.out.println(" " + te); + System.out.println(" " + Arrays.asList(foos)); + Assert.check(foos.length == 0, "Should not find any instance of @Foo"); + } + if (!hasRun) + throw new RuntimeException("The annotation processor could not find the declaration of T2, test broken!"); + } + return true; + } + + public static class T2 extends TestNonInherited { + } + + @Repeatable(FooContainer.class) + public static @interface Foo { + int value(); + } + + @Inherited + public static @interface FooContainer { + Foo[] value(); + } +} From 39fb59399ac8c31c20c4a2e4ede9527825e0613c Mon Sep 17 00:00:00 2001 From: Kumar Srinivasan Date: Mon, 21 Oct 2013 20:10:43 -0700 Subject: [PATCH 11/28] 8026758: Inefficient code in LambdaToMethod Reviewed-by: jjg, jlahoda, rfield --- .../sun/tools/javac/comp/LambdaToMethod.java | 168 ++++++++---------- 1 file changed, 72 insertions(+), 96 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index 73fb8913371..aa40ec13bdf 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -49,6 +49,7 @@ import com.sun.tools.javac.util.*; import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition; import com.sun.source.tree.MemberReferenceTree.ReferenceMode; +import java.util.EnumMap; import java.util.HashMap; import java.util.LinkedHashMap; import java.util.Map; @@ -69,6 +70,7 @@ import static com.sun.tools.javac.tree.JCTree.Tag.*; */ public class LambdaToMethod extends TreeTranslator { + private Attr attr; private JCDiagnostic.Factory diags; private Log log; private Lower lower; @@ -104,6 +106,35 @@ public class LambdaToMethod extends TreeTranslator { /** Flag for alternate metafactories indicating the lambda object requires multiple bridges */ public static final int FLAG_BRIDGES = 1 << 2; + // + protected static final Context.Key unlambdaKey = + new Context.Key(); + + public static LambdaToMethod instance(Context context) { + LambdaToMethod instance = context.get(unlambdaKey); + if (instance == null) { + instance = new LambdaToMethod(context); + } + return instance; + } + private LambdaToMethod(Context context) { + context.put(unlambdaKey, this); + diags = JCDiagnostic.Factory.instance(context); + log = Log.instance(context); + lower = Lower.instance(context); + names = Names.instance(context); + syms = Symtab.instance(context); + rs = Resolve.instance(context); + make = TreeMaker.instance(context); + types = Types.instance(context); + transTypes = TransTypes.instance(context); + analyzer = new LambdaAnalyzerPreprocessor(); + Options options = Options.instance(context); + dumpLambdaToMethodStats = options.isSet("dumpLambdaToMethodStats"); + attr = Attr.instance(context); + } + // + private class KlassInfo { /** @@ -141,37 +172,6 @@ public class LambdaToMethod extends TreeTranslator { } } - // - private static final Context.Key unlambdaKey = - new Context.Key(); - - public static LambdaToMethod instance(Context context) { - LambdaToMethod instance = context.get(unlambdaKey); - if (instance == null) { - instance = new LambdaToMethod(context); - } - return instance; - } - - private Attr attr; - - private LambdaToMethod(Context context) { - diags = JCDiagnostic.Factory.instance(context); - log = Log.instance(context); - lower = Lower.instance(context); - names = Names.instance(context); - syms = Symtab.instance(context); - rs = Resolve.instance(context); - make = TreeMaker.instance(context); - types = Types.instance(context); - transTypes = TransTypes.instance(context); - analyzer = new LambdaAnalyzerPreprocessor(); - Options options = Options.instance(context); - dumpLambdaToMethodStats = options.isSet("dumpLambdaToMethodStats"); - attr = Attr.instance(context); - } - // - // @Override public T translate(T tree) { @@ -402,21 +402,9 @@ public class LambdaToMethod extends TreeTranslator { super.visitIdent(tree); } else { LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; - if (lambdaContext.getSymbolMap(PARAM).containsKey(tree.sym)) { - Symbol translatedSym = lambdaContext.getSymbolMap(PARAM).get(tree.sym); - result = make.Ident(translatedSym).setType(tree.type); - translatedSym.setTypeAttributes(tree.sym.getRawTypeAttributes()); - } else if (lambdaContext.getSymbolMap(LOCAL_VAR).containsKey(tree.sym)) { - Symbol translatedSym = lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym); - result = make.Ident(translatedSym).setType(tree.type); - translatedSym.setTypeAttributes(tree.sym.getRawTypeAttributes()); - } else if (lambdaContext.getSymbolMap(TYPE_VAR).containsKey(tree.sym)) { - Symbol translatedSym = lambdaContext.getSymbolMap(TYPE_VAR).get(tree.sym); - result = make.Ident(translatedSym).setType(translatedSym.type); - translatedSym.setTypeAttributes(tree.sym.getRawTypeAttributes()); - } else if (lambdaContext.getSymbolMap(CAPTURED_VAR).containsKey(tree.sym)) { - Symbol translatedSym = lambdaContext.getSymbolMap(CAPTURED_VAR).get(tree.sym); - result = make.Ident(translatedSym).setType(tree.type); + JCTree ltree = lambdaContext.translate(tree); + if (ltree != null) { + result = ltree; } else { //access to untranslated symbols (i.e. compile-time constants, //members defined inside the lambda body, etc.) ) @@ -1704,20 +1692,7 @@ public class LambdaToMethod extends TreeTranslator { /** variable in the enclosing context to which this lambda is assigned */ Symbol self; - /** map from original to translated lambda parameters */ - Map lambdaParams = new LinkedHashMap(); - - /** map from original to translated lambda locals */ - Map lambdaLocals = new LinkedHashMap(); - - /** map from variables in enclosing scope to translated synthetic parameters */ - Map capturedLocals = new LinkedHashMap(); - - /** map from class symbols to translated synthetic parameters (for captured member access) */ - Map capturedThis = new LinkedHashMap(); - - /** map from original to translated lambda locals */ - Map typeVars = new LinkedHashMap(); + Map> translatedSymbols; /** the synthetic symbol for the method hoisting the translated lambda */ Symbol translatedSym; @@ -1735,6 +1710,13 @@ public class LambdaToMethod extends TreeTranslator { if (dumpLambdaToMethodStats) { log.note(tree, "lambda.stat", needsAltMetafactory(), translatedSym); } + translatedSymbols = new EnumMap<>(LambdaSymbolKind.class); + + translatedSymbols.put(PARAM, new LinkedHashMap()); + translatedSymbols.put(LOCAL_VAR, new LinkedHashMap()); + translatedSymbols.put(CAPTURED_VAR, new LinkedHashMap()); + translatedSymbols.put(CAPTURED_THIS, new LinkedHashMap()); + translatedSymbols.put(TYPE_VAR, new LinkedHashMap()); } /** @@ -1786,27 +1768,22 @@ public class LambdaToMethod extends TreeTranslator { } void addSymbol(Symbol sym, LambdaSymbolKind skind) { - Map transMap = null; + Map transMap = getSymbolMap(skind); Name preferredName; switch (skind) { case CAPTURED_THIS: - transMap = capturedThis; - preferredName = names.fromString("encl$" + capturedThis.size()); + preferredName = names.fromString("encl$" + transMap.size()); break; case CAPTURED_VAR: - transMap = capturedLocals; - preferredName = names.fromString("cap$" + capturedLocals.size()); + preferredName = names.fromString("cap$" + transMap.size()); break; case LOCAL_VAR: - transMap = lambdaLocals; preferredName = sym.name; break; case PARAM: - transMap = lambdaParams; preferredName = sym.name; break; case TYPE_VAR: - transMap = typeVars; preferredName = sym.name; break; default: throw new AssertionError(); @@ -1816,29 +1793,22 @@ public class LambdaToMethod extends TreeTranslator { } } - Map getSymbolMap(LambdaSymbolKind... skinds) { - LinkedHashMap translationMap = new LinkedHashMap(); - for (LambdaSymbolKind skind : skinds) { - switch (skind) { - case CAPTURED_THIS: - translationMap.putAll(capturedThis); - break; - case CAPTURED_VAR: - translationMap.putAll(capturedLocals); - break; - case LOCAL_VAR: - translationMap.putAll(lambdaLocals); - break; - case PARAM: - translationMap.putAll(lambdaParams); - break; - case TYPE_VAR: - translationMap.putAll(typeVars); - break; - default: throw new AssertionError(); + Map getSymbolMap(LambdaSymbolKind skind) { + Map m = translatedSymbols.get(skind); + Assert.checkNonNull(m); + return m; + } + + JCTree translate(JCIdent lambdaIdent) { + for (Map m : translatedSymbols.values()) { + if (m.containsKey(lambdaIdent.sym)) { + Symbol tSym = m.get(lambdaIdent.sym); + JCTree t = make.Ident(tSym).setType(lambdaIdent.type); + tSym.setTypeAttributes(lambdaIdent.sym.getRawTypeAttributes()); + return t; } } - return translationMap; + return null; } /** @@ -1869,10 +1839,12 @@ public class LambdaToMethod extends TreeTranslator { // // 1) reference to enclosing contexts captured by the lambda expression // 2) enclosing locals captured by the lambda expression - for (Symbol thisSym : getSymbolMap(CAPTURED_VAR, PARAM).values()) { + for (Symbol thisSym : getSymbolMap(CAPTURED_VAR).values()) { + params.append(make.VarDef((VarSymbol) thisSym, null)); + } + for (Symbol thisSym : getSymbolMap(PARAM).values()) { params.append(make.VarDef((VarSymbol) thisSym, null)); } - syntheticParams = params.toList(); //prepend synthetic args to translated lambda method signature @@ -1964,12 +1936,16 @@ public class LambdaToMethod extends TreeTranslator { } // + /* + * These keys provide mappings for various translated lambda symbols + * and the prevailing order must be maintained. + */ enum LambdaSymbolKind { - CAPTURED_VAR, - CAPTURED_THIS, - LOCAL_VAR, - PARAM, - TYPE_VAR; + PARAM, // original to translated lambda parameters + LOCAL_VAR, // original to translated lambda locals + CAPTURED_VAR, // variables in enclosing scope to translated synthetic parameters + CAPTURED_THIS, // class symbols to translated synthetic parameters (for captured member access) + TYPE_VAR; // original to translated lambda type variables } /** From d8d3ea2057e019d2bdb0690e1664c07cded7084d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Joel=20Borggr=C3=A9n-Franck?= Date: Tue, 22 Oct 2013 10:08:49 +0200 Subject: [PATCH 12/28] 8026857: AnnoConstruct.getAnnotationsByType does not search supertype for inherited annotations if @SomeContainer({}) is present An empty container should not stop javac from looking at supertypes for inherited repeating annotations Reviewed-by: jjg --- .../sun/tools/javac/code/AnnoConstruct.java | 31 ++++--- .../model/element/TestEmptyContainer.java | 82 +++++++++++++++++++ 2 files changed, 102 insertions(+), 11 deletions(-) create mode 100644 langtools/test/tools/javac/processing/model/element/TestEmptyContainer.java diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java b/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java index 4d298f9b0c8..d38905456da 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/AnnoConstruct.java @@ -117,17 +117,13 @@ public abstract class AnnoConstruct implements AnnotatedConstruct { annoType.isAnnotationPresent(Inherited.class)) return getInheritedAnnotations(annoType); - // Pack them in an array - Attribute[] contained0 = null; - if (container != null) - contained0 = unpackAttributes(container); - ListBuffer compounds = new ListBuffer<>(); - if (contained0 != null) { - for (Attribute a : contained0) - if (a instanceof Attribute.Compound) - compounds = compounds.append((Attribute.Compound)a); - } - Attribute.Compound[] contained = compounds.toArray(new Attribute.Compound[compounds.size()]); + Attribute.Compound[] contained = unpackContained(container); + + // In case of an empty legacy container we might need to look for + // inherited annos as well + if (direct == null && contained.length == 0 && + annoType.isAnnotationPresent(Inherited.class)) + return getInheritedAnnotations(annoType); int size = (direct == null ? 0 : 1) + contained.length; @SuppressWarnings("unchecked") // annoType is the Class for A @@ -159,6 +155,19 @@ public abstract class AnnoConstruct implements AnnotatedConstruct { return arr; } + private Attribute.Compound[] unpackContained(Attribute.Compound container) { + // Pack them in an array + Attribute[] contained0 = null; + if (container != null) + contained0 = unpackAttributes(container); + ListBuffer compounds = new ListBuffer<>(); + if (contained0 != null) { + for (Attribute a : contained0) + if (a instanceof Attribute.Compound) + compounds = compounds.append((Attribute.Compound)a); + } + return compounds.toArray(new Attribute.Compound[compounds.size()]); + } // This method is part of the javax.lang.model API, do not use this in javac code. public A getAnnotation(Class annoType) { diff --git a/langtools/test/tools/javac/processing/model/element/TestEmptyContainer.java b/langtools/test/tools/javac/processing/model/element/TestEmptyContainer.java new file mode 100644 index 00000000000..10a09a43284 --- /dev/null +++ b/langtools/test/tools/javac/processing/model/element/TestEmptyContainer.java @@ -0,0 +1,82 @@ +/* + * Copyright (c) 2013, 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 8026857 + * @summary Test that an empty container does not stop us from looking at + * supertypes for inherited repeated annotations. + * @library /tools/javac/lib + * @build JavacTestingAbstractProcessor TestEmptyContainer + * @compile -processor TestEmptyContainer -proc:only TestEmptyContainer.java + */ + +import com.sun.tools.javac.util.Assert; + +import java.lang.annotation.*; +import java.util.Arrays; +import java.util.Set; +import javax.annotation.processing.*; +import javax.lang.model.element.Element; +import javax.lang.model.element.TypeElement; + +import static javax.lang.model.util.ElementFilter.*; + +@TestEmptyContainer.Foo(1) +public class TestEmptyContainer extends JavacTestingAbstractProcessor { + public boolean process(Set annotations, + RoundEnvironment roundEnv) { + if (!roundEnv.processingOver()) { + boolean hasRun = false; + for (Element element : roundEnv.getRootElements()) + for (TypeElement te : typesIn(element.getEnclosedElements())) + if (te.getQualifiedName().contentEquals("TestEmptyContainer.T2")) { + hasRun = true; + Foo[] foos = te.getAnnotationsByType(Foo.class); + System.out.println(" " + te); + System.out.println(" " + Arrays.asList(foos)); + Assert.check(foos.length == 1, "Should find one @Foo"); + Assert.check(foos[0].value() == 1, "Should find @Foo(1)"); + } + if (!hasRun) + throw new RuntimeException("Annotation processor couldn't find class T2, test broken!"); + } + return true; + } + + // This empty container should not stop us from finding @Foo(1) on TestEmptyContainer above + @TestEmptyContainer.FooContainer({}) + public static class T2 extends TestEmptyContainer { + } + + @Repeatable(FooContainer.class) + @Inherited + public static @interface Foo { + int value(); + } + + @Inherited + public static @interface FooContainer { + Foo[] value(); + } +} From 48b6b38e527e97c78ea6b40b2d6c0adbf0e569fa Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Tue, 22 Oct 2013 13:54:49 +0100 Subject: [PATCH 13/28] 8025290: javac implicit versus explicit lambda compilation error Reviewed-by: jjg, dlsmith --- .../com/sun/tools/javac/comp/Attr.java | 17 +++++ .../com/sun/tools/javac/comp/Check.java | 2 +- .../com/sun/tools/javac/comp/Infer.java | 8 +- .../sun/tools/javac/util/JavacMessages.java | 30 ++++---- .../ExplicitVSImplicitLambdaTest.java | 73 +++++++++++++++++++ 5 files changed, 111 insertions(+), 19 deletions(-) create mode 100644 langtools/test/tools/javac/lambda/T8025290/ExplicitVSImplicitLambdaTest.java diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java index 16c5e5f6dfb..67d99066280 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java @@ -250,6 +250,14 @@ public class Attr extends JCTree.Visitor { Type owntype = found; if (!owntype.hasTag(ERROR) && !resultInfo.pt.hasTag(METHOD) && !resultInfo.pt.hasTag(FORALL)) { if (allowPoly && inferenceContext.free(found)) { + if ((ownkind & ~resultInfo.pkind) == 0) { + owntype = resultInfo.check(tree, inferenceContext.asFree(owntype)); + } else { + log.error(tree.pos(), "unexpected.type", + kindNames(resultInfo.pkind), + kindName(ownkind)); + owntype = types.createErrorType(owntype); + } inferenceContext.addFreeTypeListener(List.of(found, resultInfo.pt), new FreeTypeListener() { @Override public void typesInferred(InferenceContext inferenceContext) { @@ -511,6 +519,15 @@ public class Attr extends JCTree.Visitor { protected ResultInfo dup(CheckContext newContext) { return new ResultInfo(pkind, pt, newContext); } + + @Override + public String toString() { + if (pt != null) { + return pt.toString(); + } else { + return ""; + } + } } class RecoveryInfo extends ResultInfo { diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java index 073bbbc9c66..74c1540640b 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java @@ -528,7 +528,7 @@ public class Check { inferenceContext.addFreeTypeListener(List.of(req), new FreeTypeListener() { @Override public void typesInferred(InferenceContext inferenceContext) { - checkType(pos, found, inferenceContext.asInstType(req), checkContext); + checkType(pos, inferenceContext.asInstType(found), inferenceContext.asInstType(req), checkContext); } }); } diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Infer.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Infer.java index 61f75f068fa..2fe9474ab13 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Infer.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Infer.java @@ -1768,9 +1768,11 @@ public class Infer { public Type apply(Type t) { if (t.hasTag(TYPEVAR)) { TypeVar tv = (TypeVar)t; - return tv.isCaptured() ? - new CapturedUndetVar((CapturedType)tv, types) : - new UndetVar(tv, types); + if (tv.isCaptured()) { + return new CapturedUndetVar((CapturedType)tv, types); + } else { + return new UndetVar(tv, types); + } } else { return t.map(this); } diff --git a/langtools/src/share/classes/com/sun/tools/javac/util/JavacMessages.java b/langtools/src/share/classes/com/sun/tools/javac/util/JavacMessages.java index 77d2ac80530..75e132eeecf 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/util/JavacMessages.java +++ b/langtools/src/share/classes/com/sun/tools/javac/util/JavacMessages.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2005, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2005, 2013, 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 @@ -184,19 +184,19 @@ public class JavacMessages implements Messages { String key, Object... args) { String msg = null; - for (List l = bundles; l.nonEmpty() && msg == null; l = l.tail) { - ResourceBundle rb = l.head; - try { - msg = rb.getString(key); - } - catch (MissingResourceException e) { - // ignore, try other bundles in list - } - } - if (msg == null) { - msg = "compiler message file broken: key=" + key + - " arguments={0}, {1}, {2}, {3}, {4}, {5}, {6}, {7}"; - } - return MessageFormat.format(msg, args); + for (List l = bundles; l.nonEmpty() && msg == null; l = l.tail) { + ResourceBundle rb = l.head; + try { + msg = rb.getString(key); + } + catch (MissingResourceException e) { + // ignore, try other bundles in list + } + } + if (msg == null) { + msg = "compiler message file broken: key=" + key + + " arguments={0}, {1}, {2}, {3}, {4}, {5}, {6}, {7}"; + } + return MessageFormat.format(msg, args); } } diff --git a/langtools/test/tools/javac/lambda/T8025290/ExplicitVSImplicitLambdaTest.java b/langtools/test/tools/javac/lambda/T8025290/ExplicitVSImplicitLambdaTest.java new file mode 100644 index 00000000000..0bb8edbc589 --- /dev/null +++ b/langtools/test/tools/javac/lambda/T8025290/ExplicitVSImplicitLambdaTest.java @@ -0,0 +1,73 @@ +/* + * Copyright (c) 2013, 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 8025290 + * @summary javac implicit versus explicit lambda compilation error + * @compile ExplicitVSImplicitLambdaTest.java + */ + +import java.util.function.*; + +public class ExplicitVSImplicitLambdaTest { + private void test() + { + /* in the explicit case "e" is inferred to String so we can use a String + * only method. + */ + MyComparator.mycomparing1((String e) -> e.concat("")); + MyComparator.mycomparing2((String e) -> e.concat("")); + MyComparator.mycomparing3((String e) -> e.concat("")); + MyComparator.mycomparing4((String e) -> e.concat("")); + + /* in the implicit case "e" is inferred to Object so toString() is OK. + */ + MyComparator.mycomparing1((e) -> e.toString()); + MyComparator.mycomparing2((e) -> e.toString()); + MyComparator.mycomparing3((e) -> e.toString()); + MyComparator.mycomparing4((e) -> e.toString()); + } +} + +interface MyComparator { + public static > MyComparator mycomparing1( + Function keyExtractor) { + return null; + } + + public static > MyComparator mycomparing2( + Function keyExtractor) { + return null; + } + + public static > MyComparator mycomparing3( + Function keyExtractor) { + return null; + } + + public static > MyComparator mycomparing4( + Function keyExtractor) { + return null; + } +} From 69709943c351f2f6ab956c5ac531d4eee3a4538f Mon Sep 17 00:00:00 2001 From: Robert Field Date: Tue, 22 Oct 2013 16:53:21 -0700 Subject: [PATCH 14/28] 8023668: Desugar serializable lambda bodies using more robust naming scheme Lambda / bridged method-reference naming overhaul Reviewed-by: ksrini, briangoetz --- .../sun/tools/javac/comp/LambdaToMethod.java | 282 +++++++++++++----- .../javac/MethodParameters/LambdaTest.out | 6 +- .../javac/T8019486/WrongLVTForLambdaTest.java | 14 +- .../TestSerializedLambdaNameStability.java | 166 +++++++++++ .../after/TESTNameOfCapturedArgs.java | 48 +++ .../after/TESTOrderOfCapturedArgs.java | 49 +++ .../lambdaNaming/after/TESTTargetName.java | 47 +++ .../lambdaNaming/after/TESTTargetType.java | 40 +++ .../after/TESTTypesOfCapturedArgs.java | 48 +++ .../after/TESTVariableAssignmentTarget.java | 54 ++++ .../before/TESTNameOfCapturedArgs.java | 48 +++ .../before/TESTOrderOfCapturedArgs.java | 49 +++ .../lambdaNaming/before/TESTTargetName.java | 47 +++ .../lambdaNaming/before/TESTTargetType.java | 40 +++ .../before/TESTTypesOfCapturedArgs.java | 48 +++ .../before/TESTVariableAssignmentTarget.java | 54 ++++ 16 files changed, 962 insertions(+), 78 deletions(-) create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/TestSerializedLambdaNameStability.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/after/TESTNameOfCapturedArgs.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/after/TESTOrderOfCapturedArgs.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetName.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetType.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTypesOfCapturedArgs.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/after/TESTVariableAssignmentTarget.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/before/TESTNameOfCapturedArgs.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/before/TESTOrderOfCapturedArgs.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetName.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetType.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTypesOfCapturedArgs.java create mode 100644 langtools/test/tools/javac/lambda/lambdaNaming/before/TESTVariableAssignmentTarget.java diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index aa40ec13bdf..fe41cf5a034 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -30,7 +30,6 @@ import com.sun.tools.javac.tree.JCTree.JCMemberReference.ReferenceKind; import com.sun.tools.javac.tree.TreeMaker; import com.sun.tools.javac.tree.TreeTranslator; import com.sun.tools.javac.code.Attribute; -import com.sun.tools.javac.code.Flags; import com.sun.tools.javac.code.Kinds; import com.sun.tools.javac.code.Scope; import com.sun.tools.javac.code.Symbol; @@ -576,10 +575,10 @@ public class LambdaToMethod extends TreeTranslator { DiagnosticPosition pos, List staticArgs, MethodType indyType) { String functionalInterfaceClass = classSig(targetType); String functionalInterfaceMethodName = samSym.getSimpleName().toString(); - String functionalInterfaceMethodSignature = methodSig(types.erasure(samSym.type)); + String functionalInterfaceMethodSignature = typeSig(types.erasure(samSym.type)); String implClass = classSig(types.erasure(refSym.owner.type)); String implMethodName = refSym.getQualifiedName().toString(); - String implMethodSignature = methodSig(types.erasure(refSym.type)); + String implMethodSignature = typeSig(types.erasure(refSym.type)); JCExpression kindTest = eqTest(syms.intType, deserGetter("getImplMethodKind", syms.intType), make.Literal(implMethodKind)); ListBuffer serArgs = new ListBuffer<>(); @@ -1087,8 +1086,21 @@ public class LambdaToMethod extends TreeTranslator { * keep the count of lambda expression defined in given context (used to * generate unambiguous names for serializable lambdas) */ - private Map serializableLambdaCounts = - new HashMap(); + private class SyntheticMethodNameCounter { + private Map map = new HashMap<>(); + int getIndex(StringBuilder buf) { + String temp = buf.toString(); + Integer count = map.get(temp); + if (count == null) { + count = 0; + } + ++count; + map.put(temp, count); + return count; + } + } + private SyntheticMethodNameCounter syntheticMethodNameCounts = + new SyntheticMethodNameCounter(); private Map localClassDefs; @@ -1122,13 +1134,13 @@ public class LambdaToMethod extends TreeTranslator { @Override public void visitClassDef(JCClassDecl tree) { List prevStack = frameStack; - Map prevSerializableLambdaCount = - serializableLambdaCounts; + SyntheticMethodNameCounter prevSyntheticMethodNameCounts = + syntheticMethodNameCounts; Map prevClinits = clinits; DiagnosticSource prevSource = log.currentSource(); try { log.useSource(tree.sym.sourcefile); - serializableLambdaCounts = new HashMap(); + syntheticMethodNameCounts = new SyntheticMethodNameCounter(); prevClinits = new HashMap(); if (tree.sym.owner.kind == MTH) { localClassDefs.put(tree.sym, tree); @@ -1154,7 +1166,7 @@ public class LambdaToMethod extends TreeTranslator { finally { log.useSource(prevSource.getFile()); frameStack = prevStack; - serializableLambdaCounts = prevSerializableLambdaCount; + syntheticMethodNameCounts = prevSyntheticMethodNameCounts; clinits = prevClinits; } } @@ -1379,50 +1391,6 @@ public class LambdaToMethod extends TreeTranslator { } } - private Name lambdaName() { - return names.lambda.append(names.fromString("" + lambdaCount++)); - } - - /** - * For a serializable lambda, generate a name which maximizes name - * stability across deserialization. - * @param owner - * @return Name to use for the synthetic lambda method name - */ - private Name serializedLambdaName(Symbol owner) { - StringBuilder buf = new StringBuilder(); - buf.append(names.lambda); - // Append the name of the method enclosing the lambda. - String methodName = owner.name.toString(); - if (methodName.equals("")) - methodName = "static"; - else if (methodName.equals("")) - methodName = "new"; - buf.append(methodName); - buf.append('$'); - // Append a hash of the enclosing method signature to differentiate - // overloaded enclosing methods. For lambdas enclosed in lambdas, - // the generated lambda method will not have type yet, but the - // enclosing method's name will have been generated with this same - // method, so it will be unique and never be overloaded. - Assert.check(owner.type != null || directlyEnclosingLambda() != null); - if (owner.type != null) { - int methTypeHash = methodSig(owner.type).hashCode(); - buf.append(Integer.toHexString(methTypeHash)); - } - buf.append('$'); - // The above appended name components may not be unique, append a - // count based on the above name components. - String temp = buf.toString(); - Integer count = serializableLambdaCounts.get(temp); - if (count == null) { - count = 0; - } - buf.append(count++); - serializableLambdaCounts.put(temp, count); - return names.fromString(buf.toString()); - } - /** * Return a valid owner given the current declaration stack * (required to skip synthetic lambda symbols) @@ -1639,19 +1607,19 @@ public class LambdaToMethod extends TreeTranslator { private abstract class TranslationContext { /** the underlying (untranslated) tree */ - T tree; + final T tree; /** points to the adjusted enclosing scope in which this lambda/mref expression occurs */ - Symbol owner; + final Symbol owner; /** the depth of this lambda expression in the frame stack */ - int depth; + final int depth; /** the enclosing translation context (set for nested lambdas/mref) */ - TranslationContext prev; + final TranslationContext prev; /** list of methods to be bridged by the meta-factory */ - List bridges; + final List bridges; TranslationContext(T tree) { this.tree = tree; @@ -1679,6 +1647,31 @@ public class LambdaToMethod extends TreeTranslator { } return false; } + + /** + * @return Name of the enclosing method to be folded into synthetic + * method name + */ + String enclosingMethodName() { + return syntheticMethodNameComponent(owner.name); + } + + /** + * @return Method name in a form that can be folded into a + * component of a synthetic method name + */ + String syntheticMethodNameComponent(Name name) { + if (name == null) { + return "null"; + } + String methodName = name.toString(); + if (methodName.equals("")) { + methodName = "static"; + } else if (methodName.equals("")) { + methodName = "new"; + } + return methodName; + } } /** @@ -1690,7 +1683,10 @@ public class LambdaToMethod extends TreeTranslator { private class LambdaTranslationContext extends TranslationContext { /** variable in the enclosing context to which this lambda is assigned */ - Symbol self; + final Symbol self; + + /** variable in the enclosing context to which this lambda is assigned */ + final Symbol assignedTo; Map> translatedSymbols; @@ -1702,11 +1698,22 @@ public class LambdaToMethod extends TreeTranslator { LambdaTranslationContext(JCLambda tree) { super(tree); Frame frame = frameStack.head; - if (frame.tree.hasTag(VARDEF)) { - self = ((JCVariableDecl)frame.tree).sym; - } - Name name = isSerializable() ? serializedLambdaName(owner) : lambdaName(); - this.translatedSym = makePrivateSyntheticMethod(0, name, null, owner.enclClass()); + switch (frame.tree.getTag()) { + case VARDEF: + assignedTo = self = ((JCVariableDecl) frame.tree).sym; + break; + case ASSIGN: + self = null; + assignedTo = TreeInfo.symbol(((JCAssign) frame.tree).getVariable()); + break; + default: + assignedTo = self = null; + break; + } + + // This symbol will be filled-in in complete + this.translatedSym = makePrivateSyntheticMethod(0, null, null, owner.enclClass()); + if (dumpLambdaToMethodStats) { log.note(tree, "lambda.stat", needsAltMetafactory(), translatedSym); } @@ -1719,6 +1726,84 @@ public class LambdaToMethod extends TreeTranslator { translatedSymbols.put(TYPE_VAR, new LinkedHashMap()); } + /** + * For a serializable lambda, generate a disambiguating string + * which maximizes stability across deserialization. + * + * @return String to differentiate synthetic lambda method names + */ + private String serializedLambdaDisambiguation() { + StringBuilder buf = new StringBuilder(); + // Append the enclosing method signature to differentiate + // overloaded enclosing methods. For lambdas enclosed in + // lambdas, the generated lambda method will not have type yet, + // but the enclosing method's name will have been generated + // with this same method, so it will be unique and never be + // overloaded. + Assert.check( + owner.type != null || + directlyEnclosingLambda() != null); + if (owner.type != null) { + buf.append(typeSig(owner.type)); + buf.append(":"); + } + + // Add target type info + buf.append(types.findDescriptorSymbol(tree.type.tsym).owner.flatName()); + buf.append(" "); + + // Add variable assigned to + if (assignedTo != null) { + buf.append(assignedTo.flatName()); + buf.append("="); + } + //add captured locals info: type, name, order + for (Symbol fv : getSymbolMap(CAPTURED_VAR).keySet()) { + if (fv != self) { + buf.append(typeSig(fv.type)); + buf.append(" "); + buf.append(fv.flatName()); + buf.append(","); + } + } + + return buf.toString(); + } + + /** + * For a non-serializable lambda, generate a simple method. + * + * @return Name to use for the synthetic lambda method name + */ + private Name lambdaName() { + return names.lambda.append(names.fromString(enclosingMethodName() + "$" + lambdaCount++)); + } + + /** + * For a serializable lambda, generate a method name which maximizes + * name stability across deserialization. + * + * @return Name to use for the synthetic lambda method name + */ + private Name serializedLambdaName() { + StringBuilder buf = new StringBuilder(); + buf.append(names.lambda); + // Append the name of the method enclosing the lambda. + buf.append(enclosingMethodName()); + buf.append('$'); + // Append a hash of the disambiguating string : enclosing method + // signature, etc. + String disam = serializedLambdaDisambiguation(); + buf.append(Integer.toHexString(disam.hashCode())); + buf.append('$'); + // The above appended name components may not be unique, append + // a count based on the above name components. + buf.append(syntheticMethodNameCounts.getIndex(buf)); + String result = buf.toString(); + //System.err.printf("serializedLambdaName: %s -- %s\n", result, disam); + return names.fromString(result); + } + /** * Translate a symbol of a given kind into something suitable for the * synthetic lambda body @@ -1847,6 +1932,11 @@ public class LambdaToMethod extends TreeTranslator { } syntheticParams = params.toList(); + // Compute and set the lambda name + translatedSym.name = isSerializable() + ? serializedLambdaName() + : lambdaName(); + //prepend synthetic args to translated lambda method signature translatedSym.type = types.createMethodTypeWithParameters( generatedLambdaSig(), @@ -1874,7 +1964,7 @@ public class LambdaToMethod extends TreeTranslator { this.isSuper = tree.hasKind(ReferenceKind.SUPER); this.bridgeSym = needsBridge() ? makePrivateSyntheticMethod(isSuper ? 0 : STATIC, - lambdaName().append(names.fromString("$bridge")), null, + referenceBridgeName(), null, owner.enclClass()) : null; if (dumpLambdaToMethodStats) { @@ -1888,13 +1978,71 @@ public class LambdaToMethod extends TreeTranslator { * Get the opcode associated with this method reference */ int referenceKind() { - return LambdaToMethod.this.referenceKind(needsBridge() ? bridgeSym : tree.sym); + return LambdaToMethod.this.referenceKind(needsBridge() + ? bridgeSym + : tree.sym); } boolean needsVarArgsConversion() { return tree.varargsElement != null; } + /** + * Generate a disambiguating string to increase stability (important + * if serialized) + * + * @return String to differentiate synthetic lambda method names + */ + private String referenceBridgeDisambiguation() { + StringBuilder buf = new StringBuilder(); + // Append the enclosing method signature to differentiate + // overloaded enclosing methods. + if (owner.type != null) { + buf.append(typeSig(owner.type)); + buf.append(":"); + } + + // Append qualifier type + buf.append(classSig(tree.sym.owner.type)); + + // Note static/instance + buf.append(tree.sym.isStatic()? " S " : " I "); + + // Append referenced signature + buf.append(typeSig(tree.sym.erasure(types))); + + return buf.toString(); + } + + /** + * Construct a unique stable name for the method reference bridge + * + * @return Name to use for the synthetic method name + */ + private Name referenceBridgeName() { + StringBuilder buf = new StringBuilder(); + // Append lambda ID, this is semantically significant + buf.append(names.lambda); + // Note that it is a method reference bridge + buf.append("MR$"); + // Append the enclosing method name + buf.append(enclosingMethodName()); + buf.append('$'); + // Append the referenced method name + buf.append(syntheticMethodNameComponent(tree.sym.name)); + buf.append('$'); + // Append a hash of the disambiguating string : enclosing method + // signature, etc. + String disam = referenceBridgeDisambiguation(); + buf.append(Integer.toHexString(disam.hashCode())); + buf.append('$'); + // The above appended name components may not be unique, append + // a count based on the above name components. + buf.append(syntheticMethodNameCounts.getIndex(buf)); + String result = buf.toString(); + return names.fromString(result); + } + /** * @return Is this an array operation like clone() */ @@ -1954,7 +2102,7 @@ public class LambdaToMethod extends TreeTranslator { * **************************************************************** */ - private String methodSig(Type type) { + private String typeSig(Type type) { L2MSignatureGenerator sg = new L2MSignatureGenerator(); sg.assembleSig(type); return sg.toString(); diff --git a/langtools/test/tools/javac/MethodParameters/LambdaTest.out b/langtools/test/tools/javac/MethodParameters/LambdaTest.out index 7cfdfe9058a..da662e11958 100644 --- a/langtools/test/tools/javac/MethodParameters/LambdaTest.out +++ b/langtools/test/tools/javac/MethodParameters/LambdaTest.out @@ -1,7 +1,7 @@ class LambdaTest -- LambdaTest.() LambdaTest.foo(i) -LambdaTest.lambda$1(arg0, arg1)/*synthetic*/ -LambdaTest.lambda$0(arg0)/*synthetic*/ +LambdaTest.lambda$static$1(arg0)/*synthetic*/ +LambdaTest.lambda$null$0(arg0, arg1)/*synthetic*/ static interface LambdaTest$I -- inner -LambdaTest$I.m(x) \ No newline at end of file +LambdaTest$I.m(x) diff --git a/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java b/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java index 9d7d32abb3b..22eeae60fc6 100644 --- a/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java +++ b/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java @@ -61,8 +61,6 @@ public class WrongLVTForLambdaTest { {9, 0}, //number -> number / 1 }; - static final String methodToLookFor = "lambda$0"; - public static void main(String[] args) throws Exception { new WrongLVTForLambdaTest().run(); } @@ -70,7 +68,7 @@ public class WrongLVTForLambdaTest { void run() throws Exception { compileTestClass(); checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri()), methodToLookFor); + "Foo.class").toUri())); } void compileTestClass() throws Exception { @@ -79,12 +77,12 @@ public class WrongLVTForLambdaTest { ToolBox.javac(javacSuccessArgs); } - void checkClassFile(final File cfile, String methodToFind) throws Exception { + void checkClassFile(final File cfile) throws Exception { ClassFile classFile = ClassFile.read(cfile); - boolean methodFound = false; + int methodsFound = 0; for (Method method : classFile.methods) { - if (method.getName(classFile.constant_pool).equals(methodToFind)) { - methodFound = true; + if (method.getName(classFile.constant_pool).startsWith("lambda$")) { + ++methodsFound; Code_attribute code = (Code_attribute) method.attributes.get("Code"); LineNumberTable_attribute lnt = (LineNumberTable_attribute) code.attributes.get("LineNumberTable"); @@ -101,7 +99,7 @@ public class WrongLVTForLambdaTest { } } } - Assert.check(methodFound, "The seek method was not found"); + Assert.check(methodsFound == 1, "Expected to find one lambda method, found " + methodsFound); } void error(String msg) { diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/TestSerializedLambdaNameStability.java b/langtools/test/tools/javac/lambda/lambdaNaming/TestSerializedLambdaNameStability.java new file mode 100644 index 00000000000..55302b028d2 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/TestSerializedLambdaNameStability.java @@ -0,0 +1,166 @@ +/* + * Copyright (c) 2013, 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 8023668 + * @summary Desugar serializable lambda bodies using more robust naming scheme + * @library /tools/javac/lib + * @build ToolBox + * @run main TestSerializedLambdaNameStability + */ + +import java.io.*; +import java.lang.reflect.Constructor; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; +import java.nio.file.*; + +public class TestSerializedLambdaNameStability { + + final ClassLoader writingClassLoader; + final ClassLoader clonedClassLoader; + final ClassLoader checkingClassLoader; + + TestSerializedLambdaNameStability() { + writingClassLoader = new TestClassLoader("before"); + clonedClassLoader = new TestClassLoader("before"); + checkingClassLoader = new TestClassLoader("after"); + } + + public static void main(String... args) throws Exception { + new TestSerializedLambdaNameStability().doit("NameOfCapturedArgs", true); + new TestSerializedLambdaNameStability().doit("TypesOfCapturedArgs", true); + new TestSerializedLambdaNameStability().doit("OrderOfCapturedArgs", true); + new TestSerializedLambdaNameStability().doit("VariableAssignmentTarget", false); + new TestSerializedLambdaNameStability().doit("TargetName", true); + new TestSerializedLambdaNameStability().doit("TargetType", true); + } + + public void doit(String name, boolean expectFail) throws Exception { + String iName = "I" + name; + String testName = "TEST" + name; + Class kw = writingClassLoader.loadClass(testName); + Object instw = getInstance(kw); + Method mw = getMethod(kw, "write", ObjectOutput.class); + ByteArrayOutputStream baos = new ByteArrayOutputStream(); + try (ObjectOutput out = new ObjectOutputStream(baos)) { + mw.invoke(instw, out); + } + byte[] ser = baos.toByteArray(); + + // Read and check clone + readCheck(iName, testName, clonedClassLoader, ser); + System.err.printf("cloned test readCheck %s\n", testName); + + // Read and check other + if (expectFail) { + try { + readCheck(iName, testName, checkingClassLoader, ser); + } catch (InvocationTargetException ite) { + Throwable underlying = ite; + while (underlying != null && !(underlying instanceof IllegalArgumentException)) { + underlying = underlying.getCause(); + } + if (underlying != null) { + if (underlying.getMessage().contains("deserialization")) { + System.err.printf("PASS: other test %s got expected exception %s\n", testName, underlying); + return; + } + } + System.err.printf("FAIL: other test %s got unexpected exception %s\n", testName, ite); + throw new Exception("unexpected exception ", ite); + } + System.err.printf("FAIL: other test %s expected an exception", testName); + throw new Exception("expected an exception" + testName); + } else { + readCheck(iName, testName, checkingClassLoader, ser); + System.err.printf("PASS: other test %s readCheck\n", testName); + } + } + + void readCheck(String iName, String testName, ClassLoader loader, byte[] ser) throws Exception { + Class k = loader.loadClass(testName); + Object inst = getInstance(k); + Method mrc = getMethod(k, "readCheck", ObjectInput.class); + ByteArrayInputStream bais = new ByteArrayInputStream(ser); + try (ObjectInput in = new ObjectInputStream(bais)) { + mrc.invoke(inst, in); + } + } + + Method getMethod(Class k, String name, Class argTypes) throws Exception { + Method meth = k.getDeclaredMethod(name, argTypes); + meth.setAccessible(true); + return meth; + } + + Object getInstance(Class k) throws Exception { + Constructor cons = k.getConstructors()[0]; + cons.setAccessible(true); + return cons.newInstance(); + } + + static class TestClassLoader extends ClassLoader { + static final String compiledDir = System.getProperty("user.dir"); + static final String sourceBaseDir = System.getProperty("test.src"); + + final String context; + + public TestClassLoader(String context) { + super(); + this.context = context; + } + + @Override + public Class findClass(String name) throws ClassNotFoundException { + byte[] b; + + try { + b = loadClassData(name); + } catch (Throwable th) { + // th.printStackTrace(); + throw new ClassNotFoundException("Loading error", th); + } + return defineClass(name, b, 0, b.length); + } + + private byte[] loadClassData(String name) throws Exception { + String srcName; + if (name.startsWith("TEST")) + srcName = name; + else if (name.startsWith("I")) + srcName = "TEST" + name.substring(1); + else + throw new Exception("Did not expect to load " + name); + Path srcFile = Paths.get(sourceBaseDir, context, srcName + ".java"); + String testSource = new String(Files.readAllBytes(srcFile)); + ToolBox.JavaToolArgs javacSuccessArgs = + new ToolBox.JavaToolArgs().setSources(testSource); + ToolBox.javac(javacSuccessArgs); + Path cfFile = Paths.get(compiledDir, name + ".class"); + byte[] bytes = Files.readAllBytes(cfFile); + return bytes; + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTNameOfCapturedArgs.java b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTNameOfCapturedArgs.java new file mode 100644 index 00000000000..b66e6c24ccc --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTNameOfCapturedArgs.java @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface INameOfCapturedArgs extends Serializable { + int get(); +} + +class TESTNameOfCapturedArgs { + + public TESTNameOfCapturedArgs() { + } + + public void write(ObjectOutput out) throws IOException { + int y = 44; + INameOfCapturedArgs res = () -> y; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + INameOfCapturedArgs lam = (INameOfCapturedArgs) in.readObject(); + int val = lam.get(); + if (val != 44) { + throw new IllegalArgumentException("Expected 44"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTOrderOfCapturedArgs.java b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTOrderOfCapturedArgs.java new file mode 100644 index 00000000000..68c915211fc --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTOrderOfCapturedArgs.java @@ -0,0 +1,49 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface IOrderOfCapturedArgs extends Serializable { + String get(); +} + +class TESTOrderOfCapturedArgs { + + public TESTOrderOfCapturedArgs() { + } + + public void write(ObjectOutput out) throws IOException { + String a = "fu"; + String b = "bar"; + IOrderOfCapturedArgs res = () -> b + a; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + IOrderOfCapturedArgs lam = (IOrderOfCapturedArgs) in.readObject(); + Object val = lam.get(); + if (!val.equals("fubar")) { + throw new IllegalArgumentException("Expected 'fubar'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetName.java b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetName.java new file mode 100644 index 00000000000..183a03f6373 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetName.java @@ -0,0 +1,47 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface ITargetName extends Serializable { + String get(); +} + +class TESTTargetName { + + public TESTTargetName() { + } + + public void write(ObjectOutput out) throws IOException { + ITargetName resist = () -> "fubar"; + out.writeObject(resist); + } + + public void readCheck(ObjectInput in) throws Exception { + ITargetName lam = (ITargetName) in.readObject(); + Object val = lam.get(); + if (!val.equals("fubar")) { + throw new IllegalArgumentException("Expected 'fubar'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetType.java b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetType.java new file mode 100644 index 00000000000..c7b3d9f699a --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTargetType.java @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; +import java.util.function.*; + +class TESTTargetType { + + public TESTTargetType() { + } + + public void write(ObjectOutput out) throws IOException { + Object res = (Predicate & Serializable) ((str) -> str.length() > 3); + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + in.readObject(); + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTypesOfCapturedArgs.java b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTypesOfCapturedArgs.java new file mode 100644 index 00000000000..7805575d7cc --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTTypesOfCapturedArgs.java @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface ITypesOfCapturedArgs extends Serializable { + Object get(); +} + +class TESTTypesOfCapturedArgs { + + public TESTTypesOfCapturedArgs() { + } + + public void write(ObjectOutput out) throws IOException { + Object x = "hi"; + ITypesOfCapturedArgs res = () -> x; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + ITypesOfCapturedArgs lam = (ITypesOfCapturedArgs) in.readObject(); + Object val = lam.get(); + if (!val.equals("hi")) { + throw new IllegalArgumentException("Expected 'hi'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTVariableAssignmentTarget.java b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTVariableAssignmentTarget.java new file mode 100644 index 00000000000..18c0dcc2a26 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/after/TESTVariableAssignmentTarget.java @@ -0,0 +1,54 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface IVariableAssignmentTarget extends Serializable { + String get(); +} + +class TESTVariableAssignmentTarget { + + public TESTVariableAssignmentTarget() { + } + + public void write(ObjectOutput out) throws IOException { + IVariableAssignmentTarget res2 = () -> "bar"; + IVariableAssignmentTarget res1 = () -> "fu"; + out.writeObject(res1); + out.writeObject(res2); + } + + public void readCheck(ObjectInput in) throws Exception { + IVariableAssignmentTarget lam = (IVariableAssignmentTarget) in.readObject(); + Object val = lam.get(); + if (!val.equals("fu")) { + throw new IllegalArgumentException("Expected 'fu'"); + } + lam = (IVariableAssignmentTarget) in.readObject(); + val = lam.get(); + if (!val.equals("bar")) { + throw new IllegalArgumentException("Expected 'bar'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTNameOfCapturedArgs.java b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTNameOfCapturedArgs.java new file mode 100644 index 00000000000..67b9a678222 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTNameOfCapturedArgs.java @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface INameOfCapturedArgs extends Serializable { + int get(); +} + +class TESTNameOfCapturedArgs { + + public TESTNameOfCapturedArgs() { + } + + public void write(ObjectOutput out) throws IOException { + int x = 44; + INameOfCapturedArgs res = () -> x; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + INameOfCapturedArgs lam = (INameOfCapturedArgs) in.readObject(); + int val = lam.get(); + if (val != 44) { + throw new IllegalArgumentException("Expected 44"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTOrderOfCapturedArgs.java b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTOrderOfCapturedArgs.java new file mode 100644 index 00000000000..1c8461f4f49 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTOrderOfCapturedArgs.java @@ -0,0 +1,49 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface IOrderOfCapturedArgs extends Serializable { + String get(); +} + +class TESTOrderOfCapturedArgs { + + public TESTOrderOfCapturedArgs() { + } + + public void write(ObjectOutput out) throws IOException { + String a = "fu"; + String b = "bar"; + IOrderOfCapturedArgs res = () -> a + b; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + IOrderOfCapturedArgs lam = (IOrderOfCapturedArgs) in.readObject(); + Object val = lam.get(); + if (!val.equals("fubar")) { + throw new IllegalArgumentException("Expected 'fubar'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetName.java b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetName.java new file mode 100644 index 00000000000..7c6d0fd5879 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetName.java @@ -0,0 +1,47 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface ITargetName extends Serializable { + String get(); +} + +class TESTTargetName { + + public TESTTargetName() { + } + + public void write(ObjectOutput out) throws IOException { + ITargetName res = () -> "fubar"; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + ITargetName lam = (ITargetName) in.readObject(); + Object val = lam.get(); + if (!val.equals("fubar")) { + throw new IllegalArgumentException("Expected 'fubar'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetType.java b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetType.java new file mode 100644 index 00000000000..c459a8760f0 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTargetType.java @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; +import java.util.function.*; + +class TESTTargetType { + + public TESTTargetType() { + } + + public void write(ObjectOutput out) throws IOException { + Object res = (Function & Serializable) ((str) -> str.length() > 3); + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + in.readObject(); + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTypesOfCapturedArgs.java b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTypesOfCapturedArgs.java new file mode 100644 index 00000000000..4116f2e7d0a --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTTypesOfCapturedArgs.java @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface ITypesOfCapturedArgs extends Serializable { + Object get(); +} + +class TESTTypesOfCapturedArgs { + + public TESTTypesOfCapturedArgs() { + } + + public void write(ObjectOutput out) throws IOException { + String x = "hi"; + ITypesOfCapturedArgs res = () -> x; + out.writeObject(res); + } + + public void readCheck(ObjectInput in) throws Exception { + ITypesOfCapturedArgs lam = (ITypesOfCapturedArgs) in.readObject(); + Object val = lam.get(); + if (!val.equals("hi")) { + throw new IllegalArgumentException("Expected 'hi'"); + } + } +} diff --git a/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTVariableAssignmentTarget.java b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTVariableAssignmentTarget.java new file mode 100644 index 00000000000..76b8f2c0135 --- /dev/null +++ b/langtools/test/tools/javac/lambda/lambdaNaming/before/TESTVariableAssignmentTarget.java @@ -0,0 +1,54 @@ +/* + * Copyright (c) 2013, 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. + */ + +import java.io.*; + +interface IVariableAssignmentTarget extends Serializable { + String get(); +} + +class TESTVariableAssignmentTarget { + + public TESTVariableAssignmentTarget() { + } + + public void write(ObjectOutput out) throws IOException { + IVariableAssignmentTarget res1 = () -> "fu"; + IVariableAssignmentTarget res2 = () -> "bar"; + out.writeObject(res1); + out.writeObject(res2); + } + + public void readCheck(ObjectInput in) throws Exception { + IVariableAssignmentTarget lam = (IVariableAssignmentTarget) in.readObject(); + Object val = lam.get(); + if (!val.equals("fu")) { + throw new IllegalArgumentException("Expected 'fu'"); + } + lam = (IVariableAssignmentTarget) in.readObject(); + val = lam.get(); + if (!val.equals("bar")) { + throw new IllegalArgumentException("Expected 'bar'"); + } + } +} From 53d2f8ae319e29b9fc5a758649c5fb3c0bdd0003 Mon Sep 17 00:00:00 2001 From: Jonathan Gibbons Date: Tue, 22 Oct 2013 17:42:10 -0700 Subject: [PATCH 15/28] 8027119: Cleanup javadoc comments for taglet API Reviewed-by: mduigou --- langtools/src/share/classes/com/sun/javadoc/Tag.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/langtools/src/share/classes/com/sun/javadoc/Tag.java b/langtools/src/share/classes/com/sun/javadoc/Tag.java index 1d8c5907be2..32088845ef8 100644 --- a/langtools/src/share/classes/com/sun/javadoc/Tag.java +++ b/langtools/src/share/classes/com/sun/javadoc/Tag.java @@ -72,7 +72,7 @@ public interface Tag { * kind() == name(); * the following table lists those cases where there is more * than one tag of a given kind: - *

    + * * * * From 5ea357c66efc2a95ee467d2e96406d32ad85855d Mon Sep 17 00:00:00 2001 From: Jan Lahoda Date: Wed, 23 Oct 2013 07:50:04 +0200 Subject: [PATCH 16/28] 8026508: Invokedynamic instructions don't get line number table entries Setting or correcting positions for many trees produced by LambdaToMethod. Reviewed-by: vromero, rfield --- .../sun/tools/javac/comp/LambdaToMethod.java | 103 ++++++++++++------ ...daTest.java => WrongLNTForLambdaTest.java} | 75 +++++++++++-- 2 files changed, 133 insertions(+), 45 deletions(-) rename langtools/test/tools/javac/T8019486/{WrongLVTForLambdaTest.java => WrongLNTForLambdaTest.java} (55%) diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index fe41cf5a034..cef43efdf92 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -156,12 +156,15 @@ public class LambdaToMethod extends TreeTranslator { */ private final VarSymbol deserParamSym; - private KlassInfo(Symbol kSym) { + private final JCClassDecl clazz; + + private KlassInfo(JCClassDecl clazz) { + this.clazz = clazz; appendedMethodList = new ListBuffer<>(); deserializeCases = new HashMap>(); MethodType type = new MethodType(List.of(syms.serializedLambdaType), syms.objectType, List.nil(), syms.methodClass); - deserMethodSym = makePrivateSyntheticMethod(STATIC, names.deserializeLambda, type, kSym); + deserMethodSym = makePrivateSyntheticMethod(STATIC, names.deserializeLambda, type, clazz.sym); deserParamSym = new VarSymbol(FINAL, names.fromString("lambda"), syms.serializedLambdaType, deserMethodSym); } @@ -221,10 +224,16 @@ public class LambdaToMethod extends TreeTranslator { } KlassInfo prevKlassInfo = kInfo; try { - kInfo = new KlassInfo(tree.sym); + kInfo = new KlassInfo(tree); super.visitClassDef(tree); if (!kInfo.deserializeCases.isEmpty()) { - kInfo.addMethod(makeDeserializeMethod(tree.sym)); + int prevPos = make.pos; + try { + make.at(tree); + kInfo.addMethod(makeDeserializeMethod(tree.sym)); + } finally { + make.at(prevPos); + } } //add all translated instance methods here List newMethods = kInfo.appendedMethodList.toList(); @@ -400,14 +409,21 @@ public class LambdaToMethod extends TreeTranslator { if (context == null || !analyzer.lambdaIdentSymbolFilter(tree.sym)) { super.visitIdent(tree); } else { - LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; - JCTree ltree = lambdaContext.translate(tree); - if (ltree != null) { - result = ltree; - } else { - //access to untranslated symbols (i.e. compile-time constants, - //members defined inside the lambda body, etc.) ) - super.visitIdent(tree); + int prevPos = make.pos; + try { + make.at(tree); + + LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; + JCTree ltree = lambdaContext.translate(tree); + if (ltree != null) { + result = ltree; + } else { + //access to untranslated symbols (i.e. compile-time constants, + //members defined inside the lambda body, etc.) ) + super.visitIdent(tree); + } + } finally { + make.at(prevPos); } } } @@ -417,11 +433,21 @@ public class LambdaToMethod extends TreeTranslator { LambdaTranslationContext lambdaContext = (LambdaTranslationContext)context; if (context != null && lambdaContext.getSymbolMap(LOCAL_VAR).containsKey(tree.sym)) { JCExpression init = translate(tree.init); - result = make.VarDef((VarSymbol)lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym), init); + int prevPos = make.pos; + try { + result = make.at(tree).VarDef((VarSymbol)lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym), init); + } finally { + make.at(prevPos); + } } else if (context != null && lambdaContext.getSymbolMap(TYPE_VAR).containsKey(tree.sym)) { JCExpression init = translate(tree.init); VarSymbol xsym = (VarSymbol)lambdaContext.getSymbolMap(TYPE_VAR).get(tree.sym); - result = make.VarDef(xsym, init); + int prevPos = make.pos; + try { + result = make.at(tree).VarDef(xsym, init); + } finally { + make.at(prevPos); + } // Replace the entered symbol for this variable Scope sc = tree.sym.owner.members(); if (sc != null) { @@ -448,23 +474,28 @@ public class LambdaToMethod extends TreeTranslator { boolean isLambda_void = expr.type.hasTag(VOID); boolean isTarget_void = restype.hasTag(VOID); boolean isTarget_Void = types.isSameType(restype, types.boxedClass(syms.voidType).type); - if (isTarget_void) { - //target is void: - // BODY; - JCStatement stat = make.Exec(expr); - return make.Block(0, List.of(stat)); - } else if (isLambda_void && isTarget_Void) { - //void to Void conversion: - // BODY; return null; - ListBuffer stats = new ListBuffer<>(); - stats.append(make.Exec(expr)); - stats.append(make.Return(make.Literal(BOT, null).setType(syms.botType))); - return make.Block(0, stats.toList()); - } else { - //non-void to non-void conversion: - // return (TYPE)BODY; - JCExpression retExpr = transTypes.coerce(attrEnv, expr, restype); - return make.at(retExpr).Block(0, List.of(make.Return(retExpr))); + int prevPos = make.pos; + try { + if (isTarget_void) { + //target is void: + // BODY; + JCStatement stat = make.at(expr).Exec(expr); + return make.Block(0, List.of(stat)); + } else if (isLambda_void && isTarget_Void) { + //void to Void conversion: + // BODY; return null; + ListBuffer stats = new ListBuffer<>(); + stats.append(make.at(expr).Exec(expr)); + stats.append(make.Return(make.Literal(BOT, null).setType(syms.botType))); + return make.Block(0, stats.toList()); + } else { + //non-void to non-void conversion: + // return (TYPE)BODY; + JCExpression retExpr = transTypes.coerce(attrEnv, expr, restype); + return make.at(retExpr).Block(0, List.of(make.Return(retExpr))); + } + } finally { + make.at(prevPos); } } @@ -966,8 +997,14 @@ public class LambdaToMethod extends TreeTranslator { } } if (context.isSerializable()) { - addDeserializationCase(refKind, refSym, tree.type, samSym, - tree, staticArgs, indyType); + int prevPos = make.pos; + try { + make.at(kInfo.clazz); + addDeserializationCase(refKind, refSym, tree.type, samSym, + tree, staticArgs, indyType); + } finally { + make.at(prevPos); + } } } diff --git a/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java b/langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java similarity index 55% rename from langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java rename to langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java index 22eeae60fc6..525e20e074b 100644 --- a/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java +++ b/langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java @@ -25,11 +25,11 @@ /* * @test - * @bug 8019486 + * @bug 8019486 8026861 * @summary javac, generates erroneous LVT for a test case with lambda code * @library /tools/javac/lib * @build ToolBox - * @run main WrongLVTForLambdaTest + * @run main WrongLNTForLambdaTest */ import java.io.File; @@ -41,7 +41,7 @@ import com.sun.tools.classfile.LineNumberTable_attribute; import com.sun.tools.classfile.Method; import com.sun.tools.javac.util.Assert; -public class WrongLVTForLambdaTest { +public class WrongLNTForLambdaTest { static final String testSource = /* 01 */ "import java.util.List;\n" + @@ -54,21 +54,72 @@ public class WrongLVTForLambdaTest { /* 08 */ " final List numbersPlusOne = \n" + /* 09 */ " numbers.stream().map(number -> number / 1).collect(Collectors.toList());\n" + /* 10 */ " }\n" + - /* 11 */ "}"; + /* 11 */ " void variablesInLambdas(int value) {\n" + + /* 12 */ " Runnable r1 = () -> {\n" + + /* 13 */ " int i = value;\n" + + /* 14 */ " class FooBar {\n" + + /* 15 */ " public void run() {\n" + + /* 16 */ " T t = null;\n" + + /* 17 */ " }\n" + + /* 18 */ " }\n" + + /* 19 */ " };\n" + + /* 20 */ " Runnable r2 = () -> System.err.println(1);\n" + + /* 21 */ " Runnable r3 = (Runnable & java.io.Serializable) this::foo;\n" + + /* 22 */ " Runnable r4 = super :: notify;\n" + + /* 23 */ " }\n" + + /* 24 */ " private void foo() {}\n" + + /* 25 */ "}"; - static final int[][] expectedLNT = { + static final int[][] simpleLambdaExpectedLNT = { // {line-number, start-pc}, {9, 0}, //number -> number / 1 }; + static final int[][] lambdaWithVarsExpectedLNT = { + // {line-number, start-pc}, + {13, 0}, //number -> number / 1 + {19, 2}, //number -> number / 1 + }; + + static final int[][] insideLambdaWithVarsExpectedLNT = { + // {line-number, start-pc}, + {16, 0}, //number -> number / 1 + {17, 2}, //number -> number / 1 + }; + + static final int[][] lambdaVoid2VoidExpectedLNT = { + // {line-number, start-pc}, + {20, 0}, //number -> number / 1 + }; + + static final int[][] deserializeExpectedLNT = { + // {line-number, start-pc}, + {05, 0}, //number -> number / 1 + }; + + static final int[][] lambdaBridgeExpectedLNT = { + // {line-number, start-pc}, + {22, 0}, //number -> number / 1 + }; + public static void main(String[] args) throws Exception { - new WrongLVTForLambdaTest().run(); + new WrongLNTForLambdaTest().run(); } void run() throws Exception { compileTestClass(); checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri())); + "Foo.class").toUri()), "lambda$bar$0", simpleLambdaExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "lambda$variablesInLambdas$1", lambdaWithVarsExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo$1FooBar.class").toUri()), "run", insideLambdaWithVarsExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "lambda$variablesInLambdas$2", lambdaVoid2VoidExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "$deserializeLambda$", deserializeExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "lambda$MR$variablesInLambdas$notify$8bc4f5bd$1", lambdaBridgeExpectedLNT); } void compileTestClass() throws Exception { @@ -77,12 +128,12 @@ public class WrongLVTForLambdaTest { ToolBox.javac(javacSuccessArgs); } - void checkClassFile(final File cfile) throws Exception { + void checkClassFile(final File cfile, String methodToFind, int[][] expectedLNT) throws Exception { ClassFile classFile = ClassFile.read(cfile); - int methodsFound = 0; + boolean methodFound = false; for (Method method : classFile.methods) { - if (method.getName(classFile.constant_pool).startsWith("lambda$")) { - ++methodsFound; + if (method.getName(classFile.constant_pool).equals(methodToFind)) { + methodFound = true; Code_attribute code = (Code_attribute) method.attributes.get("Code"); LineNumberTable_attribute lnt = (LineNumberTable_attribute) code.attributes.get("LineNumberTable"); @@ -99,7 +150,7 @@ public class WrongLVTForLambdaTest { } } } - Assert.check(methodsFound == 1, "Expected to find one lambda method, found " + methodsFound); + Assert.check(methodFound, "The seek method was not found"); } void error(String msg) { From c9ceea7ac714ad1c231ddc9da3591855a59a5d9e Mon Sep 17 00:00:00 2001 From: Robert Field Date: Wed, 23 Oct 2013 10:28:10 -0700 Subject: [PATCH 17/28] 8022720: Method refeerences - private method should be accessible (nested classes) Reviewed-by: jjg, ksrini --- .../sun/tools/javac/comp/LambdaToMethod.java | 21 ++++--- .../MethodInvoker.java | 52 ++++++++++++++++++ .../MethodSupplier.java | 52 ++++++++++++++++++ .../privateMethodReferences/ThirdClass.java | 55 +++++++++++++++++++ 4 files changed, 172 insertions(+), 8 deletions(-) create mode 100644 langtools/test/tools/javac/lambda/privateMethodReferences/MethodInvoker.java create mode 100644 langtools/test/tools/javac/lambda/privateMethodReferences/MethodSupplier.java create mode 100644 langtools/test/tools/javac/lambda/privateMethodReferences/ThirdClass.java diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index cef43efdf92..dc9c0444b55 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -2087,13 +2087,6 @@ public class LambdaToMethod extends TreeTranslator { return tree.sym.owner == syms.arrayClass; } - boolean isPrivateConstructor() { - //hack needed to workaround 292 bug (8005122) - //when 292 issue is fixed we should simply remove this - return tree.sym.name == names.init && - (tree.sym.flags() & PRIVATE) != 0; - } - boolean receiverAccessible() { //hack needed to workaround 292 bug (7087658) //when 292 issue is fixed we should remove this and change the backend @@ -2101,13 +2094,25 @@ public class LambdaToMethod extends TreeTranslator { return tree.ownerAccessible; } + /** + * The VM does not support access across nested classes (8010319). + * Were that ever to change, this should be removed. + */ + boolean isPrivateInOtherClass() { + return (tree.sym.flags() & PRIVATE) != 0 && + !types.isSameType( + types.erasure(tree.sym.enclClass().asType()), + types.erasure(owner.enclClass().asType())); + } + /** * Does this reference needs a bridge (i.e. var args need to be * expanded or "super" is used) */ final boolean needsBridge() { return isSuper || needsVarArgsConversion() || isArrayOp() || - isPrivateConstructor() || !receiverAccessible(); + isPrivateInOtherClass() || + !receiverAccessible(); } Type generatedRefSig() { diff --git a/langtools/test/tools/javac/lambda/privateMethodReferences/MethodInvoker.java b/langtools/test/tools/javac/lambda/privateMethodReferences/MethodInvoker.java new file mode 100644 index 00000000000..0ebcd4cff18 --- /dev/null +++ b/langtools/test/tools/javac/lambda/privateMethodReferences/MethodInvoker.java @@ -0,0 +1,52 @@ +/* + * Copyright (c) 2013, 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 8022720 + * @summary private method should be accessible (nested classes) + * @run main MethodInvoker + */ + +interface MyFunctionalInterface1 { + int invokeMethodReference(); +} + +public class MethodInvoker { + public static void main(String[] args) throws Exception { + MethodInvoker.invoke(); + System.out.println("Passed!"); + } + public static void invoke() throws Exception { + MethodSupplier ms = new MethodSupplier(); + MyFunctionalInterface1 fi = ms::m; + if (fi.invokeMethodReference() != 123) { + throw new Exception(); + } + } + static class MethodSupplier { + private int m() { + return 123; + } + } +} diff --git a/langtools/test/tools/javac/lambda/privateMethodReferences/MethodSupplier.java b/langtools/test/tools/javac/lambda/privateMethodReferences/MethodSupplier.java new file mode 100644 index 00000000000..191b40118ac --- /dev/null +++ b/langtools/test/tools/javac/lambda/privateMethodReferences/MethodSupplier.java @@ -0,0 +1,52 @@ +/* + * Copyright (c) 2013, 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 8022720 + * @summary private method should be accessible (nested classes) + * @run main MethodSupplier + */ + +interface MyFunctionalInterface2 { + int invokeMethodReference(); +} + +public class MethodSupplier { + public static void main(String[] args) throws Exception { + MethodInvoker.invoke(); + System.out.println("Passed!"); + } + private int m() { + return 4321; + } + static class MethodInvoker { + public static void invoke() throws Exception { + MethodSupplier ms = new MethodSupplier(); + MyFunctionalInterface2 fi = ms::m; + if (fi.invokeMethodReference() != 4321) { + throw new Exception(); + } + } + } +} diff --git a/langtools/test/tools/javac/lambda/privateMethodReferences/ThirdClass.java b/langtools/test/tools/javac/lambda/privateMethodReferences/ThirdClass.java new file mode 100644 index 00000000000..a5e4d71171d --- /dev/null +++ b/langtools/test/tools/javac/lambda/privateMethodReferences/ThirdClass.java @@ -0,0 +1,55 @@ +/* + * Copyright (c) 2013, 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 8022720 + * @summary private method should be accessible (nested classes) + * @run main ThirdClass + */ + +interface MyFunctionalInterface3 { + int invokeMethodReference(); +} + +public class ThirdClass { + public static void main(String[] args) throws Exception { + MethodInvoker.invoke(); + System.out.println("Passed!"); + } + static class MethodSupplier { + private int m() { + return 999; + } + } + + static class MethodInvoker { + public static void invoke() throws Exception { + MethodSupplier ms = new MethodSupplier(); + MyFunctionalInterface3 fi = ms::m; + if (fi.invokeMethodReference() != 999) { + throw new Exception(); + } + } + } +} From 80787f47a9c755f1a04e69e23633b3f0ba100641 Mon Sep 17 00:00:00 2001 From: Bhavesh Patel Date: Wed, 23 Oct 2013 13:54:13 -0700 Subject: [PATCH 18/28] 8026770: javadoc creates invalid HTML in profile summary pages Reviewed-by: jjg --- .../formats/html/HtmlDocletWriter.java | 3 +- .../html/ProfilePackageWriterImpl.java | 5 ++- .../javadoc/testProfiles/TestProfiles.java | 37 ++++++++++++++++++- 3 files changed, 41 insertions(+), 4 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java index b06dc156097..d3ea1a5f187 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java @@ -391,8 +391,7 @@ public class HtmlDocletWriter extends HtmlDocWriter { tbody.addContent(tr); } table.addContent(tbody); - Content li = HtmlTree.LI(HtmlStyle.blockList, table); - summaryContentTree.addContent(li); + summaryContentTree.addContent(table); } } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java index dea7479619c..314e75f8196 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ProfilePackageWriterImpl.java @@ -175,8 +175,11 @@ public class ProfilePackageWriterImpl extends HtmlDocletWriter */ public void addClassesSummary(ClassDoc[] classes, String label, String tableSummary, String[] tableHeader, Content packageSummaryContentTree) { + HtmlTree li = new HtmlTree(HtmlTag.LI); + li.addStyle(HtmlStyle.blockList); addClassesSummary(classes, label, tableSummary, tableHeader, - packageSummaryContentTree, profileValue); + li, profileValue); + packageSummaryContentTree.addContent(li); } /** diff --git a/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java b/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java index 0dc858dddd6..a21a18e4054 100644 --- a/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java +++ b/langtools/test/com/sun/javadoc/testProfiles/TestProfiles.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8006124 8009684 8016921 8023700 8024096 8008164 8026567 + * @bug 8006124 8009684 8016921 8023700 8024096 8008164 8026567 8026770 * @summary Test javadoc support for profiles. * @author Bhavesh Patel, Evgeniya Stepanova * @library ../lib/ @@ -85,6 +85,20 @@ public class TestProfiles extends JavadocTester { "

    pkg2

    " }, + {PROFILE_BUG_ID + FS + "compact2-summary.html", + "
    {@code kind() } {@code name() }
    {@code @throws } {@code @throws }
    " + }, + {PROFILE_BUG_ID + FS + "compact2-summary.html", + "
    " + }, // Tests for profileName-package-summary.html listing the summary for a // package in a profile. {PROFILE_BUG_ID + FS + "pkg5" + FS + "compact3-package-summary.html", @@ -94,6 +108,12 @@ public class TestProfiles extends JavadocTester { {PROFILE_BUG_ID + FS + "pkg5" + FS + "compact3-package-summary.html", "
    compact3
    " }, + {PROFILE_BUG_ID + FS + "pkg5" + FS + "compact3-package-summary.html", + "
      " + NL + "
    • " + NL + + "
    " + }, //Test for "overview-frame.html" showing the "All Profiles" link. {PROFILE_BUG_ID + FS + "overview-frame.html", "Anno1Pkg4" }, {PROFILE_BUG_ID + FS + "compact1-summary.html","
  • Use
  • " + }, + {PROFILE_BUG_ID + FS + "compact2-summary.html", + "
      " + NL + "
    • " + NL + + "

      " + + "pkg2

      " + NL + "
    • " + NL + + "
    " + }, + {PROFILE_BUG_ID + FS + "pkg5" + FS + "compact3-package-summary.html", + "
      " + NL + "
    • " + NL + + "
    • " + NL + + "
    " } }; private static final String[][] PACKAGES_TEST = { From 237d1940c6e0aba905c33f0be51ef5170ffdb7ca Mon Sep 17 00:00:00 2001 From: Jan Lahoda Date: Wed, 23 Oct 2013 23:02:17 +0200 Subject: [PATCH 19/28] 8027191: Fix for JDK-8026861 refers to an incorrect bug number Reverting changeset e7c8a164d7bd, so that it can be applied again with a correct bug number Reviewed-by: jjg --- .../sun/tools/javac/comp/LambdaToMethod.java | 103 ++++++------------ ...daTest.java => WrongLVTForLambdaTest.java} | 75 ++----------- 2 files changed, 45 insertions(+), 133 deletions(-) rename langtools/test/tools/javac/T8019486/{WrongLNTForLambdaTest.java => WrongLVTForLambdaTest.java} (55%) diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index dc9c0444b55..17461a43d94 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -156,15 +156,12 @@ public class LambdaToMethod extends TreeTranslator { */ private final VarSymbol deserParamSym; - private final JCClassDecl clazz; - - private KlassInfo(JCClassDecl clazz) { - this.clazz = clazz; + private KlassInfo(Symbol kSym) { appendedMethodList = new ListBuffer<>(); deserializeCases = new HashMap>(); MethodType type = new MethodType(List.of(syms.serializedLambdaType), syms.objectType, List.nil(), syms.methodClass); - deserMethodSym = makePrivateSyntheticMethod(STATIC, names.deserializeLambda, type, clazz.sym); + deserMethodSym = makePrivateSyntheticMethod(STATIC, names.deserializeLambda, type, kSym); deserParamSym = new VarSymbol(FINAL, names.fromString("lambda"), syms.serializedLambdaType, deserMethodSym); } @@ -224,16 +221,10 @@ public class LambdaToMethod extends TreeTranslator { } KlassInfo prevKlassInfo = kInfo; try { - kInfo = new KlassInfo(tree); + kInfo = new KlassInfo(tree.sym); super.visitClassDef(tree); if (!kInfo.deserializeCases.isEmpty()) { - int prevPos = make.pos; - try { - make.at(tree); - kInfo.addMethod(makeDeserializeMethod(tree.sym)); - } finally { - make.at(prevPos); - } + kInfo.addMethod(makeDeserializeMethod(tree.sym)); } //add all translated instance methods here List newMethods = kInfo.appendedMethodList.toList(); @@ -409,21 +400,14 @@ public class LambdaToMethod extends TreeTranslator { if (context == null || !analyzer.lambdaIdentSymbolFilter(tree.sym)) { super.visitIdent(tree); } else { - int prevPos = make.pos; - try { - make.at(tree); - - LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; - JCTree ltree = lambdaContext.translate(tree); - if (ltree != null) { - result = ltree; - } else { - //access to untranslated symbols (i.e. compile-time constants, - //members defined inside the lambda body, etc.) ) - super.visitIdent(tree); - } - } finally { - make.at(prevPos); + LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; + JCTree ltree = lambdaContext.translate(tree); + if (ltree != null) { + result = ltree; + } else { + //access to untranslated symbols (i.e. compile-time constants, + //members defined inside the lambda body, etc.) ) + super.visitIdent(tree); } } } @@ -433,21 +417,11 @@ public class LambdaToMethod extends TreeTranslator { LambdaTranslationContext lambdaContext = (LambdaTranslationContext)context; if (context != null && lambdaContext.getSymbolMap(LOCAL_VAR).containsKey(tree.sym)) { JCExpression init = translate(tree.init); - int prevPos = make.pos; - try { - result = make.at(tree).VarDef((VarSymbol)lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym), init); - } finally { - make.at(prevPos); - } + result = make.VarDef((VarSymbol)lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym), init); } else if (context != null && lambdaContext.getSymbolMap(TYPE_VAR).containsKey(tree.sym)) { JCExpression init = translate(tree.init); VarSymbol xsym = (VarSymbol)lambdaContext.getSymbolMap(TYPE_VAR).get(tree.sym); - int prevPos = make.pos; - try { - result = make.at(tree).VarDef(xsym, init); - } finally { - make.at(prevPos); - } + result = make.VarDef(xsym, init); // Replace the entered symbol for this variable Scope sc = tree.sym.owner.members(); if (sc != null) { @@ -474,28 +448,23 @@ public class LambdaToMethod extends TreeTranslator { boolean isLambda_void = expr.type.hasTag(VOID); boolean isTarget_void = restype.hasTag(VOID); boolean isTarget_Void = types.isSameType(restype, types.boxedClass(syms.voidType).type); - int prevPos = make.pos; - try { - if (isTarget_void) { - //target is void: - // BODY; - JCStatement stat = make.at(expr).Exec(expr); - return make.Block(0, List.of(stat)); - } else if (isLambda_void && isTarget_Void) { - //void to Void conversion: - // BODY; return null; - ListBuffer stats = new ListBuffer<>(); - stats.append(make.at(expr).Exec(expr)); - stats.append(make.Return(make.Literal(BOT, null).setType(syms.botType))); - return make.Block(0, stats.toList()); - } else { - //non-void to non-void conversion: - // return (TYPE)BODY; - JCExpression retExpr = transTypes.coerce(attrEnv, expr, restype); - return make.at(retExpr).Block(0, List.of(make.Return(retExpr))); - } - } finally { - make.at(prevPos); + if (isTarget_void) { + //target is void: + // BODY; + JCStatement stat = make.Exec(expr); + return make.Block(0, List.of(stat)); + } else if (isLambda_void && isTarget_Void) { + //void to Void conversion: + // BODY; return null; + ListBuffer stats = new ListBuffer<>(); + stats.append(make.Exec(expr)); + stats.append(make.Return(make.Literal(BOT, null).setType(syms.botType))); + return make.Block(0, stats.toList()); + } else { + //non-void to non-void conversion: + // return (TYPE)BODY; + JCExpression retExpr = transTypes.coerce(attrEnv, expr, restype); + return make.at(retExpr).Block(0, List.of(make.Return(retExpr))); } } @@ -997,14 +966,8 @@ public class LambdaToMethod extends TreeTranslator { } } if (context.isSerializable()) { - int prevPos = make.pos; - try { - make.at(kInfo.clazz); - addDeserializationCase(refKind, refSym, tree.type, samSym, - tree, staticArgs, indyType); - } finally { - make.at(prevPos); - } + addDeserializationCase(refKind, refSym, tree.type, samSym, + tree, staticArgs, indyType); } } diff --git a/langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java b/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java similarity index 55% rename from langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java rename to langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java index 525e20e074b..22eeae60fc6 100644 --- a/langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java +++ b/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java @@ -25,11 +25,11 @@ /* * @test - * @bug 8019486 8026861 + * @bug 8019486 * @summary javac, generates erroneous LVT for a test case with lambda code * @library /tools/javac/lib * @build ToolBox - * @run main WrongLNTForLambdaTest + * @run main WrongLVTForLambdaTest */ import java.io.File; @@ -41,7 +41,7 @@ import com.sun.tools.classfile.LineNumberTable_attribute; import com.sun.tools.classfile.Method; import com.sun.tools.javac.util.Assert; -public class WrongLNTForLambdaTest { +public class WrongLVTForLambdaTest { static final String testSource = /* 01 */ "import java.util.List;\n" + @@ -54,72 +54,21 @@ public class WrongLNTForLambdaTest { /* 08 */ " final List numbersPlusOne = \n" + /* 09 */ " numbers.stream().map(number -> number / 1).collect(Collectors.toList());\n" + /* 10 */ " }\n" + - /* 11 */ " void variablesInLambdas(int value) {\n" + - /* 12 */ " Runnable r1 = () -> {\n" + - /* 13 */ " int i = value;\n" + - /* 14 */ " class FooBar {\n" + - /* 15 */ " public void run() {\n" + - /* 16 */ " T t = null;\n" + - /* 17 */ " }\n" + - /* 18 */ " }\n" + - /* 19 */ " };\n" + - /* 20 */ " Runnable r2 = () -> System.err.println(1);\n" + - /* 21 */ " Runnable r3 = (Runnable & java.io.Serializable) this::foo;\n" + - /* 22 */ " Runnable r4 = super :: notify;\n" + - /* 23 */ " }\n" + - /* 24 */ " private void foo() {}\n" + - /* 25 */ "}"; + /* 11 */ "}"; - static final int[][] simpleLambdaExpectedLNT = { + static final int[][] expectedLNT = { // {line-number, start-pc}, {9, 0}, //number -> number / 1 }; - static final int[][] lambdaWithVarsExpectedLNT = { - // {line-number, start-pc}, - {13, 0}, //number -> number / 1 - {19, 2}, //number -> number / 1 - }; - - static final int[][] insideLambdaWithVarsExpectedLNT = { - // {line-number, start-pc}, - {16, 0}, //number -> number / 1 - {17, 2}, //number -> number / 1 - }; - - static final int[][] lambdaVoid2VoidExpectedLNT = { - // {line-number, start-pc}, - {20, 0}, //number -> number / 1 - }; - - static final int[][] deserializeExpectedLNT = { - // {line-number, start-pc}, - {05, 0}, //number -> number / 1 - }; - - static final int[][] lambdaBridgeExpectedLNT = { - // {line-number, start-pc}, - {22, 0}, //number -> number / 1 - }; - public static void main(String[] args) throws Exception { - new WrongLNTForLambdaTest().run(); + new WrongLVTForLambdaTest().run(); } void run() throws Exception { compileTestClass(); checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri()), "lambda$bar$0", simpleLambdaExpectedLNT); - checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri()), "lambda$variablesInLambdas$1", lambdaWithVarsExpectedLNT); - checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo$1FooBar.class").toUri()), "run", insideLambdaWithVarsExpectedLNT); - checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri()), "lambda$variablesInLambdas$2", lambdaVoid2VoidExpectedLNT); - checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri()), "$deserializeLambda$", deserializeExpectedLNT); - checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri()), "lambda$MR$variablesInLambdas$notify$8bc4f5bd$1", lambdaBridgeExpectedLNT); + "Foo.class").toUri())); } void compileTestClass() throws Exception { @@ -128,12 +77,12 @@ public class WrongLNTForLambdaTest { ToolBox.javac(javacSuccessArgs); } - void checkClassFile(final File cfile, String methodToFind, int[][] expectedLNT) throws Exception { + void checkClassFile(final File cfile) throws Exception { ClassFile classFile = ClassFile.read(cfile); - boolean methodFound = false; + int methodsFound = 0; for (Method method : classFile.methods) { - if (method.getName(classFile.constant_pool).equals(methodToFind)) { - methodFound = true; + if (method.getName(classFile.constant_pool).startsWith("lambda$")) { + ++methodsFound; Code_attribute code = (Code_attribute) method.attributes.get("Code"); LineNumberTable_attribute lnt = (LineNumberTable_attribute) code.attributes.get("LineNumberTable"); @@ -150,7 +99,7 @@ public class WrongLNTForLambdaTest { } } } - Assert.check(methodFound, "The seek method was not found"); + Assert.check(methodsFound == 1, "Expected to find one lambda method, found " + methodsFound); } void error(String msg) { From 14fe29f8459b9f5318ca7f45e05a527b2a9b5b88 Mon Sep 17 00:00:00 2001 From: Jan Lahoda Date: Wed, 23 Oct 2013 07:50:04 +0200 Subject: [PATCH 20/28] 8026861: Wrong LineNumberTable for variable declarations in lambdas Setting or correcting positions for many trees produced by LambdaToMethod. Reviewed-by: vromero, rfield --- .../sun/tools/javac/comp/LambdaToMethod.java | 103 ++++++++++++------ ...daTest.java => WrongLNTForLambdaTest.java} | 75 +++++++++++-- 2 files changed, 133 insertions(+), 45 deletions(-) rename langtools/test/tools/javac/T8019486/{WrongLVTForLambdaTest.java => WrongLNTForLambdaTest.java} (55%) diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java index 17461a43d94..dc9c0444b55 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java @@ -156,12 +156,15 @@ public class LambdaToMethod extends TreeTranslator { */ private final VarSymbol deserParamSym; - private KlassInfo(Symbol kSym) { + private final JCClassDecl clazz; + + private KlassInfo(JCClassDecl clazz) { + this.clazz = clazz; appendedMethodList = new ListBuffer<>(); deserializeCases = new HashMap>(); MethodType type = new MethodType(List.of(syms.serializedLambdaType), syms.objectType, List.nil(), syms.methodClass); - deserMethodSym = makePrivateSyntheticMethod(STATIC, names.deserializeLambda, type, kSym); + deserMethodSym = makePrivateSyntheticMethod(STATIC, names.deserializeLambda, type, clazz.sym); deserParamSym = new VarSymbol(FINAL, names.fromString("lambda"), syms.serializedLambdaType, deserMethodSym); } @@ -221,10 +224,16 @@ public class LambdaToMethod extends TreeTranslator { } KlassInfo prevKlassInfo = kInfo; try { - kInfo = new KlassInfo(tree.sym); + kInfo = new KlassInfo(tree); super.visitClassDef(tree); if (!kInfo.deserializeCases.isEmpty()) { - kInfo.addMethod(makeDeserializeMethod(tree.sym)); + int prevPos = make.pos; + try { + make.at(tree); + kInfo.addMethod(makeDeserializeMethod(tree.sym)); + } finally { + make.at(prevPos); + } } //add all translated instance methods here List newMethods = kInfo.appendedMethodList.toList(); @@ -400,14 +409,21 @@ public class LambdaToMethod extends TreeTranslator { if (context == null || !analyzer.lambdaIdentSymbolFilter(tree.sym)) { super.visitIdent(tree); } else { - LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; - JCTree ltree = lambdaContext.translate(tree); - if (ltree != null) { - result = ltree; - } else { - //access to untranslated symbols (i.e. compile-time constants, - //members defined inside the lambda body, etc.) ) - super.visitIdent(tree); + int prevPos = make.pos; + try { + make.at(tree); + + LambdaTranslationContext lambdaContext = (LambdaTranslationContext) context; + JCTree ltree = lambdaContext.translate(tree); + if (ltree != null) { + result = ltree; + } else { + //access to untranslated symbols (i.e. compile-time constants, + //members defined inside the lambda body, etc.) ) + super.visitIdent(tree); + } + } finally { + make.at(prevPos); } } } @@ -417,11 +433,21 @@ public class LambdaToMethod extends TreeTranslator { LambdaTranslationContext lambdaContext = (LambdaTranslationContext)context; if (context != null && lambdaContext.getSymbolMap(LOCAL_VAR).containsKey(tree.sym)) { JCExpression init = translate(tree.init); - result = make.VarDef((VarSymbol)lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym), init); + int prevPos = make.pos; + try { + result = make.at(tree).VarDef((VarSymbol)lambdaContext.getSymbolMap(LOCAL_VAR).get(tree.sym), init); + } finally { + make.at(prevPos); + } } else if (context != null && lambdaContext.getSymbolMap(TYPE_VAR).containsKey(tree.sym)) { JCExpression init = translate(tree.init); VarSymbol xsym = (VarSymbol)lambdaContext.getSymbolMap(TYPE_VAR).get(tree.sym); - result = make.VarDef(xsym, init); + int prevPos = make.pos; + try { + result = make.at(tree).VarDef(xsym, init); + } finally { + make.at(prevPos); + } // Replace the entered symbol for this variable Scope sc = tree.sym.owner.members(); if (sc != null) { @@ -448,23 +474,28 @@ public class LambdaToMethod extends TreeTranslator { boolean isLambda_void = expr.type.hasTag(VOID); boolean isTarget_void = restype.hasTag(VOID); boolean isTarget_Void = types.isSameType(restype, types.boxedClass(syms.voidType).type); - if (isTarget_void) { - //target is void: - // BODY; - JCStatement stat = make.Exec(expr); - return make.Block(0, List.of(stat)); - } else if (isLambda_void && isTarget_Void) { - //void to Void conversion: - // BODY; return null; - ListBuffer stats = new ListBuffer<>(); - stats.append(make.Exec(expr)); - stats.append(make.Return(make.Literal(BOT, null).setType(syms.botType))); - return make.Block(0, stats.toList()); - } else { - //non-void to non-void conversion: - // return (TYPE)BODY; - JCExpression retExpr = transTypes.coerce(attrEnv, expr, restype); - return make.at(retExpr).Block(0, List.of(make.Return(retExpr))); + int prevPos = make.pos; + try { + if (isTarget_void) { + //target is void: + // BODY; + JCStatement stat = make.at(expr).Exec(expr); + return make.Block(0, List.of(stat)); + } else if (isLambda_void && isTarget_Void) { + //void to Void conversion: + // BODY; return null; + ListBuffer stats = new ListBuffer<>(); + stats.append(make.at(expr).Exec(expr)); + stats.append(make.Return(make.Literal(BOT, null).setType(syms.botType))); + return make.Block(0, stats.toList()); + } else { + //non-void to non-void conversion: + // return (TYPE)BODY; + JCExpression retExpr = transTypes.coerce(attrEnv, expr, restype); + return make.at(retExpr).Block(0, List.of(make.Return(retExpr))); + } + } finally { + make.at(prevPos); } } @@ -966,8 +997,14 @@ public class LambdaToMethod extends TreeTranslator { } } if (context.isSerializable()) { - addDeserializationCase(refKind, refSym, tree.type, samSym, - tree, staticArgs, indyType); + int prevPos = make.pos; + try { + make.at(kInfo.clazz); + addDeserializationCase(refKind, refSym, tree.type, samSym, + tree, staticArgs, indyType); + } finally { + make.at(prevPos); + } } } diff --git a/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java b/langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java similarity index 55% rename from langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java rename to langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java index 22eeae60fc6..525e20e074b 100644 --- a/langtools/test/tools/javac/T8019486/WrongLVTForLambdaTest.java +++ b/langtools/test/tools/javac/T8019486/WrongLNTForLambdaTest.java @@ -25,11 +25,11 @@ /* * @test - * @bug 8019486 + * @bug 8019486 8026861 * @summary javac, generates erroneous LVT for a test case with lambda code * @library /tools/javac/lib * @build ToolBox - * @run main WrongLVTForLambdaTest + * @run main WrongLNTForLambdaTest */ import java.io.File; @@ -41,7 +41,7 @@ import com.sun.tools.classfile.LineNumberTable_attribute; import com.sun.tools.classfile.Method; import com.sun.tools.javac.util.Assert; -public class WrongLVTForLambdaTest { +public class WrongLNTForLambdaTest { static final String testSource = /* 01 */ "import java.util.List;\n" + @@ -54,21 +54,72 @@ public class WrongLVTForLambdaTest { /* 08 */ " final List numbersPlusOne = \n" + /* 09 */ " numbers.stream().map(number -> number / 1).collect(Collectors.toList());\n" + /* 10 */ " }\n" + - /* 11 */ "}"; + /* 11 */ " void variablesInLambdas(int value) {\n" + + /* 12 */ " Runnable r1 = () -> {\n" + + /* 13 */ " int i = value;\n" + + /* 14 */ " class FooBar {\n" + + /* 15 */ " public void run() {\n" + + /* 16 */ " T t = null;\n" + + /* 17 */ " }\n" + + /* 18 */ " }\n" + + /* 19 */ " };\n" + + /* 20 */ " Runnable r2 = () -> System.err.println(1);\n" + + /* 21 */ " Runnable r3 = (Runnable & java.io.Serializable) this::foo;\n" + + /* 22 */ " Runnable r4 = super :: notify;\n" + + /* 23 */ " }\n" + + /* 24 */ " private void foo() {}\n" + + /* 25 */ "}"; - static final int[][] expectedLNT = { + static final int[][] simpleLambdaExpectedLNT = { // {line-number, start-pc}, {9, 0}, //number -> number / 1 }; + static final int[][] lambdaWithVarsExpectedLNT = { + // {line-number, start-pc}, + {13, 0}, //number -> number / 1 + {19, 2}, //number -> number / 1 + }; + + static final int[][] insideLambdaWithVarsExpectedLNT = { + // {line-number, start-pc}, + {16, 0}, //number -> number / 1 + {17, 2}, //number -> number / 1 + }; + + static final int[][] lambdaVoid2VoidExpectedLNT = { + // {line-number, start-pc}, + {20, 0}, //number -> number / 1 + }; + + static final int[][] deserializeExpectedLNT = { + // {line-number, start-pc}, + {05, 0}, //number -> number / 1 + }; + + static final int[][] lambdaBridgeExpectedLNT = { + // {line-number, start-pc}, + {22, 0}, //number -> number / 1 + }; + public static void main(String[] args) throws Exception { - new WrongLVTForLambdaTest().run(); + new WrongLNTForLambdaTest().run(); } void run() throws Exception { compileTestClass(); checkClassFile(new File(Paths.get(System.getProperty("user.dir"), - "Foo.class").toUri())); + "Foo.class").toUri()), "lambda$bar$0", simpleLambdaExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "lambda$variablesInLambdas$1", lambdaWithVarsExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo$1FooBar.class").toUri()), "run", insideLambdaWithVarsExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "lambda$variablesInLambdas$2", lambdaVoid2VoidExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "$deserializeLambda$", deserializeExpectedLNT); + checkClassFile(new File(Paths.get(System.getProperty("user.dir"), + "Foo.class").toUri()), "lambda$MR$variablesInLambdas$notify$8bc4f5bd$1", lambdaBridgeExpectedLNT); } void compileTestClass() throws Exception { @@ -77,12 +128,12 @@ public class WrongLVTForLambdaTest { ToolBox.javac(javacSuccessArgs); } - void checkClassFile(final File cfile) throws Exception { + void checkClassFile(final File cfile, String methodToFind, int[][] expectedLNT) throws Exception { ClassFile classFile = ClassFile.read(cfile); - int methodsFound = 0; + boolean methodFound = false; for (Method method : classFile.methods) { - if (method.getName(classFile.constant_pool).startsWith("lambda$")) { - ++methodsFound; + if (method.getName(classFile.constant_pool).equals(methodToFind)) { + methodFound = true; Code_attribute code = (Code_attribute) method.attributes.get("Code"); LineNumberTable_attribute lnt = (LineNumberTable_attribute) code.attributes.get("LineNumberTable"); @@ -99,7 +150,7 @@ public class WrongLVTForLambdaTest { } } } - Assert.check(methodsFound == 1, "Expected to find one lambda method, found " + methodsFound); + Assert.check(methodFound, "The seek method was not found"); } void error(String msg) { From 017ea0892285df2e6a8cf0b38d7b8eec565f6037 Mon Sep 17 00:00:00 2001 From: Jan Lahoda Date: Wed, 23 Oct 2013 15:45:18 -0700 Subject: [PATCH 21/28] 8026936: Initialize LamdbaToMethod lazily and as required Reviewed-by: jjg, rfield --- .../sun/tools/javac/main/JavaCompiler.java | 34 ++++++++++++++----- 1 file changed, 25 insertions(+), 9 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java b/langtools/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java index 89b95f0bbe9..e592147e256 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java +++ b/langtools/src/share/classes/com/sun/tools/javac/main/JavaCompiler.java @@ -271,10 +271,6 @@ public class JavaCompiler { */ protected TransTypes transTypes; - /** The lambda translator. - */ - protected LambdaToMethod lambdaToMethod; - /** The syntactic sugar desweetener. */ protected Lower lower; @@ -388,8 +384,6 @@ public class JavaCompiler { options = Options.instance(context); - lambdaToMethod = LambdaToMethod.instance(context); - verbose = options.isSet(VERBOSE); sourceOutput = options.isSet(PRINTSOURCE); // used to be -s stubOutput = options.isSet("-stubs"); @@ -1393,6 +1387,7 @@ public class JavaCompiler { */ class ScanNested extends TreeScanner { Set> dependencies = new LinkedHashSet>(); + protected boolean hasLambdas; @Override public void visitClassDef(JCClassDecl node) { Type st = types.supertype(node.sym.type); @@ -1402,7 +1397,18 @@ public class JavaCompiler { Env stEnv = enter.getEnv(c); if (stEnv != null && env != stEnv) { if (dependencies.add(stEnv)) { - scan(stEnv.tree); + boolean prevHasLambdas = hasLambdas; + try { + scan(stEnv.tree); + } finally { + /* + * ignore any updates to hasLambdas made during + * the nested scan, this ensures an initalized + * LambdaToMethod is available only to those + * classes that contain lambdas + */ + hasLambdas = prevHasLambdas; + } } envForSuperTypeFound = true; } @@ -1410,6 +1416,16 @@ public class JavaCompiler { } super.visitClassDef(node); } + @Override + public void visitLambda(JCLambda tree) { + hasLambdas = true; + super.visitLambda(tree); + } + @Override + public void visitReference(JCMemberReference tree) { + hasLambdas = true; + super.visitReference(tree); + } } ScanNested scanner = new ScanNested(); scanner.scan(env.tree); @@ -1468,11 +1484,11 @@ public class JavaCompiler { env.tree = transTypes.translateTopLevelClass(env.tree, localMake); compileStates.put(env, CompileState.TRANSTYPES); - if (source.allowLambda()) { + if (source.allowLambda() && scanner.hasLambdas) { if (shouldStop(CompileState.UNLAMBDA)) return; - env.tree = lambdaToMethod.translateTopLevelClass(env, env.tree, localMake); + env.tree = LambdaToMethod.instance(context).translateTopLevelClass(env, env.tree, localMake); compileStates.put(env, CompileState.UNLAMBDA); } From fb5a684124e6162b04c3f0a57b9c8ba4a4a2ea80 Mon Sep 17 00:00:00 2001 From: Eric McCorkle Date: Wed, 23 Oct 2013 23:20:32 -0400 Subject: [PATCH 22/28] 8006732: support correct bytecode storage of type annotations in multicatch Fix issue with annotations being added before attribution, which causes multicatch not to work right and several tests to fail. Reviewed-by: jfranck, jjg --- .../com/sun/tools/javac/comp/Attr.java | 7 +- .../com/sun/tools/javac/comp/MemberEnter.java | 169 ++++++++++-------- .../classes/com/sun/tools/javac/jvm/Gen.java | 14 +- .../failures/CantAnnotateStaticClass2.java | 1 - .../failures/CantAnnotateStaticClass2.out | 10 +- .../newlocations/MultiCatch.java | 1 - .../referenceinfos/MultiCatch.java | 1 - 7 files changed, 110 insertions(+), 93 deletions(-) diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java index 67d99066280..e8af8251c6f 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Attr.java @@ -4096,8 +4096,9 @@ public class Attr extends JCTree.Visitor { } private static List fromAnnotations(List annotations) { - if (annotations.isEmpty()) + if (annotations.isEmpty()) { return List.nil(); + } ListBuffer buf = new ListBuffer<>(); for (JCAnnotation anno : annotations) { @@ -4109,6 +4110,10 @@ public class Attr extends JCTree.Visitor { // Any better solutions? buf.append((Attribute.TypeCompound) anno.attribute); } + // Eventually we will want to throw an exception here, but + // we can't do that just yet, because it gets triggered + // when attempting to attach an annotation that isn't + // defined. } return buf.toList(); } diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/MemberEnter.java b/langtools/src/share/classes/com/sun/tools/javac/comp/MemberEnter.java index ce865333873..95f10e43e87 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/MemberEnter.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/MemberEnter.java @@ -573,45 +573,51 @@ public class MemberEnter extends JCTree.Visitor implements Completer { Env localEnv = methodEnv(tree, env); - DiagnosticPosition prevLintPos = deferredLintHandler.setPos(tree.pos()); + annotate.enterStart(); try { - // Compute the method type - m.type = signature(m, tree.typarams, tree.params, - tree.restype, tree.recvparam, - tree.thrown, - localEnv); - } finally { - deferredLintHandler.setPos(prevLintPos); - } + DiagnosticPosition prevLintPos = deferredLintHandler.setPos(tree.pos()); + try { + // Compute the method type + m.type = signature(m, tree.typarams, tree.params, + tree.restype, tree.recvparam, + tree.thrown, + localEnv); + } finally { + deferredLintHandler.setPos(prevLintPos); + } - if (types.isSignaturePolymorphic(m)) { - m.flags_field |= SIGNATURE_POLYMORPHIC; - } + if (types.isSignaturePolymorphic(m)) { + m.flags_field |= SIGNATURE_POLYMORPHIC; + } - // Set m.params - ListBuffer params = new ListBuffer(); - JCVariableDecl lastParam = null; - for (List l = tree.params; l.nonEmpty(); l = l.tail) { - JCVariableDecl param = lastParam = l.head; - params.append(Assert.checkNonNull(param.sym)); - } - m.params = params.toList(); + // Set m.params + ListBuffer params = new ListBuffer(); + JCVariableDecl lastParam = null; + for (List l = tree.params; l.nonEmpty(); l = l.tail) { + JCVariableDecl param = lastParam = l.head; + params.append(Assert.checkNonNull(param.sym)); + } + m.params = params.toList(); - // mark the method varargs, if necessary - if (lastParam != null && (lastParam.mods.flags & Flags.VARARGS) != 0) - m.flags_field |= Flags.VARARGS; + // mark the method varargs, if necessary + if (lastParam != null && (lastParam.mods.flags & Flags.VARARGS) != 0) + m.flags_field |= Flags.VARARGS; - localEnv.info.scope.leave(); - if (chk.checkUnique(tree.pos(), m, enclScope)) { + localEnv.info.scope.leave(); + if (chk.checkUnique(tree.pos(), m, enclScope)) { enclScope.enter(m); - } - annotateLater(tree.mods.annotations, localEnv, m, tree.pos()); - // Visit the signature of the method. Note that - // TypeAnnotate doesn't descend into the body. - typeAnnotate(tree, localEnv, m, tree.pos()); + } - if (tree.defaultValue != null) - annotateDefaultValueLater(tree.defaultValue, localEnv, m); + annotateLater(tree.mods.annotations, localEnv, m, tree.pos()); + // Visit the signature of the method. Note that + // TypeAnnotate doesn't descend into the body. + typeAnnotate(tree, localEnv, m, tree.pos()); + + if (tree.defaultValue != null) + annotateDefaultValueLater(tree.defaultValue, localEnv, m); + } finally { + annotate.enterDone(); + } } /** Create a fresh environment for method bodies. @@ -639,61 +645,68 @@ public class MemberEnter extends JCTree.Visitor implements Completer { localEnv.info.staticLevel++; } DiagnosticPosition prevLintPos = deferredLintHandler.setPos(tree.pos()); + annotate.enterStart(); try { - if (TreeInfo.isEnumInit(tree)) { - attr.attribIdentAsEnumType(localEnv, (JCIdent)tree.vartype); - } else { - attr.attribType(tree.vartype, localEnv); - if (tree.nameexpr != null) { - attr.attribExpr(tree.nameexpr, localEnv); - MethodSymbol m = localEnv.enclMethod.sym; - if (m.isConstructor()) { - Type outertype = m.owner.owner.type; - if (outertype.hasTag(TypeTag.CLASS)) { - checkType(tree.vartype, outertype, "incorrect.constructor.receiver.type"); - checkType(tree.nameexpr, outertype, "incorrect.constructor.receiver.name"); + try { + if (TreeInfo.isEnumInit(tree)) { + attr.attribIdentAsEnumType(localEnv, (JCIdent)tree.vartype); + } else { + attr.attribType(tree.vartype, localEnv); + if (tree.nameexpr != null) { + attr.attribExpr(tree.nameexpr, localEnv); + MethodSymbol m = localEnv.enclMethod.sym; + if (m.isConstructor()) { + Type outertype = m.owner.owner.type; + if (outertype.hasTag(TypeTag.CLASS)) { + checkType(tree.vartype, outertype, "incorrect.constructor.receiver.type"); + checkType(tree.nameexpr, outertype, "incorrect.constructor.receiver.name"); + } else { + log.error(tree, "receiver.parameter.not.applicable.constructor.toplevel.class"); + } } else { - log.error(tree, "receiver.parameter.not.applicable.constructor.toplevel.class"); + checkType(tree.vartype, m.owner.type, "incorrect.receiver.type"); + checkType(tree.nameexpr, m.owner.type, "incorrect.receiver.name"); } - } else { - checkType(tree.vartype, m.owner.type, "incorrect.receiver.type"); - checkType(tree.nameexpr, m.owner.type, "incorrect.receiver.name"); } } + } finally { + deferredLintHandler.setPos(prevLintPos); } - } finally { - deferredLintHandler.setPos(prevLintPos); - } - if ((tree.mods.flags & VARARGS) != 0) { - //if we are entering a varargs parameter, we need to replace its type - //(a plain array type) with the more precise VarargsType --- we need - //to do it this way because varargs is represented in the tree as a modifier - //on the parameter declaration, and not as a distinct type of array node. - ArrayType atype = (ArrayType)tree.vartype.type.unannotatedType(); - tree.vartype.type = atype.makeVarargs(); - } - Scope enclScope = enter.enterScope(env); - VarSymbol v = - new VarSymbol(0, tree.name, tree.vartype.type, enclScope.owner); - v.flags_field = chk.checkFlags(tree.pos(), tree.mods.flags, v, tree); - tree.sym = v; - if (tree.init != null) { - v.flags_field |= HASINIT; - if ((v.flags_field & FINAL) != 0 && - needsLazyConstValue(tree.init)) { - Env initEnv = getInitEnv(tree, env); - initEnv.info.enclVar = v; - v.setLazyConstValue(initEnv(tree, initEnv), attr, tree); + if ((tree.mods.flags & VARARGS) != 0) { + //if we are entering a varargs parameter, we need to + //replace its type (a plain array type) with the more + //precise VarargsType --- we need to do it this way + //because varargs is represented in the tree as a + //modifier on the parameter declaration, and not as a + //distinct type of array node. + ArrayType atype = (ArrayType)tree.vartype.type.unannotatedType(); + tree.vartype.type = atype.makeVarargs(); } + Scope enclScope = enter.enterScope(env); + VarSymbol v = + new VarSymbol(0, tree.name, tree.vartype.type, enclScope.owner); + v.flags_field = chk.checkFlags(tree.pos(), tree.mods.flags, v, tree); + tree.sym = v; + if (tree.init != null) { + v.flags_field |= HASINIT; + if ((v.flags_field & FINAL) != 0 && + needsLazyConstValue(tree.init)) { + Env initEnv = getInitEnv(tree, env); + initEnv.info.enclVar = v; + v.setLazyConstValue(initEnv(tree, initEnv), attr, tree); + } + } + if (chk.checkUnique(tree.pos(), v, enclScope)) { + chk.checkTransparentVar(tree.pos(), v, enclScope); + enclScope.enter(v); + } + annotateLater(tree.mods.annotations, localEnv, v, tree.pos()); + typeAnnotate(tree.vartype, env, v, tree.pos()); + v.pos = tree.pos; + } finally { + annotate.enterDone(); } - if (chk.checkUnique(tree.pos(), v, enclScope)) { - chk.checkTransparentVar(tree.pos(), v, enclScope); - enclScope.enter(v); - } - annotateLater(tree.mods.annotations, localEnv, v, tree.pos()); - typeAnnotate(tree.vartype, env, v, tree.pos()); - v.pos = tree.pos; } // where void checkType(JCTree tree, Type type, String diag) { diff --git a/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java b/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java index 34e61a011ee..2e03adab15c 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java +++ b/langtools/src/share/classes/com/sun/tools/javac/jvm/Gen.java @@ -1652,9 +1652,10 @@ public class Gen extends JCTree.Visitor { startpc, end, code.curCP(), catchType); if (subCatch.type.isAnnotated()) { - // All compounds share the same position, simply update the - // first one. - subCatch.type.getAnnotationMirrors().head.position.type_index = catchType; + for (Attribute.TypeCompound tc : + subCatch.type.getAnnotationMirrors()) { + tc.position.type_index = catchType; + } } } gaps = gaps.tail; @@ -1668,9 +1669,10 @@ public class Gen extends JCTree.Visitor { startpc, endpc, code.curCP(), catchType); if (subCatch.type.isAnnotated()) { - // All compounds share the same position, simply update the - // first one. - subCatch.type.getAnnotationMirrors().head.position.type_index = catchType; + for (Attribute.TypeCompound tc : + subCatch.type.getAnnotationMirrors()) { + tc.position.type_index = catchType; + } } } } diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.java b/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.java index f53340f6366..e01fa1f72d4 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.java +++ b/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.java @@ -3,7 +3,6 @@ * @bug 8006733 8006775 * @summary Ensure behavior for nested types is correct. * @author Werner Dietl - * @ignore * @compile/fail/ref=CantAnnotateStaticClass2.out -XDrawDiagnostics CantAnnotateStaticClass2.java */ diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.out b/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.out index 977dd690624..869059ce7f0 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.out +++ b/langtools/test/tools/javac/annotations/typeAnnotations/failures/CantAnnotateStaticClass2.out @@ -24,10 +24,6 @@ CantAnnotateStaticClass2.java:57:12: compiler.err.cant.type.annotate.scoping.1: CantAnnotateStaticClass2.java:58:12: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:65:12: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:66:12: compiler.err.cant.type.annotate.scoping.1: @Top.TB -CantAnnotateStaticClass2.java:105:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB -CantAnnotateStaticClass2.java:107:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB -CantAnnotateStaticClass2.java:112:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB -CantAnnotateStaticClass2.java:114:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:120:14: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:121:14: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:128:14: compiler.err.cant.type.annotate.scoping.1: @Top.TB @@ -50,6 +46,10 @@ CantAnnotateStaticClass2.java:165:22: compiler.err.cant.type.annotate.scoping.1: CantAnnotateStaticClass2.java:167:22: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:169:22: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:171:22: compiler.err.cant.type.annotate.scoping.1: @Top.TB +CantAnnotateStaticClass2.java:105:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB +CantAnnotateStaticClass2.java:107:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB +CantAnnotateStaticClass2.java:112:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB +CantAnnotateStaticClass2.java:114:18: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:184:35: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:186:41: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:187:41: compiler.err.cant.type.annotate.scoping.1: @Top.TB @@ -62,4 +62,4 @@ CantAnnotateStaticClass2.java:201:41: compiler.err.cant.type.annotate.scoping.1: CantAnnotateStaticClass2.java:202:44: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:203:44: compiler.err.cant.type.annotate.scoping.1: @Top.TB CantAnnotateStaticClass2.java:204:49: compiler.err.cant.type.annotate.scoping: @Top.TA,@Top.TB,@Top.TC -64 errors \ No newline at end of file +64 errors diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/MultiCatch.java b/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/MultiCatch.java index 59b02d7b105..8a1545d2f34 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/MultiCatch.java +++ b/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/MultiCatch.java @@ -25,7 +25,6 @@ import java.lang.annotation.*; /* * @test - * @ignore 8008762 Type annotations failures * @bug 8006775 * @summary new type annotation location: multicatch * @author Werner Dietl diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/MultiCatch.java b/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/MultiCatch.java index aaa2faac469..7c16a0d7b66 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/MultiCatch.java +++ b/langtools/test/tools/javac/annotations/typeAnnotations/referenceinfos/MultiCatch.java @@ -25,7 +25,6 @@ import static com.sun.tools.classfile.TypeAnnotation.TargetType.*; /* * @test - * @ignore 8008762 Type annotation failures * @bug 8006732 8006775 * @summary Test population of reference info for multicatch exception parameters * @author Werner Dietl From e62fb20a596112ad6fea0b1376f58299d310db8e Mon Sep 17 00:00:00 2001 From: Eric McCorkle Date: Thu, 24 Oct 2013 01:27:10 -0400 Subject: [PATCH 23/28] 8023682: Incorrect attributes emitted for anonymous class declaration Cause javac to emit type annotations on new instruction as well as anonymous class supertype for annotated anonymous classes. Reviewed-by: jjg, jfranck --- .../sun/tools/javac/code/TypeAnnotations.java | 24 +++++++++++++++---- .../com/sun/tools/javac/comp/Check.java | 3 +-- .../failures/TypeOnAnonClass.java | 13 ++++++++++ .../failures/TypeOnAnonClass.out | 2 ++ .../common/arrays/DeclarationAnnotation.out | 3 ++- .../newlocations/AnonymousClass.java | 5 +--- 6 files changed, 39 insertions(+), 11 deletions(-) create mode 100644 langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.java create mode 100644 langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.out diff --git a/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java b/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java index a9030cac1f1..4f809260129 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java +++ b/langtools/src/share/classes/com/sun/tools/javac/code/TypeAnnotations.java @@ -370,9 +370,9 @@ public class TypeAnnotations { sym.appendUniqueTypeAttributes(typeAnnotations); if (sym.getKind() == ElementKind.PARAMETER || - sym.getKind() == ElementKind.LOCAL_VARIABLE || - sym.getKind() == ElementKind.RESOURCE_VARIABLE || - sym.getKind() == ElementKind.EXCEPTION_PARAMETER) { + sym.getKind() == ElementKind.LOCAL_VARIABLE || + sym.getKind() == ElementKind.RESOURCE_VARIABLE || + sym.getKind() == ElementKind.EXCEPTION_PARAMETER) { // Make sure all type annotations from the symbol are also // on the owner. sym.owner.appendUniqueTypeAttributes(sym.getRawTypeAttributes()); @@ -1221,6 +1221,22 @@ public class TypeAnnotations { super.visitTypeParameter(tree); } + private void copyNewClassAnnotationsToOwner(JCNewClass tree) { + Symbol sym = tree.def.sym; + TypeAnnotationPosition pos = new TypeAnnotationPosition(); + ListBuffer newattrs = + new ListBuffer(); + + for (Attribute.TypeCompound old : sym.getRawTypeAttributes()) { + newattrs.append(new Attribute.TypeCompound(old.type, old.values, + pos)); + } + + pos.type = TargetType.NEW; + pos.pos = tree.pos; + sym.owner.appendUniqueTypeAttributes(newattrs.toList()); + } + @Override public void visitNewClass(JCNewClass tree) { if (tree.def != null && @@ -1239,7 +1255,7 @@ public class TypeAnnotations { } Type before = classdecl.sym.type; separateAnnotationsKinds(classdecl, tree.clazz.type, classdecl.sym, pos); - + copyNewClassAnnotationsToOwner(tree); // classdecl.sym.type now contains an annotated type, which // is not what we want there. // TODO: should we put this type somewhere in the superclass/interface? diff --git a/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java b/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java index 74c1540640b..7e63150ed0d 100644 --- a/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java +++ b/langtools/src/share/classes/com/sun/tools/javac/comp/Check.java @@ -3011,7 +3011,6 @@ public class Check { boolean annotationApplicable(JCAnnotation a, Symbol s) { Attribute.Array arr = getAttributeTargetAttribute(a.annotationType.type.tsym); Name[] targets; - if (arr == null) { targets = defaultTargetMetaInfo(a, s); } else { @@ -3028,7 +3027,7 @@ public class Check { } for (Name target : targets) { if (target == names.TYPE) - { if (s.kind == TYP) return true; } + { if (s.kind == TYP && !s.isAnonymous()) return true; } else if (target == names.FIELD) { if (s.kind == VAR && s.owner.kind != MTH) return true; } else if (target == names.METHOD) diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.java b/langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.java new file mode 100644 index 00000000000..1c5f6ee1fcc --- /dev/null +++ b/langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.java @@ -0,0 +1,13 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8023682 + * @summary Cannot annotate an anonymous class with a target type of TYPE + * @compile/fail/ref=TypeOnAnonClass.out -XDrawDiagnostics TypeOnAnonClass.java + */ +import java.lang.annotation.*; + +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.TYPE) +@interface X {} +interface Foo {} +class TypeOnAnonClass { void m() { new @X Foo() {}; } } diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.out b/langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.out new file mode 100644 index 00000000000..f5a986dd956 --- /dev/null +++ b/langtools/test/tools/javac/annotations/typeAnnotations/failures/TypeOnAnonClass.out @@ -0,0 +1,2 @@ +TypeOnAnonClass.java:13:40: compiler.err.annotation.type.not.applicable +1 error diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/failures/common/arrays/DeclarationAnnotation.out b/langtools/test/tools/javac/annotations/typeAnnotations/failures/common/arrays/DeclarationAnnotation.out index 91c76a572e9..490f378c755 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/failures/common/arrays/DeclarationAnnotation.out +++ b/langtools/test/tools/javac/annotations/typeAnnotations/failures/common/arrays/DeclarationAnnotation.out @@ -1,4 +1,5 @@ DeclarationAnnotation.java:10:21: compiler.err.annotation.type.not.applicable DeclarationAnnotation.java:11:21: compiler.err.annotation.type.not.applicable DeclarationAnnotation.java:12:21: compiler.err.annotation.type.not.applicable -3 errors \ No newline at end of file +DeclarationAnnotation.java:16:21: compiler.err.annotation.type.not.applicable +4 errors diff --git a/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/AnonymousClass.java b/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/AnonymousClass.java index e4136c29b7a..d43141e245e 100644 --- a/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/AnonymousClass.java +++ b/langtools/test/tools/javac/annotations/typeAnnotations/newlocations/AnonymousClass.java @@ -32,12 +32,9 @@ import java.lang.annotation.*; */ class AnonymousClass { Object o1 = new @TA Object() { }; - // Declaration annotations are also allowed. - Object o2 = new @TA @DA Object() { }; + Object o2 = new @TA Object() { }; } -@interface DA { } - @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface TA { } From 3033b82865324ba267a1e9c8cac8756513036ab1 Mon Sep 17 00:00:00 2001 From: Bhavesh Patel Date: Thu, 24 Oct 2013 11:22:50 -0700 Subject: [PATCH 24/28] 8006248: Since addition of -Xdoclint, javadoc ignores unknown tags Reviewed-by: jjg --- .../formats/html/ConfigurationImpl.java | 2 +- .../toolkit/taglets/TagletManager.java | 4 + .../com/sun/tools/doclint/Checker.java | 19 +++ .../com/sun/tools/doclint/DocLint.java | 6 + .../classes/com/sun/tools/doclint/Env.java | 11 ++ .../classes/com/sun/tools/javadoc/DocEnv.java | 11 +- .../com/sun/tools/javadoc/RootDocImpl.java | 4 +- .../javadoc/testCustomTag/TagTestClass.java | 31 +++++ .../javadoc/testCustomTag/TestCustomTag.java | 109 ++++++++++++++++++ .../testCustomTag/taglets/CustomTag.java | 59 ++++++++++ .../test/tools/doclint/CustomTagTest.java | 19 +++ .../test/tools/doclint/CustomTagTest.out | 8 ++ .../tools/doclint/CustomTagTestWithOption.out | 5 + .../test/tools/doclint/DocLintTester.java | 2 + 14 files changed, 286 insertions(+), 4 deletions(-) create mode 100644 langtools/test/com/sun/javadoc/testCustomTag/TagTestClass.java create mode 100644 langtools/test/com/sun/javadoc/testCustomTag/TestCustomTag.java create mode 100644 langtools/test/com/sun/javadoc/testCustomTag/taglets/CustomTag.java create mode 100644 langtools/test/tools/doclint/CustomTagTest.java create mode 100644 langtools/test/tools/doclint/CustomTagTest.out create mode 100644 langtools/test/tools/doclint/CustomTagTestWithOption.out diff --git a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ConfigurationImpl.java b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ConfigurationImpl.java index 5d84c8064db..5d5132a7ad0 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ConfigurationImpl.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/formats/html/ConfigurationImpl.java @@ -284,7 +284,7 @@ public class ConfigurationImpl extends Configuration { setTopFile(root); if (root instanceof RootDocImpl) { - ((RootDocImpl) root).initDocLint(doclintOpts); + ((RootDocImpl) root).initDocLint(doclintOpts, tagletManager.getCustomTagNames()); } } diff --git a/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/taglets/TagletManager.java b/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/taglets/TagletManager.java index b6af4746e03..d1aa9b727de 100644 --- a/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/taglets/TagletManager.java +++ b/langtools/src/share/classes/com/sun/tools/doclets/internal/toolkit/taglets/TagletManager.java @@ -205,6 +205,10 @@ public class TagletManager { } } + public Set getCustomTagNames() { + return customTags.keySet(); + } + /** * Add a new Taglet. Print a message to indicate whether or not * the Taglet was registered properly. diff --git a/langtools/src/share/classes/com/sun/tools/doclint/Checker.java b/langtools/src/share/classes/com/sun/tools/doclint/Checker.java index d3ed9c26bde..0498c98dd6b 100644 --- a/langtools/src/share/classes/com/sun/tools/doclint/Checker.java +++ b/langtools/src/share/classes/com/sun/tools/doclint/Checker.java @@ -71,6 +71,8 @@ import com.sun.source.doctree.SinceTree; import com.sun.source.doctree.StartElementTree; import com.sun.source.doctree.TextTree; import com.sun.source.doctree.ThrowsTree; +import com.sun.source.doctree.UnknownBlockTagTree; +import com.sun.source.doctree.UnknownInlineTagTree; import com.sun.source.doctree.ValueTree; import com.sun.source.doctree.VersionTree; import com.sun.source.util.DocTreePath; @@ -841,6 +843,23 @@ public class Checker extends DocTreePathScanner { } } + @Override + public Void visitUnknownBlockTag(UnknownBlockTagTree tree, Void ignore) { + checkUnknownTag(tree, tree.getTagName()); + return super.visitUnknownBlockTag(tree, ignore); + } + + @Override + public Void visitUnknownInlineTag(UnknownInlineTagTree tree, Void ignore) { + checkUnknownTag(tree, tree.getTagName()); + return super.visitUnknownInlineTag(tree, ignore); + } + + private void checkUnknownTag(DocTree tree, String tagName) { + if (env.customTags != null && !env.customTags.contains(tagName)) + env.messages.error(SYNTAX, tree, "dc.tag.unknown", tagName); + } + @Override public Void visitValue(ValueTree tree, Void ignore) { ReferenceTree ref = tree.getReference(); diff --git a/langtools/src/share/classes/com/sun/tools/doclint/DocLint.java b/langtools/src/share/classes/com/sun/tools/doclint/DocLint.java index a7b6fb6a15c..972a0e05ddd 100644 --- a/langtools/src/share/classes/com/sun/tools/doclint/DocLint.java +++ b/langtools/src/share/classes/com/sun/tools/doclint/DocLint.java @@ -78,6 +78,8 @@ public class DocLint implements Plugin { public static final String XMSGS_CUSTOM_PREFIX = "-Xmsgs:"; private static final String STATS = "-stats"; public static final String XIMPLICIT_HEADERS = "-XimplicitHeaders:"; + public static final String XCUSTOM_TAGS_PREFIX = "-XcustomTags:"; + public static final String TAGS_SEPARATOR = ","; // public static void main(String... args) { @@ -199,6 +201,8 @@ public class DocLint implements Plugin { env.messages.setOptions(null); } else if (arg.startsWith(XMSGS_CUSTOM_PREFIX)) { env.messages.setOptions(arg.substring(arg.indexOf(":") + 1)); + } else if (arg.startsWith(XCUSTOM_TAGS_PREFIX)) { + env.setCustomTags(arg.substring(arg.indexOf(":") + 1)); } else if (arg.equals("-h") || arg.equals("-help") || arg.equals("--help") || arg.equals("-?") || arg.equals("-usage")) { needHelp = true; @@ -262,6 +266,8 @@ public class DocLint implements Plugin { } else if (arg.matches(XIMPLICIT_HEADERS + "[1-6]")) { char ch = arg.charAt(arg.length() - 1); env.setImplicitHeaders(Character.digit(ch, 10)); + } else if (arg.startsWith(XCUSTOM_TAGS_PREFIX)) { + env.setCustomTags(arg.substring(arg.indexOf(":") + 1)); } else throw new IllegalArgumentException(arg); } diff --git a/langtools/src/share/classes/com/sun/tools/doclint/Env.java b/langtools/src/share/classes/com/sun/tools/doclint/Env.java index 0af19be96a8..8ef21dbe0bb 100644 --- a/langtools/src/share/classes/com/sun/tools/doclint/Env.java +++ b/langtools/src/share/classes/com/sun/tools/doclint/Env.java @@ -27,6 +27,7 @@ package com.sun.tools.doclint; import java.util.Set; +import java.util.LinkedHashSet; import javax.lang.model.element.Element; import javax.lang.model.element.ElementKind; @@ -86,6 +87,8 @@ public class Env { int implicitHeaderLevel = 0; + Set customTags; + // Utility classes DocTrees trees; Elements elements; @@ -135,6 +138,14 @@ public class Env { implicitHeaderLevel = n; } + void setCustomTags(String cTags) { + customTags = new LinkedHashSet(); + for (String s : cTags.split(DocLint.TAGS_SEPARATOR)) { + if (!s.isEmpty()) + customTags.add(s); + } + } + /** Set the current declaration and its doc comment. */ void setCurrent(TreePath path, DocCommentTree comment) { currPath = path; diff --git a/langtools/src/share/classes/com/sun/tools/javadoc/DocEnv.java b/langtools/src/share/classes/com/sun/tools/javadoc/DocEnv.java index d0ff14b70ea..a294373af29 100644 --- a/langtools/src/share/classes/com/sun/tools/javadoc/DocEnv.java +++ b/langtools/src/share/classes/com/sun/tools/javadoc/DocEnv.java @@ -800,7 +800,7 @@ public class DocEnv { return result; } - void initDoclint(Collection opts) { + void initDoclint(Collection opts, Collection customTagNames) { ArrayList doclintOpts = new ArrayList(); for (String opt: opts) { @@ -814,6 +814,15 @@ public class DocEnv { return; } + String sep = ""; + StringBuilder customTags = new StringBuilder(); + for (String customTag : customTagNames) { + customTags.append(sep); + customTags.append(customTag); + sep = DocLint.TAGS_SEPARATOR; + } + doclintOpts.add(DocLint.XCUSTOM_TAGS_PREFIX + customTags.toString()); + JavacTask t = BasicJavacTask.instance(context); doclint = new DocLint(); // standard doclet normally generates H1, H2 diff --git a/langtools/src/share/classes/com/sun/tools/javadoc/RootDocImpl.java b/langtools/src/share/classes/com/sun/tools/javadoc/RootDocImpl.java index 89a086aa8f8..5d02145a827 100644 --- a/langtools/src/share/classes/com/sun/tools/javadoc/RootDocImpl.java +++ b/langtools/src/share/classes/com/sun/tools/javadoc/RootDocImpl.java @@ -377,8 +377,8 @@ public class RootDocImpl extends DocImpl implements RootDoc { return env.fileManager; } - public void initDocLint(Collection opts) { - env.initDoclint(opts); + public void initDocLint(Collection opts, Collection customTagNames) { + env.initDoclint(opts, customTagNames); } public boolean showTagMessages() { diff --git a/langtools/test/com/sun/javadoc/testCustomTag/TagTestClass.java b/langtools/test/com/sun/javadoc/testCustomTag/TagTestClass.java new file mode 100644 index 00000000000..6bf742d36fb --- /dev/null +++ b/langtools/test/com/sun/javadoc/testCustomTag/TagTestClass.java @@ -0,0 +1,31 @@ +/* + * Copyright (c) 2013, 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. + */ + +/** + * @customTag A custom tag. + * @unknownTag An unknown tag + */ +public class TagTestClass { + + public void method(){} +} diff --git a/langtools/test/com/sun/javadoc/testCustomTag/TestCustomTag.java b/langtools/test/com/sun/javadoc/testCustomTag/TestCustomTag.java new file mode 100644 index 00000000000..0e931df8645 --- /dev/null +++ b/langtools/test/com/sun/javadoc/testCustomTag/TestCustomTag.java @@ -0,0 +1,109 @@ +/* + * Copyright (c) 2013, 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 8006248 + * @summary Test custom tag. Verify that an unknown tag generates appropriate warnings. + * @author Bhavesh Patel + * @library ../lib/ + * @build JavadocTester taglets.CustomTag TestCustomTag + * @run main TestCustomTag + */ + +public class TestCustomTag extends JavadocTester { + + //Test information. + private static final String BUG_ID = "8006248"; + + //Javadoc arguments. + private static final String[] ARGS = new String[] { + "-Xdoclint:none", "-d", BUG_ID, "-tagletpath", SRC_DIR, + "-taglet", "taglets.CustomTag", "-sourcepath", + SRC_DIR, SRC_DIR + FS + "TagTestClass.java" + }; + + private static final String[] ARGS1 = new String[] { + "-d", BUG_ID + "-1", "-tagletpath", SRC_DIR, "-taglet", "taglets.CustomTag", + "-sourcepath", SRC_DIR, SRC_DIR + FS + "TagTestClass.java" + }; + private static final String[] ARGS2 = new String[] { + "-Xdoclint:none", "-d", BUG_ID + "-2", "-sourcepath", + SRC_DIR, SRC_DIR + FS + "TagTestClass.java" + }; + + private static final String[] ARGS3 = new String[] { + "-d", BUG_ID + "-3", "-sourcepath", SRC_DIR, SRC_DIR + FS + "TagTestClass.java" + }; + + //Input for string search tests. + private static final String[][] TEST = new String[][] { + {WARNING_OUTPUT, "warning - @unknownTag is an unknown tag." + } + }; + + private static final String[][] TEST1 = new String[][] { + {ERROR_OUTPUT, "error: unknown tag: unknownTag" + } + }; + private static final String[][] TEST2 = new String[][] { + {WARNING_OUTPUT, "warning - @customTag is an unknown tag." + }, + {WARNING_OUTPUT, "warning - @unknownTag is an unknown tag." + } + }; + + private static final String[][] TEST3 = new String[][] { + {ERROR_OUTPUT, "error: unknown tag: customTag" + }, + {ERROR_OUTPUT, "error: unknown tag: unknownTag" + } + }; + + /** + * The entry point of the test. + * @param args the array of command line arguments. + */ + public static void main(String[] args) { + TestCustomTag tester = new TestCustomTag(); + run(tester, ARGS, TEST, NO_TEST); + run(tester, ARGS1, TEST1, NO_TEST); + run(tester, ARGS2, TEST2, NO_TEST); + run(tester, ARGS3, TEST3, NO_TEST); + tester.printSummary(); + } + + /** + * {@inheritDoc} + */ + public String getBugId() { + return BUG_ID; + } + + /** + * {@inheritDoc} + */ + public String getBugName() { + return getClass().getName(); + } +} diff --git a/langtools/test/com/sun/javadoc/testCustomTag/taglets/CustomTag.java b/langtools/test/com/sun/javadoc/testCustomTag/taglets/CustomTag.java new file mode 100644 index 00000000000..d38e5533f96 --- /dev/null +++ b/langtools/test/com/sun/javadoc/testCustomTag/taglets/CustomTag.java @@ -0,0 +1,59 @@ +/* + * Copyright (c) 2013, 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. + */ + +package taglets; + +import com.sun.tools.doclets.internal.toolkit.*; +import com.sun.tools.doclets.internal.toolkit.taglets.*; +import com.sun.tools.doclets.internal.toolkit.util.*; + +import com.sun.javadoc.*; +import java.util.*; + +public class CustomTag extends BaseTaglet { + + public CustomTag() { + name = "customTag"; + } + + public static void register(Map tagletMap) { + CustomTag tag = new CustomTag(); + Taglet t = (Taglet) tagletMap.get(tag.getName()); + if (t != null) { + tagletMap.remove(tag.getName()); + } + tagletMap.put(tag.getName(), tag); + } + + /** + * {@inheritDoc} + */ + public Content getTagletOutput(Tag tag, TagletWriter writer) { + ArrayList inlineTags = new ArrayList(); + inlineTags.add(new TextTag(tag.holder(), "
    Custom Tag:
    ")); + inlineTags.addAll(Arrays.asList(tag.inlineTags())); + inlineTags.add(new TextTag(tag.holder(), "
    ")); + return writer.commentTagsToOutput(tag, + (Tag[]) inlineTags.toArray(new Tag[] {})); + } +} diff --git a/langtools/test/tools/doclint/CustomTagTest.java b/langtools/test/tools/doclint/CustomTagTest.java new file mode 100644 index 00000000000..3340dfb878c --- /dev/null +++ b/langtools/test/tools/doclint/CustomTagTest.java @@ -0,0 +1,19 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8006248 + * @summary DocLint should report unknown tags + * @build DocLintTester + * @run main DocLintTester CustomTagTest.java + * @run main DocLintTester -XcustomTags: -ref CustomTagTest.out CustomTagTest.java + * @run main DocLintTester -XcustomTags:customTag -ref CustomTagTestWithOption.out CustomTagTest.java + * @run main DocLintTester -XcustomTags:customTag,anotherCustomTag -ref CustomTagTestWithOption.out CustomTagTest.java + * @author bpatel + */ + +/** + * @customTag Text for a custom tag. + * @unknownTag Text for an unknown tag. + */ +public class CustomTagTest { +} + diff --git a/langtools/test/tools/doclint/CustomTagTest.out b/langtools/test/tools/doclint/CustomTagTest.out new file mode 100644 index 00000000000..063e313071e --- /dev/null +++ b/langtools/test/tools/doclint/CustomTagTest.out @@ -0,0 +1,8 @@ +CustomTagTest.java:14: error: unknown tag: customTag + * @customTag Text for a custom tag. + ^ +CustomTagTest.java:15: error: unknown tag: unknownTag + * @unknownTag Text for an unknown tag. + ^ +2 errors + diff --git a/langtools/test/tools/doclint/CustomTagTestWithOption.out b/langtools/test/tools/doclint/CustomTagTestWithOption.out new file mode 100644 index 00000000000..09150146e0c --- /dev/null +++ b/langtools/test/tools/doclint/CustomTagTestWithOption.out @@ -0,0 +1,5 @@ +CustomTagTest.java:15: error: unknown tag: unknownTag + * @unknownTag Text for an unknown tag. + ^ +1 error + diff --git a/langtools/test/tools/doclint/DocLintTester.java b/langtools/test/tools/doclint/DocLintTester.java index 78c052a695c..2691731c8e2 100644 --- a/langtools/test/tools/doclint/DocLintTester.java +++ b/langtools/test/tools/doclint/DocLintTester.java @@ -58,6 +58,8 @@ public class DocLintTester { badArgs = true; } else if (arg.startsWith("-Xmsgs")) { opts.add(arg); + } else if (arg.startsWith("-XcustomTags")) { + opts.add(arg); } else if (arg.startsWith("-")) { opts.add(arg); if (i < args.length - 1 && !args[i+1].startsWith("-")) From e49bc26ebca65eb038f214ff28f2bfcfe079f6dd Mon Sep 17 00:00:00 2001 From: Robert Field Date: Thu, 24 Oct 2013 16:52:27 -0700 Subject: [PATCH 25/28] 8027220: DefaultMethodsTest: Change test to match spec Reviewed-by: ksrini --- .../openjdk/tests/separate/TestHarness.java | 2 +- .../openjdk/tests/vm/DefaultMethodsTest.java | 26 +++++++++---------- 2 files changed, 14 insertions(+), 14 deletions(-) diff --git a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java index 18ea7cd9f62..85705bb7ed1 100644 --- a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java +++ b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java @@ -300,7 +300,7 @@ public class TestHarness { if (verboseLocal.get() == Boolean.TRUE) { System.out.println(e.getCause()); } - assertEquals(e.getCause().getClass(), exceptionType); + assertTrue(exceptionType.isAssignableFrom(e.getCause().getClass())); } compiler.cleanup(); } diff --git a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java index 77500f255e9..3b8002f4722 100644 --- a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java +++ b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2013, 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 @@ -255,7 +255,7 @@ public class DefaultMethodsTest extends TestHarness { * interface J { default int m() { return 88; } } * class C implements I, J {} * - * TEST: C c = new C(); c.m() throws AME + * TEST: C c = new C(); c.m() throws ICCE */ public void testConflict() { // debugTest(); @@ -263,7 +263,7 @@ public class DefaultMethodsTest extends TestHarness { Interface J = new Interface("J", DefaultMethod.std("88")); Class C = new Class("C", I, J); - assertThrows(AbstractMethodError.class, C); + assertThrows(IncompatibleClassChangeError.class, C); } /** @@ -271,14 +271,14 @@ public class DefaultMethodsTest extends TestHarness { * interface J { default int m() { return 88; } } * class C implements I, J {} * - * TEST: C c = new C(); c.m() throws AME + * TEST: C c = new C(); c.m() == 88 */ public void testAmbiguousReabstract() { Interface I = new Interface("I", AbstractMethod.std()); Interface J = new Interface("J", DefaultMethod.std("88")); Class C = new Class("C", I, J); - assertThrows(AbstractMethodError.class, C); + assertInvokeVirtualEquals(88, C); } /** @@ -555,8 +555,8 @@ public class DefaultMethodsTest extends TestHarness { * interface I extends J, K { int m() default { J.super.m(); } } * class C implements I {} * - * TEST: C c = new C(); c.m() throws AME - * TODO: add case for K k = new C(); k.m() throws AME + * TEST: C c = new C(); c.m() throws ICCE + * TODO: add case for K k = new C(); k.m() throws ICCE */ public void testSuperConflict() { // debugTest(); @@ -571,7 +571,7 @@ public class DefaultMethodsTest extends TestHarness { I.addCompilationDependency(Jstub.findMethod(stdMethodName)); Class C = new Class("C", I); - assertThrows(AbstractMethodError.class, C); + assertThrows(IncompatibleClassChangeError.class, C); } /** @@ -579,8 +579,8 @@ public class DefaultMethodsTest extends TestHarness { * interface J extends I { default int m() { return 55; } } * class C implements I, J { public int m() { return I.super.m(); } } * - * TEST: C c = new C(); c.m() throws AME - * TODO: add case for J j = new C(); j.m() throws AME + * TEST: C c = new C(); c.m() == 99 + * TODO: add case for J j = new C(); j.m() == ??? */ public void testSuperDisqual() { Interface I = new Interface("I", DefaultMethod.std("99")); @@ -590,7 +590,7 @@ public class DefaultMethodsTest extends TestHarness { AccessFlag.PUBLIC)); C.addCompilationDependency(I.findMethod(stdMethodName)); - assertThrows(AbstractMethodError.class, C); + assertInvokeVirtualEquals(99, C); } /** @@ -646,7 +646,7 @@ public class DefaultMethodsTest extends TestHarness { * public int m(String s) { return I.super.m(s); } * } * - * TEST: C c = new C(); c.m("string") throws AME + * TEST: C c = new C(); c.m("string") == 44 */ public void testSuperGenericDisqual() { MethodParameter t = new MethodParameter("T", "t"); @@ -661,7 +661,7 @@ public class DefaultMethodsTest extends TestHarness { "return I.super.m(s);", AccessFlag.PUBLIC, s)); C.addCompilationDependency(I.findMethod(stdMethodName)); - assertThrows(AbstractMethodError.class, C, + assertInvokeVirtualEquals(44, C, new ConcreteMethod( "int", stdMethodName, "return -1;", AccessFlag.PUBLIC, s), "-1", "\"string\""); From fa729039bef36b18d13179c0890e40394ac2e04c Mon Sep 17 00:00:00 2001 From: Mandy Chung Date: Wed, 30 Oct 2013 08:35:52 -0700 Subject: [PATCH 26/28] 8027481: jdeps to handle classes with the same package name and correct profile for javax.crypto.* Reviewed-by: alanb, dfuchs --- .../classes/com/sun/tools/jdeps/Analyzer.java | 205 ++++++++------ .../classes/com/sun/tools/jdeps/Archive.java | 15 +- .../com/sun/tools/jdeps/JdepsTask.java | 266 ++++++++++++++---- .../classes/com/sun/tools/jdeps/Profile.java | 9 +- langtools/test/tools/jdeps/Basic.java | 57 ++-- langtools/test/tools/jdeps/Test.java | 3 +- .../javax/activity/NotCompactProfile.java | 30 ++ langtools/test/tools/jdeps/p/Bar.java | 33 +++ 8 files changed, 458 insertions(+), 160 deletions(-) create mode 100644 langtools/test/tools/jdeps/javax/activity/NotCompactProfile.java create mode 100644 langtools/test/tools/jdeps/p/Bar.java diff --git a/langtools/src/share/classes/com/sun/tools/jdeps/Analyzer.java b/langtools/src/share/classes/com/sun/tools/jdeps/Analyzer.java index 0dadafb053e..fffe754bbf5 100644 --- a/langtools/src/share/classes/com/sun/tools/jdeps/Analyzer.java +++ b/langtools/src/share/classes/com/sun/tools/jdeps/Analyzer.java @@ -26,9 +26,11 @@ package com.sun.tools.jdeps; import com.sun.tools.classfile.Dependency.Location; import com.sun.tools.jdeps.PlatformClassPath.JDKArchive; +import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.Set; import java.util.SortedMap; import java.util.SortedSet; @@ -52,7 +54,7 @@ public class Analyzer { private final Type type; private final Map results = new HashMap<>(); - private final Map map = new HashMap<>(); + private final Map map = new HashMap<>(); private final Archive NOT_FOUND = new Archive(JdepsTask.getMessage("artifact.not.found")); @@ -69,6 +71,17 @@ public class Analyzer { * Performs the dependency analysis on the given archives. */ public void run(List archives) { + // build a map from Location to Archive + for (Archive archive: archives) { + for (Location l: archive.getClasses()) { + if (!map.containsKey(l)) { + map.put(l, archive); + } else { + // duplicated class warning? + } + } + } + // traverse and analyze all dependencies for (Archive archive : archives) { ArchiveDeps deps; if (type == Type.CLASS || type == Type.VERBOSE) { @@ -76,33 +89,9 @@ public class Analyzer { } else { deps = new PackageVisitor(archive); } - archive.visit(deps); + archive.visitDependences(deps); results.put(archive, deps); } - - // set the required dependencies - for (ArchiveDeps result: results.values()) { - for (Set set : result.deps.values()) { - for (String target : set) { - Archive source = getArchive(target); - if (result.archive != source) { - String profile = ""; - if (source instanceof JDKArchive) { - profile = result.profile != null ? result.profile.toString() : ""; - if (result.getTargetProfile(target) == null) { - profile += ", JDK internal API"; - // override the value if it accesses any JDK internal - result.requireArchives.put(source, profile); - continue; - } - } - if (!result.requireArchives.containsKey(source)) { - result.requireArchives.put(source, profile); - } - } - } - } - } } public boolean hasDependences(Archive archive) { @@ -117,94 +106,143 @@ public class Analyzer { * Visits the source archive to its destination archive of * a recorded dependency. */ - void visitArchiveDependence(Archive origin, Archive target, String profile); + void visitArchiveDependence(Archive origin, Archive target, Profile profile); /** * Visits a recorded dependency from origin to target which can be * a fully-qualified classname, a package name, a profile or * archive name depending on the Analyzer's type. */ - void visitDependence(String origin, Archive source, String target, Archive archive, String profile); + void visitDependence(String origin, Archive source, String target, Archive archive, Profile profile); } public void visitArchiveDependences(Archive source, Visitor v) { ArchiveDeps r = results.get(source); - for (Map.Entry e : r.requireArchives.entrySet()) { - v.visitArchiveDependence(r.archive, e.getKey(), e.getValue()); + for (ArchiveDeps.Dep d: r.requireArchives()) { + v.visitArchiveDependence(r.archive, d.archive, d.profile); } } public void visitDependences(Archive source, Visitor v) { ArchiveDeps r = results.get(source); - for (String origin : r.deps.keySet()) { - for (String target : r.deps.get(origin)) { - Archive archive = getArchive(target); - assert source == getArchive(origin); - Profile profile = r.getTargetProfile(target); - + for (Map.Entry> e: r.deps.entrySet()) { + String origin = e.getKey(); + for (ArchiveDeps.Dep d: e.getValue()) { // filter intra-dependency unless in verbose mode - if (type == Type.VERBOSE || archive != source) { - v.visitDependence(origin, source, target, archive, - profile != null ? profile.toString() : ""); + if (type == Type.VERBOSE || d.archive != source) { + v.visitDependence(origin, source, d.target, d.archive, d.profile); } } } } - public Archive getArchive(String name) { - return map.containsKey(name) ? map.get(name) : NOT_FOUND; - } - + /** + * ArchiveDeps contains the dependencies for an Archive that + * can have one or more classes. + */ private abstract class ArchiveDeps implements Archive.Visitor { final Archive archive; - final Map requireArchives; - final SortedMap> deps; - Profile profile = null; + final SortedMap> deps; ArchiveDeps(Archive archive) { this.archive = archive; - this.requireArchives = new HashMap<>(); this.deps = new TreeMap<>(); } - void add(String loc) { - Archive a = map.get(loc); - if (a == null) { - map.put(loc, archive); - } else if (a != archive) { - // duplicated class warning? - } - } - - void add(String origin, String target) { - SortedSet set = deps.get(origin); + void add(String origin, String target, Archive targetArchive, String pkgName) { + SortedSet set = deps.get(origin); if (set == null) { deps.put(origin, set = new TreeSet<>()); } - if (!set.contains(target)) { - set.add(target); - // find the corresponding profile - Profile p = getTargetProfile(target); - if (profile == null || (p != null && profile.profile < p.profile)) { - profile = p; + Profile p = targetArchive instanceof JDKArchive + ? Profile.getProfile(pkgName) : null; + set.add(new Dep(target, targetArchive, p)); + } + + /** + * Returns the list of Archive dependences. The returned + * list contains one {@code Dep} instance per one archive + * and with the minimum profile this archive depends on. + */ + List requireArchives() { + Map map = new HashMap<>(); + for (Set set: deps.values()) { + for (Dep d: set) { + if (this.archive != d.archive) { + Profile p = map.get(d.archive); + if (p == null || (d.profile != null && p.profile < d.profile.profile)) { + map.put(d.archive, d.profile); + } + } } } + List list = new ArrayList<>(); + for (Map.Entry e: map.entrySet()) { + list.add(new Dep("", e.getKey(), e.getValue())); + } + return list; + } + + /** + * Dep represents a dependence where the target can be + * a classname or packagename and the archive and profile + * the target belongs to. + */ + class Dep implements Comparable { + final String target; + final Archive archive; + final Profile profile; + Dep(String target, Archive archive, Profile p) { + this.target = target; + this.archive = archive; + this.profile = p; + } + + @Override + public boolean equals(Object o) { + if (o instanceof Dep) { + Dep d = (Dep)o; + return this.archive == d.archive && this.target.equals(d.target); + } + return false; + } + + @Override + public int hashCode() { + int hash = 3; + hash = 17 * hash + Objects.hashCode(this.archive); + hash = 17 * hash + Objects.hashCode(this.target); + return hash; + } + + @Override + public int compareTo(Dep o) { + if (this.target.equals(o.target)) { + if (this.archive == o.archive) { + return 0; + } else { + return this.archive.getFileName().compareTo(o.archive.getFileName()); + } + } + return this.target.compareTo(o.target); + } } public abstract void visit(Location o, Location t); - public abstract Profile getTargetProfile(String target); } private class ClassVisitor extends ArchiveDeps { ClassVisitor(Archive archive) { super(archive); } - public void visit(Location l) { - add(l.getClassName()); - } + @Override public void visit(Location o, Location t) { - add(o.getClassName(), t.getClassName()); - } - public Profile getTargetProfile(String target) { - int i = target.lastIndexOf('.'); - return (i > 0) ? Profile.getProfile(target.substring(0, i)) : null; + Archive targetArchive = + this.archive.getClasses().contains(t) ? this.archive : map.get(t); + if (targetArchive == null) { + map.put(t, targetArchive = NOT_FOUND); + } + + String origin = o.getClassName(); + String target = t.getClassName(); + add(origin, target, targetArchive, t.getPackageName()); } } @@ -212,18 +250,21 @@ public class Analyzer { PackageVisitor(Archive archive) { super(archive); } + @Override public void visit(Location o, Location t) { - add(packageOf(o), packageOf(t)); + Archive targetArchive = + this.archive.getClasses().contains(t) ? this.archive : map.get(t); + if (targetArchive == null) { + map.put(t, targetArchive = NOT_FOUND); + } + + String origin = packageOf(o); + String target = packageOf(t); + add(origin, target, targetArchive, t.getPackageName()); } - public void visit(Location l) { - add(packageOf(l)); - } - private String packageOf(Location loc) { - String pkg = loc.getPackageName(); + public String packageOf(Location o) { + String pkg = o.getPackageName(); return pkg.isEmpty() ? "" : pkg; } - public Profile getTargetProfile(String target) { - return Profile.getProfile(target); - } } } diff --git a/langtools/src/share/classes/com/sun/tools/jdeps/Archive.java b/langtools/src/share/classes/com/sun/tools/jdeps/Archive.java index 826c7662c92..18a4a1ee23a 100644 --- a/langtools/src/share/classes/com/sun/tools/jdeps/Archive.java +++ b/langtools/src/share/classes/com/sun/tools/jdeps/Archive.java @@ -67,6 +67,7 @@ public class Archive { deps.put(origin, set); } } + public void addClass(Location origin, Location target) { Set set = deps.get(origin); if (set == null) { @@ -76,21 +77,27 @@ public class Archive { set.add(target); } - public void visit(Visitor v) { + public Set getClasses() { + return deps.keySet(); + } + + public void visitDependences(Visitor v) { for (Map.Entry> e: deps.entrySet()) { - v.visit(e.getKey()); for (Location target : e.getValue()) { v.visit(e.getKey(), target); } } } - public String toString() { + public String getPathName() { return path != null ? path.toString() : filename; } + public String toString() { + return filename; + } + interface Visitor { - void visit(Location loc); void visit(Location origin, Location target); } } diff --git a/langtools/src/share/classes/com/sun/tools/jdeps/JdepsTask.java b/langtools/src/share/classes/com/sun/tools/jdeps/JdepsTask.java index 66cd5a40530..1b9c0810283 100644 --- a/langtools/src/share/classes/com/sun/tools/jdeps/JdepsTask.java +++ b/langtools/src/share/classes/com/sun/tools/jdeps/JdepsTask.java @@ -190,6 +190,11 @@ class JdepsTask { task.options.fullVersion = true; } }, + new HiddenOption(false, "-showlabel") { + void process(JdepsTask task, String opt, String arg) { + task.options.showLabel = true; + } + }, new HiddenOption(true, "-depth") { void process(JdepsTask task, String opt, String arg) throws BadArgs { try { @@ -279,12 +284,21 @@ class JdepsTask { private void generateDotFiles(Path dir, Analyzer analyzer) throws IOException { Path summary = dir.resolve("summary.dot"); - try (PrintWriter sw = new PrintWriter(Files.newOutputStream(summary)); - DotFileFormatter formatter = new DotFileFormatter(sw, "summary")) { - for (Archive archive : sourceLocations) { - analyzer.visitArchiveDependences(archive, formatter); + boolean verbose = options.verbose == Analyzer.Type.VERBOSE; + DotGraph graph = verbose ? new DotSummaryForPackage() + : new DotSummaryForArchive(); + for (Archive archive : sourceLocations) { + analyzer.visitArchiveDependences(archive, graph); + if (verbose || options.showLabel) { + // traverse detailed dependences to generate package-level + // summary or build labels for edges + analyzer.visitDependences(archive, graph); } } + try (PrintWriter sw = new PrintWriter(Files.newOutputStream(summary))) { + graph.writeTo(sw); + } + // output individual .dot file for each archive if (options.verbose != Analyzer.Type.SUMMARY) { for (Archive archive : sourceLocations) { if (analyzer.hasDependences(archive)) { @@ -365,17 +379,16 @@ class JdepsTask { } } } + sourceLocations.addAll(archives); List classpaths = new ArrayList<>(); // for class file lookup + classpaths.addAll(getClassPathArchives(options.classpath)); if (options.includePattern != null) { - archives.addAll(getClassPathArchives(options.classpath)); - } else { - classpaths.addAll(getClassPathArchives(options.classpath)); + archives.addAll(classpaths); } classpaths.addAll(PlatformClassPath.getArchives()); - // add all archives to the source locations for reporting - sourceLocations.addAll(archives); + // add all classpath archives to the source locations for reporting sourceLocations.addAll(classpaths); // Work queue of names of classfiles to be searched. @@ -557,6 +570,7 @@ class JdepsTask { boolean showSummary; boolean wildcard; boolean apiOnly; + boolean showLabel; String dotOutputDir; String classpath = ""; int depth = 1; @@ -627,16 +641,34 @@ class JdepsTask { return result; } + /** + * If the given archive is JDK archive and non-null Profile, + * this method returns the profile name only if -profile option is specified; + * a null profile indicates it accesses a private JDK API and this method + * will return "JDK internal API". + * + * For non-JDK archives, this method returns the file name of the archive. + */ + private String getProfileArchiveInfo(Archive source, Profile profile) { + if (options.showProfile && profile != null) + return profile.toString(); + + if (source instanceof JDKArchive) { + return profile == null ? "JDK internal API (" + source.getFileName() + ")" : ""; + } + return source.getFileName(); + } /** - * Returns the file name of the archive for non-JRE class or - * internal JRE classes. It returns empty string for SE API. + * Returns the profile name or "JDK internal API" for JDK archive; + * otherwise empty string. */ - private static String getArchiveName(Archive source, String profile) { - String name = source.getFileName(); - if (source instanceof JDKArchive) - return profile.isEmpty() ? "JDK internal API (" + name + ")" : ""; - return name; + private String profileName(Archive archive, Profile profile) { + if (archive instanceof JDKArchive) { + return Objects.toString(profile, "JDK internal API"); + } else { + return ""; + } } class RawOutputFormatter implements Analyzer.Visitor { @@ -648,21 +680,18 @@ class JdepsTask { private String pkg = ""; @Override public void visitDependence(String origin, Archive source, - String target, Archive archive, String profile) { + String target, Archive archive, Profile profile) { if (!origin.equals(pkg)) { pkg = origin; writer.format(" %s (%s)%n", origin, source.getFileName()); } - String name = (options.showProfile && !profile.isEmpty()) - ? profile - : getArchiveName(archive, profile); - writer.format(" -> %-50s %s%n", target, name); + writer.format(" -> %-50s %s%n", target, getProfileArchiveInfo(archive, profile)); } @Override - public void visitArchiveDependence(Archive origin, Archive target, String profile) { - writer.format("%s -> %s", origin, target); - if (options.showProfile && !profile.isEmpty()) { + public void visitArchiveDependence(Archive origin, Archive target, Profile profile) { + writer.format("%s -> %s", origin.getPathName(), target.getPathName()); + if (options.showProfile && profile != null) { writer.format(" (%s)%n", profile); } else { writer.format("%n"); @@ -670,19 +699,14 @@ class JdepsTask { } } - class DotFileFormatter implements Analyzer.Visitor, AutoCloseable { + class DotFileFormatter extends DotGraph implements AutoCloseable { private final PrintWriter writer; private final String name; - DotFileFormatter(PrintWriter writer, String name) { - this.writer = writer; - this.name = name; - writer.format("digraph \"%s\" {%n", name); - } DotFileFormatter(PrintWriter writer, Archive archive) { this.writer = writer; this.name = archive.getFileName(); writer.format("digraph \"%s\" {%n", name); - writer.format(" // Path: %s%n", archive.toString()); + writer.format(" // Path: %s%n", archive.getPathName()); } @Override @@ -690,39 +714,169 @@ class JdepsTask { writer.println("}"); } - private final Set edges = new HashSet<>(); - private String node = ""; @Override public void visitDependence(String origin, Archive source, - String target, Archive archive, String profile) { - if (!node.equals(origin)) { - edges.clear(); - node = origin; - } + String target, Archive archive, Profile profile) { // if -P option is specified, package name -> profile will // be shown and filter out multiple same edges. - if (!edges.contains(target)) { - StringBuilder sb = new StringBuilder(); - String name = options.showProfile && !profile.isEmpty() - ? profile - : getArchiveName(archive, profile); - writer.format(" %-50s -> %s;%n", - String.format("\"%s\"", origin), - name.isEmpty() ? String.format("\"%s\"", target) - : String.format("\"%s (%s)\"", target, name)); - edges.add(target); + String name = getProfileArchiveInfo(archive, profile); + writeEdge(writer, new Edge(origin, target, getProfileArchiveInfo(archive, profile))); + } + @Override + public void visitArchiveDependence(Archive origin, Archive target, Profile profile) { + throw new UnsupportedOperationException(); + } + } + + class DotSummaryForArchive extends DotGraph { + @Override + public void visitDependence(String origin, Archive source, + String target, Archive archive, Profile profile) { + Edge e = findEdge(source, archive); + assert e != null; + // add the dependency to the label if enabled and not compact1 + if (profile == Profile.COMPACT1) { + return; + } + e.addLabel(origin, target, profileName(archive, profile)); + } + @Override + public void visitArchiveDependence(Archive origin, Archive target, Profile profile) { + // add an edge with the archive's name with no tag + // so that there is only one node for each JDK archive + // while there may be edges to different profiles + Edge e = addEdge(origin, target, ""); + if (target instanceof JDKArchive) { + // add a label to print the profile + if (profile == null) { + e.addLabel("JDK internal API"); + } else if (options.showProfile && !options.showLabel) { + e.addLabel(profile.toString()); + } } } + } + // DotSummaryForPackage generates the summary.dot file for verbose mode + // (-v or -verbose option) that includes all class dependencies. + // The summary.dot file shows package-level dependencies. + class DotSummaryForPackage extends DotGraph { + private String packageOf(String cn) { + int i = cn.lastIndexOf('.'); + return i > 0 ? cn.substring(0, i) : ""; + } @Override - public void visitArchiveDependence(Archive origin, Archive target, String profile) { - String name = options.showProfile && !profile.isEmpty() - ? profile : ""; - writer.format(" %-30s -> \"%s\";%n", - String.format("\"%s\"", origin.getFileName()), - name.isEmpty() - ? target.getFileName() - : String.format("%s (%s)", target.getFileName(), name)); + public void visitDependence(String origin, Archive source, + String target, Archive archive, Profile profile) { + // add a package dependency edge + String from = packageOf(origin); + String to = packageOf(target); + Edge e = addEdge(from, to, getProfileArchiveInfo(archive, profile)); + + // add the dependency to the label if enabled and not compact1 + if (!options.showLabel || profile == Profile.COMPACT1) { + return; + } + + // trim the package name of origin to shorten the label + int i = origin.lastIndexOf('.'); + String n1 = i < 0 ? origin : origin.substring(i+1); + e.addLabel(n1, target, profileName(archive, profile)); + } + @Override + public void visitArchiveDependence(Archive origin, Archive target, Profile profile) { + // nop + } + } + abstract class DotGraph implements Analyzer.Visitor { + private final Set edges = new LinkedHashSet<>(); + private Edge curEdge; + public void writeTo(PrintWriter writer) { + writer.format("digraph \"summary\" {%n"); + for (Edge e: edges) { + writeEdge(writer, e); + } + writer.println("}"); + } + + void writeEdge(PrintWriter writer, Edge e) { + writer.format(" %-50s -> \"%s\"%s;%n", + String.format("\"%s\"", e.from.toString()), + e.tag.isEmpty() ? e.to + : String.format("%s (%s)", e.to, e.tag), + getLabel(e)); + } + + Edge addEdge(T origin, T target, String tag) { + Edge e = new Edge(origin, target, tag); + if (e.equals(curEdge)) { + return curEdge; + } + + if (edges.contains(e)) { + for (Edge e1 : edges) { + if (e.equals(e1)) { + curEdge = e1; + } + } + } else { + edges.add(e); + curEdge = e; + } + return curEdge; + } + + Edge findEdge(T origin, T target) { + for (Edge e : edges) { + if (e.from.equals(origin) && e.to.equals(target)) { + return e; + } + } + return null; + } + + String getLabel(Edge e) { + String label = e.label.toString(); + return label.isEmpty() ? "" : String.format("[label=\"%s\",fontsize=9]", label); + } + + class Edge { + final T from; + final T to; + final String tag; // optional tag + final StringBuilder label = new StringBuilder(); + Edge(T from, T to, String tag) { + this.from = from; + this.to = to; + this.tag = tag; + } + void addLabel(String s) { + label.append(s).append("\\n"); + } + void addLabel(String origin, String target, String profile) { + label.append(origin).append(" -> ").append(target); + if (!profile.isEmpty()) { + label.append(" (" + profile + ")"); + } + label.append("\\n"); + } + @Override @SuppressWarnings("unchecked") + public boolean equals(Object o) { + if (o instanceof DotGraph.Edge) { + DotGraph.Edge e = (DotGraph.Edge)o; + return this.from.equals(e.from) && + this.to.equals(e.to) && + this.tag.equals(e.tag); + } + return false; + } + @Override + public int hashCode() { + int hash = 7; + hash = 67 * hash + Objects.hashCode(this.from) + + Objects.hashCode(this.to) + Objects.hashCode(this.tag); + return hash; + } } } } diff --git a/langtools/src/share/classes/com/sun/tools/jdeps/Profile.java b/langtools/src/share/classes/com/sun/tools/jdeps/Profile.java index 47af4b2041f..4d358553afe 100644 --- a/langtools/src/share/classes/com/sun/tools/jdeps/Profile.java +++ b/langtools/src/share/classes/com/sun/tools/jdeps/Profile.java @@ -81,8 +81,12 @@ enum Profile { } static class PackageToProfile { + static String[] JAVAX_CRYPTO_PKGS = new String[] { + "javax.crypto", + "javax.crypto.interfaces", + "javax.crypto.spec" + }; static Map map = initProfiles(); - private static Map initProfiles() { try { String profilesProps = System.getProperty("jdeps.profiles"); @@ -103,6 +107,9 @@ enum Profile { findProfile(cf); } } + // special case for javax.crypto.* classes that are not + // included in ct.sym since they are in jce.jar + Collections.addAll(Profile.COMPACT1.packages, JAVAX_CRYPTO_PKGS); } } } catch (IOException | ConstantPoolException e) { diff --git a/langtools/test/tools/jdeps/Basic.java b/langtools/test/tools/jdeps/Basic.java index fd5242aaa55..34999aa04be 100644 --- a/langtools/test/tools/jdeps/Basic.java +++ b/langtools/test/tools/jdeps/Basic.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2013, 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,9 +23,9 @@ /* * @test - * @bug 8003562 8005428 8015912 + * @bug 8003562 8005428 8015912 8027481 * @summary Basic tests for jdeps tool - * @build Test p.Foo + * @build Test p.Foo p.Bar javax.activity.NotCompactProfile * @run main Basic */ @@ -33,10 +33,12 @@ import java.io.File; import java.io.IOException; import java.io.PrintWriter; import java.io.StringWriter; +import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.util.*; import java.util.regex.*; +import static java.nio.file.StandardCopyOption.*; public class Basic { private static boolean symbolFileExist = initProfiles(); @@ -74,23 +76,25 @@ public class Basic { new String[] {"java.lang", "p"}, new String[] {"compact1", "not found"}); // test a directory + // also test non-SE javax.activity class dependency test(new File(testDir, "p"), - new String[] {"java.lang", "java.util", "java.lang.management"}, - new String[] {"compact1", "compact1", "compact3"}); + new String[] {"java.lang", "java.util", "java.lang.management", "javax.activity", "javax.crypto"}, + new String[] {"compact1", "compact1", "compact3", testDir.getName(), "compact1"}, + new String[] {"-classpath", testDir.getPath()}); // test class-level dependency output test(new File(testDir, "Test.class"), - new String[] {"java.lang.Object", "java.lang.String", "p.Foo"}, - new String[] {"compact1", "compact1", "not found"}, + new String[] {"java.lang.Object", "java.lang.String", "p.Foo", "p.Bar"}, + new String[] {"compact1", "compact1", "not found", "not found"}, new String[] {"-verbose:class"}); // test -p option test(new File(testDir, "Test.class"), - new String[] {"p.Foo"}, - new String[] {"not found"}, + new String[] {"p.Foo", "p.Bar"}, + new String[] {"not found", "not found"}, new String[] {"-verbose:class", "-p", "p"}); // test -e option test(new File(testDir, "Test.class"), - new String[] {"p.Foo"}, - new String[] {"not found"}, + new String[] {"p.Foo", "p.Bar"}, + new String[] {"not found", "not found"}, new String[] {"-verbose:class", "-e", "p\\..*"}); test(new File(testDir, "Test.class"), new String[] {"java.lang"}, @@ -99,13 +103,34 @@ public class Basic { // test -classpath and -include options test(null, new String[] {"java.lang", "java.util", - "java.lang.management"}, - new String[] {"compact1", "compact1", "compact3"}, + "java.lang.management", "javax.crypto"}, + new String[] {"compact1", "compact1", "compact3", "compact1"}, new String[] {"-classpath", testDir.getPath(), "-include", "p.+|Test.class"}); test(new File(testDir, "Test.class"), - new String[] {"java.lang.Object", "java.lang.String", "p.Foo"}, - new String[] {"compact1", "compact1", testDir.getName()}, + new String[] {"java.lang.Object", "java.lang.String", "p.Foo", "p.Bar"}, + new String[] {"compact1", "compact1", testDir.getName(), testDir.getName()}, new String[] {"-v", "-classpath", testDir.getPath(), "Test.class"}); + + // split package p - move p/Foo.class to dir1 and p/Bar.class to dir2 + Path testClassPath = testDir.toPath(); + Path dirP = testClassPath.resolve("p"); + Path dir1 = testClassPath.resolve("dir1"); + Path subdir1P = dir1.resolve("p"); + Path dir2 = testClassPath.resolve("dir2"); + Path subdir2P = dir2.resolve("p"); + if (!Files.exists(subdir1P)) + Files.createDirectories(subdir1P); + if (!Files.exists(subdir2P)) + Files.createDirectories(subdir2P); + Files.move(dirP.resolve("Foo.class"), subdir1P.resolve("Foo.class"), REPLACE_EXISTING); + Files.move(dirP.resolve("Bar.class"), subdir2P.resolve("Bar.class"), REPLACE_EXISTING); + StringBuilder cpath = new StringBuilder(testDir.toString()); + cpath.append(File.pathSeparator).append(dir1.toString()); + cpath.append(File.pathSeparator).append(dir2.toString()); + test(new File(testDir, "Test.class"), + new String[] {"java.lang.Object", "java.lang.String", "p.Foo", "p.Bar"}, + new String[] {"compact1", "compact1", dir1.toFile().getName(), dir2.toFile().getName()}, + new String[] {"-v", "-classpath", cpath.toString(), "Test.class"}); return errors; } @@ -148,7 +173,7 @@ public class Basic { // Use the linePattern to break the given String into lines, applying // the pattern to each line to see if we have a match private static Map findDeps(String out) { - Map result = new HashMap<>(); + Map result = new LinkedHashMap<>(); Matcher lm = linePattern.matcher(out); // Line matcher Matcher pm = null; // Pattern matcher int lines = 0; diff --git a/langtools/test/tools/jdeps/Test.java b/langtools/test/tools/jdeps/Test.java index 1a7a96a69ae..d47efa67f2f 100644 --- a/langtools/test/tools/jdeps/Test.java +++ b/langtools/test/tools/jdeps/Test.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2013, 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 @@ -24,6 +24,7 @@ public class Test { public void test() { p.Foo f = new p.Foo(); + p.Bar b = new p.Bar(); } private String name() { return "this test"; diff --git a/langtools/test/tools/jdeps/javax/activity/NotCompactProfile.java b/langtools/test/tools/jdeps/javax/activity/NotCompactProfile.java new file mode 100644 index 00000000000..37e466186dd --- /dev/null +++ b/langtools/test/tools/jdeps/javax/activity/NotCompactProfile.java @@ -0,0 +1,30 @@ +/* + * Copyright (c) 2013, 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. + */ + +package javax.activity; + +public class NotCompactProfile { + public static String name() { + return "not Java SE API"; + } +} diff --git a/langtools/test/tools/jdeps/p/Bar.java b/langtools/test/tools/jdeps/p/Bar.java new file mode 100644 index 00000000000..9a99a769070 --- /dev/null +++ b/langtools/test/tools/jdeps/p/Bar.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2013, 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. + */ + +package p; + +public class Bar extends javax.activity.NotCompactProfile { + public String bar() { + return "bar"; + } + public javax.crypto.Cipher getCiper() { + return null; + } +} From 9d30f46219ed118dedb10f032c5a10786090be09 Mon Sep 17 00:00:00 2001 From: Vicente Romero Date: Wed, 30 Oct 2013 18:09:49 +0000 Subject: [PATCH 27/28] 8027327: jar files related to test test/tools/javac/ExtDirs/ExtDirTest.java should be removed from the repo Reviewed-by: ksrini --- .../test/tools/javac/ExtDirs/ExtDirTest.java | 11 +++-------- langtools/test/tools/javac/ExtDirs/ext1/pkg1.jar | Bin 957 -> 0 bytes langtools/test/tools/javac/ExtDirs/ext2/pkg2.jar | Bin 955 -> 0 bytes langtools/test/tools/javac/ExtDirs/ext3/pkg1.jar | Bin 957 -> 0 bytes langtools/test/tools/javac/ExtDirs/ext3/pkg2.jar | Bin 955 -> 0 bytes 5 files changed, 3 insertions(+), 8 deletions(-) delete mode 100644 langtools/test/tools/javac/ExtDirs/ext1/pkg1.jar delete mode 100644 langtools/test/tools/javac/ExtDirs/ext2/pkg2.jar delete mode 100644 langtools/test/tools/javac/ExtDirs/ext3/pkg1.jar delete mode 100644 langtools/test/tools/javac/ExtDirs/ext3/pkg2.jar diff --git a/langtools/test/tools/javac/ExtDirs/ExtDirTest.java b/langtools/test/tools/javac/ExtDirs/ExtDirTest.java index 22af79a9f8f..d33592f819c 100644 --- a/langtools/test/tools/javac/ExtDirs/ExtDirTest.java +++ b/langtools/test/tools/javac/ExtDirs/ExtDirTest.java @@ -112,11 +112,6 @@ public class ExtDirTest { } void createJars() throws Exception { - -// for i in 1 2 3; do -// if test ! -d ext${i}; then mkdir ext${i}; fi -// cp ${TESTSRC}${FS}ext${i}${FS}*.jar ext${i} -// done sun.tools.jar.Main jarGenerator = new sun.tools.jar.Main(System.out, System.err, "jar"); @@ -155,19 +150,19 @@ public class ExtDirTest { void compileWithExtDirs() throws Exception { -//"$javac" ${TESTTOOLVMOPTS} -d . -extdirs ext1 "${TESTSRC}${FS}ExtDirTest_1.java" +//javac -extdirs ext1 ExtDirTest_1.java ToolBox.JavaToolArgs params = new ToolBox.JavaToolArgs() .setOptions("-d", ".", "-extdirs", "ext1") .setSources(ExtDirTest_1Src); ToolBox.javac(params); -//"$javac" ${TESTTOOLVMOPTS} -d . -extdirs ext1${PS}ext2 "${TESTSRC}${FS}ExtDirTest_2.java" +//javac -extdirs ext1:ext2 ExtDirTest_2.java params.setOptions("-d", ".", "-extdirs", "ext1" + File.pathSeparator + "ext2") .setSources(ExtDirTest_2Src); ToolBox.javac(params); -//"$javac" ${TESTTOOLVMOPTS} -d . -extdirs ext3 "${TESTSRC}${FS}ExtDirTest_3.java" +//javac -extdirs ext3 ExtDirTest_3.java params.setOptions("-d", ".", "-extdirs", "ext3") .setSources(ExtDirTest_3Src); ToolBox.javac(params); diff --git a/langtools/test/tools/javac/ExtDirs/ext1/pkg1.jar b/langtools/test/tools/javac/ExtDirs/ext1/pkg1.jar deleted file mode 100644 index 3429f8e2b9f60634fe3eca6cdc78b0a6aea5cc0e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 957 zcmWIWW@Zs#-~dAR+EZ!_NI(S0_H_+$)b;dp)Ax1s^K^3!4$<>2nr60fgFaCbIP0w?!X}jLJY`XMn z%59GFY5|7_pM7qvuu!}4VV!B{)op9lG_OmQAARiL5c(-%S7yOAiA=(z)D&mA~?;y~KunbRYaj-NQNW})P< z69*p6nLp{&i39I}qOZbd&7Kk^GHJ#XW|4SthQH-QMXf2 zxA&Z4q3&+pubYHU3sp))NJL0Vx@=sPkRrga^5S{VGv3#~`=9di)k_RWNnqN=&LYt$ zutL&f6JzD31UqKt)z`|{j`b#FmMNTM6H8&6B;zovvr!lr4D1}Sm6N1b1ATM@h!K9? zh!!kTIQ^WJSeB^m;d9Q<$3Il#grE7AXOFZrd{3S93G+DTsiPNGuvVk+qz;#-u3jPA zM_$oY**n43=FBx5(j`@MH1=pvl)#jxj(GiJM~>c=5T88r=*%Z)JWn}mbzhtIY*PBl z&Vt>{j4X1N!NRdg%T~VITTrntZ$+Tfk}J0ku2v|psJ4CEl4_~HXtT*g`;8axTohzE zo3)ow_~MftD%WE(T$~b~3LIY^TW%TWJugL!h4;VW*T(5(E5aV!ifvtbtNZV%6*E(7 zjfI}L)d|e{cUCOWH1XH5@K>7-)iwBc)oTBq_0iu$BJ@{=)7=GgR{FeYKRh*kf9t#e zZ+4EX+R2a31HEg*$iNT)3=l4GFzBKM2rE#uAUoYiAEX2lX6QL6r5^{OfjXX>74G+8QYZgi#J8|IA zocWVZojC9wDEcaV*6b-!B9mrJVHSDFbxvQ`%dhk_OW_i(O_zJW#A?{(E0xKjv zHZfLiO0Z*QUVW{M?O1O@W|_iCHn9}8Niq(zIva(7!NATTvPt&mYM_s905QVP*K1JI zi4;yhXC;;;YJ2#c^Yif!)i~j2zUkQ`Ee+pOCw;;^&UxzSg%zyTC_Jgd<*BPz$o8>U zWL5S~u(fw?a*68$4K)B;%g7|cfJmUoF$GGXr~q4X4e(}U;$pxZqX-iOfw2#bUxcNg z7)A~UPz)o02}}$2_y*Yxb3G{7kbMRUHU!XMg4&H?5GW7>yjj^mYFU7g4QS2nr60fgFaCbIP0w?!X}jLJY`XMn z%59GFY5|7_pM7qvuu!}4VV!B{)op9lG_OmQAARiL5c(-%S7yOAiA=(z)D&mA~?;y~KunbRYaj-NQNW})P< z69*p6nLp{&i39I}qOZbd&7Kk^GHJ#XW|4SthQH-QMXf2 zxA&Z4q3&+pubYHU3sp))NJL0Vx@=sPkRrga^5S{VGv3#~`=9di)k_RWNnqN=&LYt$ zutL&f6JzD31UqKt)z`|{j`b#FmMNTM6H8&6B;zovvr!lr4D1}Sm6N1b1ATM@h!K9? zh!!kTIQ^WJSeB^m;d9Q<$3Il#grE7AXOFZrd{3S93G+DTsiPNGuvVk+qz;#-u3jPA zM_$oY**n43=FBx5(j`@MH1=pvl)#jxj(GiJM~>c=5T88r=*%Z)JWn}mbzhtIY*PBl z&Vt>{j4X1N!NRdg%T~VITTrntZ$+Tfk}J0ku2v|psJ4CEl4_~HXtT*g`;8axTohzE zo3)ow_~MftD%WE(T$~b~3LIY^TW%TWJugL!h4;VW*T(5(E5aV!ifvtbtNZV%6*E(7 zjfI}L)d|e{cUCOWH1XH5@K>7-)iwBc)oTBq_0iu$BJ@{=)7=GgR{FeYKRh*kf9t#e zZ+4EX+R2a31HEg*$iNT)3=l4GFzBKM2rE#uAUoYiAEX2lX6QL6r5^{OfjXX>74G+8QYZgi#J8|IA zocWVZojC9wDEcaV*6b-!B9mrJVHSDFbxvQ`%dhk_OW_i(O_zJW#A?{(E0xKjv zHZfLiO0Z*QUVW{M?O1O@W|_iCHn9}8Niq(zIva(7!NATTvPt&mYM_s905QVP*K1JI zi4;yhXC;;;YJ2#c^Yif!)i~j2zUkQ`Ee+pOCw;;^&UxzSg%zyTC_Jgd<*BPz$o8>U zWL5S~u(fw?a*68$4K)B;%g7|cfJmUoF$GGXr~q4X4e(}U;$pxZqX-iOfw2#bUxcNg z7)A~UPz)o02}}$2_y*Yxb3G{7kbMRUHU!XMg4&H?5GW7>yjj^mYFU7g4QS Date: Wed, 30 Oct 2013 14:12:16 -0400 Subject: [PATCH 28/28] 8024930: Re-enable disabled bridging tests Reviewed-by: psandoz, rfield --- .../org/openjdk/tests/separate/Compiler.java | 29 +++--- .../openjdk/tests/separate/SourceModel.java | 24 ++++- .../openjdk/tests/separate/TestHarness.java | 28 +++++- .../openjdk/tests/vm/DefaultMethodsTest.java | 99 ++++++++++--------- 4 files changed, 109 insertions(+), 71 deletions(-) diff --git a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/Compiler.java b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/Compiler.java index 4aace3da1fe..bb4cf6f9b5d 100644 --- a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/Compiler.java +++ b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/Compiler.java @@ -46,7 +46,7 @@ public class Compiler { USECACHE // Keeps results around for reuse. Only use this is // you're sure that each compilation name maps to the // same source code - }; + } private static final AtomicInteger counter = new AtomicInteger(); private static final String targetDir = "gen-separate"; @@ -85,7 +85,7 @@ public class Compiler { } public void setFlags(Flags ... flags) { - this.flags = new HashSet(Arrays.asList(flags)); + this.flags = new HashSet<>(Arrays.asList(flags)); } public void addPostprocessor(ClassFilePreprocessor cfp) { @@ -131,17 +131,10 @@ public class Compiler { outputDirs.put(type.getName(), outDir); Class superClass = type.getSuperclass(); - if (superClass != null) { - for( Map.Entry each : compileHierarchy(superClass).entrySet()) { - outputDirs.put(each.getKey(), each.getValue()); - } - } - for (Extends ext : type.getSupertypes()) { - Type iface = ext.getType(); - for( Map.Entry each : compileHierarchy(iface).entrySet()) { - outputDirs.put(each.getKey(), each.getValue()); - } - } + if (superClass != null) + outputDirs.putAll(compileHierarchy(superClass)); + for (Extends ext : type.getSupertypes()) + outputDirs.putAll(compileHierarchy(ext.getType())); return outputDirs; } @@ -157,8 +150,12 @@ public class Compiler { SourceProcessor accum = (name, src) -> { files.add(new SourceFile(name, src)); }; - for (Type dep : type.typeDependencies()) { - dep.generateAsDependency(accum, type.methodDependencies()); + Collection deps = type.typeDependencies(type.isFullCompilation()); + for (Type dep : deps) { + if (type.isFullCompilation()) + dep.generate(accum); + else + dep.generateAsDependency(accum, type.methodDependencies()); } type.generate(accum); @@ -185,7 +182,7 @@ public class Compiler { StandardLocation.CLASS_OUTPUT, Arrays.asList(destDir)); } catch (IOException e) { throw new RuntimeException( - "IOException encountered during compilation"); + "IOException encountered during compilation", e); } Boolean result = ct.call(); if (result == Boolean.FALSE) { diff --git a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/SourceModel.java b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/SourceModel.java index 22fbff7e5c4..ec7560191f0 100644 --- a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/SourceModel.java +++ b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/SourceModel.java @@ -48,7 +48,7 @@ public class SourceModel { generate(pw); return sw.toString(); } - }; + } public static class AccessFlag extends Element { private String flag; @@ -125,6 +125,7 @@ public class SourceModel { // (and thus will be present in stubs) private Set methodDependencies; private List typeDependencies; + private boolean fullCompilation; protected Type(String name, List flags, List params, @@ -214,6 +215,14 @@ public class SourceModel { methodDependencies.add(m); } + public boolean isFullCompilation() { + return fullCompilation; + } + + public void setFullCompilation(boolean fullCompilation) { + this.fullCompilation = fullCompilation; + } + // Convenience method for creating an Extends object using this // class and specified type arguments. public Extends with(String ... args) { @@ -255,14 +264,23 @@ public class SourceModel { pw.println("}"); } - public Collection typeDependencies() { + public Collection typeDependencies(boolean recursive) { HashMap dependencies = new HashMap<>(); Type superclass = getSuperclass(); if (superclass != null) { dependencies.put(superclass.getName(), superclass); + if (recursive) { + for (Type t : superclass.typeDependencies(true)) + dependencies.put(t.getName(), t); + } } - for (Extends e : getSupertypes()) + for (Extends e : getSupertypes()) { dependencies.put(e.getType().getName(), e.getType()); + if (recursive) { + for (Type t : e.getType().typeDependencies(true)) + dependencies.put(t.getName(), t); + } + } // Do these last so that they override for (Type t : this.typeDependencies) dependencies.put(t.getName(), t); diff --git a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java index 85705bb7ed1..3f34e73eb0e 100644 --- a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java +++ b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/separate/TestHarness.java @@ -198,7 +198,7 @@ public class TestHarness { assertEquals(res, value); } } catch (InvocationTargetException | IllegalAccessException e) { - fail("Unexpected exception thrown: " + e.getCause()); + fail("Unexpected exception thrown: " + e.getCause(), e.getCause()); } } @@ -227,8 +227,7 @@ public class TestHarness { * a return type of 'int', and no arguments. */ public void assertInvokeVirtualEquals(int value, Class target) { - assertInvokeVirtualEquals( - new Integer(value), target, stdCM, "-1"); + assertInvokeVirtualEquals(value, target, stdCM, "-1"); } /** @@ -260,12 +259,31 @@ public class TestHarness { Compiler compiler = compilerLocal.get(); compiler.setFlags(compilerFlags()); - assertInvokeInterfaceEquals( - new Integer(value), target, new Extends(iface), stdAM); + assertInvokeInterfaceEquals(value, target, new Extends(iface), stdAM); compiler.cleanup(); } + protected void assertInvokeInterfaceThrows(java.lang.Class errorClass, + Class target, Extends iface, AbstractMethod method, + String... args) { + try { + assertInvokeInterfaceEquals(0, target, iface, method, args); + fail("Expected exception: " + errorClass); + } + catch (AssertionError e) { + Throwable cause = e.getCause(); + if (cause == null) + throw e; + else if ((errorClass.isAssignableFrom(cause.getClass()))) { + // this is success + return; + } + else + throw e; + } + } + /** * Creates a class which calls target::method(args) via invokevirtual, * compiles and loads both the new class and 'target', and then invokes diff --git a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java index 3b8002f4722..c7c46194ab2 100644 --- a/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java +++ b/langtools/test/tools/javac/lambdaShapes/org/openjdk/tests/vm/DefaultMethodsTest.java @@ -25,18 +25,22 @@ package org.openjdk.tests.vm; -import java.lang.reflect.*; -import java.util.*; -import java.io.File; -import java.io.IOException; - -import org.testng.annotations.Test; -import org.openjdk.tests.separate.*; import org.openjdk.tests.separate.Compiler; +import org.openjdk.tests.separate.TestHarness; +import org.testng.annotations.Test; -import static org.testng.Assert.*; -import static org.openjdk.tests.separate.SourceModel.*; +import static org.openjdk.tests.separate.SourceModel.AbstractMethod; +import static org.openjdk.tests.separate.SourceModel.AccessFlag; import static org.openjdk.tests.separate.SourceModel.Class; +import static org.openjdk.tests.separate.SourceModel.ConcreteMethod; +import static org.openjdk.tests.separate.SourceModel.DefaultMethod; +import static org.openjdk.tests.separate.SourceModel.Extends; +import static org.openjdk.tests.separate.SourceModel.Interface; +import static org.openjdk.tests.separate.SourceModel.MethodParameter; +import static org.openjdk.tests.separate.SourceModel.TypeParameter; +import static org.testng.Assert.assertEquals; +import static org.testng.Assert.assertNotNull; +import static org.testng.Assert.fail; @Test(groups = "vm") public class DefaultMethodsTest extends TestHarness { @@ -186,7 +190,7 @@ public class DefaultMethodsTest extends TestHarness { * TEST: D d = new D(); d.m() == 22; * TEST: I i = new D(); i.m() == 22; */ - void testExistingInheritedOverride() { + public void testExistingInheritedOverride() { Interface I = new Interface("I", DefaultMethod.std("99")); Class C = new Class("C", I, ConcreteMethod.std("11")); Class D = new Class("D", C, ConcreteMethod.std("22")); @@ -258,7 +262,6 @@ public class DefaultMethodsTest extends TestHarness { * TEST: C c = new C(); c.m() throws ICCE */ public void testConflict() { - // debugTest(); Interface I = new Interface("I", DefaultMethod.std("99")); Interface J = new Interface("J", DefaultMethod.std("88")); Class C = new Class("C", I, J); @@ -390,19 +393,16 @@ public class DefaultMethodsTest extends TestHarness { /** * interface I { default int m(T t) { return 99; } } - * Class C implements I { public int m() { return 88; } } + * Class C implements I { public int m(String s) { return 88; } } * - * TEST: C c = new C(); c.m() == 88; - * TEST: I i = new C(); i.m() == 88; + * TEST: C c = new C(); c.m("string") == 88; + * TEST: I i = new C(); i.m("string") == 88; */ - @Test(enabled=false) public void testSelfFill() { // This test ensures that a concrete method overrides a default method // that matches at the language-level, but has a different method // signature due to erasure. - // debugTest(); - DefaultMethod dm = new DefaultMethod( "int", "m", "return 99;", new MethodParameter("T", "t")); ConcreteMethod cm = new ConcreteMethod( @@ -415,9 +415,11 @@ public class DefaultMethodsTest extends TestHarness { AbstractMethod pm = new AbstractMethod( "int", "m", new MethodParameter("T", "t")); - assertInvokeVirtualEquals(new Integer(88), C, cm, "-1", "\"string\""); - assertInvokeInterfaceEquals( - new Integer(88), C, I.with("String"), pm, "\"string\""); + assertInvokeVirtualEquals(88, C, cm, "-1", "\"string\""); + assertInvokeInterfaceEquals(99, C, I.with("String"), pm, "\"string\""); + + C.setFullCompilation(true); // Force full bridge generation + assertInvokeInterfaceEquals(88, C, I.with("String"), pm, "\"string\""); } /** @@ -485,7 +487,6 @@ public class DefaultMethodsTest extends TestHarness { * TEST: J j = new C(); j.m("A","B","C") == 88; * TEST: K k = new C(); k.m("A","B","C") == 88; */ - @Test(enabled=false) public void testBridges() { DefaultMethod dm = new DefaultMethod("int", stdMethodName, "return 99;", new MethodParameter("T", "t"), new MethodParameter("V", "v"), @@ -518,13 +519,17 @@ public class DefaultMethodsTest extends TestHarness { J.with("String", "T"), pm2); Class C = new Class("C", K.with("String"), cm); + // First, without compiler bridges String[] args = new String[] { "\"A\"", "\"B\"", "\"C\"" }; - assertInvokeInterfaceEquals(new Integer(88), C, - I.with("String", "String", "String"), pm0, args); - assertInvokeInterfaceEquals(new Integer(88), C, - J.with("String", "String"), pm1, args); - assertInvokeInterfaceEquals(new Integer(88), C, - K.with("String"), pm2, args); + assertInvokeInterfaceEquals(99, C, I.with("String", "String", "String"), pm0, args); + assertInvokeInterfaceThrows(AbstractMethodError.class, C, J.with("String", "String"), pm1, args); + assertInvokeInterfaceThrows(AbstractMethodError.class, C, K.with("String"), pm2, args); + + // Then with compiler bridges + C.setFullCompilation(true); + assertInvokeInterfaceEquals(88, C, I.with("String", "String", "String"), pm0, args); + assertInvokeInterfaceEquals(88, C, J.with("String", "String"), pm1, args); + assertInvokeInterfaceEquals(88, C, K.with("String"), pm2, args); } /** @@ -536,8 +541,6 @@ public class DefaultMethodsTest extends TestHarness { * TEST: I i = new C(); i.m() == 88; */ public void testSuperBasic() { - // debugTest(); - Interface J = new Interface("J", DefaultMethod.std("88")); Interface I = new Interface("I", J, new DefaultMethod( "int", stdMethodName, "return J.super.m();")); @@ -559,8 +562,6 @@ public class DefaultMethodsTest extends TestHarness { * TODO: add case for K k = new C(); k.m() throws ICCE */ public void testSuperConflict() { - // debugTest(); - Interface K = new Interface("K", DefaultMethod.std("99")); Interface L = new Interface("L", DefaultMethod.std("101")); Interface J = new Interface("J", K, L); @@ -635,8 +636,7 @@ public class DefaultMethodsTest extends TestHarness { AbstractMethod pm = new AbstractMethod("int", stdMethodName, new MethodParameter("String", "s")); - assertInvokeInterfaceEquals( - new Integer(88), C, new Extends(I), pm, "\"\""); + assertInvokeInterfaceEquals(88, C, new Extends(I), pm, "\"\""); } /** @@ -674,7 +674,6 @@ public class DefaultMethodsTest extends TestHarness { * class S { Object foo() { return (new D()).m(); } // link sig: ()LInteger; * TEST: S s = new S(); s.foo() == new Integer(99) */ - @Test(enabled=false) public void testCovarBridge() { Interface I = new Interface("I", new DefaultMethod( "Integer", "m", "return new Integer(88);")); @@ -692,7 +691,8 @@ public class DefaultMethodsTest extends TestHarness { S.addCompilationDependency(Dstub); S.addCompilationDependency(DstubMethod); - assertInvokeVirtualEquals(new Integer(99), S, toCall, "null"); + // NEGATIVE test for separate compilation -- dispatches to I, not C + assertInvokeVirtualEquals(88, S, toCall, "null"); } /** @@ -719,7 +719,7 @@ public class DefaultMethodsTest extends TestHarness { S.addCompilationDependency(Dstub); S.addCompilationDependency(DstubMethod); - assertInvokeVirtualEquals(new Integer(88), S, toCall, "null"); + assertInvokeVirtualEquals(88, S, toCall, "null"); } /** @@ -757,7 +757,6 @@ public class DefaultMethodsTest extends TestHarness { * Test that a erased-signature-matching method does not implement * non-language-level matching methods */ - @Test(enabled=false) public void testNonConcreteFill() { AbstractMethod ipm = new AbstractMethod("int", "m", new MethodParameter("T", "t"), @@ -781,13 +780,14 @@ public class DefaultMethodsTest extends TestHarness { new MethodParameter("T", "t"), new MethodParameter("String", "s"), new MethodParameter("String", "w")); + DefaultMethod kdm = new DefaultMethod("int", "m", "return 99;", + new MethodParameter("T", "t"), + new MethodParameter("String", "v"), + new MethodParameter("String", "w")); Interface K = new Interface("K", new TypeParameter("T"), J.with("T", "String"), - new DefaultMethod("int", "m", "return 99;", - new MethodParameter("T", "t"), - new MethodParameter("String", "v"), - new MethodParameter("String", "w"))); + kdm); Class C = new Class("C", K.with("String"), @@ -797,13 +797,18 @@ public class DefaultMethodsTest extends TestHarness { new MethodParameter("Object", "v"), new MethodParameter("String", "w"))); + // First, without compiler bridges String a = "\"\""; - assertInvokeInterfaceEquals(99, C, - K.with("String"), kpm, a, a, a); - assertInvokeInterfaceEquals(77, C, - J.with("String", "String"), jpm, a, a, a); - assertInvokeInterfaceEquals(99, C, - I.with("String", "String", "String"), ipm, a, a, a); + assertInvokeInterfaceEquals(99, C, K.with("String"), kpm, a, a, a); + assertInvokeInterfaceEquals(77, C, J.with("String", "String"), jpm, a, a, a); + assertInvokeInterfaceThrows(AbstractMethodError.class, C, I.with("String", "String", "String"), ipm, a, a, a); + + // Now, with bridges + J.setFullCompilation(true); + K.setFullCompilation(true); + assertInvokeInterfaceEquals(99, C, K.with("String"), kpm, a, a, a); + assertInvokeInterfaceEquals(77, C, J.with("String", "String"), jpm, a, a, a); + assertInvokeInterfaceEquals(99, C, I.with("String", "String", "String"), ipm, a, a, a); } public void testStrictfpDefault() {