2013-09-14 19:04:47 +01:00
|
|
|
/*
|
2015-01-05 17:35:48 -08:00
|
|
|
* Copyright (c) 2013, 2015, Oracle and/or its affiliates. All rights reserved.
|
2013-09-14 19:04:47 +01:00
|
|
|
* 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
|
2015-01-05 17:35:48 -08:00
|
|
|
* @bug 7047734 8027660 8037937 8047719 8058708 8064857
|
2014-05-29 15:28:01 +01:00
|
|
|
* @summary The LVT is not generated correctly during some try/catch scenarios
|
2013-11-01 19:08:56 +00:00
|
|
|
* javac crash while creating LVT entry for a local variable defined in
|
|
|
|
* an inner block
|
2013-09-14 19:04:47 +01:00
|
|
|
* @library /tools/javac/lib
|
2023-12-04 07:07:57 +00:00
|
|
|
* @enablePreview
|
|
|
|
* @modules java.base/jdk.internal.classfile.impl
|
2013-09-14 19:04:47 +01:00
|
|
|
* @build JavacTestingAbstractProcessor LVTHarness
|
|
|
|
* @run main LVTHarness
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.lang.annotation.Annotation;
|
|
|
|
import java.util.Set;
|
|
|
|
import java.util.Arrays;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.Collections;
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.HashSet;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
import javax.annotation.processing.RoundEnvironment;
|
|
|
|
import javax.lang.model.element.Element;
|
|
|
|
import javax.lang.model.element.TypeElement;
|
|
|
|
import javax.tools.JavaCompiler;
|
|
|
|
import javax.tools.JavaFileObject;
|
|
|
|
import javax.tools.StandardJavaFileManager;
|
|
|
|
import javax.tools.ToolProvider;
|
|
|
|
|
|
|
|
import com.sun.source.util.JavacTask;
|
2023-12-04 07:07:57 +00:00
|
|
|
import java.lang.classfile.*;
|
|
|
|
import java.lang.classfile.attribute.*;
|
2013-09-14 19:04:47 +01:00
|
|
|
|
|
|
|
import static javax.tools.StandardLocation.*;
|
2013-09-23 10:10:07 +02:00
|
|
|
import static javax.tools.JavaFileObject.Kind.SOURCE;
|
2013-09-14 19:04:47 +01:00
|
|
|
|
|
|
|
public class LVTHarness {
|
|
|
|
|
|
|
|
static int nerrors = 0;
|
|
|
|
|
|
|
|
static final JavaCompiler comp = ToolProvider.getSystemJavaCompiler();
|
|
|
|
static final StandardJavaFileManager fm = comp.getStandardFileManager(null, null, null);
|
|
|
|
|
|
|
|
public static void main(String[] args) throws Exception {
|
2014-10-29 17:25:23 -07:00
|
|
|
try {
|
|
|
|
String testDir = System.getProperty("test.src");
|
2023-09-07 15:37:25 +00:00
|
|
|
fm.setLocation(SOURCE_PATH, List.of(new File(testDir, "tests")));
|
2014-10-29 17:25:23 -07:00
|
|
|
|
|
|
|
// Make sure classes are written to scratch dir.
|
2023-09-07 15:37:25 +00:00
|
|
|
fm.setLocation(CLASS_OUTPUT, List.of(new File(".")));
|
2014-10-29 17:25:23 -07:00
|
|
|
|
|
|
|
for (JavaFileObject jfo : fm.list(SOURCE_PATH, "", Collections.singleton(SOURCE), true)) {
|
|
|
|
new LVTHarness(jfo).check();
|
|
|
|
}
|
|
|
|
if (nerrors > 0) {
|
|
|
|
throw new AssertionError("Errors were found");
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
fm.close();
|
2013-09-14 19:04:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JavaFileObject jfo;
|
2013-09-23 10:10:07 +02:00
|
|
|
Map<ElementKey, AliveRanges> aliveRangeMap = new HashMap<>();
|
2013-09-14 19:04:47 +01:00
|
|
|
Set<String> declaredKeys = new HashSet<>();
|
|
|
|
List<ElementKey> seenAliveRanges = new ArrayList<>();
|
|
|
|
|
|
|
|
protected LVTHarness(JavaFileObject jfo) {
|
|
|
|
this.jfo = jfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void check() throws Exception {
|
2013-09-23 10:10:07 +02:00
|
|
|
|
|
|
|
JavacTask ct = (JavacTask) comp.getTask(null, fm, null, Arrays.asList("-g"),
|
|
|
|
null, Arrays.asList(jfo));
|
|
|
|
System.err.println("compiling code " + jfo);
|
2013-09-14 19:04:47 +01:00
|
|
|
ct.setProcessors(Collections.singleton(new AliveRangeFinder()));
|
|
|
|
if (!ct.call()) {
|
|
|
|
throw new AssertionError("Error during compilation");
|
|
|
|
}
|
|
|
|
|
2013-09-23 10:10:07 +02:00
|
|
|
|
|
|
|
File javaFile = new File(jfo.getName());
|
|
|
|
File classFile = new File(javaFile.getName().replace(".java", ".class"));
|
|
|
|
checkClassFile(classFile);
|
2013-09-14 19:04:47 +01:00
|
|
|
|
|
|
|
//check all candidates have been used up
|
|
|
|
for (Map.Entry<ElementKey, AliveRanges> entry : aliveRangeMap.entrySet()) {
|
|
|
|
if (!seenAliveRanges.contains(entry.getKey())) {
|
|
|
|
error("Redundant @AliveRanges annotation on method " +
|
2014-05-29 15:28:01 +01:00
|
|
|
entry.getKey().elem + " with key " + entry.getKey());
|
2013-09-14 19:04:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-07 15:37:25 +00:00
|
|
|
void checkClassFile(File file) throws IOException {
|
2023-12-04 07:07:57 +00:00
|
|
|
ClassModel classFile = ClassFile.of().parse(file.toPath());
|
2013-09-14 19:04:47 +01:00
|
|
|
|
|
|
|
//lets get all the methods in the class file.
|
2023-09-07 15:37:25 +00:00
|
|
|
for (MethodModel method : classFile.methods()) {
|
2013-09-14 19:04:47 +01:00
|
|
|
for (ElementKey elementKey: aliveRangeMap.keySet()) {
|
2023-09-07 15:37:25 +00:00
|
|
|
String methodDesc = method.methodName().stringValue() +
|
|
|
|
parse(method.methodTypeSymbol().descriptorString()).replace(" ", "");
|
2013-09-14 19:04:47 +01:00
|
|
|
if (methodDesc.equals(elementKey.elem.toString())) {
|
2023-09-07 15:37:25 +00:00
|
|
|
checkMethod(method, aliveRangeMap.get(elementKey));
|
2013-09-14 19:04:47 +01:00
|
|
|
seenAliveRanges.add(elementKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-07 15:37:25 +00:00
|
|
|
void checkMethod(MethodModel method, AliveRanges ranges) {
|
|
|
|
CodeAttribute code = method.findAttribute(Attributes.CODE).orElseThrow();
|
|
|
|
LocalVariableTableAttribute lvt = code.findAttribute(Attributes.LOCAL_VARIABLE_TABLE).orElseThrow();
|
2013-09-14 19:04:47 +01:00
|
|
|
List<String> infoFromRanges = convertToStringList(ranges);
|
2023-09-07 15:37:25 +00:00
|
|
|
List<String> infoFromLVT = convertToStringList(lvt);
|
2013-09-14 19:04:47 +01:00
|
|
|
|
|
|
|
// infoFromRanges most be contained in infoFromLVT
|
|
|
|
int i = 0;
|
|
|
|
int j = 0;
|
|
|
|
while (i < infoFromRanges.size() && j < infoFromLVT.size()) {
|
|
|
|
int comparison = infoFromRanges.get(i).compareTo(infoFromLVT.get(j));
|
|
|
|
if (comparison == 0) {
|
|
|
|
i++; j++;
|
|
|
|
} else if (comparison > 0) {
|
|
|
|
j++;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i < infoFromRanges.size()) {
|
2023-09-07 15:37:25 +00:00
|
|
|
error(infoFromLVT, infoFromRanges, method.methodName().stringValue());
|
2013-09-14 19:04:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
List<String> convertToStringList(AliveRanges ranges) {
|
|
|
|
List<String> result = new ArrayList<>();
|
|
|
|
for (Annotation anno : ranges.value()) {
|
|
|
|
AliveRange range = (AliveRange)anno;
|
|
|
|
String str = formatLocalVariableData(range.varName(),
|
|
|
|
range.bytecodeStart(), range.bytecodeLength());
|
|
|
|
result.add(str);
|
|
|
|
}
|
|
|
|
Collections.sort(result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-09-07 15:37:25 +00:00
|
|
|
List<String> convertToStringList(LocalVariableTableAttribute lvt) {
|
2013-09-14 19:04:47 +01:00
|
|
|
List<String> result = new ArrayList<>();
|
2023-09-07 15:37:25 +00:00
|
|
|
for (LocalVariableInfo entry : lvt.localVariables()) {
|
|
|
|
String str = formatLocalVariableData(entry.name().stringValue(),
|
|
|
|
entry.startPc(), entry.length());
|
2013-09-14 19:04:47 +01:00
|
|
|
result.add(str);
|
|
|
|
}
|
|
|
|
Collections.sort(result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
String formatLocalVariableData(String varName, int start, int length) {
|
|
|
|
StringBuilder sb = new StringBuilder()
|
|
|
|
.append("var name: ").append(varName)
|
|
|
|
.append(" start: ").append(start)
|
|
|
|
.append(" length: ").append(length);
|
|
|
|
return sb.toString();
|
|
|
|
}
|
|
|
|
|
2015-01-05 17:35:48 -08:00
|
|
|
protected void error(List<String> infoFromLVT, List<String> infoFromRanges, String methodName) {
|
2013-09-14 19:04:47 +01:00
|
|
|
nerrors++;
|
|
|
|
System.err.printf("Error occurred while checking file: %s\n", jfo.getName());
|
2015-01-05 17:35:48 -08:00
|
|
|
System.err.printf("at method: %s\n", methodName);
|
2013-09-14 19:04:47 +01:00
|
|
|
System.err.println("The range info from the annotations is");
|
|
|
|
printStringListToErrOutput(infoFromRanges);
|
|
|
|
System.err.println();
|
|
|
|
System.err.println("And the range info from the class file is");
|
|
|
|
printStringListToErrOutput(infoFromLVT);
|
|
|
|
System.err.println();
|
|
|
|
}
|
|
|
|
|
|
|
|
void printStringListToErrOutput(List<String> list) {
|
|
|
|
for (String s : list) {
|
|
|
|
System.err.println("\t" + s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void error(String msg) {
|
|
|
|
nerrors++;
|
|
|
|
System.err.printf("Error occurred while checking file: %s\nreason: %s\n",
|
|
|
|
jfo.getName(), msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
class AliveRangeFinder extends JavacTestingAbstractProcessor {
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean process(Set<? extends TypeElement> annotations,
|
|
|
|
RoundEnvironment roundEnv) {
|
|
|
|
if (roundEnv.processingOver())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
TypeElement aliveRangeAnno = elements.getTypeElement("AliveRanges");
|
|
|
|
|
|
|
|
if (!annotations.contains(aliveRangeAnno)) {
|
|
|
|
error("no @AliveRanges annotation found in test class");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Element elem: roundEnv.getElementsAnnotatedWith(aliveRangeAnno)) {
|
|
|
|
Annotation annotation = elem.getAnnotation(AliveRanges.class);
|
|
|
|
aliveRangeMap.put(new ElementKey(elem), (AliveRanges)annotation);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class ElementKey {
|
|
|
|
|
|
|
|
String key;
|
|
|
|
Element elem;
|
|
|
|
|
|
|
|
public ElementKey(Element elem) {
|
|
|
|
this.elem = elem;
|
|
|
|
this.key = computeKey(elem);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean equals(Object obj) {
|
|
|
|
if (obj instanceof ElementKey) {
|
|
|
|
ElementKey other = (ElementKey)obj;
|
|
|
|
return other.key.equals(key);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int hashCode() {
|
|
|
|
return key.hashCode();
|
|
|
|
}
|
|
|
|
|
|
|
|
String computeKey(Element e) {
|
|
|
|
StringBuilder buf = new StringBuilder();
|
|
|
|
while (e != null) {
|
|
|
|
buf.append(e.toString());
|
|
|
|
e = e.getEnclosingElement();
|
|
|
|
}
|
|
|
|
buf.append(jfo.getName());
|
|
|
|
return buf.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String toString() {
|
|
|
|
return "Key{" + key + "}";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-07 15:37:25 +00:00
|
|
|
private String parse(String desc) {
|
|
|
|
int end = desc.indexOf(")");
|
|
|
|
if (end == -1)
|
|
|
|
throw new AssertionError();
|
|
|
|
end ++;
|
|
|
|
int p = 0;
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
int dims = 0;
|
|
|
|
|
|
|
|
while (p < end) {
|
|
|
|
String type;
|
|
|
|
switch (desc.charAt(p++)) {
|
|
|
|
case '(' -> {
|
|
|
|
sb.append('(');
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case ')' -> {
|
|
|
|
sb.append(')');
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case '[' -> {
|
|
|
|
dims++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case 'B' -> type = "byte";
|
|
|
|
case 'C' -> type = "char";
|
|
|
|
case 'D' -> type = "double";
|
|
|
|
case 'F' -> type = "float";
|
|
|
|
case 'I' -> type = "int";
|
|
|
|
case 'J' -> type = "long";
|
|
|
|
case 'L' -> {
|
|
|
|
int sep = desc.indexOf(';', p);
|
|
|
|
if (sep == -1)
|
|
|
|
throw new AssertionError();
|
|
|
|
type = desc.substring(p, sep).replace('/', '.');
|
|
|
|
p = sep + 1;
|
|
|
|
}
|
|
|
|
case 'S' -> type = "short";
|
|
|
|
case 'Z' -> type = "boolean";
|
|
|
|
case 'V' -> type = "void";
|
|
|
|
default -> throw new AssertionError();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sb.length() > 1 && sb.charAt(0) == '(')
|
|
|
|
sb.append(", ");
|
|
|
|
sb.append(type);
|
|
|
|
for ( ; dims > 0; dims-- )
|
|
|
|
sb.append("[]");
|
|
|
|
}
|
|
|
|
return sb.toString();
|
|
|
|
}
|
|
|
|
|
2013-09-14 19:04:47 +01:00
|
|
|
}
|