src/jdk/internal/dynalink/beans/BeanLinker.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

     1 /*
     2  * Copyright (c) 2010, 2013, 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
    23  * questions.
    24  */
    26 /*
    27  * This file is available under and governed by the GNU General Public
    28  * License version 2 only, as published by the Free Software Foundation.
    29  * However, the following notice accompanied the original version of this
    30  * file, and Oracle licenses the original version of this file under the BSD
    31  * license:
    32  */
    33 /*
    34    Copyright 2009-2013 Attila Szegedi
    36    Licensed under both the Apache License, Version 2.0 (the "Apache License")
    37    and the BSD License (the "BSD License"), with licensee being free to
    38    choose either of the two at their discretion.
    40    You may not use this file except in compliance with either the Apache
    41    License or the BSD License.
    43    If you choose to use this file in compliance with the Apache License, the
    44    following notice applies to you:
    46        You may obtain a copy of the Apache License at
    48            http://www.apache.org/licenses/LICENSE-2.0
    50        Unless required by applicable law or agreed to in writing, software
    51        distributed under the License is distributed on an "AS IS" BASIS,
    52        WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
    53        implied. See the License for the specific language governing
    54        permissions and limitations under the License.
    56    If you choose to use this file in compliance with the BSD License, the
    57    following notice applies to you:
    59        Redistribution and use in source and binary forms, with or without
    60        modification, are permitted provided that the following conditions are
    61        met:
    62        * Redistributions of source code must retain the above copyright
    63          notice, this list of conditions and the following disclaimer.
    64        * Redistributions in binary form must reproduce the above copyright
    65          notice, this list of conditions and the following disclaimer in the
    66          documentation and/or other materials provided with the distribution.
    67        * Neither the name of the copyright holder nor the names of
    68          contributors may be used to endorse or promote products derived from
    69          this software without specific prior written permission.
    71        THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
    72        IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
    73        TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
    74        PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL COPYRIGHT HOLDER
    75        BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
    76        CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
    77        SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
    78        BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
    79        WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
    80        OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
    81        ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    82 */
    84 package jdk.internal.dynalink.beans;
    86 import java.lang.invoke.MethodHandle;
    87 import java.lang.invoke.MethodHandles;
    88 import java.lang.invoke.MethodType;
    89 import java.lang.reflect.Array;
    90 import java.util.Collection;
    91 import java.util.List;
    92 import java.util.Map;
    93 import jdk.internal.dynalink.CallSiteDescriptor;
    94 import jdk.internal.dynalink.beans.GuardedInvocationComponent.ValidationType;
    95 import jdk.internal.dynalink.linker.GuardedInvocation;
    96 import jdk.internal.dynalink.linker.LinkerServices;
    97 import jdk.internal.dynalink.linker.TypeBasedGuardingDynamicLinker;
    98 import jdk.internal.dynalink.support.Guards;
    99 import jdk.internal.dynalink.support.Lookup;
   100 import jdk.internal.dynalink.support.TypeUtilities;
   102 /**
   103  * A class that provides linking capabilities for a single POJO class. Normally not used directly, but managed by
   104  * {@link BeansLinker}.
   105  *
   106  * @author Attila Szegedi
   107  */
   108 class BeanLinker extends AbstractJavaLinker implements TypeBasedGuardingDynamicLinker {
   109     BeanLinker(final Class<?> clazz) {
   110         super(clazz, Guards.getClassGuard(clazz), Guards.getInstanceOfGuard(clazz));
   111         if(clazz.isArray()) {
   112             // Some languages won't have a notion of manipulating collections. Exposing "length" on arrays as an
   113             // explicit property is beneficial for them.
   114             // REVISIT: is it maybe a code smell that "dyn:getLength" is not needed?
   115             setPropertyGetter("length", GET_ARRAY_LENGTH, ValidationType.IS_ARRAY);
   116         } else if(List.class.isAssignableFrom(clazz)) {
   117             setPropertyGetter("length", GET_COLLECTION_LENGTH, ValidationType.INSTANCE_OF);
   118         }
   119     }
   121     @Override
   122     public boolean canLinkType(final Class<?> type) {
   123         return type == clazz;
   124     }
   126     @Override
   127     FacetIntrospector createFacetIntrospector() {
   128         return new BeanIntrospector(clazz);
   129     }
   131     @Override
   132     protected GuardedInvocationComponent getGuardedInvocationComponent(final CallSiteDescriptor callSiteDescriptor,
   133             final LinkerServices linkerServices, final List<String> operations) throws Exception {
   134         final GuardedInvocationComponent superGic = super.getGuardedInvocationComponent(callSiteDescriptor,
   135                 linkerServices, operations);
   136         if(superGic != null) {
   137             return superGic;
   138         }
   139         if(operations.isEmpty()) {
   140             return null;
   141         }
   142         final String op = operations.get(0);
   143         // dyn:getElem(this, id)
   144         // id is typically either an int (for arrays and lists) or an object (for maps). linkerServices can provide
   145         // conversion from call site argument type though.
   146         if("getElem".equals(op)) {
   147             return getElementGetter(callSiteDescriptor, linkerServices, pop(operations));
   148         }
   149         if("setElem".equals(op)) {
   150             return getElementSetter(callSiteDescriptor, linkerServices, pop(operations));
   151         }
   152         // dyn:getLength(this) (works on Java arrays, collections, and maps)
   153         if("getLength".equals(op)) {
   154             return getLengthGetter(callSiteDescriptor);
   155         }
   156         return null;
   157     }
   159     private static MethodHandle GET_LIST_ELEMENT = Lookup.PUBLIC.findVirtual(List.class, "get",
   160             MethodType.methodType(Object.class, int.class));
   162     private static MethodHandle GET_MAP_ELEMENT = Lookup.PUBLIC.findVirtual(Map.class, "get",
   163             MethodType.methodType(Object.class, Object.class));
   165     private static MethodHandle LIST_GUARD = Guards.getInstanceOfGuard(List.class);
   166     private static MethodHandle MAP_GUARD = Guards.getInstanceOfGuard(Map.class);
   168     private GuardedInvocationComponent getElementGetter(final CallSiteDescriptor callSiteDescriptor,
   169             final LinkerServices linkerServices, final List<String> operations) throws Exception {
   170         final MethodType callSiteType = callSiteDescriptor.getMethodType();
   171         final Class<?> declaredType = callSiteType.parameterType(0);
   172         final GuardedInvocationComponent nextComponent = getGuardedInvocationComponent(callSiteDescriptor,
   173                 linkerServices, operations);
   175         // If declared type of receiver at the call site is already an array, a list or map, bind without guard. Thing
   176         // is, it'd be quite stupid of a call site creator to go though invokedynamic when it knows in advance they're
   177         // dealing with an array, or a list or map, but hey...
   178         // Note that for arrays and lists, using LinkerServices.asType() will ensure that any language specific linkers
   179         // in use will get a chance to perform any (if there's any) implicit conversion to integer for the indices.
   180         final GuardedInvocationComponent gic;
   181         final boolean isMap;
   182         if(declaredType.isArray()) {
   183             gic = new GuardedInvocationComponent(MethodHandles.arrayElementGetter(declaredType));
   184             isMap = false;
   185         } else if(List.class.isAssignableFrom(declaredType)) {
   186             gic = new GuardedInvocationComponent(GET_LIST_ELEMENT);
   187             isMap = false;
   188         } else if(Map.class.isAssignableFrom(declaredType)) {
   189             gic = new GuardedInvocationComponent(GET_MAP_ELEMENT);
   190             isMap = true;
   191         } else if(clazz.isArray()) {
   192             gic = getClassGuardedInvocationComponent(MethodHandles.arrayElementGetter(clazz), callSiteType);
   193             isMap = false;
   194         } else if(List.class.isAssignableFrom(clazz)) {
   195             gic = new GuardedInvocationComponent(GET_LIST_ELEMENT, Guards.asType(LIST_GUARD, callSiteType), List.class,
   196                     ValidationType.INSTANCE_OF);
   197             isMap = false;
   198         } else if(Map.class.isAssignableFrom(clazz)) {
   199             gic = new GuardedInvocationComponent(GET_MAP_ELEMENT, Guards.asType(MAP_GUARD, callSiteType), Map.class,
   200                     ValidationType.INSTANCE_OF);
   201             isMap = true;
   202         } else {
   203             // Can't retrieve elements for objects that are neither arrays, nor list, nor maps.
   204             return nextComponent;
   205         }
   207         // We can have "dyn:getElem:foo", especially in composites, i.e. "dyn:getElem|getProp|getMethod:foo"
   208         final String fixedKey = getFixedKey(callSiteDescriptor);
   209         // Convert the key to a number if we're working with a list or array
   210         final Object typedFixedKey;
   211         if(!isMap && fixedKey != null) {
   212             typedFixedKey = convertKeyToInteger(fixedKey, linkerServices);
   213             if(typedFixedKey == null) {
   214                 // key is not numeric, it can never succeed
   215                 return nextComponent;
   216             }
   217         } else {
   218             typedFixedKey = fixedKey;
   219         }
   221         final GuardedInvocation gi = gic.getGuardedInvocation();
   222         final Binder binder = new Binder(linkerServices, callSiteType, typedFixedKey);
   223         final MethodHandle invocation = gi.getInvocation();
   225         if(nextComponent == null) {
   226             return gic.replaceInvocation(binder.bind(invocation));
   227         }
   229         final MethodHandle checkGuard;
   230         if(invocation == GET_LIST_ELEMENT) {
   231             checkGuard = convertArgToInt(RANGE_CHECK_LIST, linkerServices, callSiteDescriptor);
   232         } else if(invocation == GET_MAP_ELEMENT) {
   233             // TODO: A more complex solution could be devised for maps, one where we do a get() first, and fold it
   234             // into a GWT that tests if it returned null, and if it did, do another GWT with containsKey()
   235             // that returns constant null (on true), or falls back to next component (on false)
   236             checkGuard = CONTAINS_MAP;
   237         } else {
   238             checkGuard = convertArgToInt(RANGE_CHECK_ARRAY, linkerServices, callSiteDescriptor);
   239         }
   240         final MethodPair matchedInvocations = matchReturnTypes(binder.bind(invocation),
   241                 nextComponent.getGuardedInvocation().getInvocation());
   242         return nextComponent.compose(matchedInvocations.guardWithTest(binder.bindTest(checkGuard)), gi.getGuard(),
   243                 gic.getValidatorClass(), gic.getValidationType());
   244     }
   246     private static String getFixedKey(final CallSiteDescriptor callSiteDescriptor) {
   247         return callSiteDescriptor.getNameTokenCount() == 2 ? null : callSiteDescriptor.getNameToken(
   248                 CallSiteDescriptor.NAME_OPERAND);
   249     }
   251     private static Object convertKeyToInteger(final String fixedKey, final LinkerServices linkerServices) throws Exception {
   252         try {
   253             if(linkerServices.canConvert(String.class, Number.class)) {
   254                 try {
   255                     final Object val = linkerServices.getTypeConverter(String.class, Number.class).invoke(fixedKey);
   256                     if(!(val instanceof Number)) {
   257                         return null; // not a number
   258                     }
   259                     final Number n = (Number)val;
   260                     if(n instanceof Integer) {
   261                         return n;
   262                     }
   263                     final int intIndex = n.intValue();
   264                     final double doubleValue = n.doubleValue();
   265                     if(intIndex != doubleValue && !Double.isInfinite(doubleValue)) { // let infinites trigger IOOBE
   266                         return null; // not an exact integer
   267                     }
   268                     return Integer.valueOf(intIndex);
   269                 } catch(Exception|Error e) {
   270                     throw e;
   271                 } catch(final Throwable t) {
   272                     throw new RuntimeException(t);
   273                 }
   274             }
   275             return Integer.valueOf(fixedKey);
   276         } catch(final NumberFormatException e) {
   277             // key is not a number
   278             return null;
   279         }
   280     }
   282     private static MethodHandle convertArgToInt(final MethodHandle mh, final LinkerServices ls, final CallSiteDescriptor desc) {
   283         final Class<?> sourceType = desc.getMethodType().parameterType(1);
   284         if(TypeUtilities.isMethodInvocationConvertible(sourceType, Number.class)) {
   285             return mh;
   286         } else if(ls.canConvert(sourceType, Number.class)) {
   287             final MethodHandle converter = ls.getTypeConverter(sourceType, Number.class);
   288             return MethodHandles.filterArguments(mh, 1, converter.asType(converter.type().changeReturnType(
   289                     mh.type().parameterType(1))));
   290         }
   291         return mh;
   292     }
   294     /**
   295      * Contains methods to adapt an item getter/setter method handle to the requested type, optionally binding it to a
   296      * fixed key first.
   297      * @author Attila Szegedi
   298      * @version $Id: $
   299      */
   300     private static class Binder {
   301         private final LinkerServices linkerServices;
   302         private final MethodType methodType;
   303         private final Object fixedKey;
   305         Binder(final LinkerServices linkerServices, final MethodType methodType, final Object fixedKey) {
   306             this.linkerServices = linkerServices;
   307             this.methodType = fixedKey == null ? methodType : methodType.insertParameterTypes(1, fixedKey.getClass());
   308             this.fixedKey = fixedKey;
   309         }
   311         /*private*/ MethodHandle bind(final MethodHandle handle) {
   312             return bindToFixedKey(linkerServices.asTypeLosslessReturn(handle, methodType));
   313         }
   315         /*private*/ MethodHandle bindTest(final MethodHandle handle) {
   316             return bindToFixedKey(Guards.asType(handle, methodType));
   317         }
   319         private MethodHandle bindToFixedKey(final MethodHandle handle) {
   320             return fixedKey == null ? handle : MethodHandles.insertArguments(handle, 1, fixedKey);
   321         }
   322     }
   324     private static MethodHandle RANGE_CHECK_ARRAY = findRangeCheck(Object.class);
   325     private static MethodHandle RANGE_CHECK_LIST = findRangeCheck(List.class);
   326     private static MethodHandle CONTAINS_MAP = Lookup.PUBLIC.findVirtual(Map.class, "containsKey",
   327             MethodType.methodType(boolean.class, Object.class));
   329     private static MethodHandle findRangeCheck(final Class<?> collectionType) {
   330         return Lookup.findOwnStatic(MethodHandles.lookup(), "rangeCheck", boolean.class, collectionType, Object.class);
   331     }
   333     @SuppressWarnings("unused")
   334     private static final boolean rangeCheck(final Object array, final Object index) {
   335         if(!(index instanceof Number)) {
   336             return false;
   337         }
   338         final Number n = (Number)index;
   339         final int intIndex = n.intValue();
   340         final double doubleValue = n.doubleValue();
   341         if(intIndex != doubleValue && !Double.isInfinite(doubleValue)) { // let infinite trigger IOOBE
   342             return false;
   343         }
   344         if(0 <= intIndex && intIndex < Array.getLength(array)) {
   345             return true;
   346         }
   347         throw new ArrayIndexOutOfBoundsException("Array index out of range: " + n);
   348     }
   350     @SuppressWarnings("unused")
   351     private static final boolean rangeCheck(final List<?> list, final Object index) {
   352         if(!(index instanceof Number)) {
   353             return false;
   354         }
   355         final Number n = (Number)index;
   356         final int intIndex = n.intValue();
   357         final double doubleValue = n.doubleValue();
   358         if(intIndex != doubleValue && !Double.isInfinite(doubleValue)) { // let infinite trigger IOOBE
   359             return false;
   360         }
   361         if(0 <= intIndex && intIndex < list.size()) {
   362             return true;
   363         }
   364         throw new IndexOutOfBoundsException("Index: " + n + ", Size: " + list.size());
   365     }
   367     private static MethodHandle SET_LIST_ELEMENT = Lookup.PUBLIC.findVirtual(List.class, "set",
   368             MethodType.methodType(Object.class, int.class, Object.class));
   370     private static MethodHandle PUT_MAP_ELEMENT = Lookup.PUBLIC.findVirtual(Map.class, "put",
   371             MethodType.methodType(Object.class, Object.class, Object.class));
   373     private GuardedInvocationComponent getElementSetter(final CallSiteDescriptor callSiteDescriptor,
   374             final LinkerServices linkerServices, final List<String> operations) throws Exception {
   375         final MethodType callSiteType = callSiteDescriptor.getMethodType();
   376         final Class<?> declaredType = callSiteType.parameterType(0);
   378         final GuardedInvocationComponent gic;
   379         // If declared type of receiver at the call site is already an array, a list or map, bind without guard. Thing
   380         // is, it'd be quite stupid of a call site creator to go though invokedynamic when it knows in advance they're
   381         // dealing with an array, or a list or map, but hey...
   382         // Note that for arrays and lists, using LinkerServices.asType() will ensure that any language specific linkers
   383         // in use will get a chance to perform any (if there's any) implicit conversion to integer for the indices.
   384         final boolean isMap;
   385         if(declaredType.isArray()) {
   386             gic = new GuardedInvocationComponent(MethodHandles.arrayElementSetter(declaredType));
   387             isMap = false;
   388         } else if(List.class.isAssignableFrom(declaredType)) {
   389             gic = new GuardedInvocationComponent(SET_LIST_ELEMENT);
   390             isMap = false;
   391         } else if(Map.class.isAssignableFrom(declaredType)) {
   392             gic = new GuardedInvocationComponent(PUT_MAP_ELEMENT);
   393             isMap = true;
   394         } else if(clazz.isArray()) {
   395             gic = getClassGuardedInvocationComponent(MethodHandles.arrayElementSetter(clazz), callSiteType);
   396             isMap = false;
   397         } else if(List.class.isAssignableFrom(clazz)) {
   398             gic = new GuardedInvocationComponent(SET_LIST_ELEMENT, Guards.asType(LIST_GUARD, callSiteType), List.class,
   399                     ValidationType.INSTANCE_OF);
   400             isMap = false;
   401         } else if(Map.class.isAssignableFrom(clazz)) {
   402             gic = new GuardedInvocationComponent(PUT_MAP_ELEMENT, Guards.asType(MAP_GUARD, callSiteType), Map.class,
   403                     ValidationType.INSTANCE_OF);
   404             isMap = true;
   405         } else {
   406             // Can't set elements for objects that are neither arrays, nor list, nor maps.
   407             gic = null;
   408             isMap = false;
   409         }
   411         // In contrast to, say, getElementGetter, we only compute the nextComponent if the target object is not a map,
   412         // as maps will always succeed in setting the element and will never need to fall back to the next component
   413         // operation.
   414         final GuardedInvocationComponent nextComponent = isMap ? null : getGuardedInvocationComponent(
   415                 callSiteDescriptor, linkerServices, operations);
   416         if(gic == null) {
   417             return nextComponent;
   418         }
   420         // We can have "dyn:setElem:foo", especially in composites, i.e. "dyn:setElem|setProp:foo"
   421         final String fixedKey = getFixedKey(callSiteDescriptor);
   422         // Convert the key to a number if we're working with a list or array
   423         final Object typedFixedKey;
   424         if(!isMap && fixedKey != null) {
   425             typedFixedKey = convertKeyToInteger(fixedKey, linkerServices);
   426             if(typedFixedKey == null) {
   427                 // key is not numeric, it can never succeed
   428                 return nextComponent;
   429             }
   430         } else {
   431             typedFixedKey = fixedKey;
   432         }
   434         final GuardedInvocation gi = gic.getGuardedInvocation();
   435         final Binder binder = new Binder(linkerServices, callSiteType, typedFixedKey);
   436         final MethodHandle invocation = gi.getInvocation();
   438         if(nextComponent == null) {
   439             return gic.replaceInvocation(binder.bind(invocation));
   440         }
   442         final MethodHandle checkGuard = convertArgToInt(invocation == SET_LIST_ELEMENT ? RANGE_CHECK_LIST :
   443             RANGE_CHECK_ARRAY, linkerServices, callSiteDescriptor);
   444         final MethodPair matchedInvocations = matchReturnTypes(binder.bind(invocation),
   445                 nextComponent.getGuardedInvocation().getInvocation());
   446         return nextComponent.compose(matchedInvocations.guardWithTest(binder.bindTest(checkGuard)), gi.getGuard(),
   447                 gic.getValidatorClass(), gic.getValidationType());
   448     }
   450     private static MethodHandle GET_ARRAY_LENGTH = Lookup.PUBLIC.findStatic(Array.class, "getLength",
   451             MethodType.methodType(int.class, Object.class));
   453     private static MethodHandle GET_COLLECTION_LENGTH = Lookup.PUBLIC.findVirtual(Collection.class, "size",
   454             MethodType.methodType(int.class));
   456     private static MethodHandle GET_MAP_LENGTH = Lookup.PUBLIC.findVirtual(Map.class, "size",
   457             MethodType.methodType(int.class));
   459     private static MethodHandle COLLECTION_GUARD = Guards.getInstanceOfGuard(Collection.class);
   461     private GuardedInvocationComponent getLengthGetter(final CallSiteDescriptor callSiteDescriptor) {
   462         assertParameterCount(callSiteDescriptor, 1);
   463         final MethodType callSiteType = callSiteDescriptor.getMethodType();
   464         final Class<?> declaredType = callSiteType.parameterType(0);
   465         // If declared type of receiver at the call site is already an array, collection, or map, bind without guard.
   466         // Thing is, it'd be quite stupid of a call site creator to go though invokedynamic when it knows in advance
   467         // they're dealing with an array, collection, or map, but hey...
   468         if(declaredType.isArray()) {
   469             return new GuardedInvocationComponent(GET_ARRAY_LENGTH.asType(callSiteType));
   470         } else if(Collection.class.isAssignableFrom(declaredType)) {
   471             return new GuardedInvocationComponent(GET_COLLECTION_LENGTH.asType(callSiteType));
   472         } else if(Map.class.isAssignableFrom(declaredType)) {
   473             return new GuardedInvocationComponent(GET_MAP_LENGTH.asType(callSiteType));
   474         }
   476         // Otherwise, create a binding based on the actual type of the argument with an appropriate guard.
   477         if(clazz.isArray()) {
   478             return new GuardedInvocationComponent(GET_ARRAY_LENGTH.asType(callSiteType), Guards.isArray(0,
   479                     callSiteType), ValidationType.IS_ARRAY);
   480         } if(Collection.class.isAssignableFrom(clazz)) {
   481             return new GuardedInvocationComponent(GET_COLLECTION_LENGTH.asType(callSiteType), Guards.asType(
   482                     COLLECTION_GUARD, callSiteType), Collection.class, ValidationType.INSTANCE_OF);
   483         } if(Map.class.isAssignableFrom(clazz)) {
   484             return new GuardedInvocationComponent(GET_MAP_LENGTH.asType(callSiteType), Guards.asType(MAP_GUARD,
   485                     callSiteType), Map.class, ValidationType.INSTANCE_OF);
   486         }
   487         // Can't retrieve length for objects that are neither arrays, nor collections, nor maps.
   488         return null;
   489     }
   491     private static void assertParameterCount(final CallSiteDescriptor descriptor, final int paramCount) {
   492         if(descriptor.getMethodType().parameterCount() != paramCount) {
   493             throw new BootstrapMethodError(descriptor.getName() + " must have exactly " + paramCount + " parameters.");
   494         }
   495     }
   496 }

mercurial