aoqi@0: /* aoqi@0: * Copyright (c) 2008, 2013, Oracle and/or its affiliates. All rights reserved. aoqi@0: * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. aoqi@0: * aoqi@0: * This code is free software; you can redistribute it and/or modify it aoqi@0: * under the terms of the GNU General Public License version 2 only, as aoqi@0: * published by the Free Software Foundation. aoqi@0: * aoqi@0: * This code is distributed in the hope that it will be useful, but WITHOUT aoqi@0: * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or aoqi@0: * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License aoqi@0: * version 2 for more details (a copy is included in the LICENSE file that aoqi@0: * accompanied this code). aoqi@0: * aoqi@0: * You should have received a copy of the GNU General Public License version aoqi@0: * 2 along with this work; if not, write to the Free Software Foundation, aoqi@0: * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. aoqi@0: * aoqi@0: * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA aoqi@0: * or visit www.oracle.com if you need additional information or have any aoqi@0: * questions. aoqi@0: */ aoqi@0: aoqi@0: import java.io.*; aoqi@0: import java.lang.annotation.ElementType; aoqi@0: aoqi@0: import com.sun.tools.classfile.*; aoqi@0: aoqi@0: /* aoqi@0: * @test Presence aoqi@0: * @bug 6843077 aoqi@0: * @summary test that all type annotations are present in the classfile aoqi@0: */ aoqi@0: aoqi@0: public class Presence { aoqi@0: public static void main(String[] args) throws Exception { aoqi@0: new Presence().run(); aoqi@0: } aoqi@0: aoqi@0: public void run() throws Exception { aoqi@0: File javaFile = writeTestFile(); aoqi@0: File classFile = compileTestFile(javaFile); aoqi@0: aoqi@0: ClassFile cf = ClassFile.read(classFile); aoqi@0: test(cf); aoqi@0: for (Field f : cf.fields) { aoqi@0: test(cf, f); aoqi@0: } aoqi@0: for (Method m: cf.methods) { aoqi@0: test(cf, m); aoqi@0: } aoqi@0: aoqi@0: countAnnotations(); aoqi@0: aoqi@0: if (errors > 0) aoqi@0: throw new Exception(errors + " errors found"); aoqi@0: System.out.println("PASSED"); aoqi@0: } aoqi@0: aoqi@0: void test(ClassFile cf) { aoqi@0: test(cf, Attribute.RuntimeVisibleTypeAnnotations, true); aoqi@0: test(cf, Attribute.RuntimeInvisibleTypeAnnotations, false); aoqi@0: } aoqi@0: aoqi@0: void test(ClassFile cf, Method m) { aoqi@0: test(cf, m, Attribute.RuntimeVisibleTypeAnnotations, true); aoqi@0: test(cf, m, Attribute.RuntimeInvisibleTypeAnnotations, false); aoqi@0: } aoqi@0: aoqi@0: void test(ClassFile cf, Field m) { aoqi@0: test(cf, m, Attribute.RuntimeVisibleTypeAnnotations, true); aoqi@0: test(cf, m, Attribute.RuntimeInvisibleTypeAnnotations, false); aoqi@0: } aoqi@0: aoqi@0: // test the result of Attributes.getIndex according to expectations aoqi@0: // encoded in the method's name aoqi@0: void test(ClassFile cf, String name, boolean visible) { aoqi@0: int index = cf.attributes.getIndex(cf.constant_pool, name); aoqi@0: if (index != -1) { aoqi@0: Attribute attr = cf.attributes.get(index); aoqi@0: assert attr instanceof RuntimeTypeAnnotations_attribute; aoqi@0: RuntimeTypeAnnotations_attribute tAttr = (RuntimeTypeAnnotations_attribute)attr; aoqi@0: all += tAttr.annotations.length; aoqi@0: if (visible) aoqi@0: visibles += tAttr.annotations.length; aoqi@0: else aoqi@0: invisibles += tAttr.annotations.length; aoqi@0: } aoqi@0: } aoqi@0: aoqi@0: // test the result of Attributes.getIndex according to expectations aoqi@0: // encoded in the method's name aoqi@0: void test(ClassFile cf, Method m, String name, boolean visible) { aoqi@0: Attribute attr = null; aoqi@0: Code_attribute cAttr = null; aoqi@0: RuntimeTypeAnnotations_attribute tAttr = null; aoqi@0: aoqi@0: // collect annotations attributes on method aoqi@0: int index = m.attributes.getIndex(cf.constant_pool, name); aoqi@0: if (index != -1) { aoqi@0: attr = m.attributes.get(index); aoqi@0: assert attr instanceof RuntimeTypeAnnotations_attribute; aoqi@0: tAttr = (RuntimeTypeAnnotations_attribute)attr; aoqi@0: all += tAttr.annotations.length; aoqi@0: if (visible) aoqi@0: visibles += tAttr.annotations.length; aoqi@0: else aoqi@0: invisibles += tAttr.annotations.length; aoqi@0: } aoqi@0: // collect annotations from method's code attribute aoqi@0: index = m.attributes.getIndex(cf.constant_pool, Attribute.Code); aoqi@0: if(index!= -1) { aoqi@0: attr = m.attributes.get(index); aoqi@0: assert attr instanceof Code_attribute; aoqi@0: cAttr = (Code_attribute)attr; aoqi@0: index = cAttr.attributes.getIndex(cf.constant_pool, name); aoqi@0: if(index!= -1) { aoqi@0: attr = cAttr.attributes.get(index); aoqi@0: assert attr instanceof RuntimeTypeAnnotations_attribute; aoqi@0: tAttr = (RuntimeTypeAnnotations_attribute)attr; aoqi@0: all += tAttr.annotations.length; aoqi@0: if (visible) aoqi@0: visibles += tAttr.annotations.length; aoqi@0: else aoqi@0: invisibles += tAttr.annotations.length; aoqi@0: } aoqi@0: } aoqi@0: } aoqi@0: aoqi@0: // test the result of Attributes.getIndex according to expectations aoqi@0: // encoded in the method's name aoqi@0: void test(ClassFile cf, Field m, String name, boolean visible) { aoqi@0: int index = m.attributes.getIndex(cf.constant_pool, name); aoqi@0: if (index != -1) { aoqi@0: Attribute attr = m.attributes.get(index); aoqi@0: assert attr instanceof RuntimeTypeAnnotations_attribute; aoqi@0: RuntimeTypeAnnotations_attribute tAttr = (RuntimeTypeAnnotations_attribute)attr; aoqi@0: all += tAttr.annotations.length; aoqi@0: if (visible) aoqi@0: visibles += tAttr.annotations.length; aoqi@0: else aoqi@0: invisibles += tAttr.annotations.length; aoqi@0: } aoqi@0: } aoqi@0: aoqi@0: File writeTestFile() throws IOException { aoqi@0: File f = new File("TestPresence.java"); aoqi@0: PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(f))); aoqi@0: out.println("import java.util.*;"); aoqi@0: out.println("import java.lang.annotation.*;"); aoqi@0: aoqi@0: out.println("class TestPresence<@TestPresence.A T extends @TestPresence.A List<@TestPresence.A String>> { "); aoqi@0: out.println(" @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER})"); aoqi@0: out.println(" @interface A { }"); aoqi@0: aoqi@0: out.println(" Map<@A String, Map<@A String, @A String>> f1;"); aoqi@0: aoqi@0: out.println(" <@A TM extends @A List<@A String>>"); aoqi@0: out.println(" Map<@A String, @A List<@A String>>"); aoqi@0: out.println(" method(@A TestPresence this, List<@A String> @A [] param1, String @A [] @A ... param2)"); aoqi@0: out.println(" throws @A Exception {"); aoqi@0: out.println(" @A String lc1 = null;"); aoqi@0: out.println(" @A List<@A String> lc2 = null;"); aoqi@0: out.println(" @A String @A [] [] @A[] lc3 = null;"); aoqi@0: out.println(" List> lc4 = null;"); aoqi@0: out.println(" Object lc5 = (@A List<@A String>) null;"); aoqi@0: out.println(" boolean lc6 = lc1 instanceof @A String;"); aoqi@0: out.println(" boolean lc7 = lc5 instanceof @A String @A [] @A [];"); aoqi@0: out.println(" new @A ArrayList<@A String>();"); aoqi@0: out.println(" Object lc8 = new @A String @A [4];"); aoqi@0: out.println(" try {"); aoqi@0: out.println(" Object lc10 = int.class;"); aoqi@0: out.println(" } catch (@A Exception e) { e.toString(); }"); aoqi@0: out.println(" return null;"); aoqi@0: out.println(" }"); aoqi@0: out.println(" void vararg1(String @A ... t) { } "); aoqi@0: out.println("}"); aoqi@0: out.close(); aoqi@0: return f; aoqi@0: } aoqi@0: aoqi@0: File compileTestFile(File f) { aoqi@0: int rc = com.sun.tools.javac.Main.compile(new String[] { "-source", "1.8", "-g", f.getPath() }); aoqi@0: if (rc != 0) aoqi@0: throw new Error("compilation failed. rc=" + rc); aoqi@0: String path = f.getPath(); aoqi@0: return new File(path.substring(0, path.length() - 5) + ".class"); aoqi@0: } aoqi@0: aoqi@0: void countAnnotations() { aoqi@0: int expected_visibles = 0, expected_invisibles = 38; aoqi@0: int expected_all = expected_visibles + expected_invisibles; aoqi@0: aoqi@0: if (expected_all != all) { aoqi@0: errors++; aoqi@0: System.err.println("expected " + expected_all aoqi@0: + " annotations but found " + all); aoqi@0: } aoqi@0: aoqi@0: if (expected_visibles != visibles) { aoqi@0: errors++; aoqi@0: System.err.println("expected " + expected_visibles aoqi@0: + " visibles annotations but found " + visibles); aoqi@0: } aoqi@0: aoqi@0: if (expected_invisibles != invisibles) { aoqi@0: errors++; aoqi@0: System.err.println("expected " + expected_invisibles aoqi@0: + " invisibles annotations but found " + invisibles); aoqi@0: } aoqi@0: aoqi@0: } aoqi@0: aoqi@0: int errors; aoqi@0: int all; aoqi@0: int visibles; aoqi@0: int invisibles; aoqi@0: }