mirror of
https://github.com/openjdk/jdk.git
synced 2026-05-16 08:29:34 +00:00
8380080: PopFrame logic assumes trailing member name arg is recoverable as local 0
Reviewed-by: dlong, sspitsyn
This commit is contained in:
parent
cbda5317d6
commit
81442e1db8
@ -1487,23 +1487,32 @@ JRT_ENTRY(void, InterpreterRuntime::member_name_arg_or_null(JavaThread* current,
|
||||
Method* method, address bcp))
|
||||
Bytecodes::Code code = Bytecodes::code_at(method, bcp);
|
||||
if (code != Bytecodes::_invokestatic) {
|
||||
current->set_vm_result_oop(nullptr);
|
||||
return;
|
||||
}
|
||||
|
||||
ConstantPool* cpool = method->constants();
|
||||
int cp_index = Bytes::get_native_u2(bcp + 1);
|
||||
Symbol* cname = cpool->klass_name_at(cpool->klass_ref_index_at(cp_index, code));
|
||||
Symbol* mname = cpool->name_ref_at(cp_index, code);
|
||||
|
||||
if (MethodHandles::has_member_arg(cname, mname)) {
|
||||
oop member_name_oop = cast_to_oop(member_name);
|
||||
if (java_lang_invoke_DirectMethodHandle::is_instance(member_name_oop)) {
|
||||
// FIXME: remove after j.l.i.InvokerBytecodeGenerator code shape is updated.
|
||||
member_name_oop = java_lang_invoke_DirectMethodHandle::member(member_name_oop);
|
||||
}
|
||||
current->set_vm_result_oop(member_name_oop);
|
||||
} else {
|
||||
if (!MethodHandles::has_member_arg(cname, mname)) {
|
||||
current->set_vm_result_oop(nullptr);
|
||||
return;
|
||||
}
|
||||
|
||||
oop member_name_oop = cast_to_oop(member_name);
|
||||
|
||||
guarantee(member_name_oop != nullptr, "member_name_oop should not be nullptr");
|
||||
guarantee(oopDesc::is_oop(member_name_oop), "member_name_oop should be an oop");
|
||||
guarantee(java_lang_invoke_MemberName::is_instance(member_name_oop) ||
|
||||
java_lang_invoke_DirectMethodHandle::is_instance(member_name_oop),
|
||||
"member_name_oop is not MemberName or DMH");
|
||||
|
||||
if (java_lang_invoke_DirectMethodHandle::is_instance(member_name_oop)) {
|
||||
member_name_oop = java_lang_invoke_DirectMethodHandle::member(member_name_oop);
|
||||
}
|
||||
current->set_vm_result_oop(member_name_oop);
|
||||
JRT_END
|
||||
#endif // INCLUDE_JVMTI
|
||||
|
||||
|
||||
@ -1,3 +1,26 @@
|
||||
/*
|
||||
* Copyright (c) 2017, 2026, 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 compiler.jsr292;
|
||||
|
||||
import java.lang.invoke.MethodHandle;
|
||||
@ -39,7 +62,7 @@ public class InvokerSignatureMismatch {
|
||||
|
||||
static void mainLink(int i) throws Throwable {
|
||||
Object name = MethodHandleHelper.internalMemberName(INT_MH);
|
||||
MethodHandleHelper.linkToStatic((float) i, name);
|
||||
MethodHandleHelper.linkToStatic(name, (float) i);
|
||||
}
|
||||
|
||||
static void mainInvoke(int i) throws Throwable {
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2015, 2025, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2015, 2026, 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
|
||||
@ -49,7 +49,7 @@ public class MethodHandleHelper {
|
||||
}
|
||||
|
||||
@ForceInline
|
||||
public static void linkToStatic(float arg, Object name) throws Throwable {
|
||||
public static void linkToStatic(Object name, float arg) throws Throwable {
|
||||
MethodHandle.linkToStatic(arg, name);
|
||||
}
|
||||
|
||||
|
||||
@ -0,0 +1,119 @@
|
||||
/*
|
||||
* Copyright (c) 2026, 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 runtime.interpreter;
|
||||
|
||||
import com.sun.jdi.Bootstrap;
|
||||
import com.sun.jdi.VirtualMachine;
|
||||
import com.sun.jdi.connect.Connector;
|
||||
import com.sun.jdi.connect.IllegalConnectorArgumentsException;
|
||||
import com.sun.jdi.connect.LaunchingConnector;
|
||||
import com.sun.jdi.connect.VMStartException;
|
||||
import com.sun.jdi.event.BreakpointEvent;
|
||||
import com.sun.jdi.event.ClassPrepareEvent;
|
||||
import com.sun.jdi.event.Event;
|
||||
import com.sun.jdi.event.EventSet;
|
||||
import com.sun.jdi.request.ClassPrepareRequest;
|
||||
import com.sun.jdi.request.EventRequestManager;
|
||||
import jdk.test.lib.Utils;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.lang.invoke.MethodHandle;
|
||||
import java.lang.invoke.MethodHandleHelper;
|
||||
import java.lang.invoke.MethodHandles;
|
||||
import java.lang.invoke.MethodType;
|
||||
import java.nio.file.Path;
|
||||
import java.util.Map;
|
||||
|
||||
/*
|
||||
* @test
|
||||
* @bug 8380080
|
||||
* @summary MemberName in local0 must be preserved when PopFrame re-executes direct MethodHandles.linkToStatic
|
||||
* @requires vm.jvmti
|
||||
* @requires vm.flavor != "zero"
|
||||
* @library ../../compiler/jsr292/patches /test/lib
|
||||
* @modules java.base/jdk.internal.vm.annotation
|
||||
* jdk.jdi
|
||||
* @build java.base/java.lang.invoke.MethodHandleHelper
|
||||
* @run driver runtime.interpreter.PopFrameMethodNameInvariantTest
|
||||
*/
|
||||
|
||||
public class PopFrameMethodNameInvariantTest {
|
||||
public static class Target {
|
||||
static void body(float x) {}
|
||||
public static void main(String[] args) throws Throwable {
|
||||
MethodHandle target = MethodHandles.lookup().findStatic(
|
||||
Target.class,
|
||||
"body",
|
||||
MethodType.methodType(void.class, float.class));
|
||||
Object name = MethodHandleHelper.internalMemberName(target);
|
||||
MethodHandleHelper.linkToStatic(name, (float)1.0);
|
||||
}
|
||||
}
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
VirtualMachine vm = getVm();
|
||||
EventRequestManager eventRequestManager = vm.eventRequestManager();
|
||||
ClassPrepareRequest classPrepareRequest = eventRequestManager.createClassPrepareRequest();
|
||||
classPrepareRequest.addClassFilter(Target.class.getName());
|
||||
classPrepareRequest.enable();
|
||||
outerLoop:
|
||||
while (true) {
|
||||
EventSet eventSet = vm.eventQueue().remove();
|
||||
for (Event event : eventSet) {
|
||||
if (event instanceof ClassPrepareEvent prepareEvent) {
|
||||
eventRequestManager.createBreakpointRequest(
|
||||
prepareEvent.referenceType()
|
||||
.methodsByName("body")
|
||||
.get(0)
|
||||
.location()
|
||||
).enable();
|
||||
}
|
||||
if (event instanceof BreakpointEvent breakpointEvent) {
|
||||
breakpointEvent.request().disable();
|
||||
breakpointEvent.thread().popFrames(breakpointEvent.thread().frame(0));
|
||||
eventSet.resume();
|
||||
break outerLoop;
|
||||
}
|
||||
}
|
||||
eventSet.resume();
|
||||
}
|
||||
int exitCode = vm.process().waitFor();
|
||||
if (exitCode != 0) {
|
||||
throw new RuntimeException("Debugee exited with code " + exitCode);
|
||||
}
|
||||
}
|
||||
|
||||
private static VirtualMachine getVm() throws IOException, IllegalConnectorArgumentsException, VMStartException {
|
||||
LaunchingConnector connector = Bootstrap.virtualMachineManager().defaultConnector();
|
||||
Map<String, Connector.Argument> argumentMap = connector.defaultArguments();
|
||||
argumentMap.get("main").setValue(Target.class.getName());
|
||||
String options = String.join(" ", Utils.getTestJavaOpts());
|
||||
String patchPath = System.getProperty("test.patch.path");
|
||||
if (patchPath != null) {
|
||||
options += " --patch-module=java.base=\"" + Path.of(patchPath, "java.base") + "\"";
|
||||
}
|
||||
argumentMap.get("options").setValue((options + " -Xint").trim());
|
||||
return connector.launch(argumentMap);
|
||||
}
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user