| /* |
| * Copyright (c) 2013, 2018, 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. |
| */ |
| |
| package vm.runtime.defmeth; |
| |
| import nsk.share.test.TestBase; |
| import vm.runtime.defmeth.shared.DefMethTest; |
| import vm.runtime.defmeth.shared.annotation.Crash; |
| import vm.runtime.defmeth.shared.annotation.KnownFailure; |
| import vm.runtime.defmeth.shared.annotation.NotApplicableFor; |
| import vm.runtime.defmeth.shared.data.*; |
| import static vm.runtime.defmeth.shared.data.method.body.CallMethod.Invoke.*; |
| import static vm.runtime.defmeth.shared.data.method.body.CallMethod.IndexbyteOp.*; |
| import vm.runtime.defmeth.shared.builder.TestBuilder; |
| import static vm.runtime.defmeth.shared.ExecutionMode.*; |
| |
| /** |
| * Scenarios on private methods in interfaces. |
| */ |
| public class PrivateMethodsTest extends DefMethTest { |
| |
| public static void main(String[] args) { |
| TestBase.runTest(new PrivateMethodsTest(), args); |
| } |
| |
| // invokevirtual & invokeinterface from same/subintf |
| // Spec change July 2013 to not allow invokevirtual or invokeinterface |
| // to even see an interface private method |
| // Throw ICCE if method resolution returns interface private method |
| |
| // Spec change JDK 11 - invokeinterface can be used for private interface |
| // methods and is now the preferred invocation bytecode - so no ICCE. |
| // Private methods are skipped during selection unless the resolved method |
| // is private. |
| // This change is not dependent on the classfile version. |
| |
| // Note on reflection testing: |
| // Reflection is only used for the initial callsite, which is not always |
| // the method of interest. For example where a default method m() calls |
| // the private interface method privateM(). It is the latter call we are |
| // really testing, but it is the call of the default method that occurs |
| // via reflection. |
| // In private cases reflection triggers a NoSuchMethodException instead of the |
| // expected IllegalAccessError. This indicates it is getDeclaredMethod() that is |
| // failing rather than the actual invoke(). Which in turn suggests the wrong class |
| // is being used, or that getMethod() is being used instead of getDeclaredMethod(). |
| |
| /* |
| * testPrivateInvokeVirtual |
| * |
| * interface I { |
| * private int privateM() { return 1; } |
| * default public int m() { return (I)this.privateM(); } // invokevirtual |
| * } |
| * class C implements I {} |
| * |
| * TEST: I o = new C(); o.m()I throws VerifyError |
| * TEST: C o = new C(); o.m()I throws VerifyError |
| */ |
| @NotApplicableFor(modes = { REDEFINITION }) // Can't redefine a class that gets error during loading |
| public void testPrivateInvokeVirtual() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("privateM", "()I") |
| .private_().returns(1).build() |
| |
| // force an invokevirtual of an IMR to test verification code |
| .defaultMethod("m", "()I") |
| .invoke(VIRTUAL, b.intfByName("I"), null, "privateM", "()I", INTERFACEMETHODREF).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().callSite(I, C, "m", "()I").throws_(VerifyError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(VerifyError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateInvokeIntf |
| * |
| * interface I { |
| * private int privateM() { return 1; } |
| * default public int m() { return (I)this.privateM(); } // invokeinterface |
| * } |
| * class C implements I {} |
| * |
| * TEST: I o = new C(); o.m()I returns 1 |
| * TEST: C o = new C(); o.m()I returns 1 |
| */ |
| public void testPrivateInvokeIntf() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("privateM", "()I") |
| .private_().returns(1).build() |
| .defaultMethod("m", "()I") |
| .invoke(INTERFACE, b.intfByName("I"), null, "privateM", "()I", CALLSITE).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(C, C, "m", "()I").returns(1).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateInvokeStatic |
| * |
| * interface I { |
| * private int privateM() { return 1; } |
| * default public int m() { return I.privateM(); } // invokestatic |
| * } |
| * class C implements I {} |
| * |
| * TEST: I o = new C(); o.m()I throws IncompatibleClassChangeError |
| * TEST: C o = new C(); o.m()I throws IncompatibleClassChangeError |
| */ |
| public void testPrivateInvokeStatic() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("privateM", "()I") |
| .private_().returns(1).build() |
| .defaultMethod("m", "()I") |
| .invoke(STATIC, b.intfByName("I"), null, "privateM", "()I", CALLSITE).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| |
| .run(); |
| } |
| |
| // call from another default method in the same interface |
| /* |
| * testPrivateCallSameClass |
| * |
| * interface I { |
| * private privateM()I { return 1; } |
| * default public int m() { return I.super.privateM(); } // invokespecial |
| * } |
| * class C implements I {} |
| * |
| * TEST: { I o = new C(); o.m()I == 1; } |
| * TEST: { C o = new C(); o.m()I == 1; } |
| */ |
| public void testPrivateCallSameClass() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("privateM", "()I") |
| .private_().returns(1).build() |
| .defaultMethod("m", "()I") |
| .invokeSpecial(b.intfByName("I"), "privateM", "()I").build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(C, C, "m", "()I").returns(1).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateCallSubIntf |
| * |
| * Attempt to call from subinterface fails |
| |
| * interface I { |
| * private privateM()I { return 1; } |
| * } |
| * J, K, L use invokespecial |
| * interface J extends I { |
| * default public int m() { return I.super.privateM(); } |
| * } |
| * interface K extends I { |
| * default public int m() { return K.super.privateM(); } |
| * } |
| * interface L extends J { |
| * default public int m() { return I.super.privateM(); } |
| * } |
| * class C implements J {} |
| * class D implements K {} |
| * class E implements L {} |
| * |
| * TEST: { J o = new C(); o.m()I throws IAE; } |
| * TEST: { C o = new C(); o.m()I throws IAE; } |
| * TEST: { K o = new D(); o.m()I throws NSME; } // does not see |
| * TEST: { D o = new D(); o.m()I throws NSME; } |
| * TEST: { L o = new E(); o.m()I throws VerifyError; } // VerifyError intfmethodref |
| * TEST: { E o = new E(); o.m()I throws VerifyError; } |
| */ |
| @NotApplicableFor(modes = { REDEFINITION }) // Can't redefine a class that gets error during loading |
| public void testPrivateCallSubIntf() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("privateM", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .defaultMethod("m", "()I") |
| .invokeSpecial(I, "privateM", "()I").build() |
| .build(); |
| |
| Interface K = b.intf("K").extend(J) |
| .defaultMethod("m", "()I") |
| .invokeSpecial(b.intfByName("K"), "privateM", "()I").build() |
| .build(); |
| |
| // L.privateM -> J -> L (I.privateM call) |
| Interface L = b.intf("L").extend(J) |
| .defaultMethod("m", "()I") |
| .invokeSpecial(I, "privateM", "()I").build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| ConcreteClass D = b.clazz("D").implement(K).build(); |
| |
| ConcreteClass E = b.clazz("E").implement(L).build(); |
| |
| b.test().callSite(J, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| |
| .test().callSite(K, D, "m", "()I").throws_(NoSuchMethodError.class).done() |
| .test().callSite(D, D, "m", "()I").throws_(NoSuchMethodError.class).done() |
| |
| .test().callSite(L, E, "m", "()I").throws_(VerifyError.class).done() |
| .test().callSite(E, E, "m", "()I").throws_(VerifyError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * Attempt to call from subclass fails |
| * |
| * interface I { |
| * private privateM()I { return 1; } |
| * } |
| * class C implements I { |
| * public int m() { return I.super.privateM(); } |
| * } |
| * class D extends C { |
| * public int m() { return I.super.privateM(); } |
| * } |
| * class E extends C { |
| * public int m() { return C.super.privateM(); } |
| * } |
| * |
| * TEST: { C o = new C(); o.m()I throws IllegalAccessError (or VerifyError) } |
| * TEST: { D o = new D(); o.m()I throws VerifyError } |
| * TEST: { E o = new E(); o.m()I throws NoSuchMethodError (or VerifyError); } |
| */ |
| @NotApplicableFor(modes = { REDEFINITION }) // Can't redefine a class that gets error during loading |
| public void testPrivateCallImplClass() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("privateM", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I) |
| .concreteMethod("m", "()I") |
| .invokeSpecial(I, "privateM", "()I").build() |
| .build(); |
| |
| ConcreteClass D = b.clazz("D").extend(C) |
| .concreteMethod("m", "()I") |
| .invokeSpecial(I, "privateM", "()I").build() |
| .build(); |
| |
| ConcreteClass E = b.clazz("E").extend(C) |
| .concreteMethod("m", "()I") |
| .invokeSpecial(C, "privateM", "()I").build() |
| .build(); |
| |
| Class eeExpectedClass; |
| Class ccExpectedClass; |
| if (factory.getVer() >= 52) { |
| eeExpectedClass = NoSuchMethodError.class; |
| ccExpectedClass = IllegalAccessError.class; |
| } else { |
| // The test gets a VerifyError in this case due to an |
| // invokespecial IMR bytecode. This was not allowed |
| // until class file version 52. (See 8030249.) |
| eeExpectedClass = VerifyError.class; |
| ccExpectedClass = VerifyError.class; |
| } |
| b.test().callSite(C, C, "m", "()I").throws_(ccExpectedClass).done() |
| .test().callSite(D, D, "m", "()I").throws_(VerifyError.class).done() |
| .test().callSite(E, E, "m", "()I").throws_(eeExpectedClass).done() |
| |
| .run(); |
| } |
| |
| // doesn't participate in default method analysis |
| // method overriding |
| |
| /* |
| * testPrivate |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * class C implements I {} |
| * |
| * TEST: { I o = new C(); o.m()I throws IllegalAccessError; } |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { C o = new C(); o.m()I throws NoSuchMethodError; } |
| */ |
| public void testPrivate() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m", "()I").throws_(expectedClass).done() |
| .test().callSite(C, C, "m", "()I").throws_(NoSuchMethodError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateVsConcrete |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * class C implements I { |
| * public int m() { return 2; } |
| * } |
| * |
| * TEST: { I o = new C(); o.m()I == IllegalAccessError; } |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { C o = new C(); o.m()I == 2; } |
| */ |
| public void testPrivateVsConcrete() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I) |
| .concreteMethod("m", "()I").returns(2).build() |
| .build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m", "()I").throws_(expectedClass).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPublicOverridePrivate |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * interface J extends I { |
| * default public int m() { return 2; } |
| * } |
| * class C implements J {} |
| * |
| * TEST: { I o = new C(); o.m()I throws IllegalAccessError; } |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { J o = new C(); o.m()I == 2; } |
| * TEST: { C o = new C(); o.m()I == 2; } |
| */ |
| public void testPublicOverridePrivate() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .defaultMethod("m", "()I") |
| .returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m", "()I").throws_(expectedClass).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateOverrideDefault |
| * |
| * interface I { |
| * default public int m() { return 1; } |
| * } |
| * interface J extends I { |
| * private int m() { return 2; } |
| * } |
| * class C implements J {} |
| * |
| * TEST: { I o = new C(); o.m()I == 1; } |
| * TEST: { J o = new C(); o.m()I == IllegalAccessError; } II J.m priv |
| * TEST: { C o = new C(); o.m()I == 1; } |
| */ |
| public void testPrivateOverrideDefault() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .defaultMethod("m", "()I") |
| .private_().returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().privateCallSite(J, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| .test().callSite(C, C, "m", "()I").returns(1).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateReabstract |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * interface J extends I { |
| * abstract public int m(); |
| * } |
| * class C implements J {} |
| * |
| * TEST: { I o = new C(); o.m()I throws IllegalAccessError; } II I.m |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { J o = new C(); o.m()I throws java/lang/AbstractMethodError; } |
| * TEST: { C o = new C(); o.m()I throws java/lang/AbstractMethodError; } |
| */ |
| public void testPrivateReabstract() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .abstractMethod("m", "()I").build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m", "()I").throws_(expectedClass).done() |
| .test().callSite(J, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateOverrideAbstract |
| * |
| * interface I { |
| * abstract public int m(); |
| * } |
| * interface J extends I { |
| * private int m() { return 1; } |
| * } |
| * class C implements J {} |
| * |
| * TEST: { I o = new C(); o.m()I throws AbstractMethodError } |
| * TEST: { J o = new C(); o.m()I throws IllegalAccessError } |
| * TEST: { C o = new C(); o.m()I throws AbstractMethodError } |
| */ |
| public void testPrivateOverrideAbstract() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .abstractMethod("m", "()I").build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| b.test().callSite(I, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .test().privateCallSite(J, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .run(); |
| } |
| |
| /* |
| * testPrivateInherited |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * class B implements I {} |
| * class C extends B {} |
| * |
| * TEST: { I o = new C(); o.m()I throws IllegalAccessError } II I.m |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { B o = new C(); o.m()I throws NoSuchMethodError } |
| * TEST: { C o = new C(); o.m()I throws NoSuchMethodError } |
| */ |
| public void testPrivateInherited() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass B = b.clazz("B").implement(I).build(); |
| ConcreteClass C = b.clazz("C").extend(B).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m","()I").throws_(expectedClass).done() |
| .test().callSite(B, C, "m","()I").throws_(NoSuchMethodError.class).done() |
| .test().callSite(C, C, "m","()I").throws_(NoSuchMethodError.class).done() |
| |
| .run(); |
| |
| } |
| |
| /* |
| * testPrivateVsConcreteInherited |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * class B { |
| * public int m() { return 2; } |
| * } |
| * class C extends B implements I {} |
| * |
| * TEST: { I o = new C(); o.m()I == throws IllegalAccessError; } |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { B o = new C(); o.m()I == 2; } |
| * TEST: { C o = new C(); o.m()I == 2; } |
| */ |
| public void testPrivateVsConcreteInherited() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass B = b.clazz("B") |
| .concreteMethod("m", "()I").returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").extend(B).implement(I).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m","()I").throws_(expectedClass).done() |
| .test().callSite(B, C, "m","()I").returns(2).done() |
| .test().callSite(C, C, "m","()I").returns(2).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testPrivateConflict |
| * |
| * Conflicting methods |
| * |
| * interface I { |
| * private int m() { return 1; } |
| * } |
| * interface J { |
| * default public int m() { return 2; } |
| * } |
| * class C implements I, J {} |
| * |
| * TEST: { I o = new C(); o.m()I throws IllegalAccessError; } |
| * -mode reflect throws NoSuchMethodException |
| * TEST: { J o = new C(); o.m()I == 2; } |
| * TEST: { C o = new C(); o.m()I == 2; } |
| */ |
| public void testPrivateConflict() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I").private_().returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J") |
| .defaultMethod("m", "()I").returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I,J).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| b.test().callSite(I, C, "m", "()I").throws_(expectedClass).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| |
| .run(); |
| } |
| /* |
| * testPrivateSuperClassMethodNoDefaultMethod |
| * |
| * interface I { |
| * public int m(); |
| * } |
| * |
| * public class A { |
| * private int m() { return 1; } |
| * } |
| * |
| * public class B extends A implements I {} |
| * |
| * public class C extends B { |
| * public int m() { return 2; } |
| * } |
| * |
| * TEST: { B b = new C(); b.m()I throws IllegalAccessError; } |
| */ |
| public void testPrivateSuperClassMethodNoDefaultMethod() { |
| TestBuilder b = factory.getBuilder(); |
| |
| ConcreteClass A = b.clazz("A") |
| .concreteMethod("m", "()I").private_().returns(1).build() |
| .build(); |
| |
| Interface I = b.intf("I") |
| .abstractMethod("m", "()I").public_().build() |
| .build(); |
| |
| ConcreteClass B = b.clazz("B").extend(A).implement(I).build(); |
| |
| ConcreteClass C = b.clazz("C").extend(B) |
| .concreteMethod("m", "()I").public_().returns(2).build() |
| .build(); |
| |
| b.test().privateCallSite(B, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| .run(); |
| |
| } |
| |
| /* |
| * testPrivateSuperClassMethodDefaultMethod |
| * |
| * interface I { |
| * public default int m() { return 3; } |
| * } |
| * |
| * public class A { |
| * private int m() { return 1; } |
| * } |
| * |
| * public class B extends A implements I {} |
| * |
| * public class C extends B { |
| * public int m() { return 2; } |
| * } |
| * |
| * TEST: { B b = new C(); b.m()I throws IllegalAccessError; } |
| */ |
| public void testPrivateSuperClassMethodDefaultMethod() { |
| TestBuilder b = factory.getBuilder(); |
| |
| ConcreteClass A = b.clazz("A") |
| .concreteMethod("m", "()I").private_().returns(1).build() |
| .build(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I").public_().returns(3).build() |
| .build(); |
| |
| ConcreteClass B = b.clazz("B").extend(A).implement(I).build(); |
| |
| ConcreteClass C = b.clazz("C").extend(B) |
| .concreteMethod("m", "()I").public_().returns(2).build() |
| .build(); |
| |
| b.test().privateCallSite(B, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| .run(); |
| } |
| |
| /* |
| * testPrivateSuperClassMethodDefaultMethodNoOverride |
| * |
| * interface I { |
| * public default int m() { return 3; } |
| * } |
| * |
| * public class A { |
| * private int m() { return 1; } |
| * } |
| * |
| * public class B extends A implements I {} |
| * |
| * public class C extends B { } |
| * |
| * TEST: { B b = new C(); b.m()I throws IllegalAccessError; } |
| */ |
| public void testPrivateSuperClassMethodDefaultMethodNoOverride() { |
| TestBuilder b = factory.getBuilder(); |
| |
| ConcreteClass A = b.clazz("A") |
| .concreteMethod("m", "()I").private_().returns(1).build() |
| .build(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I").public_().returns(3).build() |
| .build(); |
| |
| ConcreteClass B = b.clazz("B").extend(A).implement(I).build(); |
| |
| ConcreteClass C = b.clazz("C").extend(B).build(); |
| |
| b.test().privateCallSite(B, C, "m", "()I").throws_(IllegalAccessError.class).done() |
| .run(); |
| } |
| |
| } |