blob: 7572b4fefb24db1d29963c6fa6676090dfe49d07 [file] [log] [blame]
/*
* 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(); }