/* * Copyright (c) 2008, 2013, 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. */ import java.lang.annotation.*; import java.util.Map; /* * @test * @bug 8006775 * @summary new type annotation location: nested types * @author Werner Dietl * @compile NestedTypes.java */ class Outer { class Inner { class Inner2 { // m1a-c all have the same parameter type. void m1a(@A Inner2 p1a) {} void m1b(Inner.@A Inner2 p1b) {} void m1c(Outer.Inner.@A Inner2 p1c) {} // notice the difference to m1d void m1d(@A Outer.Inner.Inner2 p1d) {} // m2a-b both have the same parameter type. void m2a(@A Inner.Inner2 p2a) {} void m2b(Outer.@A Inner.Inner2 p2b) {} // The location for @A is the same in m3a-c void m3a(@A Outer p3a) {} void m3b(@A Outer.Inner p3b) {} void m3c(@A Outer.Inner.Inner2 p3c) {} // Test combinations void m4a(@A Outer p3a) {} void m4b(@A Outer. @B Inner p3b) {} void m4c(@A Outer. @B Inner. @C Inner2 p3c) {} } } void m4a(@A Map p4a) {} void m4b(Map.@B Entry p4c) {} // Illegal: // void m4b(@A Map.Entry p4b) {} // void m4c(@A Map.@B Entry p4c) {} void m4c(Map<String,String>.@B Entry<String,String> p4d) {} // Illegal: // void m4d(@A Map<String,String>.@B Entry<String,String> p4d) {} void m4e(MyList<Map.Entry> p4e) {} void m4f(MyList<Map.@B Entry> p4f) {} // Illegal: // void m4g(MyList<@A Map.Entry> p4e) {} // void m4h(MyList<@A Map.@B Entry> p4f) {} class GInner<X> { class GInner2<Y, Z> {} } static class Static {} static class GStatic<X, Y> { static class GStatic2<Z> {} } } class Test1 { // Outer.GStatic<Object,Object>.GStatic2<Object> gs; Outer.GStatic.@A GStatic2<Object> gsgood; // TODO: add failing test // Outer.@A GStatic.GStatic2<Object> gsbad; MyList<@A Outer . @B Inner. @C Inner2> f; @A Outer .GInner<Object>.GInner2<String, Integer> g; // TODO: Make sure that something like this fails gracefully: // MyList<java.@B lang.Object> pkg; @A Outer f1; @A Outer . @B Inner f2 = f1.new @B Inner(); // TODO: ensure type annos on new are stored. @A Outer . @B GInner<@C Object> f3 = f1.new @B GInner<@C Object>(); MyList<@A Outer . @B GInner<@C MyList<@D Object>>. @E GInner2<@F Integer, @G Object>> f4; // MyList<Outer.GInner<Object>.GInner2<Integer>> f4clean; @A Outer . @B GInner<@C MyList<@D Object>>. @E GInner2<@F Integer, @G Object> f4top; MyList<@A Outer . @B GInner<@C MyList<@D Object @E[] @F[]>>. @G GInner2<@H Integer, @I Object> @J[] @K[]> f4arr; @A Outer . @B GInner<@C MyList<@D Object @E[] @F[]>>. @G GInner2<@H Integer, @I Object> @J[] @K[] f4arrtop; MyList<Outer . @B Static> f5; // Illegal: // MyList<@A Outer . @B Static> f5; Outer . @B Static f6; // Illegal: // @A Outer . @B Static f6; Outer . @Bv("B") GStatic<@Cv("C") String, @Dv("D") Object> f7; // Illegal: // @Av("A") Outer . @Bv("B") GStatic<@Cv("C") String, @Dv("D") Object> f7; Outer . @Cv("Data") Static f8; // Illegal: // @A Outer . @Cv("Data") Static f8; MyList<Outer . @Cv("Data") Static> f9; // Illegal: // MyList<@A Outer . @Cv("Data") Static> f9; } class Test2 { void m() { @A Outer f1 = null; @A Outer.@B Inner f2 = null; Outer.@B Static f3 = null; // Illegal: // @A Outer.@B Static f3 = null; @A Outer.@C Inner f4 = null; Outer . @B Static f5 = null; Outer . @Cv("Data") Static f6 = null; MyList<Outer . @Cv("Data") Static> f7 = null; } } class Test3 { void monster(@A Outer p1, @A Outer.@B Inner p2, Outer.@B Static p3, @A Outer.@Cv("Test") Inner p4, Outer . @B Static p5, Outer . @Cv("Data") Static p6, MyList<Outer . @Cv("Data") Static> p7) { } } class Test4 { void m() { @A Outer p1 = new @A Outer(); @A Outer.@B Inner p2 = p1.new @B Inner(); // Illegal: // @A Outer.@B Static p3 = new @A Outer.@B Static(); // Object o3 = new @A Outer.@B Static(); @A Outer.@Cv("Test") Inner p4 = p1.new @Cv("Test") Inner(); Outer . @B Static p5 = new Outer . @B Static(); Outer . @Cv("Data") Static p6 = new Outer . @Cv("Data") Static(); MyList<Outer . @Cv("Data") Static> p7 = new MyList<Outer . @Cv("Data") Static>(); } } class MyList<K> { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface A { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface B { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface C { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface D { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface E { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface F { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface G { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface H { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface I { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface J { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface K { } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface Av { String value(); } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface Bv { String value(); } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface Cv { String value(); } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface Dv { String value(); } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface Ev { String value(); } @Target({ElementType.TYPE_USE, ElementType.TYPE_PARAMETER}) @interface Fv { String value(); }