/* * Copyright (c) 2017, 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 8180141 * @summary Missing entry in LineNumberTable for break statement that jumps out of try-finally * @enablePreview * @modules java.base/jdk.internal.classfile.impl * jdk.compiler/com.sun.tools.javac.code * jdk.compiler/com.sun.tools.javac.comp * jdk.compiler/com.sun.tools.javac.file * jdk.compiler/com.sun.tools.javac.main * jdk.compiler/com.sun.tools.javac.tree * jdk.compiler/com.sun.tools.javac.util * @compile -g MissingLNTEntryForBreakContinueTest.java * @run main MissingLNTEntryForBreakContinueTest */ import java.io.File; import java.net.URI; import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import com.sun.tools.javac.comp.Attr; import com.sun.tools.javac.comp.AttrContext; import com.sun.tools.javac.comp.Env; import com.sun.tools.javac.comp.Modules; import com.sun.tools.javac.file.JavacFileManager; import com.sun.tools.javac.main.JavaCompiler; import com.sun.tools.javac.tree.JCTree; import com.sun.tools.javac.util.Context; import com.sun.tools.javac.util.List; import java.lang.classfile.*; import java.lang.classfile.attribute.*; import static com.sun.tools.javac.util.List.of; import static com.sun.tools.javac.tree.JCTree.Tag.*; public class MissingLNTEntryForBreakContinueTest { protected ReusableJavaCompiler tool; Context context; MissingLNTEntryForBreakContinueTest() { context = new Context(); JavacFileManager.preRegister(context); MyAttr.preRegister(context); tool = new ReusableJavaCompiler(context); } public static void main(String... args) throws Throwable { new MissingLNTEntryForBreakContinueTest().test(); } void test() throws Throwable { testFor("1", "break"); testFor("2", "continue"); } void testFor(String id, String statement) throws Throwable { JavaSource source = new JavaSource(id, statement); tool.clear(); List inputs = of(source); try { tool.compile(inputs); } catch (Throwable ex) { throw new AssertionError(ex); } File testClasses = new File("."); File file = new File(testClasses, "Test" + id + ".class"); ClassModel classFile = ClassFile.of().parse(file.toPath()); for (MethodModel m : classFile.methods()) { if (m.methodName().equalsString("foo")) { CodeAttribute code = m.findAttribute(Attributes.code()).orElseThrow(); LineNumberTableAttribute lnt = code.findAttribute(Attributes.lineNumberTable()).orElseThrow(); checkLNT(lnt, MyAttr.lineNumber); } } } void checkLNT(LineNumberTableAttribute lnt, int lineToCheckFor) { for (LineNumberInfo e: lnt.lineNumbers()) { if (e.lineNumber() == lineToCheckFor) { return; } } throw new AssertionError("seek line number not found in the LNT for method foo()"); } class JavaSource extends SimpleJavaFileObject { String statement; String id; String template = "class Test#Id {\n" + " void foo(boolean condition) {\n" + " while (true) {\n" + " try {\n" + " if (condition) {\n" + " #STM;\n" + " }\n" + " } finally {\n" + " System.out.println(\"finalizer\");\n" + " }\n" + " }\n" + " }" + "}"; JavaSource(String id, String statement) { super(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE); this.statement = statement; this.id = id; } @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) { return template.replace("#Id", id).replace("#STM", statement); } } /* this class has been set up to do not depend on a fixed line number, this Attr subclass will * look for 'break' or 'continue' statements in order to find the actual line number they occupy. * This way the test can find if that line number appears in the LNT generated for a given class. */ static class MyAttr extends Attr { static int lineNumber; static void preRegister(Context context) { context.put(attrKey, (com.sun.tools.javac.util.Context.Factory) c -> new MyAttr(c)); } MyAttr(Context context) { super(context); } @Override public com.sun.tools.javac.code.Type attribStat(JCTree tree, Env env) { com.sun.tools.javac.code.Type result = super.attribStat(tree, env); if (tree.hasTag(BREAK) || tree.hasTag(CONTINUE)) { lineNumber = env.toplevel.lineMap.getLineNumber(tree.pos); } return result; } } static class ReusableJavaCompiler extends JavaCompiler { ReusableJavaCompiler(Context context) { super(context); } @Override protected void checkReusable() { // do nothing } @Override public void close() { //do nothing } void clear() { newRound(); Modules.instance(context).newRound(); } } }