8331051: Add an @since checker test for java.base module

Reviewed-by: jlahoda, jjg
This commit is contained in:
Nizar Benalla 2024-10-16 10:17:47 +00:00
parent ebc17c7c8d
commit c81aa7551c
3 changed files with 989 additions and 2 deletions

View File

@ -1,4 +1,4 @@
# Copyright (c) 2013, 2023, Oracle and/or its affiliates. All rights reserved.
# Copyright (c) 2013, 2024, 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
@ -91,7 +91,8 @@ tier3 = \
:jdk_svc \
-:jdk_svc_sanity \
-:svc_tools \
:jdk_jpackage
:jdk_jpackage \
:jdk_since_checks
# Everything not in other tiers
tier4 = \
@ -666,3 +667,7 @@ jdk_containers_extended = \
jdk_core_no_security = \
:jdk_core \
-:jdk_security
# Set of tests for `@since` checks in source code documentation
jdk_since_checks = \
tools/sincechecker/modules/java_base/CheckSince_javaBase.java

View File

@ -0,0 +1,948 @@
/*
* Copyright (c) 2024, 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.File;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.lang.Runtime.Version;
import java.net.URI;
import java.nio.file.*;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import javax.lang.model.element.*;
import javax.lang.model.util.ElementFilter;
import javax.lang.model.util.Elements;
import javax.lang.model.util.Types;
import javax.tools.*;
import javax.tools.JavaFileManager.Location;
import com.sun.source.tree.*;
import com.sun.source.util.JavacTask;
import com.sun.source.util.TreePathScanner;
import com.sun.source.util.Trees;
import com.sun.tools.javac.api.JavacTaskImpl;
import com.sun.tools.javac.code.Flags;
import com.sun.tools.javac.code.Symbol;
import com.sun.tools.javac.util.Pair;
import jtreg.SkippedException;
/*
This checker checks the values of the `@since` tag found in the documentation comment for an element against
the release in which the element first appeared.
The source code containing the documentation comments is read from `src.zip` in the release of JDK used to run the test.
The releases used to determine the expected value of `@since` tags are taken from the historical data built into `javac`.
The `@since` checker works as a two-step process:
In the first step, we process JDKs 9-current, only classfiles,
producing a map `<unique-Element-ID`> => `<version(s)-where-it-was-introduced>`.
- "version(s)", because we handle versioning of Preview API, so there may be two versions
(we use a class with two fields for preview and stable),
one when it was introduced as a preview, and one when it went out of preview. More on that below.
- For each Element, we compute the unique ID, look into the map, and if there's nothing,
record the current version as the originating version.
- At the end of this step we have a map of the Real since values
In the second step, we look at "effective" `@since` tags in the mainline sources, from `src.zip`
(if the test run doesn't have it, we throw a `jtreg.SkippedException`)
- We only check the specific MODULE whose name was passed as an argument in the test.
In that module, we look for unqualified exports and test those packages.
- The `@since` checker verifies that for every API element, the real since value and
the effective since value are the same, and reports an error if they are not.
Important note : We only check code written since JDK 9 as the releases used to determine the expected value
of @since tags are taken from the historical data built into javac which only goes back that far
note on rules for Real and effective `@since:
Real since value of an API element is computed as the oldest release in which the given API element was introduced.
That is:
- for modules, packages, classes and interfaces, the release in which the element with the given qualified name was introduced
- for constructors, the release in which the constructor with the given VM descriptor was introduced
- for methods and fields, the release in which the given method or field with the given VM descriptor became a member
of its enclosing class or interface, whether direct or inherited
Effective since value of an API element is computed as follows:
- if the given element has a @since tag in its javadoc, it is used
- in all other cases, return the effective since value of the enclosing element
Special Handling for preview method, as per JEP 12:
- When an element is still marked as preview, the `@since` should be the first JDK release where the element was added.
- If the element is no longer marked as preview, the `@since` should be the first JDK release where it was no longer preview.
note on legacy preview: Until JDK 14, the preview APIs were not marked in any machine-understandable way.
It was deprecated, and had a comment in the javadoc.
and the use of `@PreviewFeature` only became standard in JDK 17.
So the checker has an explicit knowledge of these preview elements.
note: The `<unique-Element-ID>` for methods looks like
`method: <erased-return-descriptor> <binary-name-of-enclosing-class>.<method-name>(<ParameterDescriptor>)`.
it is somewhat inspired from the VM Method Descriptors. But we use the erased return so that methods
that were later generified remain the same.
usage: the checker is run from a module specific test
`@run main SinceChecker <moduleName> [--exclude package1,package2 | --exclude package1 package2]`
*/
public class SinceChecker {
private final Map<String, Set<String>> LEGACY_PREVIEW_METHODS = new HashMap<>();
private final Map<String, IntroducedIn> classDictionary = new HashMap<>();
private final JavaCompiler tool;
private int errorCount = 0;
// packages to skip during the test
private static final Set<String> EXCLUDE_LIST = new HashSet<>();
public static class IntroducedIn {
public String introducedPreview;
public String introducedStable;
}
public static void main(String[] args) throws Exception {
if (args.length == 0) {
throw new IllegalArgumentException("Test module not specified");
}
String moduleName = args[0];
boolean excludeFlag = false;
for (int i = 1; i < args.length; i++) {
if ("--exclude".equals(args[i])) {
excludeFlag = true;
continue;
}
if (excludeFlag) {
if (args[i].contains(",")) {
EXCLUDE_LIST.addAll(Arrays.asList(args[i].split(",")));
} else {
EXCLUDE_LIST.add(args[i]);
}
}
}
SinceChecker sinceCheckerTestHelper = new SinceChecker(moduleName);
sinceCheckerTestHelper.checkModule(moduleName);
}
private void error(String message) {
System.err.println(message);
errorCount++;
}
private SinceChecker(String moduleName) throws IOException {
tool = ToolProvider.getSystemJavaCompiler();
for (int i = 9; i <= Runtime.version().feature(); i++) {
DiagnosticListener<? super JavaFileObject> noErrors = d -> {
if (!d.getCode().equals("compiler.err.module.not.found")) {
error(d.getMessage(null));
}
};
JavacTask ct = (JavacTask) tool.getTask(null,
null,
noErrors,
List.of("--add-modules", moduleName, "--release", String.valueOf(i)),
null,
Collections.singletonList(SimpleJavaFileObject.forSource(URI.create("myfo:/Test.java"), "")));
ct.analyze();
String version = String.valueOf(i);
Elements elements = ct.getElements();
elements.getModuleElement("java.base"); // forces module graph to be instantiated
elements.getAllModuleElements().forEach(me ->
processModuleElement(me, version, ct));
}
}
private void processModuleElement(ModuleElement moduleElement, String releaseVersion, JavacTask ct) {
processElement(moduleElement, moduleElement, ct.getTypes(), releaseVersion);
for (ModuleElement.ExportsDirective ed : ElementFilter.exportsIn(moduleElement.getDirectives())) {
if (ed.getTargetModules() == null) {
processPackageElement(ed.getPackage(), releaseVersion, ct);
}
}
}
private void processPackageElement(PackageElement pe, String releaseVersion, JavacTask ct) {
processElement(pe, pe, ct.getTypes(), releaseVersion);
List<TypeElement> typeElements = ElementFilter.typesIn(pe.getEnclosedElements());
for (TypeElement te : typeElements) {
processClassElement(te, releaseVersion, ct.getTypes(), ct.getElements());
}
}
/// JDK documentation only contains public and protected declarations
private boolean isDocumented(Element te) {
Set<Modifier> mod = te.getModifiers();
return mod.contains(Modifier.PUBLIC) || mod.contains(Modifier.PROTECTED);
}
private boolean isMember(Element e) {
var kind = e.getKind();
return kind.isField() || switch (kind) {
case METHOD, CONSTRUCTOR -> true;
default -> false;
};
}
private void processClassElement(TypeElement te, String version, Types types, Elements elements) {
if (!isDocumented(te)) {
return;
}
processElement(te.getEnclosingElement(), te, types, version);
elements.getAllMembers(te).stream()
.filter(this::isDocumented)
.filter(this::isMember)
.forEach(element -> processElement(te, element, types, version));
te.getEnclosedElements().stream()
.filter(element -> element.getKind().isDeclaredType())
.map(TypeElement.class::cast)
.forEach(nestedClass -> processClassElement(nestedClass, version, types, elements));
}
private void processElement(Element explicitOwner, Element element, Types types, String version) {
String uniqueId = getElementName(explicitOwner, element, types);
IntroducedIn introduced = classDictionary.computeIfAbsent(uniqueId, _ -> new IntroducedIn());
if (isPreview(element, uniqueId, version)) {
if (introduced.introducedPreview == null) {
introduced.introducedPreview = version;
}
} else {
if (introduced.introducedStable == null) {
introduced.introducedStable = version;
}
}
}
private boolean isPreview(Element el, String uniqueId, String currentVersion) {
while (el != null) {
Symbol s = (Symbol) el;
if ((s.flags() & Flags.PREVIEW_API) != 0) {
return true;
}
el = el.getEnclosingElement();
}
return LEGACY_PREVIEW_METHODS.getOrDefault(currentVersion, Set.of())
.contains(uniqueId);
}
private void checkModule(String moduleName) throws Exception {
Path home = Paths.get(System.getProperty("java.home"));
Path srcZip = home.resolve("lib").resolve("src.zip");
if (Files.notExists(srcZip)) {
//possibly running over an exploded JDK build, attempt to find a
//co-located full JDK image with src.zip:
Path testJdk = Paths.get(System.getProperty("test.jdk"));
srcZip = testJdk.getParent().resolve("images").resolve("jdk").resolve("lib").resolve("src.zip");
}
if (!Files.isReadable(srcZip)) {
throw new SkippedException("Skipping Test because src.zip wasn't found or couldn't be read");
}
URI uri = URI.create("jar:" + srcZip.toUri());
try (FileSystem zipFO = FileSystems.newFileSystem(uri, Collections.emptyMap())) {
Path root = zipFO.getRootDirectories().iterator().next();
Path moduleDirectory = root.resolve(moduleName);
try (StandardJavaFileManager fm =
tool.getStandardFileManager(null, null, null)) {
JavacTask ct = (JavacTask) tool.getTask(null,
fm,
null,
List.of("--add-modules", moduleName, "-d", "."),
null,
Collections.singletonList(SimpleJavaFileObject.forSource(URI.create("myfo:/Test.java"), "")));
ct.analyze();
Elements elements = ct.getElements();
elements.getModuleElement("java.base");
try (EffectiveSourceSinceHelper javadocHelper = EffectiveSourceSinceHelper.create(ct, List.of(root), this)) {
processModuleCheck(elements.getModuleElement(moduleName), ct, moduleDirectory, javadocHelper);
} catch (Exception e) {
e.printStackTrace();
error("Initiating javadocHelper Failed " + e);
}
if (errorCount > 0) {
throw new Exception("The `@since` checker found " + errorCount + " problems");
}
}
}
}
private boolean isExcluded(ModuleElement.ExportsDirective ed ){
return EXCLUDE_LIST.stream().anyMatch(excludePackage ->
ed.getPackage().toString().equals(excludePackage) ||
ed.getPackage().toString().startsWith(excludePackage + "."));
}
private void processModuleCheck(ModuleElement moduleElement, JavacTask ct, Path moduleDirectory, EffectiveSourceSinceHelper javadocHelper) {
if (moduleElement == null) {
error("Module element: was null because `elements.getModuleElement(moduleName)` returns null." +
"fixes are needed for this Module");
}
String moduleVersion = getModuleVersionFromFile(moduleDirectory);
checkModuleOrPackage(javadocHelper, moduleVersion, moduleElement, ct, "Module: ");
for (ModuleElement.ExportsDirective ed : ElementFilter.exportsIn(moduleElement.getDirectives())) {
if (ed.getTargetModules() == null) {
String packageVersion = getPackageVersionFromFile(moduleDirectory, ed);
if (packageVersion != null && !isExcluded(ed)) {
checkModuleOrPackage(javadocHelper, packageVersion, ed.getPackage(), ct, "Package: ");
analyzePackageCheck(ed.getPackage(), ct, javadocHelper);
} // Skip the package if packageVersion is null
}
}
}
private void checkModuleOrPackage(EffectiveSourceSinceHelper javadocHelper, String moduleVersion, Element moduleElement, JavacTask ct, String elementCategory) {
String id = getElementName(moduleElement, moduleElement, ct.getTypes());
var elementInfo = classDictionary.get(id);
if (elementInfo == null) {
error("Element :" + id + " was not mapped");
return;
}
String version = elementInfo.introducedStable;
if (moduleVersion == null) {
error("Unable to retrieve `@since` for " + elementCategory + id);
} else {
String position = javadocHelper.getElementPosition(id);
checkEquals(position, moduleVersion, version, id);
}
}
private String getModuleVersionFromFile(Path moduleDirectory) {
Path moduleInfoFile = moduleDirectory.resolve("module-info.java");
String version = null;
if (Files.exists(moduleInfoFile)) {
try {
String moduleInfoContent = Files.readString(moduleInfoFile);
var extractedVersion = extractSinceVersionFromText(moduleInfoContent);
if (extractedVersion != null) {
version = extractedVersion.toString();
}
} catch (IOException e) {
error("module-info.java not found or couldn't be opened AND this module has no unqualified exports");
}
}
return version;
}
private String getPackageVersionFromFile(Path moduleDirectory, ModuleElement.ExportsDirective ed) {
Path pkgInfo = moduleDirectory.resolve(ed.getPackage()
.getQualifiedName()
.toString()
.replace(".", File.separator)
)
.resolve("package-info.java");
if (!Files.exists(pkgInfo)) {
return null; // Skip if the file does not exist
}
String packageTopVersion = null;
try {
String packageContent = Files.readString(pkgInfo);
var extractedVersion = extractSinceVersionFromText(packageContent);
if (extractedVersion != null) {
packageTopVersion = extractedVersion.toString();
} else {
error(ed.getPackage().getQualifiedName() + ": package-info.java exists but doesn't contain @since");
}
} catch (IOException e) {
error(ed.getPackage().getQualifiedName() + ": package-info.java couldn't be opened");
}
return packageTopVersion;
}
private void analyzePackageCheck(PackageElement pe, JavacTask ct, EffectiveSourceSinceHelper javadocHelper) {
List<TypeElement> typeElements = ElementFilter.typesIn(pe.getEnclosedElements());
for (TypeElement te : typeElements) {
analyzeClassCheck(te, null, javadocHelper, ct.getTypes(), ct.getElements());
}
}
private boolean isNotCommonRecordMethod(TypeElement te, Element element, Types types) {
var isRecord = te.getKind() == ElementKind.RECORD;
if (!isRecord) {
return true;
}
String uniqueId = getElementName(te, element, types);
boolean isCommonMethod = uniqueId.endsWith(".toString()") ||
uniqueId.endsWith(".hashCode()") ||
uniqueId.endsWith(".equals(java.lang.Object)");
if (isCommonMethod) {
return false;
}
for (var parameter : te.getEnclosedElements()) {
if (parameter.getKind() == ElementKind.RECORD_COMPONENT) {
if (uniqueId.endsWith(String.format("%s.%s()", te.getSimpleName(), parameter.getSimpleName().toString()))) {
return false;
}
}
}
return true;
}
private void analyzeClassCheck(TypeElement te, String version, EffectiveSourceSinceHelper javadocHelper,
Types types, Elements elementUtils) {
String currentjdkVersion = String.valueOf(Runtime.version().feature());
if (!isDocumented(te)) {
return;
}
checkElement(te.getEnclosingElement(), te, types, javadocHelper, version, elementUtils);
te.getEnclosedElements().stream().filter(this::isDocumented)
.filter(this::isMember)
.filter(element -> isNotCommonRecordMethod(te, element, types))
.forEach(element -> checkElement(te, element, types, javadocHelper, version, elementUtils));
te.getEnclosedElements().stream()
.filter(element -> element.getKind().isDeclaredType())
.map(TypeElement.class::cast)
.forEach(nestedClass -> analyzeClassCheck(nestedClass, currentjdkVersion, javadocHelper, types, elementUtils));
}
private void checkElement(Element explicitOwner, Element element, Types types,
EffectiveSourceSinceHelper javadocHelper, String currentVersion, Elements elementUtils) {
String uniqueId = getElementName(explicitOwner, element, types);
if (element.getKind() == ElementKind.METHOD &&
element.getEnclosingElement().getKind() == ElementKind.ENUM &&
(uniqueId.contains(".values()") || uniqueId.contains(".valueOf(java.lang.String)"))) {
//mandated enum type methods
return;
}
String sinceVersion = null;
var effectiveSince = javadocHelper.effectiveSinceVersion(explicitOwner, element, types, elementUtils);
if (effectiveSince == null) {
// Skip the element if the java file doesn't exist in src.zip
return;
}
sinceVersion = effectiveSince.toString();
IntroducedIn mappedVersion = classDictionary.get(uniqueId);
if (mappedVersion == null) {
error("Element: " + uniqueId + " was not mapped");
return;
}
String realMappedVersion = null;
try {
realMappedVersion = isPreview(element, uniqueId, currentVersion) ?
mappedVersion.introducedPreview :
mappedVersion.introducedStable;
} catch (Exception e) {
error("For element " + element + "mappedVersion" + mappedVersion + " is null " + e);
}
String position = javadocHelper.getElementPosition(uniqueId);
checkEquals(position, sinceVersion, realMappedVersion, uniqueId);
}
private Version extractSinceVersionFromText(String documentation) {
Pattern pattern = Pattern.compile("@since\\s+(\\d+(?:\\.\\d+)?)");
Matcher matcher = pattern.matcher(documentation);
if (matcher.find()) {
String versionString = matcher.group(1);
try {
if (versionString.equals("1.0")) {
versionString = "1"; //ended up being necessary
} else if (versionString.startsWith("1.")) {
versionString = versionString.substring(2);
}
return Version.parse(versionString);
} catch (NumberFormatException ex) {
error("`@since` value that cannot be parsed: " + versionString);
return null;
}
} else {
return null;
}
}
private void checkEquals(String prefix, String sinceVersion, String mappedVersion, String name) {
if (sinceVersion == null || mappedVersion == null) {
error(name + ": NULL value for either real or effective `@since` . real/mapped version is="
+ mappedVersion + " while the `@since` in the source code is= " + sinceVersion);
return;
}
if (Integer.parseInt(sinceVersion) < 9) {
sinceVersion = "9";
}
if (!sinceVersion.equals(mappedVersion)) {
String message = getWrongSinceMessage(prefix, sinceVersion, mappedVersion, name);
error(message);
}
}
private static String getWrongSinceMessage(String prefix, String sinceVersion, String mappedVersion, String elementSimpleName) {
String message;
if (mappedVersion.equals("9")) {
message = elementSimpleName + ": `@since` version is " + sinceVersion + " but the element exists before JDK 10";
} else {
message = elementSimpleName + ": `@since` version: " + sinceVersion + "; should be: " + mappedVersion;
}
return prefix + message;
}
private static String getElementName(Element owner, Element element, Types types) {
String prefix = "";
String suffix = "";
ElementKind kind = element.getKind();
if (kind.isField()) {
TypeElement te = (TypeElement) owner;
prefix = "field";
suffix = ": " + te.getQualifiedName() + ":" + element.getSimpleName();
} else if (kind == ElementKind.METHOD || kind == ElementKind.CONSTRUCTOR) {
prefix = "method";
TypeElement te = (TypeElement) owner;
ExecutableElement executableElement = (ExecutableElement) element;
String returnType = types.erasure(executableElement.getReturnType()).toString();
String methodName = executableElement.getSimpleName().toString();
String descriptor = executableElement.getParameters().stream()
.map(p -> types.erasure(p.asType()).toString())
.collect(Collectors.joining(",", "(", ")"));
suffix = ": " + returnType + " " + te.getQualifiedName() + "." + methodName + descriptor;
} else if (kind.isDeclaredType()) {
if (kind.isClass()) {
prefix = "class";
} else if (kind.isInterface()) {
prefix = "interface";
}
suffix = ": " + ((TypeElement) element).getQualifiedName();
} else if (kind == ElementKind.PACKAGE) {
prefix = "package";
suffix = ": " + ((PackageElement) element).getQualifiedName();
} else if (kind == ElementKind.MODULE) {
prefix = "module";
suffix = ": " + ((ModuleElement) element).getQualifiedName();
}
return prefix + suffix;
}
//these were preview in before the introduction of the @PreviewFeature
{
LEGACY_PREVIEW_METHODS.put("9", Set.of(
"module: jdk.nio.mapmode",
"module: java.transaction.xa",
"module: jdk.unsupported.desktop",
"module: jdk.jpackage",
"module: java.net.http"
));
LEGACY_PREVIEW_METHODS.put("10", Set.of(
"module: jdk.nio.mapmode",
"module: java.transaction.xa",
"module: java.net.http",
"module: jdk.unsupported.desktop",
"module: jdk.jpackage"
));
LEGACY_PREVIEW_METHODS.put("11", Set.of(
"module: jdk.nio.mapmode",
"module: jdk.jpackage"
));
LEGACY_PREVIEW_METHODS.put("12", Set.of(
"module: jdk.nio.mapmode",
"module: jdk.jpackage",
"method: com.sun.source.tree.ExpressionTree com.sun.source.tree.BreakTree.getValue()",
"method: java.util.List com.sun.source.tree.CaseTree.getExpressions()",
"method: com.sun.source.tree.Tree com.sun.source.tree.CaseTree.getBody()",
"method: com.sun.source.tree.CaseTree.CaseKind com.sun.source.tree.CaseTree.getCaseKind()",
"class: com.sun.source.tree.CaseTree.CaseKind",
"field: com.sun.source.tree.CaseTree.CaseKind:STATEMENT",
"field: com.sun.source.tree.CaseTree.CaseKind:RULE",
"field: com.sun.source.tree.Tree.Kind:SWITCH_EXPRESSION",
"interface: com.sun.source.tree.SwitchExpressionTree",
"method: com.sun.source.tree.ExpressionTree com.sun.source.tree.SwitchExpressionTree.getExpression()",
"method: java.util.List com.sun.source.tree.SwitchExpressionTree.getCases()",
"method: java.lang.Object com.sun.source.tree.TreeVisitor.visitSwitchExpression(com.sun.source.tree.SwitchExpressionTree,java.lang.Object)",
"method: java.lang.Object com.sun.source.util.TreeScanner.visitSwitchExpression(com.sun.source.tree.SwitchExpressionTree,java.lang.Object)",
"method: java.lang.Object com.sun.source.util.SimpleTreeVisitor.visitSwitchExpression(com.sun.source.tree.SwitchExpressionTree,java.lang.Object)"
));
LEGACY_PREVIEW_METHODS.put("13", Set.of(
"module: jdk.nio.mapmode",
"module: jdk.jpackage",
"method: java.util.List com.sun.source.tree.CaseTree.getExpressions()",
"method: com.sun.source.tree.Tree com.sun.source.tree.CaseTree.getBody()",
"method: com.sun.source.tree.CaseTree.CaseKind com.sun.source.tree.CaseTree.getCaseKind()",
"class: com.sun.source.tree.CaseTree.CaseKind",
"field: com.sun.source.tree.CaseTree.CaseKind:STATEMENT",
"field: com.sun.source.tree.CaseTree.CaseKind:RULE",
"field: com.sun.source.tree.Tree.Kind:SWITCH_EXPRESSION",
"interface: com.sun.source.tree.SwitchExpressionTree",
"method: com.sun.source.tree.ExpressionTree com.sun.source.tree.SwitchExpressionTree.getExpression()",
"method: java.util.List com.sun.source.tree.SwitchExpressionTree.getCases()",
"method: java.lang.Object com.sun.source.tree.TreeVisitor.visitSwitchExpression(com.sun.source.tree.SwitchExpressionTree,java.lang.Object)",
"method: java.lang.Object com.sun.source.util.TreeScanner.visitSwitchExpression(com.sun.source.tree.SwitchExpressionTree,java.lang.Object)",
"method: java.lang.Object com.sun.source.util.SimpleTreeVisitor.visitSwitchExpression(com.sun.source.tree.SwitchExpressionTree,java.lang.Object)",
"method: java.lang.String java.lang.String.stripIndent()",
"method: java.lang.String java.lang.String.translateEscapes()",
"method: java.lang.String java.lang.String.formatted(java.lang.Object[])",
"class: javax.swing.plaf.basic.motif.MotifLookAndFeel",
"field: com.sun.source.tree.Tree.Kind:YIELD",
"interface: com.sun.source.tree.YieldTree",
"method: com.sun.source.tree.ExpressionTree com.sun.source.tree.YieldTree.getValue()",
"method: java.lang.Object com.sun.source.tree.TreeVisitor.visitYield(com.sun.source.tree.YieldTree,java.lang.Object)",
"method: java.lang.Object com.sun.source.util.SimpleTreeVisitor.visitYield(com.sun.source.tree.YieldTree,java.lang.Object)",
"method: java.lang.Object com.sun.source.util.TreeScanner.visitYield(com.sun.source.tree.YieldTree,java.lang.Object)"
));
LEGACY_PREVIEW_METHODS.put("14", Set.of(
"module: jdk.jpackage",
"class: javax.swing.plaf.basic.motif.MotifLookAndFeel",
"field: jdk.jshell.Snippet.SubKind:RECORD_SUBKIND",
"class: javax.lang.model.element.RecordComponentElement",
"method: javax.lang.model.type.TypeMirror javax.lang.model.element.RecordComponentElement.asType()",
"method: java.lang.Object javax.lang.model.element.ElementVisitor.visitRecordComponent(javax.lang.model.element.RecordComponentElement,java.lang.Object)",
"class: javax.lang.model.util.ElementScanner14",
"class: javax.lang.model.util.AbstractElementVisitor14",
"class: javax.lang.model.util.SimpleElementVisitor14",
"method: java.lang.Object javax.lang.model.util.ElementKindVisitor6.visitTypeAsRecord(javax.lang.model.element.TypeElement,java.lang.Object)",
"class: javax.lang.model.util.ElementKindVisitor14",
"method: javax.lang.model.element.RecordComponentElement javax.lang.model.util.Elements.recordComponentFor(javax.lang.model.element.ExecutableElement)",
"method: java.util.List javax.lang.model.util.ElementFilter.recordComponentsIn(java.lang.Iterable)",
"method: java.util.Set javax.lang.model.util.ElementFilter.recordComponentsIn(java.util.Set)",
"method: java.util.List javax.lang.model.element.TypeElement.getRecordComponents()",
"field: javax.lang.model.element.ElementKind:RECORD",
"field: javax.lang.model.element.ElementKind:RECORD_COMPONENT",
"field: javax.lang.model.element.ElementKind:BINDING_VARIABLE",
"field: com.sun.source.tree.Tree.Kind:RECORD",
"field: sun.reflect.annotation.TypeAnnotation.TypeAnnotationTarget:RECORD_COMPONENT",
"class: java.lang.reflect.RecordComponent",
"class: java.lang.runtime.ObjectMethods",
"field: java.lang.annotation.ElementType:RECORD_COMPONENT",
"method: boolean java.lang.Class.isRecord()",
"method: java.lang.reflect.RecordComponent[] java.lang.Class.getRecordComponents()",
"class: java.lang.Record",
"interface: com.sun.source.tree.PatternTree",
"field: com.sun.source.tree.Tree.Kind:BINDING_PATTERN",
"method: com.sun.source.tree.PatternTree com.sun.source.tree.InstanceOfTree.getPattern()",
"interface: com.sun.source.tree.BindingPatternTree",
"method: java.lang.Object com.sun.source.tree.TreeVisitor.visitBindingPattern(com.sun.source.tree.BindingPatternTree,java.lang.Object)"
));
LEGACY_PREVIEW_METHODS.put("15", Set.of(
"module: jdk.jpackage",
"field: jdk.jshell.Snippet.SubKind:RECORD_SUBKIND",
"class: javax.lang.model.element.RecordComponentElement",
"method: javax.lang.model.type.TypeMirror javax.lang.model.element.RecordComponentElement.asType()",
"method: java.lang.Object javax.lang.model.element.ElementVisitor.visitRecordComponent(javax.lang.model.element.RecordComponentElement,java.lang.Object)",
"class: javax.lang.model.util.ElementScanner14",
"class: javax.lang.model.util.AbstractElementVisitor14",
"class: javax.lang.model.util.SimpleElementVisitor14",
"method: java.lang.Object javax.lang.model.util.ElementKindVisitor6.visitTypeAsRecord(javax.lang.model.element.TypeElement,java.lang.Object)",
"class: javax.lang.model.util.ElementKindVisitor14",
"method: javax.lang.model.element.RecordComponentElement javax.lang.model.util.Elements.recordComponentFor(javax.lang.model.element.ExecutableElement)",
"method: java.util.List javax.lang.model.util.ElementFilter.recordComponentsIn(java.lang.Iterable)",
"method: java.util.Set javax.lang.model.util.ElementFilter.recordComponentsIn(java.util.Set)",
"method: java.util.List javax.lang.model.element.TypeElement.getRecordComponents()",
"field: javax.lang.model.element.ElementKind:RECORD",
"field: javax.lang.model.element.ElementKind:RECORD_COMPONENT",
"field: javax.lang.model.element.ElementKind:BINDING_VARIABLE",
"field: com.sun.source.tree.Tree.Kind:RECORD",
"field: sun.reflect.annotation.TypeAnnotation.TypeAnnotationTarget:RECORD_COMPONENT",
"class: java.lang.reflect.RecordComponent",
"class: java.lang.runtime.ObjectMethods",
"field: java.lang.annotation.ElementType:RECORD_COMPONENT",
"class: java.lang.Record",
"method: boolean java.lang.Class.isRecord()",
"method: java.lang.reflect.RecordComponent[] java.lang.Class.getRecordComponents()",
"field: javax.lang.model.element.Modifier:SEALED",
"field: javax.lang.model.element.Modifier:NON_SEALED",
"method: javax.lang.model.element.TypeElement:getPermittedSubclasses:()",
"method: java.util.List com.sun.source.tree.ClassTree.getPermitsClause()",
"method: boolean java.lang.Class.isSealed()",
"method: java.lang.constant.ClassDesc[] java.lang.Class.permittedSubclasses()",
"interface: com.sun.source.tree.PatternTree",
"field: com.sun.source.tree.Tree.Kind:BINDING_PATTERN",
"method: com.sun.source.tree.PatternTree com.sun.source.tree.InstanceOfTree.getPattern()",
"interface: com.sun.source.tree.BindingPatternTree",
"method: java.lang.Object com.sun.source.tree.TreeVisitor.visitBindingPattern(com.sun.source.tree.BindingPatternTree,java.lang.Object)"
));
LEGACY_PREVIEW_METHODS.put("16", Set.of(
"field: jdk.jshell.Snippet.SubKind:RECORD_SUBKIND",
"field: javax.lang.model.element.Modifier:SEALED",
"field: javax.lang.model.element.Modifier:NON_SEALED",
"method: javax.lang.model.element.TypeElement:getPermittedSubclasses:()",
"method: java.util.List com.sun.source.tree.ClassTree.getPermitsClause()",
"method: boolean java.lang.Class.isSealed()",
"method: java.lang.constant.ClassDesc[] java.lang.Class.permittedSubclasses()"
));
// java.lang.foreign existed since JDK 19 and wasn't annotated - went out of preview in JDK 22
LEGACY_PREVIEW_METHODS.put("19", Set.of(
"package: java.lang.foreign"
));
LEGACY_PREVIEW_METHODS.put("20", Set.of(
"package: java.lang.foreign"
));
LEGACY_PREVIEW_METHODS.put("21", Set.of(
"package: java.lang.foreign"
));
}
/**
* Helper to find javadoc and resolve @inheritDoc and the effective since version.
*/
private final class EffectiveSourceSinceHelper implements AutoCloseable {
private static final JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
private final JavaFileManager baseFileManager;
private final StandardJavaFileManager fm;
private final Set<String> seenLookupElements = new HashSet<>();
private final Map<String, Version> signature2Source = new HashMap<>();
private final Map<String, String> signature2Location = new HashMap<>();
/**
* Create the helper.
*
* @param mainTask JavacTask from which the further Elements originate
* @param sourceLocations paths where source files should be searched
* @param validator enclosing class of the helper, typically the object invoking this method
* @return a EffectiveSourceSinceHelper
*/
public static EffectiveSourceSinceHelper create(JavacTask mainTask, Collection<? extends Path> sourceLocations, SinceChecker validator) {
StandardJavaFileManager fm = compiler.getStandardFileManager(null, null, null);
try {
fm.setLocationFromPaths(StandardLocation.MODULE_SOURCE_PATH, sourceLocations);
return validator.new EffectiveSourceSinceHelper(mainTask, fm);
} catch (IOException ex) {
try {
fm.close();
} catch (IOException closeEx) {
ex.addSuppressed(closeEx);
}
throw new UncheckedIOException(ex);
}
}
private EffectiveSourceSinceHelper(JavacTask mainTask, StandardJavaFileManager fm) {
this.baseFileManager = ((JavacTaskImpl) mainTask).getContext().get(JavaFileManager.class);
this.fm = fm;
}
public Version effectiveSinceVersion(Element owner, Element element, Types typeUtils, Elements elementUtils) {
String handle = getElementName(owner, element, typeUtils);
Version since = signature2Source.get(handle);
if (since == null) {
try {
Element lookupElement = switch (element.getKind()) {
case MODULE, PACKAGE -> element;
default -> elementUtils.getOutermostTypeElement(element);
};
if (lookupElement == null)
return null;
String lookupHandle = getElementName(owner, element, typeUtils);
if (!seenLookupElements.add(lookupHandle)) {
//we've already processed this top-level, don't try to compute
//the values again:
return null;
}
Pair<JavacTask, CompilationUnitTree> source = findSource(lookupElement, elementUtils);
if (source == null)
return null;
fillElementCache(source.fst, source.snd, source.fst.getTypes(), source.fst.getElements());
since = signature2Source.get(handle);
} catch (IOException ex) {
error("JavadocHelper failed for " + element);
}
}
return since;
}
private String getElementPosition(String signature) {
return signature2Location.getOrDefault(signature, "");
}
//where:
private void fillElementCache(JavacTask task, CompilationUnitTree cut, Types typeUtils, Elements elementUtils) {
Trees trees = Trees.instance(task);
String fileName = cut.getSourceFile().getName();
new TreePathScanner<Void, Void>() {
@Override
public Void visitMethod(MethodTree node, Void p) {
handleDeclaration(node, fileName);
return null;
}
@Override
public Void visitClass(ClassTree node, Void p) {
handleDeclaration(node, fileName);
return super.visitClass(node, p);
}
@Override
public Void visitVariable(VariableTree node, Void p) {
handleDeclaration(node, fileName);
return null;
}
@Override
public Void visitModule(ModuleTree node, Void p) {
handleDeclaration(node, fileName);
return null;
}
@Override
public Void visitBlock(BlockTree node, Void p) {
return null;
}
@Override
public Void visitPackage(PackageTree node, Void p) {
if (cut.getSourceFile().isNameCompatible("package-info", JavaFileObject.Kind.SOURCE)) {
handleDeclaration(node, fileName);
}
return super.visitPackage(node, p);
}
private void handleDeclaration(Tree node, String fileName) {
Element currentElement = trees.getElement(getCurrentPath());
if (currentElement != null) {
long startPosition = trees.getSourcePositions().getStartPosition(cut, node);
long lineNumber = cut.getLineMap().getLineNumber(startPosition);
String filePathWithLineNumber = String.format("src%s:%s ", fileName, lineNumber);
signature2Source.put(getElementName(currentElement.getEnclosingElement(), currentElement, typeUtils), computeSinceVersion(currentElement, typeUtils, elementUtils));
signature2Location.put(getElementName(currentElement.getEnclosingElement(), currentElement, typeUtils), filePathWithLineNumber);
}
}
}.scan(cut, null);
}
private Version computeSinceVersion(Element element, Types types,
Elements elementUtils) {
String docComment = elementUtils.getDocComment(element);
Version version = null;
if (docComment != null) {
version = extractSinceVersionFromText(docComment);
}
if (version != null) {
return version; //explicit @since has an absolute priority
}
if (element.getKind() != ElementKind.MODULE) {
version = effectiveSinceVersion(element.getEnclosingElement().getEnclosingElement(), element.getEnclosingElement(), types, elementUtils);
}
return version;
}
private Pair<JavacTask, CompilationUnitTree> findSource(Element forElement, Elements elementUtils) throws IOException {
String moduleName = elementUtils.getModuleOf(forElement).getQualifiedName().toString();
String binaryName = switch (forElement.getKind()) {
case MODULE -> "module-info";
case PACKAGE -> ((QualifiedNameable) forElement).getQualifiedName() + ".package-info";
default -> elementUtils.getBinaryName((TypeElement) forElement).toString();
};
Location packageLocationForModule = fm.getLocationForModule(StandardLocation.MODULE_SOURCE_PATH, moduleName);
JavaFileObject jfo = fm.getJavaFileForInput(packageLocationForModule,
binaryName,
JavaFileObject.Kind.SOURCE);
if (jfo == null)
return null;
List<JavaFileObject> jfos = Arrays.asList(jfo);
JavaFileManager patchFM = moduleName != null
? new PatchModuleFileManager(baseFileManager, jfo, moduleName)
: baseFileManager;
JavacTaskImpl task = (JavacTaskImpl) compiler.getTask(null, patchFM, d -> {
}, null, null, jfos);
Iterable<? extends CompilationUnitTree> cuts = task.parse();
task.enter();
return Pair.of(task, cuts.iterator().next());
}
@Override
public void close() throws IOException {
fm.close();
}
/**
* Manages files within a patch module.
* Provides custom behavior for handling file locations within a patch module.
* Includes methods to specify module locations, infer module names and determine
* if a location belongs to the patch module path.
*/
private static final class PatchModuleFileManager
extends ForwardingJavaFileManager<JavaFileManager> {
private final JavaFileObject file;
private final String moduleName;
public PatchModuleFileManager(JavaFileManager fileManager,
JavaFileObject file,
String moduleName) {
super(fileManager);
this.file = file;
this.moduleName = moduleName;
}
@Override
public Location getLocationForModule(Location location,
JavaFileObject fo) throws IOException {
return fo == file
? PATCH_LOCATION
: super.getLocationForModule(location, fo);
}
@Override
public String inferModuleName(Location location) throws IOException {
return location == PATCH_LOCATION
? moduleName
: super.inferModuleName(location);
}
@Override
public boolean hasLocation(Location location) {
return location == StandardLocation.PATCH_MODULE_PATH ||
super.hasLocation(location);
}
private static final Location PATCH_LOCATION = new Location() {
@Override
public String getName() {
return "PATCH_LOCATION";
}
@Override
public boolean isOutputLocation() {
return false;
}
@Override
public boolean isModuleOrientedLocation() {
return false;
}
};
}
}
}

View File

@ -0,0 +1,34 @@
/*
* Copyright (c) 2024, 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 8331051
* @summary Test for `@since` for java.base module
* @library /test/lib
* /test/jdk/tools/sincechecker
* @modules jdk.compiler/com.sun.tools.javac.api
* jdk.compiler/com.sun.tools.javac.util
* jdk.compiler/com.sun.tools.javac.code
* @run main SinceChecker java.base --exclude java.lang.classfile
*/