2018-05-24 00:09:49 +00:00
|
|
|
/*
|
2021-05-11 17:14:57 +00:00
|
|
|
* Copyright (c) 2013, 2021, Oracle and/or its affiliates. All rights reserved.
|
2018-05-24 00:09:49 +00:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
*
|
|
|
|
* @modules java.base/jdk.internal.org.objectweb.asm:+open java.base/jdk.internal.org.objectweb.asm.util:+open
|
|
|
|
* @library /vmTestbase /test/lib
|
|
|
|
*
|
|
|
|
* @comment build retransform.jar in current dir
|
|
|
|
* @run driver vm.runtime.defmeth.shared.BuildJar
|
|
|
|
*
|
|
|
|
* @run driver jdk.test.lib.FileInstaller . .
|
|
|
|
* @run main/othervm/native
|
|
|
|
* -agentlib:redefineClasses
|
|
|
|
* -javaagent:retransform.jar
|
|
|
|
* vm.runtime.defmeth.StaticMethodsTest
|
|
|
|
*/
|
2018-05-24 00:09:49 +00:00
|
|
|
package vm.runtime.defmeth;
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
import java.util.Set;
|
|
|
|
|
2018-05-24 00:09:49 +00:00
|
|
|
import vm.runtime.defmeth.shared.DefMethTest;
|
|
|
|
import vm.runtime.defmeth.shared.data.*;
|
|
|
|
import vm.runtime.defmeth.shared.builder.TestBuilder;
|
|
|
|
import vm.runtime.defmeth.shared.annotation.NotApplicableFor;
|
2021-05-11 17:14:57 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
2018-05-24 00:09:49 +00:00
|
|
|
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) {
|
2021-06-01 11:09:35 +00:00
|
|
|
DefMethTest.runTest(StaticMethodsTest.class,
|
|
|
|
/* majorVer */ Set.of(MIN_MAJOR_VER, MAX_MAJOR_VER),
|
|
|
|
/* flags */ Set.of(0, ACC_SYNCHRONIZED),
|
|
|
|
/* redefine */ Set.of(false, true),
|
|
|
|
/* execMode */ Set.of(DIRECT, REFLECTION, INVOKE_EXACT, INVOKE_GENERIC, INVOKE_WITH_ARGS, INDY));
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static method in interface
|
|
|
|
/*
|
|
|
|
* testStaticMethod
|
|
|
|
*
|
|
|
|
* interface I {
|
|
|
|
* default static public int m() { return 1; }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* class C implements I {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testStaticMethod(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
Interface I = b.intf("I")
|
|
|
|
.defaultMethod("m", "()I")
|
|
|
|
.static_().public_().returns(1).build()
|
|
|
|
.build();
|
|
|
|
|
|
|
|
ConcreteClass C = b.clazz("C").implement(I).build();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "m", "()I").returns(1).done();
|
|
|
|
b.test().staticCallSite(C, "m", "()I").throws_(NoSuchMethodError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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 {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testInvokeVirtual(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "staticM", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* testInvokeIntf
|
|
|
|
*
|
|
|
|
* interface I {
|
|
|
|
* default static public int staticM() { return 1; }
|
|
|
|
* default public int m() { return ((I)this).staticM(); }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* class C implements I {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testInvokeIntf(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "staticM", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* testInvokeSpecial
|
|
|
|
*
|
|
|
|
* interface I {
|
|
|
|
* default static public int staticM() { return 1; }
|
|
|
|
* default public int m() { return I.super.staticM(); }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* class C implements I {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testInvokeSpecial(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "staticM", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* testStaticVsDefault
|
|
|
|
*
|
|
|
|
* interface I {
|
|
|
|
* default static public int m() { return 1; }
|
|
|
|
* default public int m() { return 2; }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* class C implements I {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testStaticVsDefault(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "m", "()I").throws_(ClassFormatError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
|
|
|
// 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()
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(ClassFormatError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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 {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testInvokeFromDefaultMethod(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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()
|
2021-05-11 17:14:57 +00:00
|
|
|
|
|
|
|
.test().callSite(I, "staticPrivateM", "()I", ACC_STATIC | ACC_PRIVATE).throws_(IllegalAccessError.class).done()
|
2018-05-24 00:09:49 +00:00
|
|
|
|
|
|
|
// 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()
|
2021-06-01 11:09:35 +00:00
|
|
|
.test().callSite(C, C, "invokePrivate", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testInvokeFromSubclass(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
// 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();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
// call static method from implementing subclass
|
2018-05-24 00:09:49 +00:00
|
|
|
if (factory.getVer() >=52) {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(C, C, "invokePublic", "()I").returns(1).done()
|
|
|
|
.test().callSite(C, C, "invokePrivate", "()I").throws_(IllegalAccessError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
|
|
|
// invokestatic IMR - not supported for ver < 52
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(C, C, "invokePublic", "()I").throws_(VerifyError.class).done()
|
|
|
|
.test().callSite(C, C, "invokePrivate", "()I").throws_(VerifyError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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 {}
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testNotInherited(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
Interface I = b.intf("I")
|
|
|
|
.defaultMethod("m", "()I")
|
|
|
|
.static_().public_().returns(1).build()
|
|
|
|
.build();
|
|
|
|
|
|
|
|
ConcreteClass C = b.clazz("C").implement(I).build();
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
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();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
// invokeinterface to static method ==> ICCE
|
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testDefaultVsConcrete(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(I, C, "m", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
// invokeinterface to static method ==> ICCE
|
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(C, C, "m", "()I").returns(2).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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 {
|
|
|
|
* }
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testOverrideStatic(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "m", "()I").returns(1).done();
|
2021-05-11 17:14:57 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(J, C, "m", "()I").returns(2).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").returns(2).done();
|
2021-05-11 17:14:57 +00:00
|
|
|
|
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(I, C, "m", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testOverrideDefault(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").returns(1).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
2018-05-24 00:09:49 +00:00
|
|
|
// Reflection correctly finds the static method defined in J and
|
|
|
|
// calls it with invokestatic.
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(J, C, "m", "()I").returns(2).done();
|
|
|
|
} else {
|
|
|
|
b.test().callSite(J, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testReabstract(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(I, C, "m", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
2021-05-11 17:14:57 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(J, C, "m", "()I").throws_(AbstractMethodError.class).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testOverrideAbstract(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(AbstractMethodError.class).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(AbstractMethodError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(J, C, "m", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(J, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testInheritedDefault(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(B, C, "m","()I").throws_(NoSuchMethodError.class).done();
|
|
|
|
b.test().callSite(C, C, "m","()I").throws_(NoSuchMethodError.class).done();
|
2021-05-11 17:14:57 +00:00
|
|
|
|
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(I, C, "m","()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(I, C, "m","()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* testDefaultVsConcreteInherited
|
|
|
|
*
|
|
|
|
* interface I {
|
|
|
|
* default static public int m() { return 1; }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* class B {
|
|
|
|
* public int m() { return 2; }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* class C extends B implements I {}
|
|
|
|
*
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testDefaultVsConcreteInherited(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(I, "m","()I").returns(1).done();
|
2021-05-11 17:14:57 +00:00
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(B, C, "m","()I").returns(2).done();
|
|
|
|
b.test().callSite(C, C, "m","()I").returns(2).done();
|
2021-05-11 17:14:57 +00:00
|
|
|
|
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(I, C, "m","()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(I, C, "m","()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testDefaultVsStaticConflict(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().callSite(J, C, "m", "()I").returns(2).done();
|
|
|
|
b.test().callSite(C, C, "m", "()I").returns(2).done();
|
2021-05-11 17:14:57 +00:00
|
|
|
|
|
|
|
if (factory.getExecutionMode().equals("REFLECTION")) {
|
|
|
|
b.test().callSite(I, C, "m", "()I").returns(1).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(I, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
2021-05-11 17:14:57 +00:00
|
|
|
|
2018-05-24 00:09:49 +00:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testStaticSuperClassVsDefaultSuperInterface(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
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();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
2021-06-01 11:09:35 +00:00
|
|
|
|
2018-05-24 00:09:49 +00:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testStaticLocalVsDefaultSuperInterface(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-05-11 17:14:57 +00:00
|
|
|
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();
|
2018-05-24 00:09:49 +00:00
|
|
|
} else {
|
2021-05-11 17:14:57 +00:00
|
|
|
b.test().callSite(C, C, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
|
|
|
b.test().callSite(A, A, "m", "()I").throws_(IncompatibleClassChangeError.class).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|
2021-06-01 11:09:35 +00:00
|
|
|
|
2018-05-24 00:09:49 +00:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-01 11:09:35 +00:00
|
|
|
public void testConflictingDefaultsandStaticMethod(TestBuilder b) {
|
2018-05-24 00:09:49 +00:00
|
|
|
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();
|
|
|
|
|
2021-06-01 11:09:35 +00:00
|
|
|
b.test().staticCallSite(C, "m", "()I").returns(3).done();
|
2018-05-24 00:09:49 +00:00
|
|
|
}
|
|
|
|
}
|