Skip to content

Commit 1d7bb1f

Browse files
author
Mandy Chung
committedMar 30, 2023
8304585: Method::invoke rewraps InvocationTargetException if a caller-sensitive method throws IAE
Reviewed-by: darcy, jpai, alanb
1 parent 9df2060 commit 1d7bb1f

File tree

2 files changed

+86
-6
lines changed

2 files changed

+86
-6
lines changed
 

‎src/java.base/share/classes/jdk/internal/reflect/DirectMethodHandleAccessor.java

+2-6
Original file line numberDiff line numberDiff line change
@@ -174,12 +174,8 @@ private Object invokeImpl(Object obj, Object[] args, Class<?> caller) throws Thr
174174
// caller-sensitive method is invoked through a per-caller invoker while
175175
// the target MH is always spreading the args
176176
var invoker = JLIA.reflectiveInvoker(caller);
177-
try {
178-
// invoke the target method handle via an invoker
179-
return invoker.invokeExact(target, obj, args);
180-
} catch (IllegalArgumentException e) {
181-
throw new InvocationTargetException(e);
182-
}
177+
// invoke the target method handle via an invoker
178+
return invoker.invokeExact(target, obj, args);
183179
}
184180
}
185181

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
/*
2+
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8304585
27+
* @run junit CallerSensitiveMethodInvoke
28+
* @run junit/othervm -Djdk.reflect.useDirectMethodHandle=false CallerSensitiveMethodInvoke
29+
* @summary Test Method::invoke that wraps exception in InvocationTargetException properly
30+
*/
31+
32+
import org.junit.jupiter.api.Test;
33+
import static org.junit.jupiter.api.Assertions.*;
34+
35+
import java.lang.reflect.Field;
36+
import java.lang.reflect.InvocationTargetException;
37+
import java.lang.reflect.Method;
38+
39+
public class CallerSensitiveMethodInvoke {
40+
private int value = 10;
41+
42+
private void m() throws Exception {
43+
throw new InvocationTargetException(new IllegalArgumentException("testing"));
44+
}
45+
46+
/**
47+
* Tests the Field::get method being invoked via Method::invoke which
48+
* should throw InvocationTargetException thrown by Field::get.
49+
*/
50+
@Test
51+
public void csMethodInvoke() throws ReflectiveOperationException {
52+
Field f = CallerSensitiveMethodInvoke.class.getDeclaredField("value");
53+
try {
54+
// Field::get throws IAE
55+
Method m = Field.class.getDeclaredMethod("get", Object.class);
56+
m.invoke(f, new Object()); // illegal receiver type. IAE thrown
57+
fail("should not reach here");
58+
} catch (InvocationTargetException e) {
59+
Throwable t = e.getCause();
60+
if (!(t instanceof IllegalArgumentException)) {
61+
throw new RuntimeException("Unexpected cause of InvocationTargetException: " + t);
62+
}
63+
}
64+
}
65+
66+
/**
67+
* Tests the method being invoked throws InvocationTargetException which
68+
* will be wrapped with another InvocationTargetException by Method::invoke.
69+
*/
70+
@Test
71+
public void methodInvoke() throws ReflectiveOperationException {
72+
try {
73+
// m() throws InvocationTargetException which will be wrapped by Method::invoke
74+
Method m = CallerSensitiveMethodInvoke.class.getDeclaredMethod("m");
75+
m.invoke(new CallerSensitiveMethodInvoke());
76+
fail("should not reach here");
77+
} catch (InvocationTargetException e) {
78+
Throwable t = e.getCause();
79+
if (!(t instanceof InvocationTargetException)) {
80+
throw new RuntimeException("Unexpected cause of InvocationTargetException: " + t);
81+
}
82+
}
83+
}
84+
}

0 commit comments

Comments
 (0)
Please sign in to comment.