1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 import java.lang.invoke.MethodHandle;
18 import java.lang.invoke.MethodHandles;
19 import java.lang.invoke.MethodHandles.Lookup;
20 import java.lang.invoke.MethodType;
21 import java.lang.invoke.VarHandle;
22 import java.lang.invoke.WrongMethodTypeException;
23 
24 public class Main {
main(String[] args)25   public static void main(String[] args) throws Throwable {
26     testThrowException();
27     testDropArguments();
28     testCatchException();
29     testGuardWithTest();
30     testArrayElementGetter();
31     testArrayElementSetter();
32     testIdentity();
33     testConstant();
34     testBindTo();
35     testFilterReturnValue();
36     testPermuteArguments();
37     testInvokers();
38     testSpreaders_reference();
39     testSpreaders_primitive();
40     testInvokeWithArguments();
41     testAsCollector();
42     testFilterArguments();
43     testCollectArguments();
44     testInsertArguments();
45     testFoldArguments();
46   }
47 
testThrowException()48   public static void testThrowException() throws Throwable {
49     MethodHandle handle = MethodHandles.throwException(String.class,
50         IllegalArgumentException.class);
51 
52     if (handle.type().returnType() != String.class) {
53       fail("Unexpected return type for handle: " + handle +
54           " [ " + handle.type() + "]");
55     }
56 
57     final IllegalArgumentException iae = new IllegalArgumentException("boo!");
58     try {
59       handle.invoke(iae);
60       fail("Expected an exception of type: java.lang.IllegalArgumentException");
61     } catch (IllegalArgumentException expected) {
62       if (expected != iae) {
63         fail("Wrong exception: expected " + iae + " but was " + expected);
64       }
65     }
66   }
67 
dropArguments_delegate(String message, long message2)68   public static void dropArguments_delegate(String message, long message2) {
69     System.out.println("Message: " + message + ", Message2: " + message2);
70   }
71 
testDropArguments()72   public static void testDropArguments() throws Throwable {
73     MethodHandle delegate = MethodHandles.lookup().findStatic(Main.class,
74         "dropArguments_delegate",
75         MethodType.methodType(void.class, new Class<?>[] { String.class, long.class }));
76 
77     MethodHandle transform = MethodHandles.dropArguments(delegate, 0, int.class, Object.class);
78 
79     // The transformer will accept two additional arguments at position zero.
80     try {
81       transform.invokeExact("foo", 42l);
82       fail();
83     } catch (WrongMethodTypeException expected) {
84     }
85 
86     transform.invokeExact(45, new Object(), "foo", 42l);
87     transform.invoke(45, new Object(), "foo", 42l);
88 
89     // Additional arguments at position 1.
90     transform = MethodHandles.dropArguments(delegate, 1, int.class, Object.class);
91     transform.invokeExact("foo", 45, new Object(), 42l);
92     transform.invoke("foo", 45, new Object(), 42l);
93 
94     // Additional arguments at position 2.
95     transform = MethodHandles.dropArguments(delegate, 2, int.class, Object.class);
96     transform.invokeExact("foo", 42l, 45, new Object());
97     transform.invoke("foo", 42l, 45, new Object());
98 
99     // Note that we still perform argument conversions even for the arguments that
100     // are subsequently dropped.
101     try {
102       transform.invoke("foo", 42l, 45l, new Object());
103       fail();
104     } catch (WrongMethodTypeException expected) {
105     } catch (IllegalArgumentException expected) {
106       // TODO(narayan): We currently throw the wrong type of exception here,
107       // it's IAE and should be WMTE instead.
108     }
109 
110     // Check that asType works as expected.
111     transform = MethodHandles.dropArguments(delegate, 0, int.class, Object.class);
112     transform = transform.asType(MethodType.methodType(void.class,
113           new Class<?>[] { short.class, Object.class, String.class, long.class }));
114     transform.invokeExact((short) 45, new Object(), "foo", 42l);
115 
116     // Invalid argument location, should not be allowed.
117     try {
118       MethodHandles.dropArguments(delegate, -1, int.class, Object.class);
119       fail();
120     } catch (IllegalArgumentException expected) {
121     }
122 
123     // Invalid argument location, should not be allowed.
124     try {
125       MethodHandles.dropArguments(delegate, 3, int.class, Object.class);
126       fail();
127     } catch (IllegalArgumentException expected) {
128     }
129 
130     try {
131       MethodHandles.dropArguments(delegate, 1, void.class);
132       fail();
133     } catch (IllegalArgumentException expected) {
134     }
135   }
136 
testCatchException_target(String arg1, long arg2, String exceptionMessage)137   public static String testCatchException_target(String arg1, long arg2, String exceptionMessage)
138       throws Throwable {
139     if (exceptionMessage != null) {
140       throw new IllegalArgumentException(exceptionMessage);
141     }
142 
143     System.out.println("Target: Arg1: " + arg1 + ", Arg2: " + arg2);
144     return "target";
145   }
146 
testCatchException_handler(IllegalArgumentException iae, String arg1, long arg2, String exMsg)147   public static String testCatchException_handler(IllegalArgumentException iae, String arg1, long arg2,
148       String exMsg) {
149     System.out.println("Handler: " + iae + ", Arg1: " + arg1 + ", Arg2: " + arg2 + ", ExMsg: " + exMsg);
150     return "handler1";
151   }
152 
testCatchException_handler2(IllegalArgumentException iae, String arg1)153   public static String testCatchException_handler2(IllegalArgumentException iae, String arg1) {
154     System.out.println("Handler: " + iae + ", Arg1: " + arg1);
155     return "handler2";
156   }
157 
testCatchException()158   public static void testCatchException() throws Throwable {
159     MethodHandle target = MethodHandles.lookup().findStatic(Main.class,
160         "testCatchException_target",
161         MethodType.methodType(String.class, new Class<?>[] { String.class, long.class, String.class }));
162 
163     MethodHandle handler = MethodHandles.lookup().findStatic(Main.class,
164         "testCatchException_handler",
165         MethodType.methodType(String.class, new Class<?>[] { IllegalArgumentException.class,
166             String.class, long.class, String.class }));
167 
168     MethodHandle adapter = MethodHandles.catchException(target, IllegalArgumentException.class,
169         handler);
170 
171     String returnVal = null;
172 
173     // These two should end up calling the target always. We're passing a null exception
174     // message here, which means the target will not throw.
175     returnVal = (String) adapter.invoke("foo", 42, null);
176     assertEquals("target", returnVal);
177     returnVal = (String) adapter.invokeExact("foo", 42l, (String) null);
178     assertEquals("target", returnVal);
179 
180     // We're passing a non-null exception message here, which means the target will throw,
181     // which in turn means that the handler must be called for the next two invokes.
182     returnVal = (String) adapter.invoke("foo", 42, "exceptionMessage");
183     assertEquals("handler1", returnVal);
184     returnVal = (String) adapter.invokeExact("foo", 42l, "exceptionMessage");
185     assertEquals("handler1", returnVal);
186 
187     handler = MethodHandles.lookup().findStatic(Main.class,
188         "testCatchException_handler2",
189         MethodType.methodType(String.class, new Class<?>[] { IllegalArgumentException.class,
190             String.class }));
191     adapter = MethodHandles.catchException(target, IllegalArgumentException.class, handler);
192 
193     returnVal = (String) adapter.invoke("foo", 42, "exceptionMessage");
194     assertEquals("handler2", returnVal);
195     returnVal = (String) adapter.invokeExact("foo", 42l, "exceptionMessage");
196     assertEquals("handler2", returnVal);
197 
198     // Test that the type of the invoke doesn't matter. Here we call
199     // IllegalArgumentException.toString() on the exception that was thrown by
200     // the target.
201     handler = MethodHandles.lookup().findVirtual(IllegalArgumentException.class,
202         "toString", MethodType.methodType(String.class));
203     adapter = MethodHandles.catchException(target, IllegalArgumentException.class, handler);
204 
205     returnVal = (String) adapter.invoke("foo", 42, "exceptionMessage");
206     assertEquals("java.lang.IllegalArgumentException: exceptionMessage", returnVal);
207     returnVal = (String) adapter.invokeExact("foo", 42l, "exceptionMessage2");
208     assertEquals("java.lang.IllegalArgumentException: exceptionMessage2", returnVal);
209 
210     // Check that asType works as expected.
211     adapter = MethodHandles.catchException(target, IllegalArgumentException.class,
212         handler);
213     adapter = adapter.asType(MethodType.methodType(String.class,
214           new Class<?>[] { String.class, int.class, String.class }));
215     returnVal = (String) adapter.invokeExact("foo", 42, "exceptionMessage");
216     assertEquals("java.lang.IllegalArgumentException: exceptionMessage", returnVal);
217   }
218 
testGuardWithTest_test(String arg1, long arg2)219   public static boolean testGuardWithTest_test(String arg1, long arg2) {
220     return "target".equals(arg1) && 42 == arg2;
221   }
222 
testGuardWithTest_target(String arg1, long arg2, int arg3)223   public static String testGuardWithTest_target(String arg1, long arg2, int arg3) {
224     System.out.println("target: " + arg1 + ", " + arg2  + ", " + arg3);
225     return "target";
226   }
227 
testGuardWithTest_fallback(String arg1, long arg2, int arg3)228   public static String testGuardWithTest_fallback(String arg1, long arg2, int arg3) {
229     System.out.println("fallback: " + arg1 + ", " + arg2  + ", " + arg3);
230     return "fallback";
231   }
232 
testGuardWithTest()233   public static void testGuardWithTest() throws Throwable {
234     MethodHandle test = MethodHandles.lookup().findStatic(Main.class,
235         "testGuardWithTest_test",
236         MethodType.methodType(boolean.class, new Class<?>[] { String.class, long.class }));
237 
238     final MethodType type = MethodType.methodType(String.class,
239         new Class<?>[] { String.class, long.class, int.class });
240 
241     final MethodHandle target = MethodHandles.lookup().findStatic(Main.class,
242         "testGuardWithTest_target", type);
243     final MethodHandle fallback = MethodHandles.lookup().findStatic(Main.class,
244         "testGuardWithTest_fallback", type);
245 
246     MethodHandle adapter = MethodHandles.guardWithTest(test, target, fallback);
247 
248     String returnVal = null;
249 
250     returnVal = (String) adapter.invoke("target", 42, 56);
251     assertEquals("target", returnVal);
252     returnVal = (String) adapter.invokeExact("target", 42l, 56);
253     assertEquals("target", returnVal);
254 
255     returnVal = (String) adapter.invoke("fallback", 42l, 56);
256     assertEquals("fallback", returnVal);
257     returnVal = (String) adapter.invokeExact("target", 42l, 56);
258     assertEquals("target", returnVal);
259 
260     // Check that asType works as expected.
261     adapter = adapter.asType(MethodType.methodType(String.class,
262           new Class<?>[] { String.class, int.class, int.class }));
263     returnVal = (String) adapter.invokeExact("target", 42, 56);
264     assertEquals("target", returnVal);
265   }
266 
testArrayElementGetter()267   public static void testArrayElementGetter() throws Throwable {
268     MethodHandle getter = MethodHandles.arrayElementGetter(int[].class);
269 
270     {
271       int[] array = new int[1];
272       array[0] = 42;
273       int value = (int) getter.invoke(array, 0);
274       if (value != 42) {
275         fail("Unexpected value: " + value);
276       }
277 
278       try {
279         value = (int) getter.invoke(array, -1);
280         fail();
281       } catch (ArrayIndexOutOfBoundsException expected) {
282       }
283 
284       try {
285         value = (int) getter.invoke(null, -1);
286         fail();
287       } catch (NullPointerException expected) {
288       }
289     }
290 
291     {
292       getter = MethodHandles.arrayElementGetter(long[].class);
293       long[] array = new long[1];
294       array[0] = 42;
295       long value = (long) getter.invoke(array, 0);
296       if (value != 42l) {
297         fail("Unexpected value: " + value);
298       }
299     }
300 
301     {
302       getter = MethodHandles.arrayElementGetter(short[].class);
303       short[] array = new short[1];
304       array[0] = 42;
305       short value = (short) getter.invoke(array, 0);
306       if (value != 42l) {
307         fail("Unexpected value: " + value);
308       }
309     }
310 
311     {
312       getter = MethodHandles.arrayElementGetter(char[].class);
313       char[] array = new char[1];
314       array[0] = 42;
315       char value = (char) getter.invoke(array, 0);
316       if (value != 42l) {
317         fail("Unexpected value: " + value);
318       }
319     }
320 
321     {
322       getter = MethodHandles.arrayElementGetter(byte[].class);
323       byte[] array = new byte[1];
324       array[0] = (byte) 0x8;
325       byte value = (byte) getter.invoke(array, 0);
326       if (value != (byte) 0x8) {
327         fail("Unexpected value: " + value);
328       }
329     }
330 
331     {
332       getter = MethodHandles.arrayElementGetter(boolean[].class);
333       boolean[] array = new boolean[1];
334       array[0] = true;
335       boolean value = (boolean) getter.invoke(array, 0);
336       if (!value) {
337         fail("Unexpected value: " + value);
338       }
339     }
340 
341     {
342       getter = MethodHandles.arrayElementGetter(float[].class);
343       float[] array = new float[1];
344       array[0] = 42.0f;
345       float value = (float) getter.invoke(array, 0);
346       if (value != 42.0f) {
347         fail("Unexpected value: " + value);
348       }
349     }
350 
351     {
352       getter = MethodHandles.arrayElementGetter(double[].class);
353       double[] array = new double[1];
354       array[0] = 42.0;
355       double value = (double) getter.invoke(array, 0);
356       if (value != 42.0) {
357         fail("Unexpected value: " + value);
358       }
359     }
360 
361     {
362       getter = MethodHandles.arrayElementGetter(String[].class);
363       String[] array = new String[3];
364       array[0] = "42";
365       array[1] = "48";
366       array[2] = "54";
367       String value = (String) getter.invoke(array, 0);
368       assertEquals("42", value);
369       value = (String) getter.invoke(array, 1);
370       assertEquals("48", value);
371       value = (String) getter.invoke(array, 2);
372       assertEquals("54", value);
373     }
374   }
375 
testArrayElementSetter()376   public static void testArrayElementSetter() throws Throwable {
377     MethodHandle setter = MethodHandles.arrayElementSetter(int[].class);
378 
379     {
380       int[] array = new int[2];
381       setter.invoke(array, 0, 42);
382       setter.invoke(array, 1, 43);
383 
384       if (array[0] != 42) {
385         fail("Unexpected value: " + array[0]);
386       }
387       if (array[1] != 43) {
388         fail("Unexpected value: " + array[1]);
389       }
390 
391       try {
392         setter.invoke(array, -1, 42);
393         fail();
394       } catch (ArrayIndexOutOfBoundsException expected) {
395       }
396 
397       try {
398         setter.invoke(null, 0, 42);
399         fail();
400       } catch (NullPointerException expected) {
401       }
402     }
403 
404     {
405       setter = MethodHandles.arrayElementSetter(long[].class);
406       long[] array = new long[1];
407       setter.invoke(array, 0, 42l);
408       if (array[0] != 42l) {
409         fail("Unexpected value: " + array[0]);
410       }
411     }
412 
413     {
414       setter = MethodHandles.arrayElementSetter(short[].class);
415       short[] array = new short[1];
416       setter.invoke(array, 0, (short) 42);
417       if (array[0] != 42l) {
418         fail("Unexpected value: " + array[0]);
419       }
420     }
421 
422     {
423       setter = MethodHandles.arrayElementSetter(char[].class);
424       char[] array = new char[1];
425       setter.invoke(array, 0, (char) 42);
426       if (array[0] != 42) {
427         fail("Unexpected value: " + array[0]);
428       }
429     }
430 
431     {
432       setter = MethodHandles.arrayElementSetter(byte[].class);
433       byte[] array = new byte[1];
434       setter.invoke(array, 0, (byte) 0x8);
435       if (array[0] != (byte) 0x8) {
436         fail("Unexpected value: " + array[0]);
437       }
438     }
439 
440     {
441       setter = MethodHandles.arrayElementSetter(boolean[].class);
442       boolean[] array = new boolean[1];
443       setter.invoke(array, 0, true);
444       if (!array[0]) {
445         fail("Unexpected value: " + array[0]);
446       }
447     }
448 
449     {
450       setter = MethodHandles.arrayElementSetter(float[].class);
451       float[] array = new float[1];
452       setter.invoke(array, 0, 42.0f);
453       if (array[0] != 42.0f) {
454         fail("Unexpected value: " + array[0]);
455       }
456     }
457 
458     {
459       setter = MethodHandles.arrayElementSetter(double[].class);
460       double[] array = new double[1];
461       setter.invoke(array, 0, 42.0);
462       if (array[0] != 42.0) {
463         fail("Unexpected value: " + array[0]);
464       }
465     }
466 
467     {
468       setter = MethodHandles.arrayElementSetter(String[].class);
469       String[] array = new String[3];
470       setter.invoke(array, 0, "42");
471       setter.invoke(array, 1, "48");
472       setter.invoke(array, 2, "54");
473       assertEquals("42", array[0]);
474       assertEquals("48", array[1]);
475       assertEquals("54", array[2]);
476     }
477   }
478 
testIdentity()479   public static void testIdentity() throws Throwable {
480     {
481       MethodHandle identity = MethodHandles.identity(boolean.class);
482       boolean value = (boolean) identity.invoke(false);
483       if (value) {
484         fail("Unexpected value: " + value);
485       }
486     }
487 
488     {
489       MethodHandle identity = MethodHandles.identity(byte.class);
490       byte value = (byte) identity.invoke((byte) 0x8);
491       if (value != (byte) 0x8) {
492         fail("Unexpected value: " + value);
493       }
494     }
495 
496     {
497       MethodHandle identity = MethodHandles.identity(char.class);
498       char value = (char) identity.invoke((char) -56);
499       if (value != (char) -56) {
500         fail("Unexpected value: " + value);
501       }
502     }
503 
504     {
505       MethodHandle identity = MethodHandles.identity(short.class);
506       short value = (short) identity.invoke((short) -59);
507       if (value != (short) -59) {
508         fail("Unexpected value: " + Short.toString(value));
509       }
510     }
511 
512     {
513       MethodHandle identity = MethodHandles.identity(int.class);
514       int value = (int) identity.invoke(52);
515       if (value != 52) {
516         fail("Unexpected value: " + value);
517       }
518     }
519 
520     {
521       MethodHandle identity = MethodHandles.identity(long.class);
522       long value = (long) identity.invoke(-76l);
523       if (value != (long) -76) {
524         fail("Unexpected value: " + value);
525       }
526     }
527 
528     {
529       MethodHandle identity = MethodHandles.identity(float.class);
530       float value = (float) identity.invoke(56.0f);
531       if (value != (float) 56.0f) {
532         fail("Unexpected value: " + value);
533       }
534     }
535 
536     {
537       MethodHandle identity = MethodHandles.identity(double.class);
538       double value = (double) identity.invoke((double) 72.0);
539       if (value != (double) 72.0) {
540         fail("Unexpected value: " + value);
541       }
542     }
543 
544     {
545       MethodHandle identity = MethodHandles.identity(String.class);
546       String value = (String) identity.invoke("bazman");
547       assertEquals("bazman", value);
548     }
549   }
550 
testConstant()551   public static void testConstant() throws Throwable {
552     // int constants.
553     {
554       MethodHandle constant = MethodHandles.constant(int.class, 56);
555       int value = (int) constant.invoke();
556       if (value != 56) {
557         fail("Unexpected value: " + value);
558       }
559 
560       // short constant values are converted to int.
561       constant = MethodHandles.constant(int.class, (short) 52);
562       value = (int) constant.invoke();
563       if (value != 52) {
564         fail("Unexpected value: " + value);
565       }
566 
567       // char constant values are converted to int.
568       constant = MethodHandles.constant(int.class, (char) 'b');
569       value = (int) constant.invoke();
570       if (value != (int) 'b') {
571         fail("Unexpected value: " + value);
572       }
573 
574       // int constant values are converted to int.
575       constant = MethodHandles.constant(int.class, (byte) 0x1);
576       value = (int) constant.invoke();
577       if (value != 1) {
578         fail("Unexpected value: " + value);
579       }
580 
581       // boolean, float, double and long primitive constants are not convertible
582       // to int, so the handle creation must fail with a CCE.
583       try {
584         MethodHandles.constant(int.class, false);
585         fail();
586       } catch (ClassCastException expected) {
587       }
588 
589       try {
590         MethodHandles.constant(int.class, 0.1f);
591         fail();
592       } catch (ClassCastException expected) {
593       }
594 
595       try {
596         MethodHandles.constant(int.class, 0.2);
597         fail();
598       } catch (ClassCastException expected) {
599       }
600 
601       try {
602         MethodHandles.constant(int.class, 73l);
603         fail();
604       } catch (ClassCastException expected) {
605       }
606     }
607 
608     // long constants.
609     {
610       MethodHandle constant = MethodHandles.constant(long.class, 56l);
611       long value = (long) constant.invoke();
612       if (value != 56l) {
613         fail("Unexpected value: " + value);
614       }
615 
616       constant = MethodHandles.constant(long.class, (int) 56);
617       value = (long) constant.invoke();
618       if (value != 56l) {
619         fail("Unexpected value: " + value);
620       }
621     }
622 
623     // byte constants.
624     {
625       MethodHandle constant = MethodHandles.constant(byte.class, (byte) 0x12);
626       byte value = (byte) constant.invoke();
627       if (value != (byte) 0x12) {
628         fail("Unexpected value: " + value);
629       }
630     }
631 
632     // boolean constants.
633     {
634       MethodHandle constant = MethodHandles.constant(boolean.class, true);
635       boolean value = (boolean) constant.invoke();
636       if (!value) {
637         fail("Unexpected value: " + value);
638       }
639     }
640 
641     // char constants.
642     {
643       MethodHandle constant = MethodHandles.constant(char.class, 'f');
644       char value = (char) constant.invoke();
645       if (value != 'f') {
646         fail("Unexpected value: " + value);
647       }
648     }
649 
650     // short constants.
651     {
652       MethodHandle constant = MethodHandles.constant(short.class, (short) 123);
653       short value = (short) constant.invoke();
654       if (value != (short) 123) {
655         fail("Unexpected value: " + value);
656       }
657     }
658 
659     // float constants.
660     {
661       MethodHandle constant = MethodHandles.constant(float.class, 56.0f);
662       float value = (float) constant.invoke();
663       if (value != 56.0f) {
664         fail("Unexpected value: " + value);
665       }
666     }
667 
668     // double constants.
669     {
670       MethodHandle constant = MethodHandles.constant(double.class, 256.0);
671       double value = (double) constant.invoke();
672       if (value != 256.0) {
673         fail("Unexpected value: " + value);
674       }
675     }
676 
677     // reference constants.
678     {
679       MethodHandle constant = MethodHandles.constant(String.class, "256.0");
680       String value = (String) constant.invoke();
681       assertEquals("256.0", value);
682     }
683   }
684 
testBindTo()685   public static void testBindTo() throws Throwable {
686     MethodHandle stringCharAt = MethodHandles.lookup().findVirtual(
687         String.class, "charAt", MethodType.methodType(char.class, int.class));
688 
689     char value = (char) stringCharAt.invoke("foo", 0);
690     if (value != 'f') {
691       fail("Unexpected value: " + value);
692     }
693 
694     MethodHandle bound = stringCharAt.bindTo("foo");
695     value = (char) bound.invoke(0);
696     if (value != 'f') {
697       fail("Unexpected value: " + value);
698     }
699 
700     try {
701       stringCharAt.bindTo(new Object());
702       fail();
703     } catch (ClassCastException expected) {
704     }
705 
706     bound = stringCharAt.bindTo(null);
707     try {
708       bound.invoke(0);
709       fail();
710     } catch (NullPointerException expected) {
711     }
712 
713     MethodHandle integerParseInt = MethodHandles.lookup().findStatic(
714         Integer.class, "parseInt", MethodType.methodType(int.class, String.class));
715 
716     bound = integerParseInt.bindTo("78452");
717     int intValue = (int) bound.invoke();
718     if (intValue != 78452) {
719       fail("Unexpected value: " + intValue);
720     }
721 
722     class Locals {
723         public int intValue;
724     };
725 
726     MethodHandle setter1 = MethodHandles.lookup().findSetter(Locals.class, "intValue", int.class);
727     Locals locals = new Locals();
728     setter1.invoke(locals, 1);
729     assertEquals(1, locals.intValue);
730 
731     MethodHandle setter2 =
732         MethodHandles.insertArguments(setter1.bindTo(locals), 0, Integer.valueOf(2));
733     setter2.invoke();
734     assertEquals(2, locals.intValue);
735 
736     VarHandle vh = MethodHandles.lookup().findVarHandle(Locals.class,  "intValue", int.class);
737     MethodHandle setter3 = vh.toMethodHandle(VarHandle.AccessMode.GET_AND_SET);
738     assertEquals(setter3.type().toString(),
739                  MethodType.methodType(int.class, Locals.class, int.class).toString());
740     assertEquals(2, (int) setter3.invoke(locals, 3));
741 
742     MethodHandle setter4 =
743         MethodHandles.varHandleInvoker(VarHandle.AccessMode.GET_AND_SET,
744                                        MethodType.methodType(int.class, Locals.class, int.class));
745     assertEquals(3, (int) setter4.invoke(vh, locals, 4));
746 
747     MethodHandle setter5 = setter4.bindTo(vh);
748     assertEquals(4, (int) setter5.invoke(locals, 5));
749 
750     MethodHandle setter6 = MethodHandles.insertArguments(setter5, 0, locals, 6);
751     assertEquals(5, (int) setter6.invoke());
752     assertEquals(locals.intValue, 6);
753   }
754 
filterReturnValue_target(int a)755   public static String filterReturnValue_target(int a) {
756     return "ReturnValue" + a;
757   }
758 
filterReturnValue_filter(String value)759   public static boolean filterReturnValue_filter(String value) {
760     return value.indexOf("42") != -1;
761   }
762 
filterReturnValue_intTarget(String a)763   public static int filterReturnValue_intTarget(String a) {
764     return Integer.parseInt(a);
765   }
766 
filterReturnValue_intFilter(int b)767   public static int filterReturnValue_intFilter(int b) {
768     return b + 1;
769   }
770 
filterReturnValue_voidTarget()771   public static void filterReturnValue_voidTarget() {
772   }
773 
filterReturnValue_voidFilter()774   public static int filterReturnValue_voidFilter() {
775     return 42;
776   }
777 
testFilterReturnValue()778   public static void testFilterReturnValue() throws Throwable {
779     // A target that returns a reference.
780     {
781       final MethodHandle target = MethodHandles.lookup().findStatic(Main.class,
782           "filterReturnValue_target", MethodType.methodType(String.class, int.class));
783       final MethodHandle filter = MethodHandles.lookup().findStatic(Main.class,
784           "filterReturnValue_filter", MethodType.methodType(boolean.class, String.class));
785 
786       MethodHandle adapter = MethodHandles.filterReturnValue(target, filter);
787 
788       boolean value = (boolean) adapter.invoke((int) 42);
789       if (!value) {
790         fail("Unexpected value: " + value);
791       }
792       value = (boolean) adapter.invoke((int) 43);
793       if (value) {
794         fail("Unexpected value: " + value);
795       }
796     }
797 
798     // A target that returns a primitive.
799     {
800       final MethodHandle target = MethodHandles.lookup().findStatic(Main.class,
801           "filterReturnValue_intTarget", MethodType.methodType(int.class, String.class));
802       final MethodHandle filter = MethodHandles.lookup().findStatic(Main.class,
803           "filterReturnValue_intFilter", MethodType.methodType(int.class, int.class));
804 
805       MethodHandle adapter = MethodHandles.filterReturnValue(target, filter);
806 
807       int value = (int) adapter.invoke("56");
808       if (value != 57) {
809         fail("Unexpected value: " + value);
810       }
811     }
812 
813     // A target that returns void.
814     {
815       final MethodHandle target = MethodHandles.lookup().findStatic(Main.class,
816           "filterReturnValue_voidTarget", MethodType.methodType(void.class));
817       final MethodHandle filter = MethodHandles.lookup().findStatic(Main.class,
818           "filterReturnValue_voidFilter", MethodType.methodType(int.class));
819 
820       MethodHandle adapter = MethodHandles.filterReturnValue(target, filter);
821 
822       int value = (int) adapter.invoke();
823       if (value != 42) {
824         fail("Unexpected value: " + value);
825       }
826     }
827   }
828 
permuteArguments_callee(boolean a, byte b, char c, short d, int e, long f, float g, double h)829   public static void permuteArguments_callee(boolean a, byte b, char c,
830       short d, int e, long f, float g, double h) {
831     if (a == true && b == (byte) 'b' && c == 'c' && d == (short) 56 &&
832         e == 78 && f == (long) 97 && g == 98.0f && f == 97.0) {
833       return;
834     }
835 
836     fail("Unexpected arguments: " + a + ", " + b + ", " + c
837         + ", " + d + ", " + e + ", " + f + ", " + g + ", " + h);
838   }
839 
permuteArguments_boxingCallee(boolean a, Integer b)840   public static void permuteArguments_boxingCallee(boolean a, Integer b) {
841     if (a && b.intValue() == 42) {
842       return;
843     }
844 
845     fail("Unexpected arguments: " + a + ", " + b);
846   }
847 
testPermuteArguments()848   public static void testPermuteArguments() throws Throwable {
849     {
850       final MethodHandle target = MethodHandles.lookup().findStatic(
851           Main.class, "permuteArguments_callee",
852           MethodType.methodType(void.class, new Class<?>[] {
853             boolean.class, byte.class, char.class, short.class, int.class,
854             long.class, float.class, double.class }));
855 
856       final MethodType newType = MethodType.methodType(void.class, new Class<?>[] {
857         double.class, float.class, long.class, int.class, short.class, char.class,
858         byte.class, boolean.class });
859 
860       final MethodHandle permutation = MethodHandles.permuteArguments(
861           target, newType, new int[] { 7, 6, 5, 4, 3, 2, 1, 0 });
862 
863       permutation.invoke((double) 97.0, (float) 98.0f, (long) 97, 78,
864           (short) 56, 'c', (byte) 'b', (boolean) true);
865 
866       // The permutation array was not of the right length.
867       try {
868         MethodHandles.permuteArguments(target, newType,
869             new int[] { 7 });
870         fail();
871       } catch (IllegalArgumentException expected) {
872       }
873 
874       // The permutation array has an element that's out of bounds
875       // (there's no argument with idx == 8).
876       try {
877         MethodHandles.permuteArguments(target, newType,
878             new int[] { 8, 6, 5, 4, 3, 2, 1, 0 });
879         fail();
880       } catch (IllegalArgumentException expected) {
881       }
882 
883       // The permutation array maps to an incorrect type.
884       try {
885         MethodHandles.permuteArguments(target, newType,
886             new int[] { 7, 7, 5, 4, 3, 2, 1, 0 });
887         fail();
888       } catch (IllegalArgumentException expected) {
889       }
890     }
891 
892     // Tests for reference arguments as well as permutations that
893     // repeat arguments.
894     {
895       final MethodHandle target = MethodHandles.lookup().findVirtual(
896           String.class, "concat", MethodType.methodType(String.class, String.class));
897 
898       final MethodType newType = MethodType.methodType(String.class, String.class,
899           String.class);
900 
901       assertEquals("foobar", (String) target.invoke("foo", "bar"));
902 
903       MethodHandle permutation = MethodHandles.permuteArguments(target,
904           newType, new int[] { 1, 0 });
905       assertEquals("barfoo", (String) permutation.invoke("foo", "bar"));
906 
907       permutation = MethodHandles.permuteArguments(target, newType, new int[] { 0, 0 });
908       assertEquals("foofoo", (String) permutation.invoke("foo", "bar"));
909 
910       permutation = MethodHandles.permuteArguments(target, newType, new int[] { 1, 1 });
911       assertEquals("barbar", (String) permutation.invoke("foo", "bar"));
912     }
913 
914     // Tests for boxing and unboxing.
915     {
916       final MethodHandle target = MethodHandles.lookup().findStatic(
917           Main.class, "permuteArguments_boxingCallee",
918           MethodType.methodType(void.class, new Class<?>[] { boolean.class, Integer.class }));
919 
920       final MethodType newType = MethodType.methodType(void.class,
921           new Class<?>[] { Integer.class, boolean.class });
922 
923       MethodHandle permutation = MethodHandles.permuteArguments(target,
924           newType, new int[] { 1, 0 });
925 
926       permutation.invoke(42, true);
927       permutation.invoke(42, Boolean.TRUE);
928       permutation.invoke(Integer.valueOf(42), true);
929       permutation.invoke(Integer.valueOf(42), Boolean.TRUE);
930     }
931   }
932 
returnBar()933   private static Object returnBar() {
934     return "bar";
935   }
936 
testInvokers()937   public static void testInvokers() throws Throwable {
938     final MethodType targetType = MethodType.methodType(String.class, String.class);
939     final MethodHandle target = MethodHandles.lookup().findVirtual(
940         String.class, "concat", targetType);
941 
942     MethodHandle invoker = MethodHandles.invoker(target.type());
943     assertEquals("barbar", (String) invoker.invoke(target, "bar", "bar"));
944     assertEquals("barbar", (String) invoker.invoke(target, (Object) returnBar(), "bar"));
945     try {
946       String foo = (String) invoker.invoke(target, "bar", "bar", 24);
947       fail();
948     } catch (WrongMethodTypeException expected) {
949     }
950 
951     MethodHandle exactInvoker = MethodHandles.exactInvoker(target.type());
952     assertEquals("barbar", (String) exactInvoker.invoke(target, "bar", "bar"));
953     assertEquals("barbar", (String) exactInvoker.invoke(target, (Object) returnBar(), "bar"));
954     try {
955       assertEquals("barbar", (String) invoker.invoke(target, (Object) Integer.valueOf(7), "bar"));
956       fail();
957     } catch (ClassCastException expected) {
958     }
959     try {
960       assertEquals("barbar", (String) invoker.invoke(target, (Object) null, "bar"));
961       fail();
962     } catch (NullPointerException expected) {
963     }
964     exactInvoker.invoke(target, (Object) returnBar(), "bar");
965 
966     try {
967       String foo = (String) exactInvoker.invoke(target, "bar", "bar", 24);
968       fail();
969     } catch (WrongMethodTypeException expected) {
970     }
971   }
972 
spreadReferences(String a, String b, String c)973   public static int spreadReferences(String a, String b, String c) {
974     System.out.println("a: " + a + ", b:" + b + ", c: " + c);
975     return 42;
976   }
977 
spreadReferences_Unbox(String a, int b)978   public static int spreadReferences_Unbox(String a, int b) {
979     System.out.println("a: " + a + ", b:" + b);
980     return 43;
981   }
982 
testSpreaders_reference()983   public static void testSpreaders_reference() throws Throwable {
984     MethodType methodType = MethodType.methodType(int.class,
985         new Class<?>[] { String.class, String.class, String.class });
986     MethodHandle delegate = MethodHandles.lookup().findStatic(
987         Main.class, "spreadReferences", methodType);
988 
989     // Basic checks on array lengths.
990     //
991     // Array size = 0
992     MethodHandle mhAsSpreader = delegate.asSpreader(String[].class, 0);
993     int ret = (int) mhAsSpreader.invoke("a", "b", "c", new String[] {});
994     assertEquals(42, ret);
995     // Array size = 1
996     mhAsSpreader = delegate.asSpreader(String[].class, 1);
997     ret = (int) mhAsSpreader.invoke("a", "b", new String[] { "c" });
998     assertEquals(42, ret);
999     // Array size = 2
1000     mhAsSpreader = delegate.asSpreader(String[].class, 2);
1001     ret = (int) mhAsSpreader.invoke("a", new String[] { "b", "c" });
1002     assertEquals(42, ret);
1003     // Array size = 3
1004     mhAsSpreader = delegate.asSpreader(String[].class, 3);
1005     ret = (int) mhAsSpreader.invoke(new String[] { "a", "b", "c"});
1006     assertEquals(42, ret);
1007 
1008     // Exception case, array size = 4 is illegal.
1009     try {
1010       delegate.asSpreader(String[].class, 4);
1011       fail();
1012     } catch (IllegalArgumentException expected) {
1013     }
1014 
1015     // Exception case, calling with an arg of the wrong size.
1016     // Array size = 3
1017     mhAsSpreader = delegate.asSpreader(String[].class, 3);
1018     try {
1019       ret = (int) mhAsSpreader.invoke(new String[] { "a", "b"});
1020     } catch (IllegalArgumentException expected) {
1021     }
1022 
1023     // Various other hijinks, pass as Object[] arrays, Object etc.
1024     mhAsSpreader = delegate.asSpreader(Object[].class, 2);
1025     ret = (int) mhAsSpreader.invoke("a", new String[] { "b", "c" });
1026     assertEquals(42, ret);
1027 
1028     mhAsSpreader = delegate.asSpreader(Object[].class, 2);
1029     ret = (int) mhAsSpreader.invoke("a", new Object[] { "b", "c" });
1030     assertEquals(42, ret);
1031 
1032     mhAsSpreader = delegate.asSpreader(Object[].class, 2);
1033     ret = (int) mhAsSpreader.invoke("a", (Object) new Object[] { "b", "c" });
1034     assertEquals(42, ret);
1035 
1036     // Test implicit unboxing.
1037     MethodType methodType2 = MethodType.methodType(int.class,
1038         new Class<?>[] { String.class, int.class });
1039     MethodHandle delegate2 = MethodHandles.lookup().findStatic(
1040         Main.class, "spreadReferences_Unbox", methodType2);
1041 
1042     // .. with an Integer[] array.
1043     mhAsSpreader = delegate2.asSpreader(Integer[].class, 1);
1044     ret = (int) mhAsSpreader.invoke("a", new Integer[] { 43 });
1045     assertEquals(43, ret);
1046 
1047     // .. with an Integer[] array declared as an Object[] argument type.
1048     mhAsSpreader = delegate2.asSpreader(Object[].class, 1);
1049     ret = (int) mhAsSpreader.invoke("a", new Integer[] { 43 });
1050     assertEquals(43, ret);
1051 
1052     // .. with an Object[] array.
1053     mhAsSpreader = delegate2.asSpreader(Object[].class, 1);
1054     ret = (int) mhAsSpreader.invoke("a", new Object[] { Integer.valueOf(43)});
1055     assertEquals(43, ret);
1056 
1057     // -- Part 2--
1058     // Run a subset of these tests on MethodHandles.spreadInvoker, which only accepts
1059     // a trailing argument type of Object[].
1060     MethodHandle spreadInvoker = MethodHandles.spreadInvoker(methodType2, 1);
1061     ret = (int) spreadInvoker.invoke(delegate2, "a", new Object[] { Integer.valueOf(43)});
1062     assertEquals(43, ret);
1063 
1064     ret = (int) spreadInvoker.invoke(delegate2, "a", new Integer[] { 43 });
1065     assertEquals(43, ret);
1066 
1067     // NOTE: Annoyingly, the second argument here is leadingArgCount and not
1068     // arrayLength.
1069     spreadInvoker = MethodHandles.spreadInvoker(methodType, 3);
1070     ret = (int) spreadInvoker.invoke(delegate, "a", "b", "c", new String[] {});
1071     assertEquals(42, ret);
1072 
1073     spreadInvoker = MethodHandles.spreadInvoker(methodType, 0);
1074     ret = (int) spreadInvoker.invoke(delegate, new String[] { "a", "b", "c" });
1075     assertEquals(42, ret);
1076 
1077     // Exact invokes: Double check that the expected parameter type is
1078     // Object[] and not T[].
1079     try {
1080       spreadInvoker.invokeExact(delegate, new String[] { "a", "b", "c" });
1081       fail();
1082     } catch (WrongMethodTypeException expected) {
1083     }
1084 
1085     ret = (int) spreadInvoker.invoke(delegate, new Object[] { "a", "b", "c" });
1086     assertEquals(42, ret);
1087   }
1088 
spreadBoolean(String a, Boolean b, boolean c)1089   public static int spreadBoolean(String a, Boolean b, boolean c) {
1090     System.out.println("a: " + a + ", b:" + b + ", c: " + c);
1091     return 44;
1092   }
1093 
spreadByte(String a, Byte b, byte c, short d, int e, long f, float g, double h)1094   public static int spreadByte(String a, Byte b, byte c,
1095       short d, int e, long f, float g, double h) {
1096     System.out.println("a: " + a + ", b:" + b + ", c: " + c +
1097         ", d: " + d + ", e: " + e + ", f:" + f + ", g: " + g +
1098         ", h: " + h);
1099     return 45;
1100   }
1101 
spreadChar(String a, Character b, char c, int d, long e, float f, double g)1102   public static int spreadChar(String a, Character b, char c,
1103       int d, long e, float f, double g) {
1104     System.out.println("a: " + a + ", b:" + b + ", c: " + c +
1105         ", d: " + d + ", e: " + e + ", f:" + f + ", g: " + g);
1106     return 46;
1107   }
1108 
spreadShort(String a, Short b, short c, int d, long e, float f, double g)1109   public static int spreadShort(String a, Short b, short c,
1110       int d, long e, float f, double g) {
1111     System.out.println("a: " + a + ", b:" + b + ", c: " + c +
1112         ", d: " + d + ", e: " + e + ", f:" + f + ", g:" + g);
1113     return 47;
1114   }
1115 
spreadInt(String a, Integer b, int c, long d, float e, double f)1116   public static int spreadInt(String a, Integer b, int c,
1117       long d, float e, double f) {
1118     System.out.println("a: " + a + ", b:" + b + ", c: " + c +
1119         ", d: " + d + ", e: " + e + ", f:" + f);
1120     return 48;
1121   }
1122 
spreadLong(String a, Long b, long c, float d, double e)1123   public static int spreadLong(String a, Long b, long c, float d, double e) {
1124     System.out.println("a: " + a + ", b:" + b + ", c: " + c +
1125         ", d: " + d + ", e: " + e);
1126     return 49;
1127   }
1128 
spreadFloat(String a, Float b, float c, double d)1129   public static int spreadFloat(String a, Float b, float c, double d) {
1130     System.out.println("a: " + a + ", b:" + b + ", c: " + c + ", d: " + d);
1131     return 50;
1132   }
1133 
spreadDouble(String a, Double b, double c)1134   public static int spreadDouble(String a, Double b, double c) {
1135     System.out.println("a: " + a + ", b:" + b + ", c: " + c);
1136     return 51;
1137   }
1138 
testSpreaders_primitive()1139   public static void testSpreaders_primitive() throws Throwable {
1140     // boolean[]
1141     // ---------------------
1142     MethodType type = MethodType.methodType(int.class,
1143         new Class<?>[] { String.class, Boolean.class, boolean.class });
1144     MethodHandle delegate = MethodHandles.lookup().findStatic(
1145         Main.class, "spreadBoolean", type);
1146 
1147     MethodHandle spreader = delegate.asSpreader(boolean[].class, 2);
1148     int ret = (int) spreader.invokeExact("a", new boolean[] { true, false });
1149     assertEquals(44, ret);
1150     ret = (int) spreader.invoke("a", new boolean[] { true, false });
1151     assertEquals(44, ret);
1152 
1153     // boolean can't be cast to String (the first argument to the method).
1154     try {
1155       delegate.asSpreader(boolean[].class, 3);
1156       fail();
1157     } catch (WrongMethodTypeException expected) {
1158     }
1159 
1160     // int can't be cast to boolean to supply the last argument to the method.
1161     try {
1162       delegate.asSpreader(int[].class, 1);
1163       fail();
1164     } catch (WrongMethodTypeException expected) {
1165     }
1166 
1167     // byte[]
1168     // ---------------------
1169     type = MethodType.methodType(int.class,
1170         new Class<?>[] {
1171           String.class, Byte.class, byte.class,
1172           short.class, int.class, long.class,
1173           float.class, double.class });
1174     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadByte", type);
1175 
1176     spreader = delegate.asSpreader(byte[].class, 7);
1177     ret = (int) spreader.invokeExact("a",
1178         new byte[] { 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7 });
1179     assertEquals(45, ret);
1180     ret = (int) spreader.invoke("a",
1181         new byte[] { 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7 });
1182     assertEquals(45, ret);
1183 
1184     // char[]
1185     // ---------------------
1186     type = MethodType.methodType(int.class,
1187         new Class<?>[] {
1188           String.class, Character.class,char.class,
1189           int.class, long.class, float.class, double.class });
1190     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadChar", type);
1191 
1192     spreader = delegate.asSpreader(char[].class, 6);
1193     ret = (int) spreader.invokeExact("a",
1194         new char[] { '1', '2', '3', '4', '5', '6' });
1195     assertEquals(46, ret);
1196     ret = (int) spreader.invokeExact("a",
1197         new char[] { '1', '2', '3', '4', '5', '6' });
1198     assertEquals(46, ret);
1199 
1200     // short[]
1201     // ---------------------
1202     type = MethodType.methodType(int.class,
1203         new Class<?>[] {
1204           String.class, Short.class, short.class,
1205           int.class, long.class, float.class, double.class });
1206     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadShort", type);
1207 
1208     spreader = delegate.asSpreader(short[].class, 6);
1209     ret = (int) spreader.invokeExact("a",
1210         new short[] { 0x1, 0x2, 0x3, 0x4, 0x5, 0x6 });
1211     assertEquals(47, ret);
1212     ret = (int) spreader.invoke("a",
1213         new short[] { 0x1, 0x2, 0x3, 0x4, 0x5, 0x6 });
1214     assertEquals(47, ret);
1215 
1216     // int[]
1217     // ---------------------
1218     type = MethodType.methodType(int.class,
1219         new Class<?>[] {
1220           String.class, Integer.class, int.class,
1221           long.class, float.class, double.class });
1222     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadInt", type);
1223 
1224     spreader = delegate.asSpreader(int[].class, 5);
1225     ret = (int) spreader.invokeExact("a", new int[] { 1, 2, 3, 4, 5 });
1226     assertEquals(48, ret);
1227     ret = (int) spreader.invokeExact("a", new int[] { 1, 2, 3, 4, 5 });
1228     assertEquals(48, ret);
1229 
1230     // long[]
1231     // ---------------------
1232     type = MethodType.methodType(int.class,
1233         new Class<?>[] {
1234           String.class, Long.class, long.class, float.class, double.class });
1235     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadLong", type);
1236 
1237     spreader = delegate.asSpreader(long[].class, 4);
1238     ret = (int) spreader.invokeExact("a",
1239         new long[] { 0x1, 0x2, 0x3, 0x4 });
1240     assertEquals(49, ret);
1241     ret = (int) spreader.invoke("a",
1242         new long[] { 0x1, 0x2, 0x3, 0x4 });
1243     assertEquals(49, ret);
1244 
1245     // float[]
1246     // ---------------------
1247     type = MethodType.methodType(int.class,
1248         new Class<?>[] {
1249           String.class, Float.class, float.class, double.class });
1250     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadFloat", type);
1251 
1252     spreader = delegate.asSpreader(float[].class, 3);
1253     ret = (int) spreader.invokeExact("a",
1254         new float[] { 1.0f, 2.0f, 3.0f });
1255     assertEquals(50, ret);
1256     ret = (int) spreader.invokeExact("a",
1257         new float[] { 1.0f, 2.0f, 3.0f });
1258     assertEquals(50, ret);
1259 
1260     // double[]
1261     // ---------------------
1262     type = MethodType.methodType(int.class,
1263         new Class<?>[] { String.class, Double.class, double.class });
1264     delegate = MethodHandles.lookup().findStatic(Main.class, "spreadDouble", type);
1265 
1266     spreader = delegate.asSpreader(double[].class, 2);
1267     ret = (int) spreader.invokeExact("a", new double[] { 1.0, 2.0 });
1268     assertEquals(51, ret);
1269     ret = (int) spreader.invokeExact("a", new double[] { 1.0, 2.0 });
1270     assertEquals(51, ret);
1271   }
1272 
testInvokeWithArguments()1273   public static void testInvokeWithArguments() throws Throwable {
1274     MethodType methodType = MethodType.methodType(int.class,
1275         new Class<?>[] { String.class, String.class, String.class });
1276     MethodHandle handle = MethodHandles.lookup().findStatic(
1277         Main.class, "spreadReferences", methodType);
1278 
1279     Object ret = handle.invokeWithArguments(new Object[] { "a", "b", "c"});
1280     assertEquals(42, (int) ret);
1281     handle.invokeWithArguments(new String[] { "a", "b", "c" });
1282     assertEquals(42, (int) ret);
1283 
1284     // Pass in an array that's too small. Should throw an IAE.
1285     try {
1286       handle.invokeWithArguments(new Object[] { "a", "b" });
1287       fail();
1288     } catch (IllegalArgumentException expected) {
1289     } catch (WrongMethodTypeException expected) {
1290     }
1291 
1292     // Test implicit unboxing.
1293     MethodType methodType2 = MethodType.methodType(int.class,
1294         new Class<?>[] { String.class, int.class });
1295     MethodHandle handle2 = MethodHandles.lookup().findStatic(
1296         Main.class, "spreadReferences_Unbox", methodType2);
1297 
1298     ret = (int) handle2.invokeWithArguments(new Object[] { "a", 43 });
1299     assertEquals(43, (int) ret);
1300   }
1301 
collectBoolean(String a, boolean[] b)1302   public static int collectBoolean(String a, boolean[] b) {
1303     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1304     return 44;
1305   }
1306 
collectByte(String a, byte[] b)1307   public static int collectByte(String a, byte[] b) {
1308     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1309     return 45;
1310   }
1311 
collectChar(String a, char[] b)1312   public static int collectChar(String a, char[] b) {
1313     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1314     return 46;
1315   }
1316 
collectShort(String a, short[] b)1317   public static int collectShort(String a, short[] b) {
1318     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1319     return 47;
1320   }
1321 
collectInt(String a, int[] b)1322   public static int collectInt(String a, int[] b) {
1323     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1324     return 48;
1325   }
1326 
collectLong(String a, long[] b)1327   public static int collectLong(String a, long[] b) {
1328     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1329     return 49;
1330   }
1331 
collectFloat(String a, float[] b)1332   public static int collectFloat(String a, float[] b) {
1333     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1334     return 50;
1335   }
1336 
collectDouble(String a, double[] b)1337   public static int collectDouble(String a, double[] b) {
1338     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1339     return 51;
1340   }
1341 
collectCharSequence(String a, CharSequence[] b)1342   public static int collectCharSequence(String a, CharSequence[] b) {
1343     System.out.println("a: " + a + ", b:" + b[0] + ", c: " + b[1]);
1344     return 99;
1345   }
1346 
testAsCollector()1347   public static void testAsCollector() throws Throwable {
1348     // Reference arrays.
1349     // -------------------
1350     MethodHandle trailingRef = MethodHandles.lookup().findStatic(
1351         Main.class, "collectCharSequence",
1352         MethodType.methodType(int.class, String.class, CharSequence[].class));
1353 
1354     // int[] is not convertible to CharSequence[].class.
1355     try {
1356       trailingRef.asCollector(int[].class, 1);
1357       fail();
1358     } catch (IllegalArgumentException expected) {
1359     }
1360 
1361     // Object[] is not convertible to CharSequence[].class.
1362     try {
1363       trailingRef.asCollector(Object[].class, 1);
1364       fail();
1365     } catch (IllegalArgumentException expected) {
1366     }
1367 
1368     // String[].class is convertible to CharSequence.class
1369     MethodHandle collector = trailingRef.asCollector(String[].class, 2);
1370     assertEquals(99, (int) collector.invoke("a", "b", "c"));
1371 
1372     // Too few arguments should fail with a WMTE.
1373     try {
1374       collector.invoke("a", "b");
1375       fail();
1376     } catch (WrongMethodTypeException expected) {
1377     }
1378 
1379     // Too many arguments should fail with a WMTE.
1380     try {
1381       collector.invoke("a", "b", "c", "d");
1382       fail();
1383     } catch (WrongMethodTypeException expected) {
1384     }
1385 
1386     // Checks on other array types.
1387 
1388     MethodHandle target = MethodHandles.lookup().findStatic(
1389         Main.class, "collectBoolean",
1390         MethodType.methodType(int.class, String.class, boolean[].class));
1391     assertEquals(44, (int) target.asCollector(boolean[].class, 2).invoke("a", true, false));
1392 
1393     target = MethodHandles.lookup().findStatic(Main.class, "collectByte",
1394         MethodType.methodType(int.class, String.class, byte[].class));
1395     assertEquals(45, (int) target.asCollector(byte[].class, 2).invoke("a", (byte) 1, (byte) 2));
1396 
1397     target = MethodHandles.lookup().findStatic(Main.class, "collectChar",
1398         MethodType.methodType(int.class, String.class, char[].class));
1399     assertEquals(46, (int) target.asCollector(char[].class, 2).invoke("a", 'a', 'b'));
1400 
1401     target = MethodHandles.lookup().findStatic(Main.class, "collectShort",
1402         MethodType.methodType(int.class, String.class, short[].class));
1403     assertEquals(47, (int) target.asCollector(short[].class, 2).invoke("a", (short) 3, (short) 4));
1404 
1405     target = MethodHandles.lookup().findStatic(Main.class, "collectInt",
1406         MethodType.methodType(int.class, String.class, int[].class));
1407     assertEquals(48, (int) target.asCollector(int[].class, 2).invoke("a", 42, 43));
1408 
1409     target = MethodHandles.lookup().findStatic(Main.class, "collectLong",
1410         MethodType.methodType(int.class, String.class, long[].class));
1411     assertEquals(49, (int) target.asCollector(long[].class, 2).invoke("a", 100, 99));
1412 
1413     target = MethodHandles.lookup().findStatic(Main.class, "collectFloat",
1414         MethodType.methodType(int.class, String.class, float[].class));
1415     assertEquals(50, (int) target.asCollector(float[].class, 2).invoke("a", 8.9f, 9.1f));
1416 
1417     target = MethodHandles.lookup().findStatic(Main.class, "collectDouble",
1418         MethodType.methodType(int.class, String.class, double[].class));
1419     assertEquals(51, (int) target.asCollector(double[].class, 2).invoke("a", 6.7, 7.8));
1420   }
1421 
filter1(char a)1422   public static String filter1(char a) {
1423     return String.valueOf(a);
1424   }
1425 
filter2(String b)1426   public static char filter2(String b) {
1427     return b.charAt(0);
1428   }
1429 
badFilter1(char a, char b)1430   public static String badFilter1(char a, char b) {
1431     return "bad";
1432   }
1433 
filterTarget(String a, char b, String c, char d)1434   public static int filterTarget(String a, char b, String c, char d) {
1435     System.out.println("a: " + a + ", b: " + b + ", c:" + c + ", d:" + d);
1436     return 56;
1437   }
1438 
testFilterArguments()1439   public static void testFilterArguments() throws Throwable {
1440     MethodHandle filter1 = MethodHandles.lookup().findStatic(
1441         Main.class, "filter1", MethodType.methodType(String.class, char.class));
1442     MethodHandle filter2 = MethodHandles.lookup().findStatic(
1443         Main.class, "filter2", MethodType.methodType(char.class, String.class));
1444 
1445     MethodHandle target = MethodHandles.lookup().findStatic(
1446         Main.class, "filterTarget", MethodType.methodType(int.class,
1447           String.class, char.class, String.class, char.class));
1448 
1449     // In all the cases below, the values printed will be 'a', 'b', 'c', 'd'.
1450 
1451     // Filter arguments [0, 1] - all other arguments are passed through
1452     // as is.
1453     MethodHandle adapter = MethodHandles.filterArguments(
1454         target, 0, filter1, filter2);
1455     assertEquals(56, (int) adapter.invokeExact('a', "bXXXX", "c", 'd'));
1456 
1457     // Filter arguments [1, 2].
1458     adapter = MethodHandles.filterArguments(target, 1, filter2, filter1);
1459     assertEquals(56, (int) adapter.invokeExact("a", "bXXXX", 'c', 'd'));
1460 
1461     // Filter arguments [2, 3].
1462     adapter = MethodHandles.filterArguments(target, 2, filter1, filter2);
1463     assertEquals(56, (int) adapter.invokeExact("a", 'b', 'c', "dXXXXX"));
1464 
1465     // Try out a few error cases :
1466 
1467     // The return types of the filter doesn't align with the expected argument
1468     // type of the target.
1469     try {
1470       adapter = MethodHandles.filterArguments(target, 2, filter2, filter1);
1471       fail();
1472     } catch (IllegalArgumentException expected) {
1473     }
1474 
1475     // There are more filters than arguments.
1476     try {
1477       adapter = MethodHandles.filterArguments(target, 3, filter2, filter1);
1478       fail();
1479     } catch (IllegalArgumentException expected) {
1480     }
1481 
1482     // We pass in an obviously bogus position.
1483     try {
1484       adapter = MethodHandles.filterArguments(target, -1, filter2, filter1);
1485       fail();
1486     } catch (ArrayIndexOutOfBoundsException expected) {
1487     }
1488 
1489     // We pass in a function that has more than one argument.
1490     MethodHandle badFilter1 = MethodHandles.lookup().findStatic(
1491         Main.class, "badFilter1",
1492         MethodType.methodType(String.class, char.class, char.class));
1493 
1494     try {
1495       adapter = MethodHandles.filterArguments(target, 0, badFilter1, filter2);
1496       fail();
1497     } catch (IllegalArgumentException expected) {
1498     }
1499   }
1500 
voidFilter(char a, char b)1501   static void voidFilter(char a, char b) {
1502     System.out.println("voidFilter");
1503   }
1504 
filter(char a, char b)1505   static String filter(char a, char b) {
1506     return String.valueOf(a) + "+" + b;
1507   }
1508 
badFilter(char a, char b)1509   static char badFilter(char a, char b) {
1510     return 0;
1511   }
1512 
target(String a, String b, String c)1513   static int target(String a, String b, String c) {
1514     System.out.println("a: " + a + ", b: " + b + ", c: " + c);
1515     return 57;
1516   }
1517 
testCollectArguments()1518   public static void testCollectArguments() throws Throwable {
1519     // Test non-void filters.
1520     MethodHandle filter = MethodHandles.lookup().findStatic(
1521         Main.class, "filter",
1522         MethodType.methodType(String.class, char.class, char.class));
1523 
1524     MethodHandle target = MethodHandles.lookup().findStatic(
1525         Main.class, "target",
1526         MethodType.methodType(int.class, String.class, String.class, String.class));
1527 
1528     // Filter at position 0.
1529     MethodHandle adapter = MethodHandles.collectArguments(target, 0, filter);
1530     assertEquals(57, (int) adapter.invokeExact('a', 'b', "c", "d"));
1531 
1532     // Filter at position 1.
1533     adapter = MethodHandles.collectArguments(target, 1, filter);
1534     assertEquals(57, (int) adapter.invokeExact("a", 'b', 'c', "d"));
1535 
1536     // Filter at position 2.
1537     adapter = MethodHandles.collectArguments(target, 2, filter);
1538     assertEquals(57, (int) adapter.invokeExact("a", "b", 'c', 'd'));
1539 
1540     // Test void filters. Note that we're passing in one more argument
1541     // than usual because the filter returns nothing - we have to invoke with
1542     // the full set of filter args and the full set of target args.
1543     filter = MethodHandles.lookup().findStatic(Main.class, "voidFilter",
1544         MethodType.methodType(void.class, char.class, char.class));
1545     adapter = MethodHandles.collectArguments(target, 0, filter);
1546     assertEquals(57, (int) adapter.invokeExact('a', 'b', "a", "b", "c"));
1547 
1548     adapter = MethodHandles.collectArguments(target, 1, filter);
1549     assertEquals(57, (int) adapter.invokeExact("a", 'a', 'b', "b", "c"));
1550 
1551     // Test out a few failure cases.
1552     filter = MethodHandles.lookup().findStatic(
1553         Main.class, "filter",
1554         MethodType.methodType(String.class, char.class, char.class));
1555 
1556     // Bogus filter position.
1557     try {
1558       adapter = MethodHandles.collectArguments(target, 3, filter);
1559       fail();
1560     } catch (IndexOutOfBoundsException | IllegalArgumentException expected) {
1561     }
1562 
1563     // Mismatch in filter return type.
1564     filter = MethodHandles.lookup().findStatic(
1565         Main.class, "badFilter",
1566         MethodType.methodType(char.class, char.class, char.class));
1567     try {
1568       adapter = MethodHandles.collectArguments(target, 0, filter);
1569       fail();
1570     } catch (IllegalArgumentException expected) {
1571     }
1572   }
1573 
insertReceiver(String a, int b, Integer c, String d)1574   static int insertReceiver(String a, int b, Integer c, String d) {
1575     System.out.println("a: " + a + ", b:" + b + ", c:" + c + ", d:" + d);
1576     return 73;
1577   }
1578 
testInsertArguments()1579   public static void testInsertArguments() throws Throwable {
1580     MethodHandle target = MethodHandles.lookup().findStatic(
1581         Main.class, "insertReceiver",
1582         MethodType.methodType(int.class,
1583           String.class, int.class, Integer.class, String.class));
1584 
1585     // Basic single element array inserted at position 0.
1586     MethodHandle adapter = MethodHandles.insertArguments(
1587         target, 0, new Object[] { "foo" });
1588     assertEquals(73, (int) adapter.invokeExact(45, Integer.valueOf(56), "bar"));
1589 
1590     // Exercise unboxing.
1591     adapter = MethodHandles.insertArguments(
1592         target, 1, new Object[] { Integer.valueOf(56), 57 });
1593     assertEquals(73, (int) adapter.invokeExact("foo", "bar"));
1594 
1595     // Exercise a widening conversion.
1596     adapter = MethodHandles.insertArguments(
1597         target, 1, new Object[] { (short) 56, Integer.valueOf(57) });
1598     assertEquals(73, (int) adapter.invokeExact("foo", "bar"));
1599 
1600     // Insert an argument at the last position.
1601     adapter = MethodHandles.insertArguments(
1602         target, 3, new Object[] { "bar" });
1603     assertEquals(73, (int) adapter.invokeExact("foo", 45, Integer.valueOf(46)));
1604 
1605     // Exercise a few error cases.
1606 
1607     // A reference type that can't be cast to another reference type.
1608     try {
1609       MethodHandles.insertArguments(target, 3, new Object[] { new Object() });
1610       fail();
1611     } catch (ClassCastException expected) {
1612     }
1613 
1614     // A boxed type that can't be unboxed correctly.
1615     try {
1616       MethodHandles.insertArguments(target, 1, new Object[] { Long.valueOf(56) });
1617       fail();
1618     } catch (ClassCastException expected) {
1619     }
1620   }
1621 
foldFilter(char a, char b)1622   public static String foldFilter(char a, char b) {
1623     return String.valueOf(a) + "+" + b;
1624   }
1625 
voidFoldFilter(String e, char a, char b)1626   public static void voidFoldFilter(String e, char a, char b) {
1627     System.out.println(String.valueOf(a) + "+" + b);
1628   }
1629 
foldTarget(String a, char b, char c, String d)1630   public static int foldTarget(String a, char b, char c, String d) {
1631     System.out.println("a: " + a + " ,b:" + b + " ,c:" + c + " ,d:" + d);
1632     return 89;
1633   }
1634 
mismatchedVoidFilter(Integer a)1635   public static void mismatchedVoidFilter(Integer a) {
1636   }
1637 
mismatchedNonVoidFilter(char a, char b)1638   public static Integer mismatchedNonVoidFilter(char a, char b) {
1639     return null;
1640   }
1641 
testFoldArguments()1642   public static void testFoldArguments() throws Throwable {
1643     // Test non-void filters.
1644     MethodHandle filter = MethodHandles.lookup().findStatic(
1645         Main.class, "foldFilter",
1646         MethodType.methodType(String.class, char.class, char.class));
1647 
1648     MethodHandle target = MethodHandles.lookup().findStatic(
1649         Main.class, "foldTarget",
1650         MethodType.methodType(int.class, String.class,
1651           char.class, char.class, String.class));
1652 
1653     // Folder with a non-void type.
1654     MethodHandle adapter = MethodHandles.foldArguments(target, filter);
1655     assertEquals(89, (int) adapter.invokeExact('c', 'd', "e"));
1656 
1657     // Folder with a void type.
1658     filter = MethodHandles.lookup().findStatic(
1659         Main.class, "voidFoldFilter",
1660         MethodType.methodType(void.class, String.class, char.class, char.class));
1661     adapter = MethodHandles.foldArguments(target, filter);
1662     assertEquals(89, (int) adapter.invokeExact("a", 'c', 'd', "e"));
1663 
1664     // Test a few erroneous cases.
1665 
1666     filter = MethodHandles.lookup().findStatic(
1667         Main.class, "mismatchedVoidFilter",
1668         MethodType.methodType(void.class, Integer.class));
1669     try {
1670       adapter = MethodHandles.foldArguments(target, filter);
1671       fail();
1672     } catch (IllegalArgumentException expected) {
1673     }
1674 
1675     filter = MethodHandles.lookup().findStatic(
1676         Main.class, "mismatchedNonVoidFilter",
1677         MethodType.methodType(Integer.class, char.class, char.class));
1678     try {
1679       adapter = MethodHandles.foldArguments(target, filter);
1680       fail();
1681     } catch (IllegalArgumentException expected) {
1682     }
1683   }
1684 
fail()1685   public static void fail() {
1686     System.out.println("FAIL");
1687     Thread.dumpStack();
1688   }
1689 
fail(String message)1690   public static void fail(String message) {
1691     System.out.println("fail: " + message);
1692     Thread.dumpStack();
1693   }
1694 
assertEquals(int i1, int i2)1695   public static void assertEquals(int i1, int i2) {
1696     if (i1 != i2) throw new AssertionError("Expected: " + i1 + " was " + i2);
1697   }
1698 
assertEquals(String s1, String s2)1699   public static void assertEquals(String s1, String s2) {
1700     if (s1 == s2) {
1701       return;
1702     }
1703 
1704     if (s1 != null && s2 != null && s1.equals(s2)) {
1705       return;
1706     }
1707 
1708     throw new AssertionError("assertEquals s1: " + s1 + ", s2: " + s2);
1709   }
1710 }
1711