jdk-24/test/langtools/tools/javac/sym/ElementStructureTest.java
Adam Sotona 2b00ac0d02 8308753: Class-File API transition to Preview
Reviewed-by: ihse, mchung, vromero
2023-12-04 07:07:57 +00:00

693 lines
25 KiB
Java

/*
* Copyright (c) 2015, 2022, 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 8072480 8203814
* @summary Check the platform classpath contains the correct elements.
* @library /tools/lib
* @enablePreview
* @modules jdk.compiler/com.sun.tools.javac.code
* jdk.compiler/com.sun.tools.javac.api
* jdk.compiler/com.sun.tools.javac.main
* jdk.compiler/com.sun.tools.javac.platform
* jdk.compiler/com.sun.tools.javac.util
* java.base/jdk.internal.classfile.impl
* jdk.jdeps/com.sun.tools.javap
* @build toolbox.ToolBox ElementStructureTest
* @run main ElementStructureTest
*/
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Reader;
import java.io.Writer;
import java.net.URI;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.security.MessageDigest;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.ServiceLoader;
import java.util.Set;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.function.Predicate;
import java.util.regex.Pattern;
import java.util.stream.Stream;
import javax.lang.model.element.AnnotationMirror;
import javax.lang.model.element.AnnotationValue;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementVisitor;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.ModuleElement;
import javax.lang.model.element.NestingKind;
import javax.lang.model.element.PackageElement;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.TypeParameterElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.TypeMirror;
import javax.tools.FileObject;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileManager;
import javax.tools.JavaFileObject;
import javax.tools.JavaFileObject.Kind;
import javax.tools.StandardLocation;
import javax.tools.ToolProvider;
import com.sun.source.util.JavacTask;
import java.lang.classfile.ClassFile;
import com.sun.tools.javac.api.JavacTaskImpl;
import com.sun.tools.javac.code.Symbol.CompletionFailure;
import com.sun.tools.javac.platform.PlatformProvider;
import toolbox.ToolBox;
/**To generate the hash values for version N, invoke this class like:
*
* java ElementStructureTest generate-hashes $LANGTOOLS_DIR/src/jdk.compiler/share/data/symbols/include.list (<classes-for-N> N)+
*
* Where <classes-for-N> is the file produced by make/src/classes/build/tools/symbolgenerator/Probe.java.
* So, to produce hashes for 6, 7 and 8, this command can be used:
*
* java ElementStructureTest generate-hashes classes-6 6 classes-7 7 classes-8 8
*
* To inspect differences between the actual and expected output for version N, invoke this class like:
*
* java ElementStructureTest generate-output $LANGTOOLS_DIR/src/jdk.compiler/share/data/symbols/include.list (<classes-for-N> N <actual-output-file> <expected-output-file>)+
*
* For example, to get the actual and expected output for 6 in /tmp/actual and /tmp/expected, respectively:
*
* java ElementStructureTest generate-output $LANGTOOLS_DIR/src/jdk.compiler/share/data/symbols/include.list classes-6 6 /tmp/actual /tmp/expected
*/
public class ElementStructureTest {
static final byte[] hash6 = new byte[] {
(byte) 0x99, (byte) 0x34, (byte) 0x82, (byte) 0xCF,
(byte) 0xE0, (byte) 0x53, (byte) 0xF3, (byte) 0x13,
(byte) 0x4E, (byte) 0xCF, (byte) 0x49, (byte) 0x32,
(byte) 0xB7, (byte) 0x52, (byte) 0x0F, (byte) 0x68
};
static final byte[] hash7 = new byte[] {
(byte) 0x2C, (byte) 0x01, (byte) 0xC0, (byte) 0xFB,
(byte) 0xD5, (byte) 0x66, (byte) 0x0D, (byte) 0x9C,
(byte) 0x09, (byte) 0x17, (byte) 0x2F, (byte) 0x5A,
(byte) 0x3D, (byte) 0xC1, (byte) 0xFE, (byte) 0xCB
};
static final byte[] hash8 = new byte[] {
(byte) 0x10, (byte) 0xE6, (byte) 0xE8, (byte) 0x11,
(byte) 0xC8, (byte) 0x02, (byte) 0x63, (byte) 0x9B,
(byte) 0xAB, (byte) 0x11, (byte) 0x9E, (byte) 0x4F,
(byte) 0xFA, (byte) 0x00, (byte) 0x6D, (byte) 0x81
};
final static Map<String, byte[]> version2Hash = new HashMap<>();
static {
version2Hash.put("6", hash6);
version2Hash.put("7", hash7);
version2Hash.put("8", hash8);
}
public static void main(String... args) throws Exception {
if (args.length == 0) {
new ElementStructureTest().doTest();
return ;
}
switch (args[0]) {
case "generate-hashes":
new ElementStructureTest().generateHashes(args);
break;
case "generate-output":
new ElementStructureTest().generateOutput(args);
break;
default:
throw new IllegalStateException("Unrecognized request: " + args[0]);
}
}
void doTest() throws Exception {
for (PlatformProvider provider : ServiceLoader.load(PlatformProvider.class)) {
for (String ver : provider.getSupportedPlatformNames()) {
if (!version2Hash.containsKey(ver))
continue;
try (ByteArrayOutputStream baos = new ByteArrayOutputStream(); Writer output = new OutputStreamWriter(baos, "UTF-8")) {
run(output, ver);
output.close();
byte[] actual = MessageDigest.getInstance("MD5").digest(baos.toByteArray());
if (!Arrays.equals(version2Hash.get(ver), actual))
throw new AssertionError("Wrong hash: " + toHex(actual) + " for version: " + ver);
}
}
}
}
void generateHashes(String... args) throws Exception {
Predicate<String> ignoreList = constructAcceptIgnoreList(args[1]);
for (int i = 2; i < args.length; i += 2) {
try (ByteArrayOutputStream baos = new ByteArrayOutputStream(); Writer output = new OutputStreamWriter(baos, "UTF-8")) {
realClasses(args[i], ignoreList, output, args[i + 1]);
output.close();
System.err.println("version:" + args[i + 1] + "; " + toHex(MessageDigest.getInstance("MD5").digest(baos.toByteArray())));
}
}
}
void generateOutput(String... args) throws Exception {
Predicate<String> ignoreList = constructAcceptIgnoreList(args[1]);
for (int i = 2; i < args.length; i += 4) {
try (Writer actual = Files.newBufferedWriter(Paths.get(args[i + 2]));
Writer expected = Files.newBufferedWriter(Paths.get(args[i + 3]))) {
run(actual, args[i + 1]);
realClasses(args[i], ignoreList, expected, args[i + 1]);
}
}
}
Predicate<String> constructAcceptIgnoreList(String fromFiles) throws IOException {
StringBuilder acceptPattern = new StringBuilder();
StringBuilder rejectPattern = new StringBuilder();
for (String file : fromFiles.split(File.pathSeparator)) {
try (Stream<String> lines = Files.lines(Paths.get(file))) {
lines.forEach(line -> {
if (line.isEmpty())
return;
StringBuilder targetPattern;
switch (line.charAt(0)) {
case '+' -> targetPattern = acceptPattern;
case '-' -> targetPattern = rejectPattern;
default -> {
return;
}
}
line = line.substring(1);
if (line.endsWith("/")) {
line += "[^/]*";
} else {
line += "|" + line + "$[^/]*";
}
line = line.replace("/", ".");
if (targetPattern.length() != 0)
targetPattern.append("|");
targetPattern.append(line);
});
}
}
Pattern accept = Pattern.compile(acceptPattern.toString());
Pattern reject = Pattern.compile(rejectPattern.toString());
return clazzName -> accept.matcher(clazzName).matches() && !reject.matcher(clazzName).matches();
}
private static String toHex(byte[] bytes) {
StringBuilder hex = new StringBuilder();
String delim = "";
for (byte b : bytes) {
hex.append(delim);
hex.append(String.format("(byte) 0x%02X", b));
delim = ", ";
}
return hex.toString();
}
void run(Writer output, String version) throws Exception {
List<String> options = Arrays.asList("--release", version, "-classpath", "");
List<ToolBox.JavaSource> files = List.of(new ToolBox.JavaSource("Test", ""));
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
JavacTaskImpl task = (JavacTaskImpl) compiler.getTask(null, null, null, options, null, files);
task.analyze();
JavaFileManager fm = task.getContext().get(JavaFileManager.class);
for (String pack : packages(fm)) {
PackageElement packEl = task.getElements().getPackageElement(pack);
if (packEl == null) {
throw new AssertionError("Cannot find package: " + pack);
}
new ExhaustiveElementScanner(task, output, p -> true).visit(packEl);
}
}
void realClasses(String location, Predicate<String> acceptor, Writer output, String version) throws Exception {
Path classes = Paths.get(location);
Map<String, JavaFileObject> className2File = new HashMap<>();
Map<JavaFileObject, String> file2ClassName = new HashMap<>();
try (BufferedReader descIn = Files.newBufferedReader(classes)) {
String classFileData;
while ((classFileData = descIn.readLine()) != null) {
ByteArrayOutputStream data = new ByteArrayOutputStream();
for (int i = 0; i < classFileData.length(); i += 2) {
data.write(Integer.parseInt(classFileData.substring(i, i + 2), 16));
}
JavaFileObject file = new ByteArrayJavaFileObject(data.toByteArray());
try (InputStream in = new ByteArrayInputStream(data.toByteArray())) {
String name = ClassFile.of().parse(in.readAllBytes()).thisClass().name().stringValue();
className2File.put(name, file);
file2ClassName.put(file, name);
} catch (IOException ex) {
throw new IllegalStateException(ex);
}
}
}
try (JavaFileManager fm = new TestFileManager(className2File, file2ClassName)) {
JavacTaskImpl task = (JavacTaskImpl) ToolProvider.getSystemJavaCompiler().getTask(null, fm, null, Arrays.asList("-source", version), null, Arrays.asList(new ToolBox.JavaSource("Test", "")));
task.parse();
PACK: for (String pack : packages(fm)) {
PackageElement packEl = task.getElements().getPackageElement(pack);
assert packEl != null;
new ExhaustiveElementScanner(task, output, acceptor).visit(packEl);
}
}
}
Set<String> packages(JavaFileManager fm) throws IOException {
Set<String> packages = new TreeSet<>();
EnumSet<Kind> kinds = EnumSet.of(JavaFileObject.Kind.CLASS, JavaFileObject.Kind.OTHER);
for (JavaFileObject file : fm.list(StandardLocation.PLATFORM_CLASS_PATH, "", kinds, true)) {
String binary = fm.inferBinaryName(StandardLocation.PLATFORM_CLASS_PATH, file);
packages.add(binary.substring(0, binary.lastIndexOf('.')));
}
return packages;
}
final class ExhaustiveElementScanner implements ElementVisitor<Void, Void> {
final JavacTask task;
final Writer out;
final Predicate<String> acceptType;
public ExhaustiveElementScanner(JavacTask task, Writer out, Predicate<String> acceptType) {
this.task = task;
this.out = out;
this.acceptType = acceptType;
}
@Override
public Void visit(Element e, Void p) {
return e.accept(this, p);
}
@Override
public Void visit(Element e) {
return e.accept(this, null);
}
private void write(TypeMirror type) throws IOException {
try {
out.write(type.toString()
.replace("java.lang.invoke.MethodHandle$PolymorphicSignature", "java.lang.invoke.MethodHandle.PolymorphicSignature")
.replace("javax.swing.JRootPane$DefaultAction", "javax.swing.JRootPane.DefaultAction")
.replace("javax.swing.plaf.metal.MetalFileChooserUI$DirectoryComboBoxRenderer", "javax.swing.plaf.metal.MetalFileChooserUI.DirectoryComboBoxRenderer")
);
} catch (CompletionFailure cf) {
out.write("cf");
}
}
private void writeTypes(Iterable<? extends TypeMirror> types) throws IOException {
String sep = "";
for (TypeMirror type : types) {
out.write(sep);
write(type);
sep = ", ";
}
}
private void writeAnnotations(Iterable<? extends AnnotationMirror> annotations) throws IOException {
for (AnnotationMirror ann : annotations) {
out.write("@");
write(ann.getAnnotationType());
if (!ann.getElementValues().isEmpty()) {
out.write("(");
Map<ExecutableElement, AnnotationValue> valuesMap = new TreeMap<>((a1, a2) -> a1.getSimpleName().toString().compareTo(a2.getSimpleName().toString()));
valuesMap.putAll(ann.getElementValues());
for (Entry<? extends ExecutableElement, ? extends AnnotationValue> ev : valuesMap.entrySet()) {
out.write(ev.getKey().getSimpleName().toString());
out.write(" = ");
out.write(ev.getValue().toString());
}
out.write(")");
}
}
}
void analyzeElement(Element e) {
try {
write(e.asType());
writeAnnotations(e.getAnnotationMirrors());
out.write(e.getKind().toString());
out.write(e.getModifiers().toString());
out.write(e.getSimpleName().toString());
} catch (IOException ex) {
ex.printStackTrace();
}
}
boolean acceptAccess(Element e) {
return e.getModifiers().contains(Modifier.PUBLIC) || e.getModifiers().contains(Modifier.PROTECTED);
}
@Override
public Void visitExecutable(ExecutableElement e, Void p) {
if (!acceptAccess(e))
return null;
try {
analyzeElement(e);
out.write(String.valueOf(e.getDefaultValue()));
for (VariableElement param : e.getParameters()) {
visit(param, p);
}
out.write(String.valueOf(typeMirrorTranslate(e.getReceiverType())));
write(e.getReturnType());
out.write(e.getSimpleName().toString());
writeTypes(e.getThrownTypes());
for (TypeParameterElement param : e.getTypeParameters()) {
visit(param, p);
}
out.write(String.valueOf(e.isDefault()));
out.write(String.valueOf(e.isVarArgs()));
out.write("\n");
} catch (IOException ex) {
ex.printStackTrace();
}
return null;
}
/**
* Original implementation of getReceiverType returned null
* for many cases where TypeKind.NONE was specified; translate
* back to null to compare against old hashes.
*/
private TypeMirror typeMirrorTranslate(TypeMirror type) {
if (type.getKind() == javax.lang.model.type.TypeKind.NONE)
return null;
else
return type;
}
@Override
public Void visitPackage(PackageElement e, Void p) {
List<Element> types = new ArrayList<>(e.getEnclosedElements());
Collections.sort(types, (e1, e2) -> e1.getSimpleName().toString().compareTo(e2.getSimpleName().toString()));
for (Element encl : types) {
visit(encl, p);
}
return null;
}
@Override
public Void visitType(TypeElement e, Void p) {
if (!acceptAccess(e))
return null;
writeType(e);
return null;
}
void writeType(TypeElement e) {
if (!acceptType.test(task.getElements().getBinaryName(e).toString()))
return ;
try {
analyzeElement(e);
writeTypes(e.getInterfaces());
out.write(e.getNestingKind().toString());
out.write(e.getQualifiedName().toString());
write(e.getSuperclass());
for (TypeParameterElement param : e.getTypeParameters()) {
visit(param, null);
}
List<Element> defs = new ArrayList<>(e.getEnclosedElements()); //XXX: forcing ordering for members - not completely correct!
Collections.sort(defs, (e1, e2) -> e1.toString().compareTo(e2.toString()));
for (Element def : defs) {
visit(def, null);
}
out.write("\n");
} catch (IOException ex) {
ex.printStackTrace();
}
}
@Override
public Void visitVariable(VariableElement e, Void p) {
if (!acceptAccess(e))
return null;
try {
analyzeElement(e);
writeConstant(e.getConstantValue());
out.write("\n");
} catch (IOException ex) {
ex.printStackTrace();
}
return null;
}
@Override
public Void visitTypeParameter(TypeParameterElement e, Void p) {
try {
analyzeElement(e);
out.write(e.getBounds().toString());
out.write("\n");
} catch (IOException ex) {
ex.printStackTrace();
}
return null;
}
@Override
public Void visitModule(ModuleElement e, Void p) {
throw new IllegalStateException("Not supported yet.");
}
@Override
public Void visitUnknown(Element e, Void p) {
throw new IllegalStateException("Should not get here.");
}
private void writeConstant(Object value) throws IOException {
if (value instanceof Double) {
out.write(Double.toHexString((Double) value));
} else if (value instanceof Float) {
out.write(Float.toHexString((Float) value));
} else {
out.write(String.valueOf(value));
}
}
}
final class TestFileManager implements JavaFileManager {
final Map<String, JavaFileObject> className2File;
final Map<JavaFileObject, String> file2ClassName;
public TestFileManager(Map<String, JavaFileObject> className2File, Map<JavaFileObject, String> file2ClassName) {
this.className2File = className2File;
this.file2ClassName = file2ClassName;
}
@Override
public ClassLoader getClassLoader(Location location) {
return new URLClassLoader(new URL[0]);
}
@Override
public Iterable<JavaFileObject> list(Location location, String packageName, Set<Kind> kinds, boolean recurse) throws IOException {
if (location != StandardLocation.PLATFORM_CLASS_PATH || !kinds.contains(Kind.CLASS))
return Collections.emptyList();
if (!packageName.isEmpty())
packageName += ".";
List<JavaFileObject> result = new ArrayList<>();
for (Entry<String, JavaFileObject> e : className2File.entrySet()) {
String currentPackage = e.getKey().substring(0, e.getKey().lastIndexOf(".") + 1);
if (recurse ? currentPackage.startsWith(packageName) : packageName.equals(currentPackage))
result.add(e.getValue());
}
return result;
}
@Override
public String inferBinaryName(Location location, JavaFileObject file) {
return file2ClassName.get(file);
}
@Override
public boolean isSameFile(FileObject a, FileObject b) {
return a == b;
}
@Override
public boolean handleOption(String current, Iterator<String> remaining) {
return false;
}
@Override
public boolean hasLocation(Location location) {
return location == StandardLocation.PLATFORM_CLASS_PATH;
}
@Override
public JavaFileObject getJavaFileForInput(Location location, String className, Kind kind) throws IOException {
if (location != StandardLocation.PLATFORM_CLASS_PATH || kind != Kind.CLASS)
return null;
return className2File.get(className);
}
@Override
public JavaFileObject getJavaFileForOutput(Location location, String className, Kind kind, FileObject sibling) throws IOException {
throw new UnsupportedOperationException("");
}
@Override
public FileObject getFileForInput(Location location, String packageName, String relativeName) throws IOException {
return null;
}
@Override
public FileObject getFileForOutput(Location location, String packageName, String relativeName, FileObject sibling) throws IOException {
throw new UnsupportedOperationException("");
}
@Override
public void flush() throws IOException {
}
@Override
public void close() throws IOException {
}
@Override
public int isSupportedOption(String option) {
return -1;
}
}
static class ByteArrayJavaFileObject implements JavaFileObject {
private final byte[] data;
public ByteArrayJavaFileObject(byte[] data) {
this.data = data;
}
@Override
public Kind getKind() {
return Kind.CLASS;
}
@Override
public boolean isNameCompatible(String simpleName, Kind kind) {
return true;
}
@Override
public NestingKind getNestingKind() {
return null;
}
@Override
public Modifier getAccessLevel() {
return null;
}
@Override
public URI toUri() {
return null;
}
@Override
public String getName() {
return null;
}
@Override
public InputStream openInputStream() throws IOException {
return new ByteArrayInputStream(data);
}
@Override
public OutputStream openOutputStream() throws IOException {
throw new UnsupportedOperationException();
}
@Override
public Reader openReader(boolean ignoreEncodingErrors) throws IOException {
throw new UnsupportedOperationException();
}
@Override
public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
throw new UnsupportedOperationException();
}
@Override
public Writer openWriter() throws IOException {
throw new UnsupportedOperationException();
}
@Override
public long getLastModified() {
return 0;
}
@Override
public boolean delete() {
throw new UnsupportedOperationException();
}
}
}