< prev index next >

src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java

Print this page

  1 /*
  2  * Copyright (c) 2012, 2021, 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.  Oracle designates this
  8  * particular file as subject to the "Classpath" exception as provided
  9  * by Oracle in the LICENSE file that accompanied this code.
 10  *
 11  * This code is distributed in the hope that it will be useful, but WITHOUT
 12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 14  * version 2 for more details (a copy is included in the LICENSE file that
 15  * accompanied this code).
 16  *
 17  * You should have received a copy of the GNU General Public License version
 18  * 2 along with this work; if not, write to the Free Software Foundation,
 19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 20  *
 21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 22  * or visit www.oracle.com if you need additional information or have any

234 public final class LambdaMetafactory {
235 
236     private LambdaMetafactory() {}
237 
238     /** Flag for {@link #altMetafactory} indicating the lambda object
239      * must be serializable */
240     public static final int FLAG_SERIALIZABLE = 1 << 0;
241 
242     /**
243      * Flag for {@link #altMetafactory} indicating the lambda object implements
244      * other interfaces besides {@code Serializable}
245      */
246     public static final int FLAG_MARKERS = 1 << 1;
247 
248     /**
249      * Flag for alternate metafactories indicating the lambda object requires
250      * additional methods that invoke the {@code implementation}
251      */
252     public static final int FLAG_BRIDGES = 1 << 2;
253 




254     private static final Class<?>[] EMPTY_CLASS_ARRAY = new Class<?>[0];
255     private static final MethodType[] EMPTY_MT_ARRAY = new MethodType[0];
256 
257     // LambdaMetafactory bootstrap methods are startup sensitive, and may be
258     // special cased in java.lang.invoke.BootstrapMethodInvoker to ensure
259     // methods are invoked with exact type information to avoid generating
260     // code for runtime checks. Take care any changes or additions here are
261     // reflected there as appropriate.
262 
263     /**
264      * Facilitates the creation of simple "function objects" that implement one
265      * or more interfaces by delegation to a provided {@link MethodHandle},
266      * after appropriate type adaptation and partial evaluation of arguments.
267      * Typically used as a <em>bootstrap method</em> for {@code invokedynamic}
268      * call sites, to support the <em>lambda expression</em> and <em>method
269      * reference expression</em> features of the Java Programming Language.
270      *
271      * <p>This is the standard, streamlined metafactory; additional flexibility
272      * is provided by {@link #altMetafactory(MethodHandles.Lookup, String, MethodType, Object...)}.
273      * A general description of the behavior of this method is provided

319      * @throws SecurityException If a security manager is present, and it
320      *         <a href="MethodHandles.Lookup.html#secmgr">refuses access</a>
321      *         from {@code caller} to the package of {@code implementation}.
322      */
323     public static CallSite metafactory(MethodHandles.Lookup caller,
324                                        String interfaceMethodName,
325                                        MethodType factoryType,
326                                        MethodType interfaceMethodType,
327                                        MethodHandle implementation,
328                                        MethodType dynamicMethodType)
329             throws LambdaConversionException {
330         AbstractValidatingLambdaMetafactory mf;
331         mf = new InnerClassLambdaMetafactory(Objects.requireNonNull(caller),
332                                              Objects.requireNonNull(factoryType),
333                                              Objects.requireNonNull(interfaceMethodName),
334                                              Objects.requireNonNull(interfaceMethodType),
335                                              Objects.requireNonNull(implementation),
336                                              Objects.requireNonNull(dynamicMethodType),
337                                              false,
338                                              EMPTY_CLASS_ARRAY,
339                                              EMPTY_MT_ARRAY);

340         mf.validateMetafactoryArgs();
341         return mf.buildCallSite();
342     }
343 
344     /**
345      * Facilitates the creation of simple "function objects" that implement one
346      * or more interfaces by delegation to a provided {@link MethodHandle},
347      * after appropriate type adaptation and partial evaluation of arguments.
348      * Typically used as a <em>bootstrap method</em> for {@code invokedynamic}
349      * call sites, to support the <em>lambda expression</em> and <em>method
350      * reference expression</em> features of the Java Programming Language.
351      *
352      * <p>This is the general, more flexible metafactory; a streamlined version
353      * is provided by {@link #metafactory(java.lang.invoke.MethodHandles.Lookup,
354      * String, MethodType, MethodType, MethodHandle, MethodType)}.
355      * A general description of the behavior of this method is provided
356      * {@link LambdaMetafactory above}.
357      *
358      * <p>The argument list for this method includes three fixed parameters,
359      * corresponding to the parameters automatically stacked by the VM for the

365      *  CallSite altMetafactory(MethodHandles.Lookup caller,
366      *                          String interfaceMethodName,
367      *                          MethodType factoryType,
368      *                          Object... args)
369      * }</pre>
370      *
371      * <p>but it behaves as if the argument list is as follows:
372      *
373      * <pre>{@code
374      *  CallSite altMetafactory(MethodHandles.Lookup caller,
375      *                          String interfaceMethodName,
376      *                          MethodType factoryType,
377      *                          MethodType interfaceMethodType,
378      *                          MethodHandle implementation,
379      *                          MethodType dynamicMethodType,
380      *                          int flags,
381      *                          int altInterfaceCount,        // IF flags has MARKERS set
382      *                          Class... altInterfaces,       // IF flags has MARKERS set
383      *                          int altMethodCount,           // IF flags has BRIDGES set
384      *                          MethodType... altMethods      // IF flags has BRIDGES set

385      *                          )
386      * }</pre>
387      *
388      * <p>Arguments that appear in the argument list for
389      * {@link #metafactory(MethodHandles.Lookup, String, MethodType, MethodType, MethodHandle, MethodType)}
390      * have the same specification as in that method.  The additional arguments
391      * are interpreted as follows:
392      * <ul>
393      *     <li>{@code flags} indicates additional options; this is a bitwise
394      *     OR of desired flags.  Defined flags are {@link #FLAG_BRIDGES},
395      *     {@link #FLAG_MARKERS}, and {@link #FLAG_SERIALIZABLE}.</li>
396      *     <li>{@code altInterfaceCount} is the number of additional interfaces
397      *     the function object should implement, and is present if and only if the
398      *     {@code FLAG_MARKERS} flag is set.</li>
399      *     <li>{@code altInterfaces} is a variable-length list of additional
400      *     interfaces to implement, whose length equals {@code altInterfaceCount},
401      *     and is present if and only if the {@code FLAG_MARKERS} flag is set.</li>
402      *     <li>{@code altMethodCount} is the number of additional method signatures
403      *     the function object should implement, and is present if and only if
404      *     the {@code FLAG_BRIDGES} flag is set.</li>
405      *     <li>{@code altMethods} is a variable-length list of additional
406      *     methods signatures to implement, whose length equals {@code altMethodCount},
407      *     and is present if and only if the {@code FLAG_BRIDGES} flag is set.</li>




408      * </ul>
409      *
410      * <p>Each class named by {@code altInterfaces} is subject to the same
411      * restrictions as {@code Rd}, the return type of {@code factoryType},
412      * as described {@link LambdaMetafactory above}.  Each {@code MethodType}
413      * named by {@code altMethods} is subject to the same restrictions as
414      * {@code interfaceMethodType}, as described {@link LambdaMetafactory above}.
415      *
416      * <p>When FLAG_SERIALIZABLE is set in {@code flags}, the function objects
417      * will implement {@code Serializable}, and will have a {@code writeReplace}
418      * method that returns an appropriate {@link SerializedLambda}.  The
419      * {@code caller} class must have an appropriate {@code $deserializeLambda$}
420      * method, as described in {@link SerializedLambda}.
421      *



422      * <p>When the target of the {@code CallSite} returned from this method is
423      * invoked, the resulting function objects are instances of a class with
424      * the following properties:
425      * <ul>
426      *     <li>The class implements the interface named by the return type
427      *     of {@code factoryType} and any interfaces named by {@code altInterfaces}</li>
428      *     <li>The class declares methods with the name given by {@code interfaceMethodName},
429      *     and the signature given by {@code interfaceMethodType} and additional signatures
430      *     given by {@code altMethods}</li>
431      *     <li>The class may override methods from {@code Object}, and may
432      *     implement methods related to serialization.</li>
433      * </ul>
434      *
435      * @param caller Represents a lookup context with the accessibility
436      *               privileges of the caller.  Specifically, the lookup context
437      *               must have {@linkplain MethodHandles.Lookup#hasFullPrivilegeAccess()
438      *               full privilege access}.
439      *               When used with {@code invokedynamic}, this is stacked
440      *               automatically by the VM.
441      * @param interfaceMethodName The name of the method to implement.  When used with

470      * @throws SecurityException If a security manager is present, and it
471      *         <a href="MethodHandles.Lookup.html#secmgr">refuses access</a>
472      *         from {@code caller} to the package of {@code implementation}.
473      */
474     public static CallSite altMetafactory(MethodHandles.Lookup caller,
475                                           String interfaceMethodName,
476                                           MethodType factoryType,
477                                           Object... args)
478             throws LambdaConversionException {
479         Objects.requireNonNull(caller);
480         Objects.requireNonNull(interfaceMethodName);
481         Objects.requireNonNull(factoryType);
482         Objects.requireNonNull(args);
483         int argIndex = 0;
484         MethodType interfaceMethodType = extractArg(args, argIndex++, MethodType.class);
485         MethodHandle implementation = extractArg(args, argIndex++, MethodHandle.class);
486         MethodType dynamicMethodType = extractArg(args, argIndex++, MethodType.class);
487         int flags = extractArg(args, argIndex++, Integer.class);
488         Class<?>[] altInterfaces = EMPTY_CLASS_ARRAY;
489         MethodType[] altMethods = EMPTY_MT_ARRAY;


490         if ((flags & FLAG_MARKERS) != 0) {
491             int altInterfaceCount = extractArg(args, argIndex++, Integer.class);
492             if (altInterfaceCount < 0) {
493                 throw new IllegalArgumentException("negative argument count");
494             }
495             if (altInterfaceCount > 0) {
496                 altInterfaces = extractArgs(args, argIndex, Class.class, altInterfaceCount);
497                 argIndex += altInterfaceCount;
498             }
499         }
500         if ((flags & FLAG_BRIDGES) != 0) {
501             int altMethodCount = extractArg(args, argIndex++, Integer.class);
502             if (altMethodCount < 0) {
503                 throw new IllegalArgumentException("negative argument count");
504             }
505             if (altMethodCount > 0) {
506                 altMethods = extractArgs(args, argIndex, MethodType.class, altMethodCount);
507                 argIndex += altMethodCount;
508             }
509         }





510         if (argIndex < args.length) {
511             throw new IllegalArgumentException("too many arguments");
512         }
513 
514         boolean isSerializable = ((flags & FLAG_SERIALIZABLE) != 0);
515         if (isSerializable) {
516             boolean foundSerializableSupertype = Serializable.class.isAssignableFrom(factoryType.returnType());
517             for (Class<?> c : altInterfaces)
518                 foundSerializableSupertype |= Serializable.class.isAssignableFrom(c);
519             if (!foundSerializableSupertype) {
520                 altInterfaces = Arrays.copyOf(altInterfaces, altInterfaces.length + 1);
521                 altInterfaces[altInterfaces.length-1] = Serializable.class;
522             }
523         }
524 
525         AbstractValidatingLambdaMetafactory mf
526                 = new InnerClassLambdaMetafactory(caller,
527                                                   factoryType,
528                                                   interfaceMethodName,
529                                                   interfaceMethodType,
530                                                   implementation,
531                                                   dynamicMethodType,
532                                                   isSerializable,
533                                                   altInterfaces,
534                                                   altMethods);

535         mf.validateMetafactoryArgs();
536         return mf.buildCallSite();
537     }
538 
539     private static <T> T extractArg(Object[] args, int index, Class<T> type) {
540         if (index >= args.length) {
541             throw new IllegalArgumentException("missing argument");
542         }
543         Object result = Objects.requireNonNull(args[index]);
544         if (!type.isInstance(result)) {
545             throw new IllegalArgumentException("argument has wrong type");
546         }
547         return type.cast(result);
548     }
549 
550     private static <T> T[] extractArgs(Object[] args, int index, Class<T> type, int count) {
551         @SuppressWarnings("unchecked")
552         T[] result = (T[]) Array.newInstance(type, count);
553         for (int i = 0; i < count; i++) {
554             result[i] = extractArg(args, index + i, type);

  1 /*
  2  * Copyright (c) 2012, 2024, 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.  Oracle designates this
  8  * particular file as subject to the "Classpath" exception as provided
  9  * by Oracle in the LICENSE file that accompanied this code.
 10  *
 11  * This code is distributed in the hope that it will be useful, but WITHOUT
 12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 14  * version 2 for more details (a copy is included in the LICENSE file that
 15  * accompanied this code).
 16  *
 17  * You should have received a copy of the GNU General Public License version
 18  * 2 along with this work; if not, write to the Free Software Foundation,
 19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 20  *
 21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 22  * or visit www.oracle.com if you need additional information or have any

234 public final class LambdaMetafactory {
235 
236     private LambdaMetafactory() {}
237 
238     /** Flag for {@link #altMetafactory} indicating the lambda object
239      * must be serializable */
240     public static final int FLAG_SERIALIZABLE = 1 << 0;
241 
242     /**
243      * Flag for {@link #altMetafactory} indicating the lambda object implements
244      * other interfaces besides {@code Serializable}
245      */
246     public static final int FLAG_MARKERS = 1 << 1;
247 
248     /**
249      * Flag for alternate metafactories indicating the lambda object requires
250      * additional methods that invoke the {@code implementation}
251      */
252     public static final int FLAG_BRIDGES = 1 << 2;
253 
254     /** Flag for {@link #altMetafactory} indicating the lambda object
255      * must be a {@code Quotable} object, inspectable using code reflection. */
256     public static final int FLAG_QUOTABLE = 1 << 3;
257 
258     private static final Class<?>[] EMPTY_CLASS_ARRAY = new Class<?>[0];
259     private static final MethodType[] EMPTY_MT_ARRAY = new MethodType[0];
260 
261     // LambdaMetafactory bootstrap methods are startup sensitive, and may be
262     // special cased in java.lang.invoke.BootstrapMethodInvoker to ensure
263     // methods are invoked with exact type information to avoid generating
264     // code for runtime checks. Take care any changes or additions here are
265     // reflected there as appropriate.
266 
267     /**
268      * Facilitates the creation of simple "function objects" that implement one
269      * or more interfaces by delegation to a provided {@link MethodHandle},
270      * after appropriate type adaptation and partial evaluation of arguments.
271      * Typically used as a <em>bootstrap method</em> for {@code invokedynamic}
272      * call sites, to support the <em>lambda expression</em> and <em>method
273      * reference expression</em> features of the Java Programming Language.
274      *
275      * <p>This is the standard, streamlined metafactory; additional flexibility
276      * is provided by {@link #altMetafactory(MethodHandles.Lookup, String, MethodType, Object...)}.
277      * A general description of the behavior of this method is provided

323      * @throws SecurityException If a security manager is present, and it
324      *         <a href="MethodHandles.Lookup.html#secmgr">refuses access</a>
325      *         from {@code caller} to the package of {@code implementation}.
326      */
327     public static CallSite metafactory(MethodHandles.Lookup caller,
328                                        String interfaceMethodName,
329                                        MethodType factoryType,
330                                        MethodType interfaceMethodType,
331                                        MethodHandle implementation,
332                                        MethodType dynamicMethodType)
333             throws LambdaConversionException {
334         AbstractValidatingLambdaMetafactory mf;
335         mf = new InnerClassLambdaMetafactory(Objects.requireNonNull(caller),
336                                              Objects.requireNonNull(factoryType),
337                                              Objects.requireNonNull(interfaceMethodName),
338                                              Objects.requireNonNull(interfaceMethodType),
339                                              Objects.requireNonNull(implementation),
340                                              Objects.requireNonNull(dynamicMethodType),
341                                              false,
342                                              EMPTY_CLASS_ARRAY,
343                                              EMPTY_MT_ARRAY,
344                                  null);
345         mf.validateMetafactoryArgs();
346         return mf.buildCallSite();
347     }
348 
349     /**
350      * Facilitates the creation of simple "function objects" that implement one
351      * or more interfaces by delegation to a provided {@link MethodHandle},
352      * after appropriate type adaptation and partial evaluation of arguments.
353      * Typically used as a <em>bootstrap method</em> for {@code invokedynamic}
354      * call sites, to support the <em>lambda expression</em> and <em>method
355      * reference expression</em> features of the Java Programming Language.
356      *
357      * <p>This is the general, more flexible metafactory; a streamlined version
358      * is provided by {@link #metafactory(java.lang.invoke.MethodHandles.Lookup,
359      * String, MethodType, MethodType, MethodHandle, MethodType)}.
360      * A general description of the behavior of this method is provided
361      * {@link LambdaMetafactory above}.
362      *
363      * <p>The argument list for this method includes three fixed parameters,
364      * corresponding to the parameters automatically stacked by the VM for the

370      *  CallSite altMetafactory(MethodHandles.Lookup caller,
371      *                          String interfaceMethodName,
372      *                          MethodType factoryType,
373      *                          Object... args)
374      * }</pre>
375      *
376      * <p>but it behaves as if the argument list is as follows:
377      *
378      * <pre>{@code
379      *  CallSite altMetafactory(MethodHandles.Lookup caller,
380      *                          String interfaceMethodName,
381      *                          MethodType factoryType,
382      *                          MethodType interfaceMethodType,
383      *                          MethodHandle implementation,
384      *                          MethodType dynamicMethodType,
385      *                          int flags,
386      *                          int altInterfaceCount,        // IF flags has MARKERS set
387      *                          Class... altInterfaces,       // IF flags has MARKERS set
388      *                          int altMethodCount,           // IF flags has BRIDGES set
389      *                          MethodType... altMethods      // IF flags has BRIDGES set
390      *                          MethodHandle quotableField    // IF flags has QUOTABLE set
391      *                          )
392      * }</pre>
393      *
394      * <p>Arguments that appear in the argument list for
395      * {@link #metafactory(MethodHandles.Lookup, String, MethodType, MethodType, MethodHandle, MethodType)}
396      * have the same specification as in that method.  The additional arguments
397      * are interpreted as follows:
398      * <ul>
399      *     <li>{@code flags} indicates additional options; this is a bitwise
400      *     OR of desired flags.  Defined flags are {@link #FLAG_BRIDGES},
401      *     {@link #FLAG_MARKERS}, and {@link #FLAG_SERIALIZABLE}.</li>
402      *     <li>{@code altInterfaceCount} is the number of additional interfaces
403      *     the function object should implement, and is present if and only if the
404      *     {@code FLAG_MARKERS} flag is set.</li>
405      *     <li>{@code altInterfaces} is a variable-length list of additional
406      *     interfaces to implement, whose length equals {@code altInterfaceCount},
407      *     and is present if and only if the {@code FLAG_MARKERS} flag is set.</li>
408      *     <li>{@code altMethodCount} is the number of additional method signatures
409      *     the function object should implement, and is present if and only if
410      *     the {@code FLAG_BRIDGES} flag is set.</li>
411      *     <li>{@code altMethods} is a variable-length list of additional
412      *     methods signatures to implement, whose length equals {@code altMethodCount},
413      *     and is present if and only if the {@code FLAG_BRIDGES} flag is set.</li>
414      *     <li>{@code quotableField} is a
415      *     {@linkplain MethodHandles.Lookup#findGetter(Class, String, Class) getter} method handle
416      *     that is used to retrieve the string representation of the quotable lambda's associated
417      *     intermediate representation.</li>
418      * </ul>
419      *
420      * <p>Each class named by {@code altInterfaces} is subject to the same
421      * restrictions as {@code Rd}, the return type of {@code factoryType},
422      * as described {@link LambdaMetafactory above}.  Each {@code MethodType}
423      * named by {@code altMethods} is subject to the same restrictions as
424      * {@code interfaceMethodType}, as described {@link LambdaMetafactory above}.
425      *
426      * <p>When FLAG_SERIALIZABLE is set in {@code flags}, the function objects
427      * will implement {@code Serializable}, and will have a {@code writeReplace}
428      * method that returns an appropriate {@link SerializedLambda}.  The
429      * {@code caller} class must have an appropriate {@code $deserializeLambda$}
430      * method, as described in {@link SerializedLambda}.
431      *
432      * <p>When FLAG_QUOTABLE is set in {@code flags}, the function objects
433      * will implement {@code Quotable}.
434      *
435      * <p>When the target of the {@code CallSite} returned from this method is
436      * invoked, the resulting function objects are instances of a class with
437      * the following properties:
438      * <ul>
439      *     <li>The class implements the interface named by the return type
440      *     of {@code factoryType} and any interfaces named by {@code altInterfaces}</li>
441      *     <li>The class declares methods with the name given by {@code interfaceMethodName},
442      *     and the signature given by {@code interfaceMethodType} and additional signatures
443      *     given by {@code altMethods}</li>
444      *     <li>The class may override methods from {@code Object}, and may
445      *     implement methods related to serialization.</li>
446      * </ul>
447      *
448      * @param caller Represents a lookup context with the accessibility
449      *               privileges of the caller.  Specifically, the lookup context
450      *               must have {@linkplain MethodHandles.Lookup#hasFullPrivilegeAccess()
451      *               full privilege access}.
452      *               When used with {@code invokedynamic}, this is stacked
453      *               automatically by the VM.
454      * @param interfaceMethodName The name of the method to implement.  When used with

483      * @throws SecurityException If a security manager is present, and it
484      *         <a href="MethodHandles.Lookup.html#secmgr">refuses access</a>
485      *         from {@code caller} to the package of {@code implementation}.
486      */
487     public static CallSite altMetafactory(MethodHandles.Lookup caller,
488                                           String interfaceMethodName,
489                                           MethodType factoryType,
490                                           Object... args)
491             throws LambdaConversionException {
492         Objects.requireNonNull(caller);
493         Objects.requireNonNull(interfaceMethodName);
494         Objects.requireNonNull(factoryType);
495         Objects.requireNonNull(args);
496         int argIndex = 0;
497         MethodType interfaceMethodType = extractArg(args, argIndex++, MethodType.class);
498         MethodHandle implementation = extractArg(args, argIndex++, MethodHandle.class);
499         MethodType dynamicMethodType = extractArg(args, argIndex++, MethodType.class);
500         int flags = extractArg(args, argIndex++, Integer.class);
501         Class<?>[] altInterfaces = EMPTY_CLASS_ARRAY;
502         MethodType[] altMethods = EMPTY_MT_ARRAY;
503         // Getter that returns the op of a Quotable instance
504         MethodHandle quotableOpGetter = null;
505         if ((flags & FLAG_MARKERS) != 0) {
506             int altInterfaceCount = extractArg(args, argIndex++, Integer.class);
507             if (altInterfaceCount < 0) {
508                 throw new IllegalArgumentException("negative argument count");
509             }
510             if (altInterfaceCount > 0) {
511                 altInterfaces = extractArgs(args, argIndex, Class.class, altInterfaceCount);
512                 argIndex += altInterfaceCount;
513             }
514         }
515         if ((flags & FLAG_BRIDGES) != 0) {
516             int altMethodCount = extractArg(args, argIndex++, Integer.class);
517             if (altMethodCount < 0) {
518                 throw new IllegalArgumentException("negative argument count");
519             }
520             if (altMethodCount > 0) {
521                 altMethods = extractArgs(args, argIndex, MethodType.class, altMethodCount);
522                 argIndex += altMethodCount;
523             }
524         }
525         if ((flags & FLAG_QUOTABLE) != 0) {
526             quotableOpGetter = extractArg(args, argIndex++, MethodHandle.class);
527             altInterfaces = Arrays.copyOf(altInterfaces, altInterfaces.length + 1);
528             altInterfaces[altInterfaces.length-1] = InnerClassLambdaMetafactory.CodeReflectionSupport.QUOTABLE_CLASS;
529         }
530         if (argIndex < args.length) {
531             throw new IllegalArgumentException("too many arguments");
532         }
533 
534         boolean isSerializable = ((flags & FLAG_SERIALIZABLE) != 0);
535         if (isSerializable) {
536             boolean foundSerializableSupertype = Serializable.class.isAssignableFrom(factoryType.returnType());
537             for (Class<?> c : altInterfaces)
538                 foundSerializableSupertype |= Serializable.class.isAssignableFrom(c);
539             if (!foundSerializableSupertype) {
540                 altInterfaces = Arrays.copyOf(altInterfaces, altInterfaces.length + 1);
541                 altInterfaces[altInterfaces.length-1] = Serializable.class;
542             }
543         }
544 
545         AbstractValidatingLambdaMetafactory mf
546                 = new InnerClassLambdaMetafactory(caller,
547                                                   factoryType,
548                                                   interfaceMethodName,
549                                                   interfaceMethodType,
550                                                   implementation,
551                                                   dynamicMethodType,
552                                                   isSerializable,
553                                                   altInterfaces,
554                                                   altMethods,
555                                                   quotableOpGetter);
556         mf.validateMetafactoryArgs();
557         return mf.buildCallSite();
558     }
559 
560     private static <T> T extractArg(Object[] args, int index, Class<T> type) {
561         if (index >= args.length) {
562             throw new IllegalArgumentException("missing argument");
563         }
564         Object result = Objects.requireNonNull(args[index]);
565         if (!type.isInstance(result)) {
566             throw new IllegalArgumentException("argument has wrong type");
567         }
568         return type.cast(result);
569     }
570 
571     private static <T> T[] extractArgs(Object[] args, int index, Class<T> type, int count) {
572         @SuppressWarnings("unchecked")
573         T[] result = (T[]) Array.newInstance(type, count);
574         for (int i = 0; i < count; i++) {
575             result[i] = extractArg(args, index + i, type);
< prev index next >