/* * Copyright (c) 2013, 2021, 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.builder.TesterBuilder; import vm.runtime.defmeth.shared.data.*; import vm.runtime.defmeth.shared.builder.TestBuilder; import vm.runtime.defmeth.shared.annotation.NotApplicableFor; import static jdk.internal.org.objectweb.asm.Opcodes.ACC_PRIVATE; import static jdk.internal.org.objectweb.asm.Opcodes.ACC_STATIC; 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(); // call static method from another class b.test().staticCallSite(I, "staticPublicM", "()I").returns(1).done() .test().callSite(I, "staticPrivateM", "()I", ACC_STATIC | ACC_PRIVATE).throws_(IllegalAccessError.class).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(); // call static method from another class b.test().staticCallSite(I, "staticPublicM", "()I").returns(1).done() .test().callSite(I, "staticPrivateM", "()I", ACC_STATIC | ACC_PRIVATE).throws_(IllegalAccessError.class).done(); // call static method from implementing subclass if (factory.getVer() >=52) { b.test().callSite(C, C, "invokePublic", "()I").returns(1).done() .test().callSite(C, C, "invokePrivate", "()I").throws_(IllegalAccessError.class).done(); } else { // invokestatic IMR - not supported for ver < 52 b.test().callSite(C, C, "invokePublic", "()I").throws_(VerifyError.class).done() .test().callSite(C, C, "invokePrivate", "()I").throws_(VerifyError.class).done(); } b.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(); b.test().staticCallSite(I, "m", "()I").returns(1).done(); b.test().callSite(C, C, "m", "()I").throws_(NoSuchMethodError.class).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(I, C, "m", "()I").returns(1).done(); } else { // invokeinterface to static method ==> ICCE b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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")) { b.test().callSite(I, C, "m", "()I").returns(1).done(); } else { // invokeinterface to static method ==> ICCE b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.test().callSite(C, C, "m", "()I").returns(2).done(); b.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(); b.test().staticCallSite(I, "m", "()I").returns(1).done() .test().callSite(J, C, "m", "()I").returns(2).done() .test().callSite(C, C, "m", "()I").returns(2).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(I, C, "m", "()I").returns(1).done(); } else { b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); b.test().callSite(I, C, "m", "()I").returns(1).done() .test().callSite(C, C, "m", "()I").returns(1).done(); if (factory.getExecutionMode().equals("REFLECTION")) { // Reflection correctly finds the static method defined in J and // calls it with invokestatic. b.test().callSite(J, C, "m", "()I").returns(2).done(); } else { b.test().callSite(J, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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").returns(1).done(); } else { b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.test().callSite(J, C, "m", "()I").throws_(AbstractMethodError.class).done() .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done(); b.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(); b.test().callSite(I, C, "m", "()I").throws_(AbstractMethodError.class).done() .test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(J, C, "m", "()I").returns(1).done(); } else { b.test().callSite(J, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); b.test().callSite(B, C, "m","()I").throws_(NoSuchMethodError.class).done() .test().callSite(C, C, "m","()I").throws_(NoSuchMethodError.class).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(I, C, "m","()I").returns(1).done(); } else { b.test().callSite(I, C, "m","()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); b.test().staticCallSite(I, "m","()I").returns(1).done() .test().callSite(B, C, "m","()I").returns(2).done() .test().callSite(C, C, "m","()I").returns(2).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(I, C, "m","()I").returns(1).done(); } else { b.test().callSite(I, C, "m","()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); b.test().callSite(J, C, "m", "()I").returns(2).done() .test().callSite(C, C, "m", "()I").returns(2).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(I, C, "m", "()I").returns(1).done(); } else { b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); b.test().callSite(I, C, "m", "()I").returns(1).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(C, C, "m", "()I").returns(2).done(); } else { b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); b.test().callSite(I, A, "m", "()I").returns(1).done(); b.test().callSite(I, C, "m", "()I").returns(1).done(); if (factory.getExecutionMode().equals("REFLECTION")) { b.test().callSite(A, A, "m", "()I").returns(2).done(); b.test().callSite(C, C, "m", "()I").returns(2).done(); } else { b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); b.test().callSite(A, A, "m", "()I").throws_(IncompatibleClassChangeError.class).done(); } b.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(); } }