test/jdk/java/lang/invoke/SpecialInterfaceCall.java
author dholmes
Sat, 23 Jun 2018 01:32:41 -0400
changeset 50735 2f2af62dfac7
parent 50041 86c6968ff67a
permissions -rw-r--r--
8010319: Implementation of JEP 181: Nest-Based Access Control Reviewed-by: alanb, psandoz, mchung, coleenp, acorn, mcimadamore, forax, jlahoda, sspitsyn, abuckley Contributed-by: alex.buckley@oracle.com, maurizio.mimadamore@oracle.com, mandy.chung@oracle.com, tobias.hartmann@oracle.com, david.holmes@oracle.com, vladimir.x.ivanov@oracle.com, karen.kinnear@oracle.com, vladimir.kozlov@oracle.com, john.r.rose@oracle.com, daniel.smith@oracle.com, serguei.spitsyn@oracle.com, kumardotsrinivasan@gmail.com, boris.ulasevich@bell-sw.com
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     1
/*
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     2
 * Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     4
 *
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     7
 * published by the Free Software Foundation.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     8
 *
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    13
 * accompanied this code).
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    14
 *
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    18
 *
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    21
 * questions.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    22
 */
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    23
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    24
/**
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    25
 * @test
50735
2f2af62dfac7 8010319: Implementation of JEP 181: Nest-Based Access Control
dholmes
parents: 50041
diff changeset
    26
 * @bug 8200167 8010319
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    27
 * @summary Test direct and MethodHandle access to interface methods using invokespecial semantics
50735
2f2af62dfac7 8010319: Implementation of JEP 181: Nest-Based Access Control
dholmes
parents: 50041
diff changeset
    28
 * @comment This must be compiled so invokespecial is used
2f2af62dfac7 8010319: Implementation of JEP 181: Nest-Based Access Control
dholmes
parents: 50041
diff changeset
    29
 * @compile -XDdisableVirtualizedPrivateInvoke SpecialInterfaceCall.java
50016
947f79c91b35 8202465: [C1] casts should not be eliminated for interface types
vlivanov
parents: 49935
diff changeset
    30
 * @compile SpecialInterfaceCallI4.jasm
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    31
 * @run main/othervm -Xint SpecialInterfaceCall
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    32
 * @run main/othervm -Xbatch -XX:+TieredCompilation -XX:TieredStopAtLevel=1 SpecialInterfaceCall
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    33
 * @run main/othervm -Xbatch -XX:-TieredCompilation SpecialInterfaceCall
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    34
 */
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    35
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    36
import java.lang.invoke.*;
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    37
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    38
public class SpecialInterfaceCall {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    39
    interface I1 {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    40
        default void pub_m() {};
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    41
        private void priv_m() {};
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    42
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    43
    interface I2 extends I1 {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    44
        // This needs to be a public method to avoid access control issues,
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    45
        // but logically we treat it as private and emulate invokespecial
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    46
        // using MethodHandles.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    47
        default void pub_m() {};
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    48
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    49
        private void priv_m() {};
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    50
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    51
        static void invokeDirect(I2 i) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    52
            i.priv_m(); // generates invokespecial
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    53
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    54
        static void invokeSpecialMH(I2 i) throws Throwable {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    55
            // emulates behaviour of invokeDirect
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    56
            mh_I2_priv_m_from_I2.invokeExact(i);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    57
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    58
        // special case of invoking an Object method via an interface
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    59
        static void invokeSpecialObjectMH(I2 i) throws Throwable {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    60
            // emulates invokespecial of I1.toString on i, which resolves
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    61
            // to Object.toString
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    62
            String s = (String) mh_I1_toString_from_I2.invokeExact(i);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    63
        }
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    64
        // special case of invoking a final Object method via an interface
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    65
        static void invokeSpecialObjectFinalMH(I2 i) throws Throwable {
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    66
            // emulates invokespecial of I1.getClass on i, which resolves
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    67
            // to Object.getClass
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    68
            Class<?> c = (Class<?>) mh_I1_getClass_from_I2.invokeExact(i);
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    69
        }
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    70
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    71
    interface I3 extends I2 {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    72
        // Must take an I3 here rather than I2 else we get
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    73
        // WrongMethodTypeException: expected (I3)void but found (I2)void
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    74
        // Statically the receiver type is bounded by the caller type.
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    75
        static void invokeSpecialMH(I3 i) throws Throwable {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    76
            // emulates an invokespecial of ((I2)i).pub_m()
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    77
            mh_I2_pub_m_from_I3.invokeExact(i);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    78
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    79
    }
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    80
    // This interface acts like I2 but we define directInvoke* methods
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    81
    // that we will rewrite the bytecode of to use invokespecial
50016
947f79c91b35 8202465: [C1] casts should not be eliminated for interface types
vlivanov
parents: 49935
diff changeset
    82
    // (see SpecialInterfaceCallI4.jasm).
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    83
    interface I4 extends I1 {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    84
        static void invokeDirect(I4 i) {
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    85
            // invokeSpecial Object.toString()
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    86
            throw new Error("Class file for I4 is not overwritten");
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    87
        }
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    88
        static void invokeDirectFinal(I4 i) {
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
    89
            // invokeSpecial Object.getClass() - final method
50016
947f79c91b35 8202465: [C1] casts should not be eliminated for interface types
vlivanov
parents: 49935
diff changeset
    90
            throw new Error("Class file for I4 is not overwritten");
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    91
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    92
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    93
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    94
    // Concrete classes
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    95
    static class C1 implements I1 { }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    96
    static class C2 implements I2 { }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    97
    static class C3 implements I3 { }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    98
    static class C4 implements I4 { }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
    99
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   100
    // Classes that don't implement I2/I3 but do have a
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   101
    // priv_m/pub_m method in their hierarchy
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   102
    static class D1 implements I1 { }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   103
    static class E {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   104
        public void pub_m() {}
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   105
        private void priv_m() {}
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   106
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   107
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   108
    // This MH acts like the direct invokespecial in I2.invokeDirect
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   109
    static final MethodHandle mh_I2_priv_m_from_I2;
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   110
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   111
    // This MH acts like an invokespecial of I2.pub_m from I3
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   112
    static final MethodHandle mh_I2_pub_m_from_I3;
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   113
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   114
    // This MH acts likes an invokespecial of I1.toString from I2
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   115
    static final MethodHandle mh_I1_toString_from_I2;
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   116
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   117
    // This MH acts likes an invokespecial of I1.getClass from I2
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   118
    static final MethodHandle mh_I1_getClass_from_I2;
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   119
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   120
    static {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   121
        try {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   122
            MethodType mt = MethodType.methodType(void.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   123
            MethodHandles.Lookup lookup = MethodHandles.lookup();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   124
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   125
            mh_I2_priv_m_from_I2 = lookup.findSpecial(I2.class, "priv_m", mt, I2.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   126
            mh_I2_pub_m_from_I3 = lookup.findSpecial(I2.class, "pub_m", mt, I3.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   127
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   128
            mt = MethodType.methodType(String.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   129
            mh_I1_toString_from_I2 = lookup.findSpecial(I1.class, "toString", mt, I2.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   130
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   131
            mt = MethodType.methodType(Class.class);
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   132
            mh_I1_getClass_from_I2 = lookup.findSpecial(I1.class, "getClass", mt, I2.class);
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   133
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   134
        } catch (Throwable e) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   135
            throw new Error(e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   136
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   137
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   138
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   139
    static void runPositiveTests() {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   140
        shouldNotThrow(() -> I2.invokeDirect(new C2()));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   141
        shouldNotThrow(() -> I2.invokeDirect(new C3()));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   142
        shouldNotThrow(() -> I2.invokeSpecialMH(new C2()));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   143
        shouldNotThrow(() -> I2.invokeSpecialMH(new C3()));
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   144
        shouldNotThrow(() -> I2.invokeSpecialObjectMH(new C2()));
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   145
        shouldNotThrow(() -> I2.invokeSpecialObjectMH(new C3()));
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   146
        shouldNotThrow(() -> I2.invokeSpecialObjectFinalMH(new C2()));
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   147
        shouldNotThrow(() -> I2.invokeSpecialObjectFinalMH(new C3()));
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   148
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   149
        shouldNotThrow(() -> I3.invokeSpecialMH(new C3()));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   150
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   151
        shouldNotThrow(() -> I4.invokeDirect(new C4()));
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   152
        shouldNotThrow(() -> I4.invokeDirectFinal(new C4()));
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   153
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   154
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   155
    static void runNegativeTests() {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   156
        System.out.println("IAE I2.invokeDirect D1");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   157
        shouldThrowIAE(() -> I2.invokeDirect(unsafeCastI2(new D1())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   158
        System.out.println("IAE I2.invokeDirect E");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   159
        shouldThrowIAE(() -> I2.invokeDirect(unsafeCastI2(new E())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   160
        System.out.println("ICCE I2.invokeMH D1");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   161
        shouldThrowICCE(() -> I2.invokeSpecialMH(unsafeCastI2(new D1())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   162
        System.out.println("ICCE I2.invokeMH E");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   163
        shouldThrowICCE(() -> I2.invokeSpecialMH(unsafeCastI2(new E())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   164
        System.out.println("ICCE I3.invokeMH D1");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   165
        shouldThrowICCE(() -> I3.invokeSpecialMH(unsafeCastI3(new D1())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   166
        System.out.println("ICCE I3.invokeMH E");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   167
        shouldThrowICCE(() -> I3.invokeSpecialMH(unsafeCastI3(new E())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   168
        System.out.println("ICCE I3.invokeMH C2");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   169
        shouldThrowICCE(() -> I3.invokeSpecialMH(unsafeCastI3(new C2())));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   170
        System.out.println("ICCE I4.invokeDirect C1");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   171
        shouldThrowIAE(() -> I4.invokeDirect(unsafeCastI4(new C1())));
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   172
        System.out.println("ICCE I4.invokeDirectFinal C1");
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   173
        shouldThrowIAE(() -> I4.invokeDirectFinal(unsafeCastI4(new C1())));
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   174
        System.out.println("ICCE I2.invokeObjectMH C1");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   175
        shouldThrowICCE(() -> I2.invokeSpecialObjectMH(unsafeCastI2(new C1())));
50041
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   176
        System.out.println("ICCE I2.invokeObjectFinalMH C1");
86c6968ff67a 8202686: Missing test case for 8200167 - final Object methods
dholmes
parents: 50016
diff changeset
   177
        shouldThrowICCE(() -> I2.invokeSpecialObjectFinalMH(unsafeCastI2(new C1())));
49935
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   178
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   179
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   180
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   181
    static void warmup() {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   182
        for (int i = 0; i < 20_000; i++) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   183
            runPositiveTests();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   184
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   185
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   186
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   187
    public static void main(String[] args) throws Throwable {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   188
        System.out.println("UNRESOLVED:");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   189
        runNegativeTests();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   190
        runPositiveTests();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   191
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   192
        System.out.println("RESOLVED:");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   193
        runNegativeTests();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   194
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   195
        System.out.println("WARMUP:");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   196
        warmup();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   197
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   198
        System.out.println("COMPILED:");
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   199
        runNegativeTests();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   200
        runPositiveTests();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   201
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   202
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   203
    static interface Test {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   204
        void run() throws Throwable;
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   205
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   206
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   207
    static void shouldThrowICCE(Test t) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   208
        shouldThrow(IncompatibleClassChangeError.class,
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   209
                    "is not a subclass of caller class", t);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   210
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   211
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   212
    static void shouldThrowIAE(Test t) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   213
        shouldThrow(IllegalAccessError.class,
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   214
                    "must be the current class or a subtype of interface", t);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   215
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   216
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   217
    static void shouldThrow(Class<?> expectedError, String reason, Test t) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   218
        try {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   219
            t.run();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   220
        } catch (Throwable e) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   221
            if (expectedError.isInstance(e)) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   222
                if (e.getMessage().contains(reason)) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   223
                    // passed
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   224
                    System.out.println("Threw expected: " + e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   225
                    return;
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   226
                }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   227
                else {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   228
                    throw new AssertionError("Wrong exception reason: expected '" + reason
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   229
                                             + "', got '" + e.getMessage() + "'", e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   230
                }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   231
            } else {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   232
                String msg = String.format("Wrong exception thrown: expected=%s; thrown=%s",
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   233
                                           expectedError.getName(), e.getClass().getName());
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   234
                throw new AssertionError(msg, e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   235
            }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   236
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   237
        throw new AssertionError("No exception thrown: expected " + expectedError.getName());
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   238
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   239
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   240
    static void shouldNotThrow(Test t) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   241
        try {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   242
            t.run();
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   243
            // passed
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   244
        } catch (Throwable e) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   245
            throw new AssertionError("Exception was thrown: ", e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   246
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   247
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   248
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   249
    // Note: these unsafe casts are only possible for interface types
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   250
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   251
    static I2 unsafeCastI2(Object obj) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   252
        try {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   253
            MethodHandle mh = MethodHandles.identity(Object.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   254
            mh = MethodHandles.explicitCastArguments(mh, mh.type().changeReturnType(I2.class));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   255
            return (I2)mh.invokeExact((Object) obj);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   256
        } catch (Throwable e) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   257
            throw new Error(e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   258
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   259
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   260
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   261
    static I3 unsafeCastI3(Object obj) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   262
        try {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   263
            MethodHandle mh = MethodHandles.identity(Object.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   264
            mh = MethodHandles.explicitCastArguments(mh, mh.type().changeReturnType(I3.class));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   265
            return (I3)mh.invokeExact((Object) obj);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   266
        } catch (Throwable e) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   267
            throw new Error(e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   268
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   269
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   270
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   271
    static I4 unsafeCastI4(Object obj) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   272
        try {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   273
            MethodHandle mh = MethodHandles.identity(Object.class);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   274
            mh = MethodHandles.explicitCastArguments(mh, mh.type().changeReturnType(I4.class));
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   275
            return (I4)mh.invokeExact((Object) obj);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   276
        } catch (Throwable e) {
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   277
            throw new Error(e);
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   278
        }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   279
    }
2ace90aec488 8200167: Validate more special case invocations
dholmes
parents:
diff changeset
   280
}