/* * Copyright (c) 2013, 2015, 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 7047734 8027660 8037937 8047719 8058708 8064857 * @summary The LVT is not generated correctly during some try/catch scenarios * javac crash while creating LVT entry for a local variable defined in * an inner block * @library /tools/javac/lib * @enablePreview * @modules java.base/jdk.internal.classfile.impl * @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; import java.lang.classfile.*; import java.lang.classfile.attribute.*; import static javax.tools.StandardLocation.*; import static javax.tools.JavaFileObject.Kind.SOURCE; 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 { try { String testDir = System.getProperty("test.src"); fm.setLocation(SOURCE_PATH, List.of(new File(testDir, "tests"))); // Make sure classes are written to scratch dir. fm.setLocation(CLASS_OUTPUT, List.of(new File("."))); 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(); } } JavaFileObject jfo; Map aliveRangeMap = new HashMap<>(); Set declaredKeys = new HashSet<>(); List seenAliveRanges = new ArrayList<>(); protected LVTHarness(JavaFileObject jfo) { this.jfo = jfo; } protected void check() throws Exception { JavacTask ct = (JavacTask) comp.getTask(null, fm, null, Arrays.asList("-g"), null, Arrays.asList(jfo)); System.err.println("compiling code " + jfo); ct.setProcessors(Collections.singleton(new AliveRangeFinder())); if (!ct.call()) { throw new AssertionError("Error during compilation"); } File javaFile = new File(jfo.getName()); File classFile = new File(javaFile.getName().replace(".java", ".class")); checkClassFile(classFile); //check all candidates have been used up for (Map.Entry entry : aliveRangeMap.entrySet()) { if (!seenAliveRanges.contains(entry.getKey())) { error("Redundant @AliveRanges annotation on method " + entry.getKey().elem + " with key " + entry.getKey()); } } } void checkClassFile(File file) throws IOException { ClassModel classFile = ClassFile.of().parse(file.toPath()); //lets get all the methods in the class file. for (MethodModel method : classFile.methods()) { for (ElementKey elementKey: aliveRangeMap.keySet()) { String methodDesc = method.methodName().stringValue() + parse(method.methodTypeSymbol().descriptorString()).replace(" ", ""); if (methodDesc.equals(elementKey.elem.toString())) { checkMethod(method, aliveRangeMap.get(elementKey)); seenAliveRanges.add(elementKey); } } } } void checkMethod(MethodModel method, AliveRanges ranges) { CodeAttribute code = method.findAttribute(Attributes.CODE).orElseThrow(); LocalVariableTableAttribute lvt = code.findAttribute(Attributes.LOCAL_VARIABLE_TABLE).orElseThrow(); List infoFromRanges = convertToStringList(ranges); List infoFromLVT = convertToStringList(lvt); // 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()) { error(infoFromLVT, infoFromRanges, method.methodName().stringValue()); } } List convertToStringList(AliveRanges ranges) { List 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; } List convertToStringList(LocalVariableTableAttribute lvt) { List result = new ArrayList<>(); for (LocalVariableInfo entry : lvt.localVariables()) { String str = formatLocalVariableData(entry.name().stringValue(), entry.startPc(), entry.length()); 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(); } protected void error(List infoFromLVT, List infoFromRanges, String methodName) { nerrors++; System.err.printf("Error occurred while checking file: %s\n", jfo.getName()); System.err.printf("at method: %s\n", methodName); 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 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 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 + "}"; } } 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(); } }