95bf19563b
Co-authored-by: Alex Buckley <alex.buckley@oracle.com> Co-authored-by: Maurizio Mimadamore <maurizio.mimadamore@oracle.com> Co-authored-by: Mandy Chung <mandy.chung@oracle.com> Co-authored-by: Tobias Hartmann <tobias.hartmann@oracle.com> Co-authored-by: Vlaidmir Ivanov <vladimir.x.ivanov@oracle.com> Co-authored-by: Karen Kinnear <karen.kinnear@oracle.com> Co-authored-by: Vladimir Kozlov <vladimir.kozlov@oracle.com> Co-authored-by: John Rose <john.r.rose@oracle.com> Co-authored-by: Daniel Smith <daniel.smith@oracle.com> Co-authored-by: Serguei Spitsyn <serguei.spitsyn@oracle.com> Co-authored-by: Kumar Srinivasan <kumardotsrinivasan@gmail.com> Co-authored-by: Boris Ulasevich <boris.ulasevich@bell-sw.com> Reviewed-by: alanb, psandoz, mchung, coleenp, acorn, mcimadamore, forax, jlahoda, sspitsyn, abuckley
Copyright (c) 2014, 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. ABOUT Once published, it is impossible to add methods to an interface without breaking existing implementations (specifically, adding a method to an interface is not a source-compatible change). The longer the time since a library has been published, the more likely it is that this restriction will cause grief for its maintainers. The addition of closures to the Java language in JDK 8 place additional stress on the aging Collection interfaces; one of the most significant benefits of closures is that it enables the development of more powerful libraries. It would be disappointing to add a language feature that enables better libraries while at the same time not extending the core libraries to take advantage of that feature. A mechanism for adding new methods to existing interfaces is proposed, which is called virtual extension (or default) methods. Existing interfaces can be augmented without compromising backward compatibility by adding extension methods to the interface, whose declaration would contain instructions for finding the default implementation in the event that implementers do not provide a method body. A key characteristic of extension methods is that they are virtual methods just like other interface methods, but provide a default implementation in the event that the implementing class does not provide a method body. VM support is necessary to implement virtual extension methods. OVERVIEW The test suite is organized in the following manner. The tests rely on a framework to generate class hierarchies and tests directly in bytecode from a pseudo-code in Java. Pseudo-code is written using builder pattern and fluent coding style. The framework is located in src/vm/runtime/defmeth/shared and divided into /data and /builder sections. As an example, the following code: TestBuilder b = factory.getBuilder(); Interface I = b.intf("I") .defaultMethod("m", "()I").returns(1).build() .build(); ConcreteClass C = b.clazz("C").implement(I) .concreteMethod("m", "()I").returns(2).build() .build(); b.test().callSite(I, C, "m", "()I").returns(2).done() .test().callSite(C, C, "m", "()I").returns(2).done() .run(); translates into bytecode equivalent of: 2-class hierarchy: interface I { int m() default { return 1; } } class C implements I { public int m() { return 2; } } and 2 tests: Test1_I_C_m { static void test() { I i = new C(); if (i.m() != 2) throw new TestFailure(); } } Test2_C_C_m { static void test() { C c = new C(); if (c.m() != 2) throw new TestFailure(); } } TestBuilder.run() calls Test1_I_C_m.test() and Test2_C_C_m.test() and performs failure reporting, if necessary. All tests are located in src/vm/runtime/defmeth and are grouped according to the area they excercise. The test groups are: - AccessibilityFlagsTest - BasicTest - ConflictingDefaultsTest - DefaultVsAbstractTest - MethodResolutionTest - ObjectMethodOverridesTest - PrivateMethodsTest - RedefineTest - StaticMethodsTest - StressTest - SuperCallTest Each test group can be executed in different modes. For each mode there's a corresponding scenario in src/vm/runtime/defmeth/scenarios. Scenarios are organized in the following manner: .../scenarios/[test_group]_[majorVer]_[methodFlags]_[invocationType]_[shouldRedefine] where majorVer - major version of class files for generated concrete classes values: ver49, ver50, ver51, ver52 methodFlags - additional access flags for methods in generated classes values: none == no additional flags sync == ACC_SYNCHRONIZED strict == ACC_STRICT syncstrict == ACC_SYNCHRONIZED | ACC_STRICT invocationType - how methods in test hiearchies are invoked during testing values: direct - using invoke* bytecodes reflect - using Reflection API invoke - using invokedynamic & java.lang.invoke API (MethodHandles/JSR292) redefine - whether to preload and redefine classes before running individual tests values: redefine, noredefine testGroup - name of test group being used values: BasicTests/BridgeMethod/etc STRESS TESTING Stress test differs from other scenarios - it has only 2 modes: redefine and noredefine. Stress scenario is the following: - in multiple threads (5 by default)... - ... continuously run random vm.runtime.defmeth.* tests ... - ... in random configurations ... - ... until predefined period of time is over... - ... or any failures occured. HOW TO RUN Directly from command-line: $ java -cp ${VMTESTBASE}/bin/classes vm.runtime.defmeth.shared.DefMethTest Specify testing mode: -flags <int> additional access flags on default methods (default: 0) -ver <int> class file major version (default: 52) -redefine <boolean> redefine classes during execution (default: false) -mode [direct|reflect|invoke] specify method invocation mechanism (default: direct): - direct - invoke* instructions in bytecode - reflect - Reflection API - invoke - invokedynamic & MethodHandle.invoke* -execMode [DIRECT|REFLECTION|INVOKE_EXACT|INVOKE_GENERIC|INVOKE_WITH_ARGS|INDY] specify concrete execution mode Execution-specific flags: -list <boolean> list available tests -filter <regex> filter tests by name (default: .* ) If you run tests directly from command line, in order to make "-redefine true", StressTest or RedefineTest work, additional steps are necessary: add -agentlib:redefineClasses to JVM options set correct LD_LIBRARY_PATH: LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:${VM_TESTBASE}/bin/lib/${PLATFORM}/vm/runtime/defmeth/shared/ Also, it is possible to run any test group directly: $ java -cp ${VMTESTBASE}/bin/classes vm.runtime.defmeth.BasicTest StressTest has some specific options: -stressTime <long> Stress execution time in seconds (default: 60) -stressThreadsFactor <int> Stress threads factor (default: 1) -seed <int> force deterministic behavior (default: 0) -redefine <boolean> use scenarios w/ class redefinition (default: false) -ver <int> minimum class file version to be used in the tests (default: 49) -ignoreTestFailures ignore failures of individual tests To simplify failure analysis, the framework has some additional flags to produce diagnostics output: -Dvm.runtime.defmeth.printTests print pseudo-code for each test; -Dvm.runtime.defmeth.printAssembly print bytecode assembly for all generated class files; -Dvm.runtime.defmeth.printASMify print "asmified" version of generated class files; very useful when preparing reduced test cases. -Dvm.runtime.defmeth.dumpClasses dump class files under DUMP_CLASS_FILES in <test_name> folder -Dvm.runtime.defmeth.printStackTrace print full stack traces for all errors and test failures -Dvm.runtime.defmeth.traceClassRedefinition trace class redefinition during testing LINKS [1] "Design and Implementation of Default Methods in Hotspot JVM", by Keith McGuigan, 09/18/2012 http://cr.openjdk.java.net/~kamg/default_methods_in_hotspot.txt [2] "Featherweight Defenders: A formal model for virtual extension methods in Java", by Brian Goetz, Robert Field, 03/27/2012 http://cr.openjdk.java.net/~briangoetz/lambda/featherweight-defenders.pdf [3] "Interface evolution via virtual extension methods", by Brian Goetz, 4th draft, 06/2011 http://cr.openjdk.java.net/~briangoetz/lambda/Defender%20Methods%20v4.pdf