| /* |
| * 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.data.*; |
| import vm.runtime.defmeth.shared.builder.TestBuilder; |
| import vm.runtime.defmeth.shared.annotation.NotApplicableFor; |
| import static vm.runtime.defmeth.shared.data.method.body.CallMethod.Invoke.*; |
| import static vm.runtime.defmeth.shared.data.method.body.CallMethod.IndexbyteOp.*; |
| import static vm.runtime.defmeth.shared.ExecutionMode.*; |
| |
| /* |
| * Scenarios on static methods in interfaces. |
| */ |
| public class StaticMethodsTest extends DefMethTest { |
| |
| public static void main(String[] args) { |
| TestBase.runTest(new StaticMethodsTest(), args); |
| } |
| |
| // static method in interface |
| /* |
| * testStaticMethod |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testStaticMethod() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().staticCallSite(I, "m", "()I").returns(1).done() |
| |
| .run(); |
| } |
| |
| // invoke[virtual|interface|special] from same/subintf |
| /* |
| * testInvokeVirtual |
| * |
| * interface I { |
| * default static public int staticM() { return 1; } |
| * default public int m() { return ((I)this).staticM(); } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testInvokeVirtual() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("staticM", "()I") |
| .static_().public_().returns(1).build() |
| |
| // force an invokevirtual MR of staticM() |
| .defaultMethod("m", "()I") |
| .invoke(VIRTUAL, b.intfByName("I"), null, "staticM", "()I", METHODREF).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().staticCallSite(I, "staticM", "()I").returns(1).done() |
| |
| .test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testInvokeIntf |
| * |
| * interface I { |
| * default static public int staticM() { return 1; } |
| * default public int m() { return ((I)this).staticM(); } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testInvokeIntf() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("staticM", "()I") |
| .static_().public_().returns(1).build() |
| |
| .defaultMethod("m", "()I") |
| .invoke(INTERFACE, b.intfByName("I"), null, "staticM", "()I", CALLSITE).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().staticCallSite(I, "staticM", "()I").returns(1).done() |
| |
| .test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testInvokeSpecial |
| * |
| * interface I { |
| * default static public int staticM() { return 1; } |
| * default public int m() { return I.super.staticM(); } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testInvokeSpecial() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("staticM", "()I") |
| .static_().public_().returns(1).build() |
| |
| .defaultMethod("m", "()I") |
| .invoke(SPECIAL, b.intfByName("I"), null, "staticM", "()I", CALLSITE).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().staticCallSite(I, "staticM", "()I").returns(1).done() |
| |
| .test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| |
| .run(); |
| } |
| |
| /* |
| * testStaticVsDefault |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * default public int m() { return 2; } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testStaticVsDefault() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .defaultMethod("m", "()I") |
| .public_().returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| b.test().staticCallSite(I, "m", "()I").throws_(ClassFormatError.class).done() |
| |
| // FIXME: throws exception during an attempt to lookup Test2.test() method |
| |
| // Invalid test. ClassFormatError is thrown at verification time, rather |
| // than execution time. |
| // .test().callSite(I, C, "m", "()I").throws_(ClassFormatError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(ClassFormatError.class).done() |
| |
| .run(); |
| } |
| |
| // call static method from default method |
| /* |
| * testInvokeFromDefaultMethod |
| * |
| * interface I { |
| * default static public int staticPublicM() { return 1; } |
| * default public int invokePublic() { return I.staticPublicM(); } |
| * default static private int staticPrivateM() { return 1; } |
| * default public int invokePrivate() { return I.staticPrivateM(); } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testInvokeFromDefaultMethod() throws Exception { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("staticPublicM", "()I") |
| .static_().public_().returns(1).build() |
| .defaultMethod("invokePublic", "()I") |
| .invokeStatic(b.intfByName("I"), "staticPublicM", "()I").build() |
| |
| .defaultMethod("staticPrivateM", "()I") |
| .static_().private_().returns(1).build() |
| .defaultMethod("invokePrivate", "()I") |
| .invokeStatic(b.intfByName("I"), "staticPrivateM", "()I").build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| Class expectedClass; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedClass = NoSuchMethodException.class; |
| } else { |
| expectedClass = IllegalAccessError.class; |
| } |
| |
| // call static method from another class |
| b.test().staticCallSite(I, "staticPublicM", "()I").returns(1).done() |
| .test().staticCallSite(I, "staticPrivateM", "()I").throws_(expectedClass).done() |
| |
| // call public static method from default method |
| .test().callSite(I, C, "invokePublic", "()I").returns(1).done() |
| .test().callSite(C, C, "invokePublic", "()I").returns(1).done() |
| |
| // call private static method from default method |
| .test().callSite(I, C, "invokePrivate", "()I").returns(1).done() |
| .test().callSite(C, C, "invokePrivate", "()I").returns(1).done() |
| |
| .run(); |
| } |
| |
| // call static method from implementing subclass |
| /* |
| * testInvokeFromSubclass |
| * |
| * interface I { |
| * default static public int staticPublicM() { return 1; } |
| * default static private int staticPrivateM() { return 1; } |
| * } |
| * |
| * class C implements I { |
| * public int invokePublic() { return I.staticPublicM(); } |
| * public int invokePrivate() { return I.staticPublicM(); } |
| * |
| * I.staticPublicM(); ==> returns 1; |
| * I.staticPrivateM(); ==> Either NSME or IAE depending on execution mode |
| * C c = new C(); c.invokePublic(); ==> returns 1 or if -ver < 52 IAE or VerifyError |
| * C c = new C(); c.invokePrivate() ==> IAE or if -ver < 52, IAE or VerifyError |
| * } |
| */ |
| |
| @NotApplicableFor(modes = { REDEFINITION }) // Can't redefine a class that gets error during loading |
| public void testInvokeFromSubclass() throws Exception { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("staticPublicM", "()I") |
| .static_().public_().returns(1).build() |
| |
| .defaultMethod("staticPrivateM", "()I") |
| .static_().private_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I) |
| .concreteMethod("invokePublic", "()I") |
| .invokeStatic(b.intfByName("I"), "staticPublicM", "()I").build() |
| .concreteMethod("invokePrivate", "()I") |
| .invokeStatic(b.intfByName("I"), "staticPrivateM", "()I").build() |
| .build(); |
| |
| Class expectedError1; |
| if (factory.getExecutionMode().equals("REFLECTION")) { |
| expectedError1 = NoSuchMethodException.class; |
| } else { |
| expectedError1 = IllegalAccessError.class; |
| } |
| |
| // Adjust for -ver < 52 |
| if (factory.getVer() >=52) { |
| // call static method from another class |
| b.test().staticCallSite(I, "staticPublicM", "()I").returns(1).done() |
| .test().staticCallSite(I, "staticPrivateM", "()I").throws_(expectedError1).done() |
| |
| // call static method from implementing subclass |
| .test().callSite(C, C, "invokePublic", "()I").returns(1).done() |
| .test().callSite(C, C, "invokePrivate", "()I").throws_(IllegalAccessError.class).done() |
| |
| .run(); |
| } else { |
| // call static method from another class |
| b.test().staticCallSite(I, "staticPublicM", "()I").returns(1).done() |
| .test().staticCallSite(I, "staticPrivateM", "()I").throws_(expectedError1).done() |
| |
| // call static method from implementing subclass |
| // invokestatic IMR - not supported for ver < 52 |
| .test().callSite(C, C, "invokePublic", "()I").throws_(VerifyError.class).done() |
| .test().callSite(C, C, "invokePrivate", "()I").throws_(VerifyError.class).done() |
| |
| .run(); |
| } |
| } |
| |
| // static method doesn't participate in default method analysis: |
| // method overriding |
| /* |
| * testNotInherited |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * class C implements I {} |
| */ |
| public void testNotInherited() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().staticCallSite(I, "m", "()I").returns(1).done() |
| // invokeinterface to static method ==> ICCE |
| .test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(NoSuchMethodError.class).done() |
| .run(); |
| } else { |
| b.test().staticCallSite(I, "m", "()I").returns(1).done() |
| .test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(C, C, "m", "()I").throws_(NoSuchMethodError.class).done() |
| .run(); |
| } |
| } |
| |
| /* |
| * testDefaultVsConcrete |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * class C implements I { |
| * public int m() { return 2; } |
| * } |
| * TEST: I o = new C(); o.m()I throws ICCE |
| * TEST: C o = new C(); o.m()I == 2 |
| */ |
| public void testDefaultVsConcrete() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(I) |
| .concreteMethod("m", "()I").returns(2).build() |
| .build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| // invokeinterface to static method ==> ICCE |
| b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done().run(); |
| } else { |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done().run(); |
| } |
| |
| } |
| |
| /* |
| * TEST: StaticMethodsTest.testOverrideStatic |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * interface J extends I { |
| * default public int m() { return 2; } |
| * } |
| * |
| * class C implements J { |
| * } |
| */ |
| public void testOverrideStatic() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().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(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().staticCallSite(I, "m", "()I").returns(1).done() |
| .test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| .run(); |
| } else { |
| b.test().staticCallSite(I, "m", "()I").returns(1).done() |
| .test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| .run(); |
| } |
| |
| } |
| |
| /* |
| * testOverrideDefault |
| * |
| * interface I { |
| * default public int m() { return 1; } |
| * } |
| * |
| * interface J extends I { |
| * default static public 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 == ICCE |
| * TEST: C o = new C(); o.m()I == 1 |
| */ |
| public void testOverrideDefault() { |
| 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") |
| .static_().public_().returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(J, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").returns(1).done() |
| |
| .run(); |
| |
| } else { |
| // Reflection correctly finds the static method defined in J and |
| // calls it with invokestatic. |
| |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(1).done() |
| |
| .run(); |
| } |
| } |
| |
| /* |
| * testReabstract |
| * |
| * interface I { |
| * default static public 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 ICCE |
| * -mode reflect returns 1 |
| * TEST: J o = new C(); o.m()I throws AME |
| * TEST: C o = new C(); o.m()I throws AME |
| */ |
| public void testReabstract() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .abstractMethod("m", "()I").build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(J, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .run(); |
| } else { |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(J, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .run(); |
| } |
| } |
| |
| /* |
| * testOverrideAbstract |
| * |
| * interface I { |
| * abstract public int m(); |
| * } |
| * |
| * interface J extends I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * class C implements J {} |
| * |
| * TEST: I o = new C(); o.m()I throws AME |
| * TEST: J o = new C(); o.m()I throws ICCE |
| * -mode reflect returns 1 |
| * TEST: C o = new C(); o.m()I throws AME |
| */ |
| public void testOverrideAbstract() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .abstractMethod("m", "()I").build() |
| .build(); |
| |
| Interface J = b.intf("J").extend(I) |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").implement(J).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(I, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .test().callSite(J, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| |
| .run(); |
| } else { |
| b.test().callSite(I, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| .test().callSite(J, C, "m", "()I").returns(1).done() |
| .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done() |
| |
| .run(); |
| } |
| } |
| |
| /* |
| * testInheritedDefault |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * class B implements I {} |
| * |
| * class C extends B {} |
| * |
| * TEST: I o = new C(); o.m()I throws IncompatibleClassChangeError |
| * -mode reflect returns 1 |
| * TEST: B o = new C(); o.m()I throws NoSuchMethodError |
| * TEST: C o = new C(); o.m()I throws NoSuchMethodError |
| */ |
| public void testInheritedDefault() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass B = b.clazz("B").implement(I).build(); |
| ConcreteClass C = b.clazz("C").extend(B).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(I, C, "m","()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(B, C, "m","()I").throws_(NoSuchMethodError.class).done() |
| .test().callSite(C, C, "m","()I").throws_(NoSuchMethodError.class).done() |
| .run(); |
| } else { |
| b.test().callSite(I, C, "m","()I").returns(1).done() |
| .test().callSite(B, C, "m","()I").throws_(NoSuchMethodError.class).done() |
| .test().callSite(C, C, "m","()I").throws_(NoSuchMethodError.class).done() |
| .run(); |
| } |
| |
| } |
| |
| /* |
| * testDefaultVsConcreteInherited |
| * |
| * interface I { |
| * default static public int m() { return 1; } |
| * } |
| * |
| * class B { |
| * public int m() { return 2; } |
| * } |
| * |
| * class C extends B implements I {} |
| * |
| */ |
| public void testDefaultVsConcreteInherited() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().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(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().staticCallSite(I, "m","()I").returns(1).done() |
| .test().callSite(I, C, "m","()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(B, C, "m","()I").returns(2).done() |
| .test().callSite(C, C, "m","()I").returns(2).done() |
| .run(); |
| } else { |
| b.test().staticCallSite(I, "m","()I").returns(1).done() |
| .test().callSite(I, C, "m","()I").returns(1).done() |
| .test().callSite(B, C, "m","()I").returns(2).done() |
| .test().callSite(C, C, "m","()I").returns(2).done() |
| .run(); |
| } |
| |
| } |
| |
| /* |
| * testDefaultVsStaticConflict |
| * |
| * interface I { |
| * default static public 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 ICCE |
| * -mode reflect returns 1 |
| * TEST: J o = new C(); o.m()I == 2 |
| * TEST: C o = new C(); o.m()I == 2 |
| */ |
| public void testDefaultVsStaticConflict() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .static_().public_().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(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| .run(); |
| } else { |
| b.test().callSite(I, C, "m", "()I").returns(1).done() |
| .test().callSite(J, C, "m", "()I").returns(2).done() |
| .test().callSite(C, C, "m", "()I").returns(2).done() |
| .run(); |
| } |
| |
| } |
| /* |
| * testStaticSuperClassVsDefaultSuperInterface |
| * |
| * interface I { |
| * default public int m() { return 1; } |
| * } |
| * |
| * class A { |
| * public static int m() { return 2; } |
| * } |
| * |
| * class C extends A implements I {} |
| * |
| * TEST: C o = new C(); o.m()I throws ICCE |
| * -mode reflect returns 2 |
| * TEST: I o = new C(); o.m()I == 1 |
| */ |
| public void testStaticSuperClassVsDefaultSuperInterface() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass A = b.clazz("A") |
| .concreteMethod("m", "()I") |
| .static_().public_().returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").extend(A).implement(I).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(I, C, "m", "()I").returns(1).done() |
| .run(); |
| } else { |
| b.test().callSite(C, C, "m", "()I").returns(2).done() |
| .test().callSite(I, C, "m", "()I").returns(1).done() |
| .run(); |
| } |
| } |
| /* |
| * testStaticLocalVsDefaultSuperInterface |
| * |
| * interface I { |
| * default public int m() { return 1; } |
| * } |
| * |
| * class A implements I { |
| * public static int m() { return 2; } |
| * } |
| * |
| * class C extends A implements I {} |
| * |
| * TEST: A o = new A(); o.m()I throws ICCE |
| * -mode reflect returns 2 |
| * TEST: I o = new A(); o.m()I == 1 |
| */ |
| public void testStaticLocalVsDefaultSuperInterface() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .public_().returns(1).build() |
| .build(); |
| |
| ConcreteClass A = b.clazz("A").implement(I) |
| .concreteMethod("m", "()I") |
| .static_().public_().returns(2).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").extend(A).implement(I).build(); |
| |
| if (!factory.getExecutionMode().equals("REFLECTION")) { |
| b.test().callSite(A, A, "m", "()I").throws_(IncompatibleClassChangeError.class).done() |
| .test().callSite(I, A, "m", "()I").returns(1).done() |
| .run(); |
| } else { |
| b.test().callSite(A, A, "m", "()I").returns(2).done() |
| .test().callSite(I, A, "m", "()I").returns(1).done() |
| .run(); |
| } |
| } |
| /* |
| * testConflictingDefaultsandStaticMethod |
| * @bug 8033150 |
| * |
| * interface I { |
| * default public int m() { return 1; } |
| * } |
| * |
| * interface J { |
| * default public int m() { return 2; } |
| * } |
| * |
| * class A implements I, J { |
| * public static int m() { return 3; } |
| * } |
| * |
| * class C extends A {} |
| * |
| * TEST: C.m(); should call A.m, return value = 3 |
| */ |
| public void testConflictingDefaultsandStaticMethod() { |
| TestBuilder b = factory.getBuilder(); |
| |
| Interface I = b.intf("I") |
| .defaultMethod("m", "()I") |
| .public_().returns(1).build() |
| .build(); |
| |
| Interface J = b.intf("J") |
| .defaultMethod("m", "()I") |
| .public_().returns(2).build() |
| .build(); |
| |
| ConcreteClass A = b.clazz("A").implement(I,J) |
| .concreteMethod("m", "()I") |
| .static_().public_().returns(3).build() |
| .build(); |
| |
| ConcreteClass C = b.clazz("C").extend(A).build(); |
| |
| b.test().staticCallSite(C, "m", "()I").returns(3).done() |
| .run(); |
| } |
| } |