From fbe28e4ee1f1ff7fb617c2e1f96c04f4b371fa2b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Tue, 27 Jul 2021 19:28:54 +0000 Subject: [PATCH] 8270866: NPE in DocTreePath.getTreePath() Reviewed-by: jjg --- .../formats/html/AbstractMemberWriter.java | 4 +- .../formats/html/HtmlDocletWriter.java | 18 +---- .../formats/html/SubWriterHolderWriter.java | 10 +-- .../toolkit/taglets/InheritDocTaglet.java | 1 - .../doclets/toolkit/taglets/ParamTaglet.java | 4 +- .../doclets/toolkit/taglets/ReturnTaglet.java | 4 +- .../doclets/toolkit/util/CommentHelper.java | 38 ++++------ .../doclet/testInherited/TestInherited.java | 71 ++++++++++++++++++- 8 files changed, 90 insertions(+), 60 deletions(-) diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractMemberWriter.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractMemberWriter.java index 285e43cc6aa..f717cdf2cf4 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractMemberWriter.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractMemberWriter.java @@ -335,7 +335,7 @@ public abstract class AbstractMemberWriter implements MemberSummaryWriter, Membe : HtmlLinkInfo.Kind.MEMBER, te, element, typeContent); Content desc = new ContentBuilder(); - writer.addSummaryLinkComment(this, element, desc); + writer.addSummaryLinkComment(element, desc); useTable.addRow(summaryType, typeContent, desc); } contentTree.add(useTable); @@ -363,7 +363,7 @@ public abstract class AbstractMemberWriter implements MemberSummaryWriter, Membe addSummaryLink(tElement, member, summaryLink); rowContents.add(summaryLink); Content desc = new ContentBuilder(); - writer.addSummaryLinkComment(this, member, firstSentenceTrees, desc); + writer.addSummaryLinkComment(member, firstSentenceTrees, desc); rowContents.add(desc); table.addRow(member, rowContents); } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java index c95b735b709..508d62a04d8 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java @@ -1162,7 +1162,7 @@ public class HtmlDocletWriter { public void addInlineComment(Element element, DocTree tag, Content htmltree) { CommentHelper ch = utils.getCommentHelper(element); List description = ch.getDescription(tag); - addCommentTags(element, tag, description, false, false, false, htmltree); + addCommentTags(element, description, false, false, false, htmltree); } /** @@ -1249,22 +1249,6 @@ public class HtmlDocletWriter { */ private void addCommentTags(Element element, List tags, boolean depr, boolean first, boolean inSummary, Content htmltree) { - addCommentTags(element, null, tags, depr, first, inSummary, htmltree); - } - - /** - * Adds the comment tags. - * - * @param element for which the comment tags will be generated - * @param holderTag the block tag context for the inline tags - * @param tags the first sentence tags for the doc - * @param depr true if it is deprecated - * @param first true if the first sentence tags should be added - * @param inSummary true if the comment tags are added into the summary section - * @param htmltree the documentation tree to which the comment tags will be added - */ - private void addCommentTags(Element element, DocTree holderTag, List tags, boolean depr, - boolean first, boolean inSummary, Content htmltree) { if (options.noComment()) { return; } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/SubWriterHolderWriter.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/SubWriterHolderWriter.java index 5fe0a1edd43..5f09e20fd74 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/SubWriterHolderWriter.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/SubWriterHolderWriter.java @@ -38,7 +38,6 @@ import jdk.javadoc.internal.doclets.formats.html.markup.HtmlId; import jdk.javadoc.internal.doclets.formats.html.markup.HtmlStyle; import jdk.javadoc.internal.doclets.formats.html.markup.TagName; import jdk.javadoc.internal.doclets.formats.html.markup.HtmlTree; -import jdk.javadoc.internal.doclets.formats.html.markup.RawHtml; import jdk.javadoc.internal.doclets.toolkit.Content; import jdk.javadoc.internal.doclets.toolkit.util.DocPath; @@ -136,25 +135,22 @@ public abstract class SubWriterHolderWriter extends HtmlDocletWriter { /** * Add the summary link for the member. * - * @param mw the writer for the member being documented * @param member the member to be documented * @param contentTree the content tree to which the link will be added */ - public void addSummaryLinkComment(AbstractMemberWriter mw, Element member, Content contentTree) { + public void addSummaryLinkComment(Element member, Content contentTree) { List tags = utils.getFirstSentenceTrees(member); - addSummaryLinkComment(mw, member, tags, contentTree); + addSummaryLinkComment(member, tags, contentTree); } /** * Add the summary link comment. * - * @param mw the writer for the member being documented * @param member the member being documented * @param firstSentenceTags the first sentence tags for the member to be documented * @param tdSummary the content tree to which the comment will be added */ - public void addSummaryLinkComment(AbstractMemberWriter mw, - Element member, List firstSentenceTags, Content tdSummary) { + public void addSummaryLinkComment(Element member, List firstSentenceTags, Content tdSummary) { addIndexComment(member, firstSentenceTags, tdSummary); } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/InheritDocTaglet.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/InheritDocTaglet.java index 730de373b65..3e02a52da55 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/InheritDocTaglet.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/InheritDocTaglet.java @@ -97,7 +97,6 @@ public class InheritDocTaglet extends BaseTaglet { if (!inheritedDoc.inlineTags.isEmpty()) { replacement = writer.commentTagsToOutput(inheritedDoc.holder, inheritedDoc.holderTag, inheritedDoc.inlineTags, isFirstSentence); - ch.setOverrideElement(inheritedDoc.holder); } } else { diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ParamTaglet.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ParamTaglet.java index b0a980397b5..57303ffaa1d 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ParamTaglet.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ParamTaglet.java @@ -206,9 +206,7 @@ public class ParamTaglet extends BaseTaglet implements InheritableTaglet { String lname = kind != ParamKind.TYPE_PARAMETER ? utils.getSimpleName(e) : utils.getTypeName(e.asType(), false); - CommentHelper ch = utils.getCommentHelper(holder); - ch.setOverrideElement(inheritedDoc.holder); - Content content = processParamTag(holder, kind, writer, + Content content = processParamTag(inheritedDoc.holder, kind, writer, (ParamTree) inheritedDoc.holderTag, lname, alreadyDocumented.isEmpty()); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ReturnTaglet.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ReturnTaglet.java index b44f803c712..2f1e6d9827f 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ReturnTaglet.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/ReturnTaglet.java @@ -121,9 +121,7 @@ public class ReturnTaglet extends BaseTaglet implements InheritableTaglet { Input input = new DocFinder.Input(utils, holder, this); DocFinder.Output inheritedDoc = DocFinder.search(writer.configuration(), input); if (inheritedDoc.holderTag != null) { - CommentHelper ch = utils.getCommentHelper(input.element); - ch.setOverrideElement(inheritedDoc.holder); - return writer.returnTagOutput(holder, (ReturnTree) inheritedDoc.holderTag, false); + return writer.returnTagOutput(inheritedDoc.holder, (ReturnTree) inheritedDoc.holderTag, false); } return null; } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java index 65bcdce9181..03193f4de9d 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java @@ -165,26 +165,8 @@ public class CommentHelper { } return configuration.docEnv.getTypeUtils().asElement(symbol); } - // case A: the element contains no comments associated and - // the comments need to be copied from ancestor - // case B: the element has @inheritDoc, then the ancestral comment - // as appropriate has to be copied over. - - // Case A. - if (dcTree == null && overriddenElement != null) { - CommentHelper ovch = utils.getCommentHelper(overriddenElement); - return ovch.getElement(rtree); - } - if (dcTree == null) { - return null; - } - DocTreePath docTreePath = DocTreePath.getPath(path, dcTree, rtree); + DocTreePath docTreePath = getDocTreePath(rtree); if (docTreePath == null) { - // Case B. - if (overriddenElement != null) { - CommentHelper ovch = utils.getCommentHelper(overriddenElement); - return ovch.getElement(rtree); - } return null; } DocTrees doctrees = configuration.docEnv.getDocTrees(); @@ -192,10 +174,7 @@ public class CommentHelper { } public TypeMirror getType(ReferenceTree rtree) { - // Workaround for JDK-8269706 - if (path == null || dcTree == null || rtree == null) - return null; - DocTreePath docTreePath = DocTreePath.getPath(path, dcTree, rtree); + DocTreePath docTreePath = getDocTreePath(rtree); if (docTreePath != null) { DocTrees doctrees = configuration.docEnv.getDocTrees(); return doctrees.getType(docTreePath); @@ -700,9 +679,18 @@ public class CommentHelper { } public DocTreePath getDocTreePath(DocTree dtree) { - if (path == null || dcTree == null || dtree == null) + if (dcTree == null && overriddenElement != null) { + // This is an inherited comment, return path from ancestor. + return configuration.utils.getCommentHelper(overriddenElement).getDocTreePath(dtree); + } else if (path == null || dcTree == null || dtree == null) { return null; - return DocTreePath.getPath(path, dcTree, dtree); + } + DocTreePath dtPath = DocTreePath.getPath(path, dcTree, dtree); + if (dtPath == null && overriddenElement != null) { + // The overriding element has a doc tree, but it doesn't contain what we're looking for. + return configuration.utils.getCommentHelper(overriddenElement).getDocTreePath(dtree); + } + return dtPath; } public Element getOverriddenElement() { diff --git a/test/langtools/jdk/javadoc/doclet/testInherited/TestInherited.java b/test/langtools/jdk/javadoc/doclet/testInherited/TestInherited.java index 21f34669097..21391201a03 100644 --- a/test/langtools/jdk/javadoc/doclet/testInherited/TestInherited.java +++ b/test/langtools/jdk/javadoc/doclet/testInherited/TestInherited.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8269722 + * @bug 8269722 8270866 * @summary NPE in HtmlDocletWriter, reporting errors on inherited tags * @library /tools/lib ../../lib * @modules jdk.javadoc/jdk.javadoc.internal.tool @@ -67,6 +67,14 @@ public class TestInherited extends JavadocTester { "-Xdoclint:-missing", "-XDdoe", src.resolve("BadParam.java").toString()); checkExit(Exit.OK); + checkOutput("BadParam.Base.html", true, """ +
Parameters:
+
i - a < b
+ """); + checkOutput("BadParam.Sub.html", true, """ +
Parameters:
+
i - a < b
+ """); } @Test @@ -91,5 +99,64 @@ public class TestInherited extends JavadocTester { "-Xdoclint:-missing", src.resolve("BadReturn.java").toString()); checkExit(Exit.OK); + checkOutput("BadReturn.Base.html", true, """ +
Returns:
+
a < b
+ """); + checkOutput("BadReturn.Sub.html", true, """ +
Returns:
+
a < b
+ """); } -} \ No newline at end of file + + @Test + public void testBadInheritedReference(Path base) throws Exception { + Path src = base.resolve("src"); + tb.writeJavaFiles(src, """ + public class BadReference { + public interface Intf { + /** + * {@link NonExistingClass} + */ + public void m(); + } + + public static class Impl1 implements Intf { + public void m() { } + } + + public static class Impl2 implements Intf { + /** + * {@inheritDoc} + */ + public void m() { } + } + + // subclass has doc comment but inherits main description + public static class Impl3 implements Intf { + /** + * @since 1 + */ + public void m() { } + } + } + """); + + javadoc("-d", base.resolve("out").toString(), + "-Xdoclint:-reference", + src.resolve("BadReference.java").toString()); + checkExit(Exit.OK); + checkOutput("BadReference.Intf.html", true, """ +
NonExistingClass
+ """); + checkOutput("BadReference.Impl1.html", true, """ +
NonExistingClass
+ """); + checkOutput("BadReference.Impl2.html", true, """ +
NonExistingClass
+ """); + checkOutput("BadReference.Impl3.html", true, """ +
NonExistingClass
+ """); + } +}