src/jdk/internal/dynalink/beans/SingleDynamicMethod.java

Wed, 20 Aug 2014 10:26:01 +0200

author
attila
date
Wed, 20 Aug 2014 10:26:01 +0200
changeset 963
e2497b11a021
parent 962
ac62e33a99b0
child 1205
4112748288bb
child 1239
e1146c9cc758
permissions
-rw-r--r--

8027043: Turn global accesses into MethodHandle.constant, with one chance of reassignment, e.g. x = value occuring once in the global scope is ok, twice is not.
8027958: NASHORN TEST: Create tests to test markdown javascript engine work with Nashorn
8028345: Remove nashorn repo "bin" scripts to avoid confusion with JDK bin launcher programs
8029090: Developers should be able to pass nashorn properties and enable/disable JFR from command line
8030169: Need regression test for bug JDK-8010731
8033105: Make sure Nashorn test harness can run zlib benchmark
8033334: Make sure that scope depth information is maintained in the RecompilableScriptFunctionDatas, to avoid unnecessary slow proto linkage when doing on demand compilation
8034206: Make parts of code pipeline reusable in order to facilitate faster warmup and faster lazy compilation.
8035820: Optimistic recompilation
8035836: Array performance improvements
8036127: Prototype filter needs to be applied to getter guard as well, not just getter
8036986: Test should check that correctly type is returned running with optimistic. If optimistic assumption was wrong we should get the right one.
8037086: Check that deoptimizing recompilations are correct
8037177: -Dnashorn.optimistic should be enabled by default, meaning that it has to be explicitly set to false to run with the jdk 8 style conservative types
8037534: Use scope types to determine optimistic types
8037572: Add more test cases to check static types
8037967: Broke the build, by commiting without saving the last review comment
8038223: Symbol trace debug output takes time
8038396: fix for the compiler expression evaluator to be more inquisitive about types
8038398: OptimisticRecompilationTest fails on staging repo nashorn/jdk9/nashorn due to test framework
8038406: Testability: as a first step of moving loggers away from the process global space, the Debug object now supports logging POJOs from log entries as an event queue, which can be introspected from test scripts. This is way better than screen scraping brittle and subject-to-change log output.
8038413: NPE in unboxInteger
8038416: Access to undefined scoped variables deoptimized too much
8038426: Move all loggers from process wide scope into Global scope
8038799: Guard and unbox boxed primitives types on setting them in Properties to avoid megamorphisism
8038945: Simplify strict undefined checks
8039044: Expand undefined intrinsics for all commutative combinators of scrict undefined checks
8039746: Transform applies to calls wherever possible, for ScriptFunctions and JSObjects.
8040024: BranchOptimizer produces bad code for NaN FP comparison
8040089: Apply to call transform was incomplete. Now passes all tests and performance is back
8040093: Make sure that optimistic splitting works in optimistic types
8040102: Remove all references to Unsafe and definition of anonymous clases from the code
8040655: When processing a RewriteException debug object, the return value has already been reset to null. We need to catch this value before that.
8041434: Add synchronization to the common global constants structure
8041625: AccessorProperty currentType must only by Object.class when non-primitive, and scoping followup problem for lazily generated with bodies
8041905: Fix apply2call bug that prevented avatar.js unit tests from running correctly
8041995: Problems when loading tree expressions with several optimistic program points when optimistically initializing ObjectNodes
8042118: Separate types from symbols
8043002: Improve performance of Nashorn equality operators
8043003: Use strongly referenced generic invokers
8043004: Reduce variability at JavaAdapter call sites
8043132: Nashorn : all tests failed with java.security.AccessControlException
8043133: Fix corner cases of JDK-8041995
8043137: Collapse long sequences of NOP in Nashorn bytecode output
8043232: Index selection of overloaded java new constructors
8043235: Type-based optimizations interfere with continuation methods
8043431: Fix yet another corner case of JDK-8041995
8043504: Octane test harness was missing argument to print_always at one callsite, causing erroneous logging
8043605: Enable history for empty property maps
8043608: Make equality tests inline better
8043611: Move timing dependent benchmark for apply2call specialization to currently_failing. It is dependent that nothing takes machine time when doing the two runs, causing spurious assertions. Suggest running octane.raytrace manually instead to verify that this works, or incorporating it in the nightly test suite
8043632: Parallelize class installation and various script fixes.
8043633: In order to remove global state outside of contexts, make sure Timing class is an instance and not a static global collection of data. Move into Context. Move -Dnashorn.timing to an official logging option.
8043956: Make code caching work with optimistic typing and lazy compilation
8044012: Integrate the latest best known performance flags int ant octane jobs, and make sure that it's easy to compare 'ant octane-nashorn' and 'ant octane-v8' at the push of a button. (or rather; the entry of a command line)
8044102: Ensure bechmark exclude list for Octane benchmarks is in only one place, project.properties, and fix benchmark harness
8044154: Nashorn : all tests failed with java.security.AccessControlException
8044171: Make optimistic exception handlers smaller
8044502: Get rid of global optimistic flag
8044518: Ensure exceptions related to optimistic recompilation are not serializable
8044533: Deoptimizing negation produces wrong result for zero
8044534: Constant folding for unary + should produce int for boolean literals
8044760: Avoid PropertyMap duplicate for global instances
8044786: Some tests fail with non-optimistic compilation
8044803: Unnecessary restOf check
8044816: On-demand compiled top-level program doesn't need :createProgramFunction
8044851: nashorn properties leak memory
8046013: TypeError: Cannot apply "with" to non script object
8046014: MultiGlobalCompiledScript should cache :createProgramFunction handle
8046025: AccessorProperty.getGetter is not threadsafe
8046026: CompiledFunction.relinkComposableInvoker assert is being hit
8046201: Avoid repeated flattening of nested ConsStrings
8046215: Running uncompilable scripts throws NullPointerException
8046898: Make sure that lazy compilation is the default, remove redundant "enable lazy compilation" flags, added warning message if compile logging is enabled and lazy is switched off. Verified existing test suite code coverage equivalence between lazy and eager.
8046905: apply on apply is broken
8046921: Deoptimization type information peristence
8047035: (function() "hello")() crashes in Lexer with jdk9
8047057: Add a regression test for the passing test cases from JDK-8042304
8047067: all eval arguments need to be copied in Lower
8047078: Fuzzing bug discovered when ArrayLiteralNodes weren't immutable
8047166: 'do with({}) break ; while(0);' crashes in CodeGenerator
8047331: Assertion in CompiledFunction when running earley-boyer after Merge
8047357: More precise synthetic return + unreachable throw
8047359: large string size RangeError should be thrown rather than reporting negative length
8047369: Add regression tests for passing test cases of JDK-8024971
8047371: local variable declaration in TypeEvaluator should use ScriptObject.addOwnProperty instead of .set
8047728: (function(x){var o={x:0}; with(o){delete x} return o.x})() evaluates to 0 instead of undefined
8047959: bindings created for declarations in eval code are not mutable
8048009: Type info caching accidentally defeated
8048071: eval within 'with' statement does not use correct scope if with scope expression has a copy of eval
8048079: Persistent code store is broken after optimistic types merge
8048505: ScriptingFunctions.readFully couldn't handle file names represented as ConsStrings
8048586: String concatenation with optimistic types is slow
8048718: JSON.parse('{"0":0, "64":0}') throws ArrayindexOutOfBoundsException
8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
8049086: Minor API convenience functions on "Java" object
8049222: JSType class exposes public mutable arrays
8049223: RewriteException class exposes public mutable arrays
8049242: Explicit constructor overload selection should work with StaticClass as well
8049318: Test hideLocationProperties.js fails on Window due to backslash in path
8049524: Global object initialization via javax.script API should be minimal
8050432: javax.script.filename variable should not be enumerable with nashorn engine's ENGINE_SCOPE bindings
8050964: OptimisticTypesPersistence.java should use java.util.Date instead of java.sql.Date
8051019: Separate src and test execution sandbox directories
8051346: Test262 tests for ECMAScript 5 now in branch "es5-tests"
8051439: Wrong type calculated for ADD operator with undefined operand
8051839: GuardedInvocation needs to clone an argument
8053908: jdeps is not PATH on Mac, results in ant clean test failure on Mac
8053910: ScriptObjectMirror causing havoc with Invocation interface
8053913: Auto format caused warning in CompositeTypeBasedGuardingDynamicLinker
8054223: Nashorn: AssertionError when use __DIR__ and ScriptEngine.eval()
8054411: Add nashorn.args.prepend system property
8054503: test/script/external/test262/test/suite/ch12/12.6/12.6.4/12.6.4-2.js fails with tip
8054651: Global.initConstructor and ScriptFunction.getPrototype(Object) can have stricter types
8054898: Avoid creation of empty type info files
8054993: type info cache may be disabled for test262 and tests explicitly changing that property should use @fork
8055034: jjs exits interactive mode if exception was thrown when trying to print value of last evaluated expression
8055042: Compile-time expression evaluator was missing variables
8055107: Extension directives to turn on callsite profiling, tracing, AST print and other debug features locally
8055139: test/script/trusted/JDK-8055107.js fails with access control exception
8055186: Backport Nashorn optimistic typing to 8u repository
8055529: Clean up the bin directory
Reviewed-by: jlaskey, lagergren, sundar
Contributed-by: marcus.largergren@oracle.com, hannes.wallnoefer@oracle.com, sundararajan.athijegannathan@oracle.com

attila@404 1 /*
attila@404 2 * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved.
attila@404 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
attila@404 4 *
attila@404 5 * This code is free software; you can redistribute it and/or modify it
attila@404 6 * under the terms of the GNU General Public License version 2 only, as
attila@404 7 * published by the Free Software Foundation. Oracle designates this
attila@404 8 * particular file as subject to the "Classpath" exception as provided
attila@404 9 * by Oracle in the LICENSE file that accompanied this code.
attila@404 10 *
attila@404 11 * This code is distributed in the hope that it will be useful, but WITHOUT
attila@404 12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
attila@404 13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
attila@404 14 * version 2 for more details (a copy is included in the LICENSE file that
attila@404 15 * accompanied this code).
attila@404 16 *
attila@404 17 * You should have received a copy of the GNU General Public License version
attila@404 18 * 2 along with this work; if not, write to the Free Software Foundation,
attila@404 19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
attila@404 20 *
attila@404 21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
attila@404 22 * or visit www.oracle.com if you need additional information or have any
attila@404 23 * questions.
attila@404 24 */
attila@404 25
attila@404 26 /*
attila@404 27 * This file is available under and governed by the GNU General Public
attila@404 28 * License version 2 only, as published by the Free Software Foundation.
attila@404 29 * However, the following notice accompanied the original version of this
attila@404 30 * file, and Oracle licenses the original version of this file under the BSD
attila@404 31 * license:
attila@404 32 */
attila@404 33 /*
attila@404 34 Copyright 2009-2013 Attila Szegedi
attila@404 35
attila@404 36 Licensed under both the Apache License, Version 2.0 (the "Apache License")
attila@404 37 and the BSD License (the "BSD License"), with licensee being free to
attila@404 38 choose either of the two at their discretion.
attila@404 39
attila@404 40 You may not use this file except in compliance with either the Apache
attila@404 41 License or the BSD License.
attila@404 42
attila@404 43 If you choose to use this file in compliance with the Apache License, the
attila@404 44 following notice applies to you:
attila@404 45
attila@404 46 You may obtain a copy of the Apache License at
attila@404 47
attila@404 48 http://www.apache.org/licenses/LICENSE-2.0
attila@404 49
attila@404 50 Unless required by applicable law or agreed to in writing, software
attila@404 51 distributed under the License is distributed on an "AS IS" BASIS,
attila@404 52 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
attila@404 53 implied. See the License for the specific language governing
attila@404 54 permissions and limitations under the License.
attila@404 55
attila@404 56 If you choose to use this file in compliance with the BSD License, the
attila@404 57 following notice applies to you:
attila@404 58
attila@404 59 Redistribution and use in source and binary forms, with or without
attila@404 60 modification, are permitted provided that the following conditions are
attila@404 61 met:
attila@404 62 * Redistributions of source code must retain the above copyright
attila@404 63 notice, this list of conditions and the following disclaimer.
attila@404 64 * Redistributions in binary form must reproduce the above copyright
attila@404 65 notice, this list of conditions and the following disclaimer in the
attila@404 66 documentation and/or other materials provided with the distribution.
attila@404 67 * Neither the name of the copyright holder nor the names of
attila@404 68 contributors may be used to endorse or promote products derived from
attila@404 69 this software without specific prior written permission.
attila@404 70
attila@404 71 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
attila@404 72 IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
attila@404 73 TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
attila@404 74 PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL COPYRIGHT HOLDER
attila@404 75 BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
attila@404 76 CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
attila@404 77 SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
attila@404 78 BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
attila@404 79 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
attila@404 80 OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
attila@404 81 ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
attila@404 82 */
attila@404 83
attila@404 84 package jdk.internal.dynalink.beans;
attila@404 85
attila@404 86 import java.lang.invoke.MethodHandle;
attila@404 87 import java.lang.invoke.MethodHandles;
attila@404 88 import java.lang.invoke.MethodType;
attila@404 89 import java.lang.reflect.Array;
attila@404 90 import java.util.StringTokenizer;
attila@404 91 import jdk.internal.dynalink.CallSiteDescriptor;
attila@404 92 import jdk.internal.dynalink.linker.LinkerServices;
attila@404 93 import jdk.internal.dynalink.support.Guards;
attila@622 94 import jdk.internal.dynalink.support.Lookup;
attila@404 95
attila@404 96 /**
attila@404 97 * Base class for dynamic methods that dispatch to a single target Java method or constructor. Handles adaptation of the
attila@404 98 * target method to a call site type (including mapping variable arity methods to a call site signature with different
attila@404 99 * arity).
attila@404 100 * @author Attila Szegedi
attila@404 101 * @version $Id: $
attila@404 102 */
attila@404 103 abstract class SingleDynamicMethod extends DynamicMethod {
attila@622 104
attila@622 105 private static final MethodHandle CAN_CONVERT_TO = Lookup.findOwnStatic(MethodHandles.lookup(), "canConvertTo", boolean.class, LinkerServices.class, Class.class, Object.class);
attila@622 106
attila@962 107 SingleDynamicMethod(final String name) {
attila@404 108 super(name);
attila@404 109 }
attila@404 110
attila@404 111 /**
attila@404 112 * Returns true if this method is variable arity.
attila@404 113 * @return true if this method is variable arity.
attila@404 114 */
attila@404 115 abstract boolean isVarArgs();
attila@404 116
attila@404 117 /**
attila@404 118 * Returns this method's native type.
attila@404 119 * @return this method's native type.
attila@404 120 */
attila@404 121 abstract MethodType getMethodType();
attila@404 122
attila@404 123 /**
attila@404 124 * Given a specified lookup, returns a method handle to this method's target.
attila@404 125 * @param lookup the lookup to use.
attila@404 126 * @return the handle to this method's target method.
attila@404 127 */
attila@404 128 abstract MethodHandle getTarget(MethodHandles.Lookup lookup);
attila@404 129
attila@404 130 @Override
attila@962 131 MethodHandle getInvocation(final CallSiteDescriptor callSiteDescriptor, final LinkerServices linkerServices) {
attila@404 132 return getInvocation(getTarget(callSiteDescriptor.getLookup()), callSiteDescriptor.getMethodType(),
attila@404 133 linkerServices);
attila@404 134 }
attila@404 135
attila@404 136 @Override
attila@962 137 SingleDynamicMethod getMethodForExactParamTypes(final String paramTypes) {
attila@404 138 return typeMatchesDescription(paramTypes, getMethodType()) ? this : null;
attila@404 139 }
attila@404 140
attila@404 141 @Override
attila@962 142 boolean contains(final SingleDynamicMethod method) {
attila@404 143 return getMethodType().parameterList().equals(method.getMethodType().parameterList());
attila@404 144 }
attila@404 145
attila@962 146 static String getMethodNameWithSignature(final MethodType type, final String methodName) {
attila@404 147 final String typeStr = type.toString();
attila@404 148 final int retTypeIndex = typeStr.lastIndexOf(')') + 1;
attila@404 149 int secondParamIndex = typeStr.indexOf(',') + 1;
attila@404 150 if(secondParamIndex == 0) {
attila@404 151 secondParamIndex = retTypeIndex - 1;
attila@404 152 }
attila@404 153 return typeStr.substring(retTypeIndex) + " " + methodName + "(" + typeStr.substring(secondParamIndex, retTypeIndex);
attila@404 154 }
attila@404 155
attila@404 156 /**
attila@404 157 * Given a method handle and a call site type, adapts the method handle to the call site type. Performs type
attila@404 158 * conversions as needed using the specified linker services, and in case that the method handle is a vararg
attila@963 159 * collector, matches it to the arity of the call site. The type of the return value is only changed if it can be
attila@963 160 * converted using a conversion that loses neither precision nor magnitude, see
attila@963 161 * {@link LinkerServices#asTypeLosslessReturn(MethodHandle, MethodType)}.
attila@404 162 * @param target the method handle to adapt
attila@404 163 * @param callSiteType the type of the call site
attila@404 164 * @param linkerServices the linker services used for type conversions
attila@404 165 * @return the adapted method handle.
attila@404 166 */
attila@962 167 static MethodHandle getInvocation(final MethodHandle target, final MethodType callSiteType, final LinkerServices linkerServices) {
attila@404 168 final MethodType methodType = target.type();
attila@404 169 final int paramsLen = methodType.parameterCount();
attila@404 170 final boolean varArgs = target.isVarargsCollector();
attila@404 171 final MethodHandle fixTarget = varArgs ? target.asFixedArity() : target;
attila@404 172 final int fixParamsLen = varArgs ? paramsLen - 1 : paramsLen;
attila@404 173 final int argsLen = callSiteType.parameterCount();
attila@404 174 if(argsLen < fixParamsLen) {
attila@404 175 // Less actual arguments than number of fixed declared arguments; can't invoke.
attila@404 176 return null;
attila@404 177 }
attila@404 178 // Method handle has the same number of fixed arguments as the call site type
attila@404 179 if(argsLen == fixParamsLen) {
attila@404 180 // Method handle that matches the number of actual arguments as the number of fixed arguments
attila@404 181 final MethodHandle matchedMethod;
attila@404 182 if(varArgs) {
attila@404 183 // If vararg, add a zero-length array of the expected type as the last argument to signify no variable
attila@404 184 // arguments.
attila@404 185 matchedMethod = MethodHandles.insertArguments(fixTarget, fixParamsLen, Array.newInstance(
attila@404 186 methodType.parameterType(fixParamsLen).getComponentType(), 0));
attila@404 187 } else {
attila@404 188 // Otherwise, just use the method
attila@404 189 matchedMethod = fixTarget;
attila@404 190 }
attila@404 191 return createConvertingInvocation(matchedMethod, linkerServices, callSiteType);
attila@404 192 }
attila@404 193
attila@404 194 // What's below only works for varargs
attila@404 195 if(!varArgs) {
attila@404 196 return null;
attila@404 197 }
attila@404 198
attila@404 199 final Class<?> varArgType = methodType.parameterType(fixParamsLen);
attila@404 200 // Handle a somewhat sinister corner case: caller passes exactly one argument in the vararg position, and we
attila@404 201 // must handle both a prepacked vararg array as well as a genuine 1-long vararg sequence.
attila@404 202 if(argsLen == paramsLen) {
attila@404 203 final Class<?> callSiteLastArgType = callSiteType.parameterType(fixParamsLen);
attila@404 204 if(varArgType.isAssignableFrom(callSiteLastArgType)) {
attila@404 205 // Call site signature guarantees we'll always be passed a single compatible array; just link directly
attila@404 206 // to the method, introducing necessary conversions. Also, preserve it being a variable arity method.
attila@404 207 return createConvertingInvocation(target, linkerServices, callSiteType).asVarargsCollector(
attila@404 208 callSiteLastArgType);
attila@404 209 }
attila@622 210
attila@622 211 // This method handle takes the single argument and packs it into a newly allocated single-element array. It
attila@622 212 // will be used when the incoming argument can't be converted to the vararg array type (the "vararg packer"
attila@622 213 // method).
attila@622 214 final MethodHandle varArgCollectingInvocation = createConvertingInvocation(collectArguments(fixTarget,
attila@622 215 argsLen), linkerServices, callSiteType);
attila@622 216
attila@622 217 // Is call site type assignable from an array type (e.g. Object:int[], or Object[]:String[])
attila@622 218 final boolean isAssignableFromArray = callSiteLastArgType.isAssignableFrom(varArgType);
attila@622 219 // Do we have a custom conversion that can potentially convert the call site type to an array?
attila@622 220 final boolean isCustomConvertible = linkerServices.canConvert(callSiteLastArgType, varArgType);
attila@622 221 if(!isAssignableFromArray && !isCustomConvertible) {
attila@622 222 // Call site signature guarantees the argument can definitely not be converted to an array (i.e. it is
attila@622 223 // primitive), and no conversion can help with it either. Link immediately to a vararg-packing method
attila@622 224 // handle.
attila@622 225 return varArgCollectingInvocation;
attila@404 226 }
attila@622 227
attila@622 228 // This method handle employs language-specific conversions to convert the last argument into an array of
attila@622 229 // vararg type.
attila@622 230 final MethodHandle arrayConvertingInvocation = createConvertingInvocation(MethodHandles.filterArguments(
attila@622 231 fixTarget, fixParamsLen, linkerServices.getTypeConverter(callSiteLastArgType, varArgType)),
attila@622 232 linkerServices, callSiteType);
attila@622 233
attila@622 234 // This method handle determines whether the value can be converted to the array of vararg type using a
attila@622 235 // language-specific conversion.
attila@622 236 final MethodHandle canConvertArgToArray = MethodHandles.insertArguments(CAN_CONVERT_TO, 0, linkerServices,
attila@622 237 varArgType);
attila@622 238
attila@622 239 // This one adjusts the previous one for the location of the argument and the call site type.
attila@622 240 final MethodHandle canConvertLastArgToArray = MethodHandles.dropArguments(canConvertArgToArray, 0,
attila@622 241 MethodType.genericMethodType(fixParamsLen).parameterList()).asType(callSiteType.changeReturnType(boolean.class));
attila@622 242
attila@622 243 // This one takes the previous ones and combines them into a method handle that converts the argument into
attila@622 244 // a vararg array when it can, otherwise falls back to the vararg packer.
attila@622 245 final MethodHandle convertToArrayWhenPossible = MethodHandles.guardWithTest(canConvertLastArgToArray,
attila@622 246 arrayConvertingInvocation, varArgCollectingInvocation);
attila@622 247
attila@622 248 if(isAssignableFromArray) {
attila@622 249 return MethodHandles.guardWithTest(
attila@622 250 // Is incoming parameter already a compatible array?
attila@622 251 Guards.isInstance(varArgType, fixParamsLen, callSiteType),
attila@622 252 // Yes: just pass it to the method
attila@622 253 createConvertingInvocation(fixTarget, linkerServices, callSiteType),
attila@622 254 // No: either go through a custom conversion, or if it is not possible, go directly to the
attila@622 255 // vararg packer.
attila@622 256 isCustomConvertible ? convertToArrayWhenPossible : varArgCollectingInvocation);
attila@622 257 }
attila@622 258
attila@622 259 // Just do the custom conversion with fallback to the vararg packer logic.
attila@622 260 assert isCustomConvertible;
attila@622 261 return convertToArrayWhenPossible;
attila@404 262 }
attila@404 263
attila@404 264 // Remaining case: more than one vararg.
attila@404 265 return createConvertingInvocation(collectArguments(fixTarget, argsLen), linkerServices, callSiteType);
attila@404 266 }
attila@404 267
attila@622 268 @SuppressWarnings("unused")
attila@962 269 private static boolean canConvertTo(final LinkerServices linkerServices, final Class<?> to, final Object obj) {
attila@622 270 return obj == null ? false : linkerServices.canConvert(obj.getClass(), to);
attila@622 271 }
attila@622 272
attila@404 273 /**
attila@404 274 * Creates a method handle out of the original target that will collect the varargs for the exact component type of
attila@404 275 * the varArg array. Note that this will nicely trigger language-specific type converters for exactly those varargs
attila@404 276 * for which it is necessary when later passed to linkerServices.convertArguments().
attila@404 277 *
attila@404 278 * @param target the original method handle
attila@404 279 * @param parameterCount the total number of arguments in the new method handle
attila@404 280 * @return a collecting method handle
attila@404 281 */
attila@962 282 static MethodHandle collectArguments(final MethodHandle target, final int parameterCount) {
attila@404 283 final MethodType methodType = target.type();
attila@404 284 final int fixParamsLen = methodType.parameterCount() - 1;
attila@404 285 final Class<?> arrayType = methodType.parameterType(fixParamsLen);
attila@404 286 return target.asCollector(arrayType, parameterCount - fixParamsLen);
attila@404 287 }
attila@404 288
attila@404 289 private static MethodHandle createConvertingInvocation(final MethodHandle sizedMethod,
attila@404 290 final LinkerServices linkerServices, final MethodType callSiteType) {
attila@963 291 return linkerServices.asTypeLosslessReturn(sizedMethod, callSiteType);
attila@404 292 }
attila@404 293
attila@962 294 private static boolean typeMatchesDescription(final String paramTypes, final MethodType type) {
attila@404 295 final StringTokenizer tok = new StringTokenizer(paramTypes, ", ");
attila@404 296 for(int i = 1; i < type.parameterCount(); ++i) { // i = 1 as we ignore the receiver
attila@404 297 if(!(tok.hasMoreTokens() && typeNameMatches(tok.nextToken(), type.parameterType(i)))) {
attila@404 298 return false;
attila@404 299 }
attila@404 300 }
attila@404 301 return !tok.hasMoreTokens();
attila@404 302 }
attila@404 303
attila@962 304 private static boolean typeNameMatches(final String typeName, final Class<?> type) {
attila@404 305 return typeName.equals(typeName.indexOf('.') == -1 ? type.getSimpleName() : type.getCanonicalName());
attila@404 306 }
attila@404 307 }

mercurial