1 /*
   2  * Copyright 2003 Sun Microsystems, Inc.  All Rights Reserved.
   3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   4  *
   5  * This code is free software; you can redistribute it and/or modify it
   6  * under the terms of the GNU General Public License version 2 only, as
   7  * published by the Free Software Foundation.
   8  *
   9  * This code is distributed in the hope that it will be useful, but WITHOUT
  10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  12  * version 2 for more details (a copy is included in the LICENSE file that
  13  * accompanied this code).
  14  *
  15  * You should have received a copy of the GNU General Public License version
  16  * 2 along with this work; if not, write to the Free Software Foundation,
  17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  18  *
  19  * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
  20  * CA 95054 USA or visit www.sun.com if you need additional information or
  21  * have any questions.
  22  */
  23 
  24 /*
  25 * @test
  26 * @bug 4891872
  27 * @summary Some tests for the generic core reflection api.
  28 * @author Gilad Bracha
  29 * @compile -source 1.5 TestC2.java
  30 * @run main/othervm -ea TestC2
  31 */
  32 
  33 
  34 import java.lang.reflect.*;
  35 
  36 
  37 abstract class C0<T> {
  38 
  39     public T ft;
  40     public C0<T> fc1t;
  41     public C0 fc1;
  42 
  43     public C0(){}
  44     public C0(T t) {}
  45 
  46     public abstract  C0<T> mc1t(T t, C0<T> c1t, C0 c1);
  47 
  48     public abstract C0 mc1();
  49 
  50     public abstract T mt(T t);
  51 
  52 }
  53 
  54 interface I1<X1, X2> extends I3 {
  55 
  56     X1 foo(X2 x2);
  57 }
  58 
  59 interface I2<E1, E2 extends Throwable, E3> {
  60 
  61 
  62     E1 bar(E3 e3) throws E2;
  63 
  64 }
  65 
  66 interface I3 {
  67 
  68 
  69 }
  70 
  71 
  72 abstract class C2<T1 extends C2<T1, T2, T3>, T2 extends C0<T2>,
  73                                                         T3 extends Throwable>
  74     extends C0<T1>
  75     implements I1<T1, T2>, I2<T1, T3, T2>, I3
  76 {
  77 
  78     public T1 ft;
  79     public C0<String> fc1t;
  80     public C0 fc1;
  81     public int fi;
  82 
  83     public C2(T2 t2) {}
  84     public <T> C2(T t) {}
  85     public <T1, T2, T3, T4> C2(T1 t1, T2 t2, T4 t4) {}
  86     public C2() throws T3 {}
  87 
  88     public abstract <T>  C0<T> mc1t(T3 t3, C0<T> c1t, C0 c1);
  89 
  90     public abstract <E, R> C0 mc1(E e);
  91 
  92     public abstract T1 mt(T2 t);
  93 
  94 }
  95 
  96 public class TestC2 {
  97 
  98     static Class<C2> cls = C2.class;
  99 
 100 
 101     public static void main(String[] args) throws Throwable {
 102         testSuperclass();
 103         testSuperInterfaces();
 104         testTypeParameters();
 105         testMethods();
 106         testConstructors();
 107         testFields();
 108     }
 109 
 110     static void testSuperclass() {
 111 
 112         System.out.println("testing superclass");
 113         Type sc = cls.getGenericSuperclass();
 114         assert
 115             sc instanceof ParameterizedType :
 116             "Superclass of C2 should be a parameterized type";
 117         ParameterizedType psc = (ParameterizedType) sc;
 118         assert
 119             ((psc.getRawType() == C0.class) ) :
 120             "The raw generic superclass of C2 should be C0";
 121 
 122         Type[] tas = psc.getActualTypeArguments();
 123         assert
 124             tas.length == 1 :
 125             "Superclass of C2 should have one type argument";
 126 
 127         Type t = tas[0];
 128 
 129         assert
 130             t instanceof TypeVariable :
 131             "Type argument to superclass of C2 should be a type variable";
 132 
 133         TypeVariable tv = (TypeVariable) t;
 134         assert
 135             tv.getName().equals("T1") :
 136             "Name of type argument to superclass of C2 should be T1";
 137         Type[] bs = tv.getBounds();
 138         assert
 139             bs.length == 1 :
 140             "T1 has one bound (superclass)";
 141         t = bs[0];
 142         assert
 143             t instanceof ParameterizedType :
 144             "Bound of C0 should be a parameterized type";
 145         ParameterizedType pt = (ParameterizedType) t;
 146 
 147         assert
 148             ((pt.getRawType() == C2.class) ) :
 149             "The raw bound of T1 should be C2";
 150 
 151         tas = pt.getActualTypeArguments();
 152         assert
 153             tas.length == 3 :
 154             "Bound of T1 should have three type arguments";
 155         assert
 156             tas[0] instanceof TypeVariable :
 157             "First argument to bound of T1 is a type variable";
 158         assert
 159             tas[1] instanceof TypeVariable :
 160             "Second argument to bound of T1 is a type variable";
 161         assert
 162             tas[2] instanceof TypeVariable :
 163             "Third argument to bound of T1 is a type variable";
 164 
 165         TypeVariable tv1 = (TypeVariable) tas[0];
 166         TypeVariable tv2 = (TypeVariable) tas[1];
 167         TypeVariable tv3 = (TypeVariable) tas[2];
 168 
 169         assert
 170             tv1.getName().equals("T1"):
 171             "First type arg to bound of T1 is T1";
 172         assert
 173             tv2.getName().equals("T2"):
 174             "Seconmd type arg to bound of T1 is T2";
 175         assert
 176             tv3.getName().equals("T3"):
 177             "Third type arg to bound of T1 is T3";
 178 
 179 
 180     }
 181 
 182     static void testSuperInterfaces() {
 183         System.out.println("testing superinterfaces");
 184         Type[] sis = cls.getGenericInterfaces();
 185         assert
 186             ((sis.length == 3)):
 187             "C2 should have three generic superinterfaces";
 188 
 189         Type t = sis[0];
 190         assert
 191             t instanceof ParameterizedType :
 192             "First superinterface of C2 should be a parameterized type";
 193         ParameterizedType pt = (ParameterizedType) t;
 194         assert
 195             pt.getRawType() == I1.class :
 196             "First super interface of C2 is instantiation of I1";
 197         Type[] tas = pt.getActualTypeArguments();
 198         assert
 199             tas.length == 2 :
 200             "First super interface of C2 has 2 type arguments";
 201 
 202         t = sis[1];
 203         assert
 204             t instanceof ParameterizedType :
 205             "Second superinterface of C2 should be a parameterized type";
 206         pt = (ParameterizedType) t;
 207         assert
 208             pt.getRawType() == I2.class :
 209             "Second super interface of C2 is instantiation of I2";
 210         tas = pt.getActualTypeArguments();
 211         assert
 212             tas.length == 3 :
 213             "Second super interface of C2 has 3 type arguments";
 214 
 215         t = sis[2];
 216         assert
 217             t == I3.class :
 218             "Third superinterface of C2 is I3";
 219 
 220         // Test interfaces themselves
 221 
 222         TypeVariable[] tvs = I1.class.getTypeParameters();
 223         assert
 224             tvs.length == 2 :
 225             "I3 has two formal type parameters";
 226         assert
 227             tvs[0].getName().equals("X1") :
 228             "Name of first formal type arg of I1 is X1";
 229         assert
 230             tvs[1].getName().equals("X2") :
 231             "Name of second formal type arg of I1 is X2";
 232 
 233         assert
 234             I1.class.getGenericSuperclass() == I1.class.getSuperclass() :
 235             "The generic and non-generic superclasses of an interface must be the same";
 236         sis = I1.class.getGenericInterfaces();
 237         assert
 238             sis.length == 1 :
 239             "I1 has one generic superinterface";
 240         assert
 241             sis[0] == I3.class :
 242             "Superinterface of I1 is I3";
 243 
 244         tvs = I2.class.getTypeParameters();
 245         assert
 246             tvs.length == 3 :
 247             "I3 has three formal type parameters";
 248         assert
 249             tvs[0].getName().equals("E1") :
 250             "Name of first formal type arg of I2 is E1";
 251         assert
 252             tvs[1].getName().equals("E2") :
 253             "Name of second formal type arg of I2 is E2";
 254         assert
 255             tvs[2].getName().equals("E3") :
 256             "Name of third formal type arg of I2 is E3";
 257 
 258         assert
 259             I2.class.getGenericSuperclass() == I2.class.getSuperclass() :
 260             "The generic and non-generic superclasses of an interface must be the same";
 261 
 262         tvs = I3.class.getTypeParameters();
 263         assert
 264             tvs.length == 0 :
 265             "I3 has no formal type parameters";
 266 
 267         assert
 268             I3.class.getGenericSuperclass() == I3.class.getSuperclass() :
 269             "The generic and non-generic superclasses of an interface must be the same";
 270 
 271 
 272     }
 273 
 274     static void testTypeParameters() {
 275         System.out.println("testing type parameters");
 276         TypeVariable[] tvs = cls.getTypeParameters();
 277         assert
 278             tvs.length == 3 :
 279             "C2 should have three type parameters";
 280         TypeVariable tv = tvs[0];
 281         Type[] bs = tv.getBounds();
 282         assert
 283             bs.length == 1 :
 284             "T1 should have one bound";
 285         assert
 286             bs[0] instanceof ParameterizedType :
 287             "The bound of T1 should be a parameterized type";
 288 
 289         tv = tvs[1];
 290         bs = tv.getBounds();
 291         assert
 292             bs.length == 1 :
 293             "T2 should have one bound";
 294         assert
 295             bs[0] instanceof ParameterizedType :
 296             "The bound of T2 should be a parameterized type";
 297 
 298         tv = tvs[2];
 299         bs = tv.getBounds();
 300         assert
 301             bs.length == 1 :
 302             "T3 should have one bound";
 303         assert
 304             bs[0] == Throwable.class :
 305             "The bound of T3 should be Throwable";
 306     }
 307 
 308     static void testMethods() throws NoSuchMethodException {
 309         System.out.println("testing methods");
 310 
 311 
 312 
 313         Class[] params1 = new Class[3];
 314         params1[0] = Throwable.class;
 315         params1[1] = C0.class;
 316         params1[2] = C0.class;
 317 
 318         Class[] params2 = new Class[1];
 319         params2[0] = Object.class;
 320 
 321         Class[] params3 = new Class[1];
 322         params3[0] = C0.class;
 323 
 324         Method mc1t = cls.getMethod("mc1t", params1);
 325         Method mc1 = cls.getMethod("mc1", params2);
 326         Method mt = cls.getMethod("mt", params3);
 327 
 328         Type rt_mc1t = mc1t.getGenericReturnType();
 329         assert
 330             rt_mc1t  instanceof ParameterizedType :
 331             "The return type of mc1t should be a parameterized type";
 332         ParameterizedType pt = (ParameterizedType) rt_mc1t;
 333 
 334         assert
 335             pt.getRawType() == C0.class :
 336             "The raw return type of mc1t should be C0";
 337 
 338         Type[] tas = pt.getActualTypeArguments();
 339         assert
 340             tas.length == 1 :
 341             "Return type of mc1t should have one type argument";
 342         assert
 343             tas[0] instanceof TypeVariable :
 344             "Type argument of return type of mc1t is a type variable";
 345 
 346         Type rt_mc1 = mc1.getGenericReturnType();
 347         assert
 348             rt_mc1 == C0.class :
 349             "Return type of mc1 is C0";
 350 
 351         Type rt_mt = mt.getGenericReturnType();
 352         assert
 353             rt_mt instanceof TypeVariable :
 354             "Return type of mt is a type variable";
 355 
 356         Type[] pt_mc1t = mc1t.getGenericParameterTypes();
 357 
 358         assert
 359             pt_mc1t.length == 3 :
 360             "C0.mc1t has three parameters";
 361         Type p1_mc1t = pt_mc1t[0];
 362         assert p1_mc1t != null;
 363         assert
 364             p1_mc1t instanceof TypeVariable :
 365             "Generic type of the 1st parameter of mc1t(T) is a type variable";
 366         TypeVariable tv = (TypeVariable) p1_mc1t;
 367 
 368         assert
 369             tv.getName().equals("T3") :
 370             "Name of 1st type parameter of mc1t is T3, not " + tv.getName();
 371         Type[] bs = tv.getBounds();
 372         assert
 373             bs.length == 1 :
 374             "T3 should have one bound (mc1t)";
 375         assert
 376             bs[0] == Throwable.class :
 377             "The bound of T3 should be Throwable(mc1t)";
 378 
 379         Type p2_mc1t = pt_mc1t[1];
 380         assert
 381             p2_mc1t instanceof ParameterizedType :
 382             "The type of parameter 2 of mc1t is a parameterized type";
 383         pt = (ParameterizedType) p2_mc1t;
 384         assert
 385             pt.getRawType() == C0.class :
 386             "Type of parameter 2 of mc1t is instantiation of C0";
 387         assert
 388             pt.getOwnerType() == null :
 389             "Type of parameter 2 of mc1t is has null owner";
 390 
 391         tas = pt.getActualTypeArguments();
 392         assert
 393             tas.length == 1 :
 394             "The type of parameter 2 of mc1t has one type argument";
 395         Type ta = tas[0];
 396 
 397         assert
 398             ta instanceof TypeVariable :
 399             "The actual type arg of C0<T> is a type variable (mc1t)";
 400         tv = (TypeVariable) ta;
 401         assert
 402             tv.getName().equals("T") :
 403             "mc1t: Name of the type arg of C0<T> is T, not " + tv.getName();
 404         bs = tv.getBounds();
 405         assert
 406             bs.length == 1 :
 407             "mc1t: The type argument of C0<T>  should have one bound";
 408         assert
 409             bs[0] == Object.class :
 410             "mc1t: The bound of the type arg of C0<T> should be Object";
 411 
 412         Type p3_mc1t = pt_mc1t[2];
 413         assert
 414             p3_mc1t == C0.class :
 415             "Type of parameter 3 of mc1t is C0";
 416 
 417         Type[] pt_mc1 = mc1.getGenericParameterTypes();
 418         assert
 419             pt_mc1.length == 1 :
 420             "C2.mc1 has one parameter";
 421 
 422         Type[] pt_mt = mt.getGenericParameterTypes();
 423         assert
 424             pt_mt.length == 1 :
 425             "C2.mt has one parameter";
 426         Type p_mt = pt_mt[0];
 427         assert
 428             p_mt instanceof TypeVariable :
 429             "The generic type of the parameter of mt(T) is a type variable";
 430         tv = (TypeVariable) p_mt;
 431         assert
 432             tv.getName().equals("T2") :
 433             "The name of the type parameter of mt is T2, not " + tv.getName();
 434         bs = tv.getBounds();
 435         assert
 436             bs.length == 1 :
 437             "T2 should have one bound";
 438         assert
 439             bs[0] instanceof ParameterizedType:
 440             "The bound of T2 should be parameterized type";
 441 
 442         Type[] et_mc1t = mc1t.getGenericExceptionTypes();
 443         assert
 444             et_mc1t.length == 0 :
 445             "Method C0.mc1t should have no generic exception types";
 446 
 447         Type[] et_mc1 = mc1.getGenericExceptionTypes();
 448         assert
 449             et_mc1.length == 0 :
 450             "Method C0.mc1 should have no generic exception types";
 451 
 452         Type[] et_mt = mt.getGenericExceptionTypes();
 453         assert
 454             et_mt.length == 0 :
 455             "Method C0.mt should have no generic exception types";
 456 
 457 
 458         TypeVariable[] tv_mc1t = mc1t.getTypeParameters();
 459         assert
 460             tv_mc1t.length == 1 :
 461             "Method C2.mc1t should have one type parameter";
 462 
 463         TypeVariable[] tv_mc1 = mc1.getTypeParameters();
 464         assert
 465             tv_mc1.length == 2 :
 466             "Method C2.mc1 should have two type parameters";
 467 
 468         TypeVariable[] tv_mt = mt.getTypeParameters();
 469         assert
 470             tv_mt.length == 0 :
 471             "Method C2.mt should have no type parameters";
 472     }
 473 
 474 
 475     static void testFields() throws NoSuchFieldException{
 476         System.out.println("testing fields");
 477         Field ft = cls. getField("ft");
 478         Field fc1t = cls. getField("fc1t");
 479         Field fc1 = cls. getField("fc1");
 480         Field fi = cls. getField("fi");
 481 
 482         Type gt_ft = ft.getGenericType();
 483         assert
 484             gt_ft instanceof TypeVariable :
 485             "The generic type of C0.ft is a type variable";
 486         TypeVariable tv = (TypeVariable) gt_ft;
 487         assert
 488             tv.getName().equals("T1") :
 489             "The name of the type of ft is T1, not " + tv.getName();
 490         Type[] bs = tv.getBounds();
 491         assert
 492             bs.length == 1 :
 493             "The type of ft should have one bound";
 494 
 495 
 496         Type gt_fc1t = fc1t.getGenericType();
 497         assert
 498             gt_fc1t instanceof ParameterizedType :
 499             "The generic type of C0.fc1t is a parameterized type";
 500         ParameterizedType pt = (ParameterizedType) gt_fc1t;
 501         assert
 502             pt.getRawType() == C0.class :
 503             "Type of C2.fc1t is an instantiation of C0";
 504         assert
 505             pt.getOwnerType() == null :
 506             "Type of C2.fc1t is has null owner";
 507         Type[] tas = pt.getActualTypeArguments();
 508         assert
 509             tas.length == 1 :
 510             "The type of fc1t has one type argument";
 511         Type ta = tas[0];
 512 
 513         assert
 514             ta == String.class :
 515             "The actual type arg of C0<String> is String";
 516 
 517 
 518         Type gt_fc1 = fc1.getGenericType();
 519         assert
 520             gt_fc1 == C0.class :
 521             " Type of C2.fc1 should be C0";
 522 
 523         Type gt_fi = fi.getGenericType();
 524         assert
 525             gt_fi == int.class:
 526             " Type of C2.fi should be int";
 527 
 528     }
 529 
 530     static void testConstructors() throws NoSuchMethodException {
 531         System.out.println("testing constructors");
 532         Class[] params1 = new Class[1];
 533         params1[0] = C0.class;
 534         Constructor<C2> con = cls.getDeclaredConstructor(params1);
 535 
 536         Type[] pt_con = con.getGenericParameterTypes();
 537         assert
 538             pt_con.length == 1 :
 539             "Constructor C0(T) should have one generic parameter type";
 540         Type pt = pt_con[0];
 541         assert
 542             pt instanceof TypeVariable :
 543             "The generic type of the parameter of C0(T2) is a type variable";
 544         TypeVariable tv = (TypeVariable) pt;
 545         assert
 546             tv.getName().equals("T2") :
 547             "The name of the type parameter of C2 is T2, not " + tv.getName();
 548         Type[] bs = tv.getBounds();
 549         assert
 550             bs.length == 1 :
 551             "T should have one bound";
 552 
 553 
 554         Type[] et_con = con.getGenericExceptionTypes();
 555         assert
 556             et_con.length == 0 :
 557             "Constructor C2(T2) should have no generic exception types";
 558 
 559         TypeVariable[] tv_con = con.getTypeParameters();
 560         assert
 561             tv_con.length == 0 :
 562             "Constructor C2(T2) should have no type parameters";
 563 
 564 
 565         Class[] params2 = new Class[1];
 566         params2[0] = Object.class;
 567         con = cls.getDeclaredConstructor(params2);
 568 
 569         pt_con = con.getGenericParameterTypes();
 570         assert
 571             pt_con.length == 1 :
 572             "Constructor C0(T) should have one generic parameter type";
 573         pt = pt_con[0];
 574         assert
 575             pt instanceof TypeVariable :
 576             "The generic type of the parameter of C2(T) is a type variable";
 577         tv = (TypeVariable) pt;
 578         assert
 579             tv.getName().equals("T") :
 580             "The name of the type parameter of C2 is T, not " + tv.getName();
 581         bs = tv.getBounds();
 582         assert
 583             bs.length == 1 :
 584             "T should have one bound";
 585 
 586 
 587         et_con = con.getGenericExceptionTypes();
 588         assert
 589             et_con.length == 0 :
 590             "Constructor C2(T) should have no generic exception types";
 591 
 592         tv_con = con.getTypeParameters();
 593         assert
 594             tv_con.length == 1 :
 595             "Constructor C2(T) should have one type parameter";
 596 
 597         Class[] params3 = new Class[3];
 598         params3[0] = Object.class;
 599         params3[1] = Object.class;
 600         params3[2] = Object.class;
 601 
 602         con = cls.getDeclaredConstructor(params3);
 603 
 604         pt_con = con.getGenericParameterTypes();
 605         assert
 606             pt_con.length == 3 :
 607             "Constructor C2(T1,T2,T4) should have three generic parameter types";
 608         pt = pt_con[0];
 609         assert
 610             pt instanceof TypeVariable :
 611             "The generic type of the first parameter of C2(T1,T2,T4) is a type variable";
 612         tv = (TypeVariable) pt;
 613         assert
 614             tv.getName().equals("T1") :
 615             "The name of the type parameter of C2(T1,T2,T4) is T1, not " + tv.getName();
 616         bs = tv.getBounds();
 617         assert
 618             bs.length == 1 :
 619             "T should have one bound";
 620 
 621 
 622         et_con = con.getGenericExceptionTypes();
 623         assert
 624             et_con.length == 0 :
 625             "Constructor C2(T1,T2,T4) should have no generic exception types";
 626 
 627         tv_con = con.getTypeParameters();
 628         assert
 629             tv_con.length == 4 :
 630             "Constructor C2(T1,T2,T4) should have four type parameters";
 631 
 632         Class[] params4 = new Class[0];
 633         con = cls.getDeclaredConstructor(params4);
 634 
 635         pt_con = con.getGenericParameterTypes();
 636         assert
 637             pt_con.length == 0 :
 638             "Constructor C2() should have no generic parameter types";
 639 
 640 
 641         et_con = con.getGenericExceptionTypes();
 642         assert
 643             et_con.length == 1 :
 644             "Constructor C2() should have one generic exception type";
 645 
 646         tv_con = con.getTypeParameters();
 647         assert
 648             tv_con.length == 0 :
 649             "Constructor C2() should have no type parameters";
 650 
 651 
 652     }
 653 }