nashorn/src/jdk.dynalink/share/classes/jdk/dynalink/support/ChainedCallSite.java
author attila
Tue, 24 Nov 2015 10:19:34 +0100
changeset 34447 ec4c069f9436
parent 33337 nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/support/ChainedCallSite.java@af3fea63e008
child 35407 204abe4d8cbc
permissions -rw-r--r--
8141338: Move jdk.internal.dynalink package to jdk.dynalink Reviewed-by: hannesw, sundar
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     1
/*
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     2
 * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     4
 *
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    10
 *
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    15
 * accompanied this code).
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    16
 *
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    20
 *
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    23
 * questions.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    24
 */
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    25
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    26
/*
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    27
 * This file is available under and governed by the GNU General Public
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    28
 * License version 2 only, as published by the Free Software Foundation.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    29
 * However, the following notice accompanied the original version of this
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    30
 * file, and Oracle licenses the original version of this file under the BSD
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    31
 * license:
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    32
 */
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    33
/*
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    34
   Copyright 2009-2013 Attila Szegedi
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    35
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    36
   Licensed under both the Apache License, Version 2.0 (the "Apache License")
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    37
   and the BSD License (the "BSD License"), with licensee being free to
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    38
   choose either of the two at their discretion.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    39
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    40
   You may not use this file except in compliance with either the Apache
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    41
   License or the BSD License.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    42
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    43
   If you choose to use this file in compliance with the Apache License, the
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    44
   following notice applies to you:
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    45
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    46
       You may obtain a copy of the Apache License at
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    47
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    48
           http://www.apache.org/licenses/LICENSE-2.0
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    49
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    50
       Unless required by applicable law or agreed to in writing, software
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    51
       distributed under the License is distributed on an "AS IS" BASIS,
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    52
       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    53
       implied. See the License for the specific language governing
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    54
       permissions and limitations under the License.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    55
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    56
   If you choose to use this file in compliance with the BSD License, the
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    57
   following notice applies to you:
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    58
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    59
       Redistribution and use in source and binary forms, with or without
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    60
       modification, are permitted provided that the following conditions are
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    61
       met:
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    62
       * Redistributions of source code must retain the above copyright
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    63
         notice, this list of conditions and the following disclaimer.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    64
       * Redistributions in binary form must reproduce the above copyright
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    65
         notice, this list of conditions and the following disclaimer in the
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    66
         documentation and/or other materials provided with the distribution.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    67
       * Neither the name of the copyright holder nor the names of
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    68
         contributors may be used to endorse or promote products derived from
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    69
         this software without specific prior written permission.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    70
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    71
       THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    72
       IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    73
       TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    74
       PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL COPYRIGHT HOLDER
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    75
       BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    76
       CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    77
       SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    78
       BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    79
       WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    80
       OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    81
       ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    82
*/
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    83
34447
ec4c069f9436 8141338: Move jdk.internal.dynalink package to jdk.dynalink
attila
parents: 33337
diff changeset
    84
package jdk.dynalink.support;
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    85
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    86
import java.lang.invoke.MethodHandle;
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    87
import java.lang.invoke.MethodHandles;
33006
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
    88
import java.util.Arrays;
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    89
import java.util.Iterator;
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    90
import java.util.LinkedList;
34447
ec4c069f9436 8141338: Move jdk.internal.dynalink package to jdk.dynalink
attila
parents: 33337
diff changeset
    91
import jdk.dynalink.CallSiteDescriptor;
ec4c069f9436 8141338: Move jdk.internal.dynalink package to jdk.dynalink
attila
parents: 33337
diff changeset
    92
import jdk.dynalink.linker.GuardedInvocation;
ec4c069f9436 8141338: Move jdk.internal.dynalink package to jdk.dynalink
attila
parents: 33337
diff changeset
    93
import jdk.dynalink.linker.support.Lookup;
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    94
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
    95
/**
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
    96
 * A relinkable call site that implements a polymorphic inline caching strategy.
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
    97
 * It remembers up to 8 {@link GuardedInvocation}s it was linked with, and on
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
    98
 * each relink request builds a cascading chain of method handles of one
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
    99
 * invocation falling back to the next one. The number of remembered invocations
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   100
 * can be customized by overriding {@link #getMaxChainLength()} in a subclass.
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   101
 * When this call site is relinked with a new invocation and the length of the
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   102
 * chain is already at the maximum, it will throw away the oldest invocation.
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   103
 * Invocations with invalidated switch points and ones for which their
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   104
 * invalidating exception triggered are removed eagerly from the chain. The
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   105
 * invocations are never reordered; the most recently linked method handle is
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   106
 * always at the start of the chain and the least recently linked at its end.
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   107
 * The call site can be safely relinked on more than one thread concurrently.
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   108
 * Race conditions in linking are resolved by throwing away the
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   109
 * {@link GuardedInvocation} produced on the losing thread without incorporating
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   110
 * it into the chain, so it can lead to repeated linking for the same arguments.
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   111
 */
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   112
public class ChainedCallSite extends AbstractRelinkableCallSite {
32941
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   113
    private static final MethodHandle PRUNE_CATCHES;
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   114
    private static final MethodHandle PRUNE_SWITCHPOINTS;
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   115
    static {
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   116
        final MethodHandle PRUNE = Lookup.findOwnSpecial(MethodHandles.lookup(), "prune", MethodHandle.class,
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   117
                MethodHandle.class, boolean.class);
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   118
        PRUNE_CATCHES      = MethodHandles.insertArguments(PRUNE, 2, true);
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   119
        PRUNE_SWITCHPOINTS = MethodHandles.insertArguments(PRUNE, 2, false);
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   120
    }
19455
b972b61a6921 8022509: Various Dynalink security enhancements
attila
parents: 16245
diff changeset
   121
33006
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   122
    /**
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   123
     * Contains the invocations currently linked into this call site's target. They are used when we are
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   124
     * relinking to rebuild the guardWithTest chain. Valid values for this field are: {@code null} if there's
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   125
     * no linked invocations, or an instance of {@link GuardedInvocation} if there is exactly one previous
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   126
     * invocation, or an instance of {@code GuardedInvocation[]} if there is more than one previous
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   127
     * invocation.
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   128
     */
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   129
    private Object invocations;
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   130
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   131
    /**
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   132
     * Creates a new chained call site.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   133
     * @param descriptor the descriptor for the call site.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   134
     */
24778
2ff5d7041566 8044638: Tidy up Nashorn codebase for code standards
attila
parents: 24720
diff changeset
   135
    public ChainedCallSite(final CallSiteDescriptor descriptor) {
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   136
        super(descriptor);
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   137
    }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   138
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   139
    /**
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   140
     * The maximum number of method handles in the chain. Defaults to 8. You can
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   141
     * override it in a subclass if you need to change the value.
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   142
     * @return the maximum number of method handles in the chain. The return
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   143
     * value is checked, and if your override returns a value less than 1, a
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   144
     * {@link RuntimeException} will be thrown.
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   145
     */
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   146
    protected int getMaxChainLength() {
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   147
        return 8;
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   148
    }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   149
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   150
    @Override
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   151
    public void relink(final GuardedInvocation guardedInvocation, final MethodHandle relinkAndInvoke) {
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   152
        relinkInternal(guardedInvocation, relinkAndInvoke, false, false);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   153
    }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   154
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   155
    @Override
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   156
    public void resetAndRelink(final GuardedInvocation guardedInvocation, final MethodHandle relinkAndInvoke) {
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   157
        relinkInternal(guardedInvocation, relinkAndInvoke, true, false);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   158
    }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   159
24778
2ff5d7041566 8044638: Tidy up Nashorn codebase for code standards
attila
parents: 24720
diff changeset
   160
    private MethodHandle relinkInternal(final GuardedInvocation invocation, final MethodHandle relink, final boolean reset, final boolean removeCatches) {
33006
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   161
        final Object currentInvocations = invocations;
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   162
        final LinkedList<GuardedInvocation> newInvocations;
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   163
        if (currentInvocations == null || reset) {
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   164
            newInvocations = new LinkedList<>();
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   165
        } else if (currentInvocations instanceof GuardedInvocation) {
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   166
            newInvocations = new LinkedList<>();
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   167
            newInvocations.add((GuardedInvocation)currentInvocations);
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   168
        } else if (currentInvocations instanceof GuardedInvocation[]) {
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   169
            newInvocations = new LinkedList<>(Arrays.asList(((GuardedInvocation[])currentInvocations)));
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   170
        } else {
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   171
            throw new AssertionError();
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   172
        }
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   173
24720
75f8388b79df 8035836: Array performance improvements
lagergren
parents: 19630
diff changeset
   174
        // First, prune the chain of invalidated switchpoints, we always do this
75f8388b79df 8035836: Array performance improvements
lagergren
parents: 19630
diff changeset
   175
        // We also remove any catches if the remove catches flag is set
24778
2ff5d7041566 8044638: Tidy up Nashorn codebase for code standards
attila
parents: 24720
diff changeset
   176
        for(final Iterator<GuardedInvocation> it = newInvocations.iterator(); it.hasNext();) {
24720
75f8388b79df 8035836: Array performance improvements
lagergren
parents: 19630
diff changeset
   177
            final GuardedInvocation inv = it.next();
75f8388b79df 8035836: Array performance improvements
lagergren
parents: 19630
diff changeset
   178
            if(inv.hasBeenInvalidated() || (removeCatches && inv.getException() != null)) {
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   179
                it.remove();
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   180
            }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   181
        }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   182
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   183
        // prune() is allowed to invoke this method with invocation == null meaning we're just pruning the chain and not
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   184
        // adding any new invocations to it.
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   185
        if(invocation != null) {
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   186
            // Remove oldest entry if we're at max length
33330
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   187
            if(newInvocations.size() == checkMaxChainLength(getMaxChainLength())) {
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   188
                newInvocations.removeFirst();
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   189
            }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   190
            newInvocations.addLast(invocation);
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   191
        }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   192
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   193
        // prune-and-invoke is used as the fallback for invalidated switchpoints. If a switchpoint gets invalidated, we
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   194
        // rebuild the chain and get rid of all invalidated switchpoints instead of letting them linger.
32941
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   195
        final MethodHandle pruneAndInvokeSwitchPoints = makePruneAndInvokeMethod(relink, PRUNE_SWITCHPOINTS);
be82ab9eb287 8139269: Do not expose prune method handles from ChainedCallSite
attila
parents: 25865
diff changeset
   196
        final MethodHandle pruneAndInvokeCatches      = makePruneAndInvokeMethod(relink, PRUNE_CATCHES);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   197
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   198
        // Fold the new chain
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   199
        MethodHandle target = relink;
24778
2ff5d7041566 8044638: Tidy up Nashorn codebase for code standards
attila
parents: 24720
diff changeset
   200
        for(final GuardedInvocation inv: newInvocations) {
24720
75f8388b79df 8035836: Array performance improvements
lagergren
parents: 19630
diff changeset
   201
            target = inv.compose(target, pruneAndInvokeSwitchPoints, pruneAndInvokeCatches);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   202
        }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   203
33006
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   204
        switch (newInvocations.size()) {
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   205
            case 0:
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   206
                invocations = null;
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   207
                break;
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   208
            case 1:
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   209
                invocations = newInvocations.getFirst();
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   210
                break;
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   211
            default:
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   212
                invocations = newInvocations.toArray(new GuardedInvocation[newInvocations.size()]);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   213
        }
33006
99298bc38e28 8139270: Drastically reduce memory footprint of ChainedCallSite
attila
parents: 32941
diff changeset
   214
        setTarget(target);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   215
        return target;
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   216
    }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   217
33330
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   218
    private static int checkMaxChainLength(final int maxChainLength) {
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   219
        if (maxChainLength > 0) {
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   220
            return maxChainLength;
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   221
        }
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   222
        throw new RuntimeException("getMaxChainLength() returned a non-positive value");
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   223
35531ae624ef 8139304: Remove elaborate call site descriptor class hierarchy and factory for them. Remove AutoDiscovery, DefaultPrelinkFilter, and BottomGuardingDynamicLinker as they can be inlined into DynamicLinkerFactory. Remove CallerSensitiveDetector as it can be inlined into AbstractJavaLinker. Make ClassMap non-public.
attila
parents: 33006
diff changeset
   224
    }
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   225
    /**
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   226
     * Creates a method that rebuilds our call chain, pruning it of any invalidated switchpoints, and then invokes that
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   227
     * chain.
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   228
     * @param relinkAndInvoke the ultimate fallback for the chain passed from the dynamic linker.
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   229
     * @return a method handle for prune-and-invoke
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   230
     */
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   231
    private MethodHandle makePruneAndInvokeMethod(final MethodHandle relinkAndInvoke, final MethodHandle prune) {
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   232
        // Bind prune to (this, relink)
33333
0bad500ce4e0 8139590: Improve Dynalink JavaDoc
attila
parents: 33330
diff changeset
   233
        final MethodHandle boundPrune = MethodHandles.insertArguments(prune, 0, this, relinkAndInvoke);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   234
        // Make it ignore all incoming arguments
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   235
        final MethodHandle ignoreArgsPrune = MethodHandles.dropArguments(boundPrune, 0, type().parameterList());
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   236
        // Invoke prune, then invoke the call site target with original arguments
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   237
        return MethodHandles.foldArguments(MethodHandles.exactInvoker(type()), ignoreArgsPrune);
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   238
    }
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   239
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   240
    @SuppressWarnings("unused")
24778
2ff5d7041566 8044638: Tidy up Nashorn codebase for code standards
attila
parents: 24720
diff changeset
   241
    private MethodHandle prune(final MethodHandle relink, final boolean catches) {
24720
75f8388b79df 8035836: Array performance improvements
lagergren
parents: 19630
diff changeset
   242
        return relinkInternal(null, relink, false, catches);
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents:
diff changeset
   243
    }
16245
6a1c6c8bc113 8008371: Fix Dynalink compiler warnings and whitespace
attila
parents: 16234
diff changeset
   244
}