src/java.base/share/classes/java/lang/invoke/MethodTypeForm.java
author erikj
Tue, 12 Sep 2017 19:03:39 +0200
changeset 47216 71c04702a3d5
parent 39342 jdk/src/java.base/share/classes/java/lang/invoke/MethodTypeForm.java@f66a89ed6fca
child 49935 2ace90aec488
permissions -rw-r--r--
8187443: Forest Consolidation: Move files to unified layout Reviewed-by: darcy, ihse
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
     1
/*
39342
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
     2
 * Copyright (c) 2008, 2016, Oracle and/or its affiliates. All rights reserved.
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
     4
 *
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2707
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2707
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    10
 *
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    15
 * accompanied this code).
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    16
 *
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    20
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2707
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2707
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2707
diff changeset
    23
 * questions.
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    24
 */
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    25
8822
8145ab9f5f86 7012648: move JSR 292 to package java.lang.invoke and adjust names
jrose
parents: 8821
diff changeset
    26
package java.lang.invoke;
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    27
35252
754852fba784 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 32649
diff changeset
    28
import jdk.internal.vm.annotation.Stable;
8822
8145ab9f5f86 7012648: move JSR 292 to package java.lang.invoke and adjust names
jrose
parents: 8821
diff changeset
    29
import sun.invoke.util.Wrapper;
39342
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
    30
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
    31
import java.lang.ref.SoftReference;
39342
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
    32
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
    33
import static java.lang.invoke.MethodHandleStatics.newIllegalArgumentException;
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    34
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    35
/**
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    36
 * Shared information for a group of method types, which differ
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    37
 * only by reference types, and therefore share a common erasure
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    38
 * and wrapping.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    39
 * <p>
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    40
 * For an empirical discussion of the structure of method types,
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    41
 * see <a href="http://groups.google.com/group/jvm-languages/browse_thread/thread/ac9308ae74da9b7e/">
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    42
 * the thread "Avoiding Boxing" on jvm-languages</a>.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    43
 * There are approximately 2000 distinct erased method types in the JDK.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    44
 * There are a little over 10 times that number of unerased types.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    45
 * No more than half of these are likely to be loaded at once.
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    46
 * @author John Rose
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
    47
 */
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
    48
final class MethodTypeForm {
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    49
    final int[] argToSlotTable, slotToArgTable;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    50
    final long argCounts;               // packed slot & value counts
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    51
    final long primCounts;              // packed prim & double counts
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    52
    final MethodType erasedType;        // the canonical erasure
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
    53
    final MethodType basicType;         // the canonical erasure, with primitives simplified
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    54
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    55
    // Cached adapter information:
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
    56
    @Stable final SoftReference<MethodHandle>[] methodHandles;
26468
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
    57
    // Indexes into methodHandles:
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
    58
    static final int
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
    59
            MH_BASIC_INV      =  0,  // cached instance of MH.invokeBasic
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
    60
            MH_NF_INV         =  1,  // cached helper for LF.NamedFunction
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
    61
            MH_UNINIT_CS      =  2,  // uninitialized call site
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
    62
            MH_LIMIT          =  3;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
    63
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
    64
    // Cached lambda form information, for basic types only:
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
    65
    final @Stable SoftReference<LambdaForm>[] lambdaForms;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
    66
    // Indexes into lambdaForms:
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
    67
    static final int
27295
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    68
            LF_INVVIRTUAL              =  0,  // DMH invokeVirtual
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    69
            LF_INVSTATIC               =  1,
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    70
            LF_INVSPECIAL              =  2,
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    71
            LF_NEWINVSPECIAL           =  3,
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    72
            LF_INVINTERFACE            =  4,
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    73
            LF_INVSTATIC_INIT          =  5,  // DMH invokeStatic with <clinit> barrier
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    74
            LF_INTERPRET               =  6,  // LF interpreter
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    75
            LF_REBIND                  =  7,  // BoundMethodHandle
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    76
            LF_DELEGATE                =  8,  // DelegatingMethodHandle
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    77
            LF_DELEGATE_BLOCK_INLINING =  9,  // Counting DelegatingMethodHandle w/ @DontInline
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    78
            LF_EX_LINKER               = 10,  // invokeExact_MT (for invokehandle)
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    79
            LF_EX_INVOKER              = 11,  // MHs.invokeExact
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    80
            LF_GEN_LINKER              = 12,  // generic invoke_MT (for invokehandle)
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    81
            LF_GEN_INVOKER             = 13,  // generic MHs.invoke
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    82
            LF_CS_LINKER               = 14,  // linkToCallSite_CS
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    83
            LF_MH_LINKER               = 15,  // linkToCallSite_MH
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    84
            LF_GWC                     = 16,  // guardWithCatch (catchException)
8ed31033bf48 8059877: GWT branch frequencies pollution due to LF sharing
vlivanov
parents: 26480
diff changeset
    85
            LF_GWT                     = 17,  // guardWithTest
39342
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
    86
            LF_TF                      = 18,  // tryFinally
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
    87
            LF_LOOP                    = 19,  // loop
f66a89ed6fca 8143211: provide bytecode intrinsics for loop and try/finally executors
mhaupt
parents: 35252
diff changeset
    88
            LF_LIMIT                   = 20;
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    89
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
    90
    /** Return the type corresponding uniquely (1-1) to this MT-form.
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
    91
     *  It might have any primitive returns or arguments, but will have no references except Object.
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
    92
     */
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    93
    public MethodType erasedType() {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    94
        return erasedType;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    95
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
    96
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
    97
    /** Return the basic type derived from the erased type of this MT-form.
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
    98
     *  A basic type is erased (all references Object) and also has all primitive
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
    99
     *  types (except int, long, float, double, void) normalized to int.
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   100
     *  Such basic types correspond to low-level JVM calling sequences.
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   101
     */
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   102
    public MethodType basicType() {
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   103
        return basicType;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   104
    }
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   105
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   106
    private boolean assertIsBasicType() {
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   107
        // primitives must be flattened also
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   108
        assert(erasedType == basicType)
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   109
                : "erasedType: " + erasedType + " != basicType: " + basicType;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   110
        return true;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   111
    }
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   112
26468
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   113
    public MethodHandle cachedMethodHandle(int which) {
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   114
        assert(assertIsBasicType());
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   115
        SoftReference<MethodHandle> entry = methodHandles[which];
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   116
        return (entry != null) ? entry.get() : null;
26468
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   117
    }
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   118
32649
2ee9017c7597 8136583: Core libraries should use blessed modifier order
martin
parents: 27803
diff changeset
   119
    public synchronized MethodHandle setCachedMethodHandle(int which, MethodHandle mh) {
26468
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   120
        // Simulate a CAS, to avoid racy duplication of results.
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   121
        SoftReference<MethodHandle> entry = methodHandles[which];
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   122
        if (entry != null) {
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   123
            MethodHandle prev = entry.get();
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   124
            if (prev != null) {
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   125
                return prev;
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   126
            }
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   127
        }
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   128
        methodHandles[which] = new SoftReference<>(mh);
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   129
        return mh;
26468
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   130
    }
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   131
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   132
    public LambdaForm cachedLambdaForm(int which) {
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   133
        assert(assertIsBasicType());
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   134
        SoftReference<LambdaForm> entry = lambdaForms[which];
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   135
        return (entry != null) ? entry.get() : null;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   136
    }
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   137
32649
2ee9017c7597 8136583: Core libraries should use blessed modifier order
martin
parents: 27803
diff changeset
   138
    public synchronized LambdaForm setCachedLambdaForm(int which, LambdaForm form) {
24696
00479fa7c27f 8005873: JRuby test_respond_to.rb asserts with: MT-unsafe modification of inline cache
thartmann
parents: 23038
diff changeset
   139
        // Simulate a CAS, to avoid racy duplication of results.
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   140
        SoftReference<LambdaForm> entry = lambdaForms[which];
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   141
        if (entry != null) {
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   142
            LambdaForm prev = entry.get();
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   143
            if (prev != null) {
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   144
                return prev;
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   145
            }
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   146
        }
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   147
        lambdaForms[which] = new SoftReference<>(form);
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   148
        return form;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   149
    }
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   150
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   151
    /**
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   152
     * Build an MTF for a given type, which must have all references erased to Object.
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   153
     * This MTF will stand for that type and all un-erased variations.
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   154
     * Eagerly compute some basic properties of the type, common to all variations.
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   155
     */
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   156
    @SuppressWarnings({"rawtypes", "unchecked"})
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   157
    protected MethodTypeForm(MethodType erasedType) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   158
        this.erasedType = erasedType;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   159
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   160
        Class<?>[] ptypes = erasedType.ptypes();
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   161
        int ptypeCount = ptypes.length;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   162
        int pslotCount = ptypeCount;            // temp. estimate
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   163
        int rtypeCount = 1;                     // temp. estimate
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   164
        int rslotCount = 1;                     // temp. estimate
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   165
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   166
        int[] argToSlotTab = null, slotToArgTab = null;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   167
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   168
        // Walk the argument types, looking for primitives.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   169
        int pac = 0, lac = 0, prc = 0, lrc = 0;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   170
        Class<?>[] epts = ptypes;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   171
        Class<?>[] bpts = epts;
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   172
        for (int i = 0; i < epts.length; i++) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   173
            Class<?> pt = epts[i];
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   174
            if (pt != Object.class) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   175
                ++pac;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   176
                Wrapper w = Wrapper.forPrimitiveType(pt);
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   177
                if (w.isDoubleWord())  ++lac;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   178
                if (w.isSubwordOrInt() && pt != int.class) {
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   179
                    if (bpts == epts)
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   180
                        bpts = bpts.clone();
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   181
                    bpts[i] = int.class;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   182
                }
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   183
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   184
        }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   185
        pslotCount += lac;                  // #slots = #args + #longs
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   186
        Class<?> rt = erasedType.returnType();
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   187
        Class<?> bt = rt;
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   188
        if (rt != Object.class) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   189
            ++prc;          // even void.class counts as a prim here
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   190
            Wrapper w = Wrapper.forPrimitiveType(rt);
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   191
            if (w.isDoubleWord())  ++lrc;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   192
            if (w.isSubwordOrInt() && rt != int.class)
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   193
                bt = int.class;
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   194
            // adjust #slots, #args
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   195
            if (rt == void.class)
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   196
                rtypeCount = rslotCount = 0;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   197
            else
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   198
                rslotCount += lrc;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   199
        }
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   200
        if (epts == bpts && bt == rt) {
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   201
            this.basicType = erasedType;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   202
        } else {
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   203
            this.basicType = MethodType.makeImpl(bt, bpts, true);
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   204
            // fill in rest of data from the basic type:
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   205
            MethodTypeForm that = this.basicType.form();
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   206
            assert(this != that);
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   207
            this.primCounts = that.primCounts;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   208
            this.argCounts = that.argCounts;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   209
            this.argToSlotTable = that.argToSlotTable;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   210
            this.slotToArgTable = that.slotToArgTable;
26468
2d57604f9299 8050053: Improve caching of different invokers
vlivanov
parents: 26467
diff changeset
   211
            this.methodHandles = null;
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   212
            this.lambdaForms = null;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   213
            return;
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   214
        }
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   215
        if (lac != 0) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   216
            int slot = ptypeCount + lac;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   217
            slotToArgTab = new int[slot+1];
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   218
            argToSlotTab = new int[1+ptypeCount];
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   219
            argToSlotTab[0] = slot;  // argument "-1" is past end of slots
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   220
            for (int i = 0; i < epts.length; i++) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   221
                Class<?> pt = epts[i];
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   222
                Wrapper w = Wrapper.forBasicType(pt);
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   223
                if (w.isDoubleWord())  --slot;
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   224
                --slot;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   225
                slotToArgTab[slot] = i+1; // "+1" see argSlotToParameter note
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   226
                argToSlotTab[1+i]  = slot;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   227
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   228
            assert(slot == 0);  // filled the table
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   229
        } else if (pac != 0) {
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   230
            // have primitives but no long primitives; share slot counts with generic
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   231
            assert(ptypeCount == pslotCount);
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   232
            MethodTypeForm that = MethodType.genericMethodType(ptypeCount).form();
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   233
            assert(this != that);
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   234
            slotToArgTab = that.slotToArgTable;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   235
            argToSlotTab = that.argToSlotTable;
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   236
        } else {
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   237
            int slot = ptypeCount; // first arg is deepest in stack
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   238
            slotToArgTab = new int[slot+1];
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   239
            argToSlotTab = new int[1+ptypeCount];
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   240
            argToSlotTab[0] = slot;  // argument "-1" is past end of slots
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   241
            for (int i = 0; i < ptypeCount; i++) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   242
                --slot;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   243
                slotToArgTab[slot] = i+1; // "+1" see argSlotToParameter note
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   244
                argToSlotTab[1+i]  = slot;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   245
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   246
        }
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   247
        this.primCounts = pack(lrc, prc, lac, pac);
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   248
        this.argCounts = pack(rslotCount, rtypeCount, pslotCount, ptypeCount);
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   249
        this.argToSlotTable = argToSlotTab;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   250
        this.slotToArgTable = slotToArgTab;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   251
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   252
        if (pslotCount >= 256)  throw newIllegalArgumentException("too many arguments");
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   253
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   254
        // Initialize caches, but only for basic types
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   255
        assert(basicType == erasedType);
27803
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   256
        this.lambdaForms   = new SoftReference[LF_LIMIT];
d04ca9d519ce 8057020: LambdaForm caches should support eviction
vlivanov
parents: 27295
diff changeset
   257
        this.methodHandles = new SoftReference[MH_LIMIT];
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   258
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   259
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   260
    private static long pack(int a, int b, int c, int d) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   261
        assert(((a|b|c|d) & ~0xFFFF) == 0);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   262
        long hw = ((a << 16) | b), lw = ((c << 16) | d);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   263
        return (hw << 32) | lw;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   264
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   265
    private static char unpack(long packed, int word) { // word==0 => return a, ==3 => return d
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   266
        assert(word <= 3);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   267
        return (char)(packed >> ((3-word) * 16));
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   268
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   269
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   270
    public int parameterCount() {                      // # outgoing values
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   271
        return unpack(argCounts, 3);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   272
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   273
    public int parameterSlotCount() {                  // # outgoing interpreter slots
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   274
        return unpack(argCounts, 2);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   275
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   276
    public int returnCount() {                         // = 0 (V), or 1
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   277
        return unpack(argCounts, 1);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   278
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   279
    public int returnSlotCount() {                     // = 0 (V), 2 (J/D), or 1
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   280
        return unpack(argCounts, 0);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   281
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   282
    public int primitiveParameterCount() {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   283
        return unpack(primCounts, 3);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   284
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   285
    public int longPrimitiveParameterCount() {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   286
        return unpack(primCounts, 2);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   287
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   288
    public int primitiveReturnCount() {                // = 0 (obj), or 1
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   289
        return unpack(primCounts, 1);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   290
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   291
    public int longPrimitiveReturnCount() {            // = 1 (J/D), or 0
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   292
        return unpack(primCounts, 0);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   293
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   294
    public boolean hasPrimitives() {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   295
        return primCounts != 0;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   296
    }
13423
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   297
    public boolean hasNonVoidPrimitives() {
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   298
        if (primCounts == 0)  return false;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   299
        if (primitiveParameterCount() != 0)  return true;
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   300
        return (primitiveReturnCount() != 0 && returnCount() != 0);
17843fff200d 7023639: JSR 292 method handle invocation needs a fast path for compiled code
twisti
parents: 10419
diff changeset
   301
    }
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   302
    public boolean hasLongPrimitives() {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   303
        return (longPrimitiveParameterCount() | longPrimitiveReturnCount()) != 0;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   304
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   305
    public int parameterToArgSlot(int i) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   306
        return argToSlotTable[1+i];
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   307
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   308
    public int argSlotToParameter(int argSlot) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   309
        // Note:  Empty slots are represented by zero in this table.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   310
        // Valid arguments slots contain incremented entries, so as to be non-zero.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   311
        // We return -1 the caller to mean an empty slot.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   312
        return slotToArgTable[argSlot] - 1;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   313
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   314
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   315
    static MethodTypeForm findForm(MethodType mt) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   316
        MethodType erased = canonicalize(mt, ERASE, ERASE);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   317
        if (erased == null) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   318
            // It is already erased.  Make a new MethodTypeForm.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   319
            return new MethodTypeForm(mt);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   320
        } else {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   321
            // Share the MethodTypeForm with the erased version.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   322
            return erased.form();
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   323
        }
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
   324
    }
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   325
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   326
    /** Codes for {@link #canonicalize(java.lang.Class, int)}.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   327
     * ERASE means change every reference to {@code Object}.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   328
     * WRAP means convert primitives (including {@code void} to their
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   329
     * corresponding wrapper types.  UNWRAP means the reverse of WRAP.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   330
     * INTS means convert all non-void primitive types to int or long,
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   331
     * according to size.  LONGS means convert all non-void primitives
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   332
     * to long, regardless of size.  RAW_RETURN means convert a type
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   333
     * (assumed to be a return type) to int if it is smaller than an int,
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   334
     * or if it is void.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   335
     */
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   336
    public static final int NO_CHANGE = 0, ERASE = 1, WRAP = 2, UNWRAP = 3, INTS = 4, LONGS = 5, RAW_RETURN = 6;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   337
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   338
    /** Canonicalize the types in the given method type.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   339
     * If any types change, intern the new type, and return it.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   340
     * Otherwise return null.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   341
     */
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   342
    public static MethodType canonicalize(MethodType mt, int howRet, int howArgs) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   343
        Class<?>[] ptypes = mt.ptypes();
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   344
        Class<?>[] ptc = MethodTypeForm.canonicalizeAll(ptypes, howArgs);
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   345
        Class<?> rtype = mt.returnType();
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   346
        Class<?> rtc = MethodTypeForm.canonicalize(rtype, howRet);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   347
        if (ptc == null && rtc == null) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   348
            // It is already canonical.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   349
            return null;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   350
        }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   351
        // Find the erased version of the method type:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   352
        if (rtc == null)  rtc = rtype;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   353
        if (ptc == null)  ptc = ptypes;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   354
        return MethodType.makeImpl(rtc, ptc, true);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   355
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   356
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   357
    /** Canonicalize the given return or param type.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   358
     *  Return null if the type is already canonicalized.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   359
     */
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   360
    static Class<?> canonicalize(Class<?> t, int how) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   361
        Class<?> ct;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   362
        if (t == Object.class) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   363
            // no change, ever
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   364
        } else if (!t.isPrimitive()) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   365
            switch (how) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   366
                case UNWRAP:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   367
                    ct = Wrapper.asPrimitiveType(t);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   368
                    if (ct != t)  return ct;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   369
                    break;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   370
                case RAW_RETURN:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   371
                case ERASE:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   372
                    return Object.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   373
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   374
        } else if (t == void.class) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   375
            // no change, usually
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   376
            switch (how) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   377
                case RAW_RETURN:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   378
                    return int.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   379
                case WRAP:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   380
                    return Void.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   381
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   382
        } else {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   383
            // non-void primitive
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   384
            switch (how) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   385
                case WRAP:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   386
                    return Wrapper.asWrapperType(t);
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   387
                case INTS:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   388
                    if (t == int.class || t == long.class)
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   389
                        return null;  // no change
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   390
                    if (t == double.class)
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   391
                        return long.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   392
                    return int.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   393
                case LONGS:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   394
                    if (t == long.class)
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   395
                        return null;  // no change
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   396
                    return long.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   397
                case RAW_RETURN:
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   398
                    if (t == int.class || t == long.class ||
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   399
                        t == float.class || t == double.class)
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   400
                        return null;  // no change
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   401
                    // everything else returns as an int
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   402
                    return int.class;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   403
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   404
        }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   405
        // no change; return null to signify
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   406
        return null;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   407
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   408
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   409
    /** Canonicalize each param type in the given array.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   410
     *  Return null if all types are already canonicalized.
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   411
     */
26467
d69abed3a07d 8050052: Small cleanups in java.lang.invoke code
vlivanov
parents: 25859
diff changeset
   412
    static Class<?>[] canonicalizeAll(Class<?>[] ts, int how) {
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   413
        Class<?>[] cs = null;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   414
        for (int imax = ts.length, i = 0; i < imax; i++) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   415
            Class<?> c = canonicalize(ts[i], how);
9731
d0f7a3e441c4 7044892: JSR 292: API entry points sometimes throw the wrong exceptions or doesn't throw the expected one
jrose
parents: 9646
diff changeset
   416
            if (c == void.class)
d0f7a3e441c4 7044892: JSR 292: API entry points sometimes throw the wrong exceptions or doesn't throw the expected one
jrose
parents: 9646
diff changeset
   417
                c = null;  // a Void parameter was unwrapped to void; ignore
8821
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   418
            if (c != null) {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   419
                if (cs == null)
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   420
                    cs = ts.clone();
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   421
                cs[i] = c;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   422
            }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   423
        }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   424
        return cs;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   425
    }
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   426
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   427
    @Override
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   428
    public String toString() {
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   429
        return "Form"+erasedType;
2836ee97ee27 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 5506
diff changeset
   430
    }
2707
5a17df307cbc 6829144: JSR 292 JVM features need a provisional Java API
jrose
parents:
diff changeset
   431
}