From 049532b2d38e08ab1e9abacbf96d78857229542d Mon Sep 17 00:00:00 2001 From: Jonathan Gibbons Date: Mon, 20 May 2019 18:18:42 -0700 Subject: [PATCH] 8224166: Create a taglet to better handle @jls and @jvms tags Reviewed-by: erikj, darcy --- make/Docs.gmk | 7 +- .../src/classes/build/tools/taglet/JSpec.java | 192 ++++++++++++++++++ 2 files changed, 196 insertions(+), 3 deletions(-) create mode 100644 make/jdk/src/classes/build/tools/taglet/JSpec.java diff --git a/make/Docs.gmk b/make/Docs.gmk index 497edcaee77..22c6d3cbe9f 100644 --- a/make/Docs.gmk +++ b/make/Docs.gmk @@ -85,14 +85,14 @@ JAVADOC_TAGS := \ -tag param \ -tag return \ -tag throws \ + -taglet build.tools.taglet.JSpec\$$JLS \ + -taglet build.tools.taglet.JSpec\$$JVMS \ -taglet build.tools.taglet.ModuleGraph \ -taglet build.tools.taglet.ToolGuide \ -tag since \ -tag serialData \ -tag factory \ -tag see \ - -tag 'jvms:a:See The Java™ Virtual Machine Specification:' \ - -tag 'jls:a:See The Java™ Language Specification:' \ -taglet build.tools.taglet.ExtLink \ -taglet build.tools.taglet.Incubating \ -tagletpath $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes \ @@ -276,7 +276,8 @@ define SetupApiDocsGenerationBody $1_INDIRECT_EXPORTS := $$(call FindTransitiveIndirectDepsForModules, $$($1_MODULES)) $1_ALL_MODULES := $$(sort $$($1_MODULES) $$($1_INDIRECT_EXPORTS)) - $1_JAVA_ARGS := -Dextlink.spec.version=$$(VERSION_SPECIFICATION) + $1_JAVA_ARGS := -Dextlink.spec.version=$$(VERSION_SPECIFICATION) \ + -Djspec.version=$$(VERSION_SPECIFICATION) ifeq ($$(ENABLE_FULL_DOCS), true) # Tell the ModuleGraph taglet to generate html links to soon-to-be-created diff --git a/make/jdk/src/classes/build/tools/taglet/JSpec.java b/make/jdk/src/classes/build/tools/taglet/JSpec.java new file mode 100644 index 00000000000..f7f770cc09c --- /dev/null +++ b/make/jdk/src/classes/build/tools/taglet/JSpec.java @@ -0,0 +1,192 @@ +/* + * Copyright (c) 2017, 2019, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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 build.tools.taglet; + +import java.util.EnumSet; +import java.util.List; +import java.util.Set; +import java.util.regex.Matcher; +import java.util.regex.Pattern; +import javax.lang.model.element.Element; + +import com.sun.source.doctree.DocTree; +import com.sun.source.doctree.LiteralTree; +import com.sun.source.doctree.UnknownBlockTagTree; +import com.sun.source.util.SimpleDocTreeVisitor; +import jdk.javadoc.doclet.Taglet; + +import static com.sun.source.doctree.DocTree.Kind.*; + +/** + * A base class for block tags to insert a link to an external copy of JLS or JVMS. + * The tags can be used as follows: + * + *
+ * @jls section-number description
+ * 
+ * + * For example: + * + *
+ * @jls 3.4 Line Terminators
+ * 
+ * + * will produce the following HTML for a docs build configured for Java SE 12. + * + *
{@code
+ * 
See Java Language Specification: + *
3.4 Line terminators + * }
+ * + * The version of the spec must be set in the jspec.version system property. + */ +public class JSpec implements Taglet { + static final String SPEC_VERSION; + + static { + SPEC_VERSION = System.getProperty("jspec.version"); + if (SPEC_VERSION == null) { + throw new RuntimeException("jspec.version property not set"); + } + } + + public static class JLS extends JSpec { + public JLS() { + super("jls", + "Java Language Specification", + "https://docs.oracle.com/javase/specs/jls/se" + SPEC_VERSION + "/html", + "jls"); + } + } + + public static class JVMS extends JSpec { + public JVMS() { + super("jvms", + "Java Virtual Machine Specification", + "https://docs.oracle.com/javase/specs/jvms/se" + SPEC_VERSION + "/html", + "jvms"); + } + } + + private String tagName; + private String specTitle; + private String baseURL; + private String idPrefix; + + protected JSpec(String tagName, String specTitle, String baseURL, String idPrefix) { + this.tagName = tagName; + this.specTitle = specTitle; + this.baseURL = baseURL; + this.idPrefix = idPrefix; + } + + + static final Pattern TAG_PATTERN = Pattern.compile("(?s)(.+ )?(?[1-9][0-9]*)(?
[0-9.]*)( .*)?$"); + + + /** + * Returns the set of locations in which the tag may be used. + */ + @Override + public Set getAllowedLocations() { + return EnumSet.allOf(jdk.javadoc.doclet.Taglet.Location.class); + } + + @Override + public boolean isInlineTag() { + return false; + } + + @Override + public String getName() { + return tagName; + } + + @Override + public String toString(List tags, Element elem) { + + if (tags.isEmpty()) + return ""; + + StringBuilder sb = new StringBuilder(); + sb.append("
See " + specTitle + ":
\n") + .append("
\n"); + + for (DocTree tag : tags) { + + if (tag.getKind() != UNKNOWN_BLOCK_TAG) { + continue; + } + + UnknownBlockTagTree blockTag = (UnknownBlockTagTree)tag; + String tagText = blockTag.getContent().toString().trim(); + Matcher m = TAG_PATTERN.matcher(tagText); + if (m.find()) { + String chapter = m.group("chapter"); + String section = m.group("section"); + + String url = String.format("%1$s/%2$s-%3$s.html#jls-%3$s%4$s", + baseURL, idPrefix, chapter, section); + + + sb.append("") + .append(expand(blockTag)) + .append("
"); + } + + } + + sb.append("
"); + + return sb.toString(); + } + + private String expand(UnknownBlockTagTree tree) { + StringBuilder sb = new StringBuilder(); + return (new SimpleDocTreeVisitor() { + public StringBuilder defaultAction(DocTree tree, StringBuilder sb) { + return sb.append(tree.toString()); + } + + public StringBuilder visitLiteral(LiteralTree tree, StringBuilder sb) { + if (tree.getKind() == CODE) { + sb.append(""); + } + sb.append(escape(tree.getBody().toString())); + if (tree.getKind() == CODE) { + sb.append(""); + } + return sb; + } + + private String escape(String s) { + return s.replace("&", "&").replace("<", "<").replace(">", ">"); + } + }).visit(tree.getContent(), new StringBuilder()).toString(); + } +}