8304585: Method::invoke rewraps InvocationTargetException if a caller-sensitive method throws IAE
Reviewed-by: darcy, jpai, alanb
This commit is contained in:
parent
9df2060059
commit
1d7bb1ffa0
@ -174,12 +174,8 @@ class DirectMethodHandleAccessor extends MethodAccessorImpl {
|
|||||||
// caller-sensitive method is invoked through a per-caller invoker while
|
// caller-sensitive method is invoked through a per-caller invoker while
|
||||||
// the target MH is always spreading the args
|
// the target MH is always spreading the args
|
||||||
var invoker = JLIA.reflectiveInvoker(caller);
|
var invoker = JLIA.reflectiveInvoker(caller);
|
||||||
try {
|
|
||||||
// invoke the target method handle via an invoker
|
// invoke the target method handle via an invoker
|
||||||
return invoker.invokeExact(target, obj, args);
|
return invoker.invokeExact(target, obj, args);
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
throw new InvocationTargetException(e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,84 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8304585
|
||||||
|
* @run junit CallerSensitiveMethodInvoke
|
||||||
|
* @run junit/othervm -Djdk.reflect.useDirectMethodHandle=false CallerSensitiveMethodInvoke
|
||||||
|
* @summary Test Method::invoke that wraps exception in InvocationTargetException properly
|
||||||
|
*/
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.lang.reflect.InvocationTargetException;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
|
||||||
|
public class CallerSensitiveMethodInvoke {
|
||||||
|
private int value = 10;
|
||||||
|
|
||||||
|
private void m() throws Exception {
|
||||||
|
throw new InvocationTargetException(new IllegalArgumentException("testing"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the Field::get method being invoked via Method::invoke which
|
||||||
|
* should throw InvocationTargetException thrown by Field::get.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void csMethodInvoke() throws ReflectiveOperationException {
|
||||||
|
Field f = CallerSensitiveMethodInvoke.class.getDeclaredField("value");
|
||||||
|
try {
|
||||||
|
// Field::get throws IAE
|
||||||
|
Method m = Field.class.getDeclaredMethod("get", Object.class);
|
||||||
|
m.invoke(f, new Object()); // illegal receiver type. IAE thrown
|
||||||
|
fail("should not reach here");
|
||||||
|
} catch (InvocationTargetException e) {
|
||||||
|
Throwable t = e.getCause();
|
||||||
|
if (!(t instanceof IllegalArgumentException)) {
|
||||||
|
throw new RuntimeException("Unexpected cause of InvocationTargetException: " + t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the method being invoked throws InvocationTargetException which
|
||||||
|
* will be wrapped with another InvocationTargetException by Method::invoke.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void methodInvoke() throws ReflectiveOperationException {
|
||||||
|
try {
|
||||||
|
// m() throws InvocationTargetException which will be wrapped by Method::invoke
|
||||||
|
Method m = CallerSensitiveMethodInvoke.class.getDeclaredMethod("m");
|
||||||
|
m.invoke(new CallerSensitiveMethodInvoke());
|
||||||
|
fail("should not reach here");
|
||||||
|
} catch (InvocationTargetException e) {
|
||||||
|
Throwable t = e.getCause();
|
||||||
|
if (!(t instanceof InvocationTargetException)) {
|
||||||
|
throw new RuntimeException("Unexpected cause of InvocationTargetException: " + t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user