src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/InvocationPlugins.java
author dlong
Sat, 08 Dec 2018 00:56:10 -0800
changeset 52910 583fd71c47d6
parent 51436 091c0d22e735
child 54084 84f10bbf993f
permissions -rw-r--r--
8214023: Update Graal Reviewed-by: kvn
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     1
/*
52910
583fd71c47d6 8214023: Update Graal
dlong
parents: 51436
diff changeset
     2
 * Copyright (c) 2015, 2018, Oracle and/or its affiliates. All rights reserved.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     4
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     7
 * published by the Free Software Foundation.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     8
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    13
 * accompanied this code).
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    14
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    18
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    21
 * questions.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    22
 */
50858
2d3e99a72541 8205824: Update Graal
never
parents: 50609
diff changeset
    23
2d3e99a72541 8205824: Update Graal
never
parents: 50609
diff changeset
    24
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    25
package org.graalvm.compiler.nodes.graphbuilderconf;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    26
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    27
import static java.lang.String.format;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    28
import static org.graalvm.compiler.nodes.graphbuilderconf.InvocationPlugins.LateClassPlugins.CLOSED_LATE_CLASS_PLUGIN;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    29
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    30
import java.lang.reflect.Constructor;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    31
import java.lang.reflect.Method;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    32
import java.lang.reflect.Modifier;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    33
import java.lang.reflect.Type;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    34
import java.util.ArrayList;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    35
import java.util.Arrays;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    36
import java.util.Collections;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    37
import java.util.List;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    38
import java.util.Map;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    39
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    40
import jdk.internal.vm.compiler.collections.EconomicMap;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    41
import jdk.internal.vm.compiler.collections.Equivalence;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    42
import jdk.internal.vm.compiler.collections.MapCursor;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    43
import jdk.internal.vm.compiler.collections.Pair;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    44
import jdk.internal.vm.compiler.collections.UnmodifiableEconomicMap;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    45
import jdk.internal.vm.compiler.collections.UnmodifiableMapCursor;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    46
import org.graalvm.compiler.api.replacements.MethodSubstitution;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    47
import org.graalvm.compiler.api.replacements.MethodSubstitutionRegistry;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    48
import org.graalvm.compiler.bytecode.BytecodeProvider;
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
    49
import org.graalvm.compiler.core.common.SuppressFBWarnings;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    50
import org.graalvm.compiler.debug.Assertions;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    51
import org.graalvm.compiler.debug.GraalError;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    52
import org.graalvm.compiler.graph.Node;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    53
import org.graalvm.compiler.graph.iterators.NodeIterable;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    54
import org.graalvm.compiler.nodes.ValueNode;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    55
import org.graalvm.compiler.nodes.graphbuilderconf.InvocationPlugin.Receiver;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    56
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    57
import jdk.vm.ci.meta.MetaUtil;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    58
import jdk.vm.ci.meta.ResolvedJavaMethod;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    59
import jdk.vm.ci.meta.ResolvedJavaType;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    60
import jdk.vm.ci.meta.Signature;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    61
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    62
/**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    63
 * Manages a set of {@link InvocationPlugin}s.
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    64
 *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    65
 * Most plugins are registered during initialization (i.e., before
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    66
 * {@link #lookupInvocation(ResolvedJavaMethod)} or {@link #getBindings} is called). These
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    67
 * registrations can be made with {@link Registration},
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    68
 * {@link #register(InvocationPlugin, String, String, Type...)},
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    69
 * {@link #register(InvocationPlugin, Type, String, Type...)} or
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    70
 * {@link #registerOptional(InvocationPlugin, Type, String, Type...)}. Initialization is not
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    71
 * thread-safe and so must only be performed by a single thread.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    72
 *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    73
 * Plugins that are not guaranteed to be made during initialization must use
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    74
 * {@link LateRegistration}.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    75
 */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    76
public class InvocationPlugins {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    77
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    78
    public static class InvocationPluginReceiver implements InvocationPlugin.Receiver {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    79
        private final GraphBuilderContext parser;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    80
        private ValueNode[] args;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    81
        private ValueNode value;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    82
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    83
        public InvocationPluginReceiver(GraphBuilderContext parser) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    84
            this.parser = parser;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    85
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    86
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    87
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    88
        public ValueNode get(boolean performNullCheck) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    89
            assert args != null : "Cannot get the receiver of a static method";
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    90
            if (!performNullCheck) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    91
                return args[0];
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    92
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    93
            if (value == null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    94
                value = parser.nullCheckedValue(args[0]);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    95
                if (value != args[0]) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    96
                    args[0] = value;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    97
                }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    98
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    99
            return value;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   100
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   101
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   102
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   103
        public boolean isConstant() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   104
            return args[0].isConstant();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   105
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   106
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   107
        public InvocationPluginReceiver init(ResolvedJavaMethod targetMethod, ValueNode[] newArgs) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   108
            if (!targetMethod.isStatic()) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   109
                this.args = newArgs;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   110
                this.value = null;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   111
                return this;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   112
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   113
            return null;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   114
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   115
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   116
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   117
    /**
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   118
     * A symbol for an already resolved method.
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   119
     */
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   120
    public static class ResolvedJavaSymbol implements Type {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   121
        private final ResolvedJavaType resolved;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   122
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   123
        public ResolvedJavaSymbol(ResolvedJavaType type) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   124
            this.resolved = type;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   125
        }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   126
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   127
        public ResolvedJavaType getResolved() {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   128
            return resolved;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   129
        }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   130
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   131
        @Override
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   132
        public String toString() {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   133
            return resolved.toJavaName();
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   134
        }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   135
    }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   136
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   137
    /**
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   138
     * A symbol that is lazily {@linkplain OptionalLazySymbol#resolve() resolved} to a {@link Type}.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   139
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   140
    static class OptionalLazySymbol implements Type {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   141
        private static final Class<?> MASK_NULL = OptionalLazySymbol.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   142
        private final String name;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   143
        private Class<?> resolved;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   144
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   145
        OptionalLazySymbol(String name) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   146
            this.name = name;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   147
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   148
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   149
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   150
        public String getTypeName() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   151
            return name;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   152
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   153
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   154
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   155
         * Gets the resolved {@link Class} corresponding to this symbol or {@code null} if
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   156
         * resolution fails.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   157
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   158
        public Class<?> resolve() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   159
            if (resolved == null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   160
                Class<?> resolvedOrNull = resolveClass(name, true);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   161
                resolved = resolvedOrNull == null ? MASK_NULL : resolvedOrNull;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   162
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   163
            return resolved == MASK_NULL ? null : resolved;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   164
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   165
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   166
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   167
        public String toString() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   168
            return name;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   169
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   170
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   171
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   172
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   173
     * Utility for {@linkplain InvocationPlugins#register(InvocationPlugin, Class, String, Class...)
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   174
     * registration} of invocation plugins.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   175
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   176
    public static class Registration implements MethodSubstitutionRegistry {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   177
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   178
        private final InvocationPlugins plugins;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   179
        private final Type declaringType;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   180
        private final BytecodeProvider methodSubstitutionBytecodeProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   181
        private boolean allowOverwrite;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   182
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   183
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   184
        public Class<?> getReceiverType() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   185
            return Receiver.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   186
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   187
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   188
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   189
         * Creates an object for registering {@link InvocationPlugin}s for methods declared by a
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   190
         * given class.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   191
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   192
         * @param plugins where to register the plugins
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   193
         * @param declaringType the class declaring the methods for which plugins will be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   194
         *            via this object
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   195
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   196
        public Registration(InvocationPlugins plugins, Type declaringType) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   197
            this.plugins = plugins;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   198
            this.declaringType = declaringType;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   199
            this.methodSubstitutionBytecodeProvider = null;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   200
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   201
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   202
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   203
         * Creates an object for registering {@link InvocationPlugin}s for methods declared by a
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   204
         * given class.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   205
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   206
         * @param plugins where to register the plugins
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   207
         * @param declaringType the class declaring the methods for which plugins will be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   208
         *            via this object
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   209
         * @param methodSubstitutionBytecodeProvider provider used to get the bytecodes to parse for
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   210
         *            method substitutions
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   211
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   212
        public Registration(InvocationPlugins plugins, Type declaringType, BytecodeProvider methodSubstitutionBytecodeProvider) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   213
            this.plugins = plugins;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   214
            this.declaringType = declaringType;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   215
            this.methodSubstitutionBytecodeProvider = methodSubstitutionBytecodeProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   216
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   217
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   218
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   219
         * Creates an object for registering {@link InvocationPlugin}s for methods declared by a
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   220
         * given class.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   221
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   222
         * @param plugins where to register the plugins
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   223
         * @param declaringClassName the name of the class class declaring the methods for which
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   224
         *            plugins will be registered via this object
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   225
         * @param methodSubstitutionBytecodeProvider provider used to get the bytecodes to parse for
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   226
         *            method substitutions
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   227
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   228
        public Registration(InvocationPlugins plugins, String declaringClassName, BytecodeProvider methodSubstitutionBytecodeProvider) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   229
            this.plugins = plugins;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   230
            this.declaringType = new OptionalLazySymbol(declaringClassName);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   231
            this.methodSubstitutionBytecodeProvider = methodSubstitutionBytecodeProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   232
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   233
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   234
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   235
         * Configures this registration to allow or disallow overwriting of invocation plugins.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   236
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   237
        public Registration setAllowOverwrite(boolean allowOverwrite) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   238
            this.allowOverwrite = allowOverwrite;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   239
            return this;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   240
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   241
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   242
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   243
         * Registers a plugin for a method with no arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   244
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   245
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   246
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   247
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   248
        public void register0(String name, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   249
            plugins.register(plugin, false, allowOverwrite, declaringType, name);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   250
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   251
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   252
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   253
         * Registers a plugin for a method with 1 argument.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   254
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   255
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   256
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   257
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   258
        public void register1(String name, Type arg, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   259
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   260
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   261
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   262
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   263
         * Registers a plugin for a method with 2 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   264
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   265
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   266
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   267
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   268
        public void register2(String name, Type arg1, Type arg2, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   269
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg1, arg2);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   270
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   271
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   272
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   273
         * Registers a plugin for a method with 3 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   274
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   275
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   276
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   277
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   278
        public void register3(String name, Type arg1, Type arg2, Type arg3, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   279
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg1, arg2, arg3);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   280
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   281
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   282
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   283
         * Registers a plugin for a method with 4 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   284
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   285
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   286
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   287
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   288
        public void register4(String name, Type arg1, Type arg2, Type arg3, Type arg4, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   289
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg1, arg2, arg3, arg4);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   290
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   291
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   292
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   293
         * Registers a plugin for a method with 5 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   294
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   295
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   296
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   297
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   298
        public void register5(String name, Type arg1, Type arg2, Type arg3, Type arg4, Type arg5, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   299
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg1, arg2, arg3, arg4, arg5);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   300
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   301
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   302
        /**
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   303
         * Registers a plugin for a method with 6 arguments.
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   304
         *
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   305
         * @param name the name of the method
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   306
         * @param plugin the plugin to be registered
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   307
         */
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   308
        public void register6(String name, Type arg1, Type arg2, Type arg3, Type arg4, Type arg5, Type arg6, InvocationPlugin plugin) {
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   309
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg1, arg2, arg3, arg4, arg5, arg6);
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   310
        }
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   311
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   312
        /**
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   313
         * Registers a plugin for a method with 7 arguments.
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   314
         *
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   315
         * @param name the name of the method
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   316
         * @param plugin the plugin to be registered
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   317
         */
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   318
        public void register7(String name, Type arg1, Type arg2, Type arg3, Type arg4, Type arg5, Type arg6, Type arg7, InvocationPlugin plugin) {
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   319
            plugins.register(plugin, false, allowOverwrite, declaringType, name, arg1, arg2, arg3, arg4, arg5, arg6, arg7);
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   320
        }
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   321
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   322
        /**
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   323
         * Registers a plugin for an optional method with no arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   324
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   325
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   326
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   327
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   328
        public void registerOptional0(String name, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   329
            plugins.register(plugin, true, allowOverwrite, declaringType, name);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   330
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   331
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   332
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   333
         * Registers a plugin for an optional method with 1 argument.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   334
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   335
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   336
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   337
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   338
        public void registerOptional1(String name, Type arg, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   339
            plugins.register(plugin, true, allowOverwrite, declaringType, name, arg);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   340
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   341
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   342
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   343
         * Registers a plugin for an optional method with 2 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   344
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   345
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   346
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   347
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   348
        public void registerOptional2(String name, Type arg1, Type arg2, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   349
            plugins.register(plugin, true, allowOverwrite, declaringType, name, arg1, arg2);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   350
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   351
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   352
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   353
         * Registers a plugin for an optional method with 3 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   354
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   355
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   356
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   357
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   358
        public void registerOptional3(String name, Type arg1, Type arg2, Type arg3, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   359
            plugins.register(plugin, true, allowOverwrite, declaringType, name, arg1, arg2, arg3);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   360
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   361
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   362
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   363
         * Registers a plugin for an optional method with 4 arguments.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   364
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   365
         * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   366
         * @param plugin the plugin to be registered
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   367
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   368
        public void registerOptional4(String name, Type arg1, Type arg2, Type arg3, Type arg4, InvocationPlugin plugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   369
            plugins.register(plugin, true, allowOverwrite, declaringType, name, arg1, arg2, arg3, arg4);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   370
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   371
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   372
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   373
         * Registers a plugin that implements a method based on the bytecode of a substitute method.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   374
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   375
         * @param substituteDeclaringClass the class declaring the substitute method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   376
         * @param name the name of both the original and substitute method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   377
         * @param argumentTypes the argument types of the method. Element 0 of this array must be
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   378
         *            the {@link Class} value for {@link InvocationPlugin.Receiver} iff the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   379
         *            is non-static. Upon returning, element 0 will have been rewritten to
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   380
         *            {@code declaringClass}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   381
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   382
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   383
        public void registerMethodSubstitution(Class<?> substituteDeclaringClass, String name, Type... argumentTypes) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   384
            registerMethodSubstitution(substituteDeclaringClass, name, name, argumentTypes);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   385
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   386
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   387
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   388
         * Registers a plugin that implements a method based on the bytecode of a substitute method.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   389
         *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   390
         * @param substituteDeclaringClass the class declaring the substitute method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   391
         * @param name the name of both the original method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   392
         * @param substituteName the name of the substitute method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   393
         * @param argumentTypes the argument types of the method. Element 0 of this array must be
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   394
         *            the {@link Class} value for {@link InvocationPlugin.Receiver} iff the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   395
         *            is non-static. Upon returning, element 0 will have been rewritten to
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   396
         *            {@code declaringClass}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   397
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   398
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   399
        public void registerMethodSubstitution(Class<?> substituteDeclaringClass, String name, String substituteName, Type... argumentTypes) {
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   400
            MethodSubstitutionPlugin plugin = createMethodSubstitution(substituteDeclaringClass, substituteName, argumentTypes);
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   401
            plugins.register(plugin, false, allowOverwrite, declaringType, name, argumentTypes);
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   402
        }
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   403
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   404
        public MethodSubstitutionPlugin createMethodSubstitution(Class<?> substituteDeclaringClass, String substituteName, Type... argumentTypes) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   405
            assert methodSubstitutionBytecodeProvider != null : "Registration used for method substitutions requires a non-null methodSubstitutionBytecodeProvider";
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   406
            MethodSubstitutionPlugin plugin = new MethodSubstitutionPlugin(methodSubstitutionBytecodeProvider, substituteDeclaringClass, substituteName, argumentTypes);
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   407
            return plugin;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   408
        }
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   409
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   410
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   411
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   412
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   413
     * Utility for registering plugins after Graal may have been initialized. Registrations made via
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   414
     * this class are not finalized until {@link #close} is called.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   415
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   416
    public static class LateRegistration implements AutoCloseable {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   417
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   418
        private InvocationPlugins plugins;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   419
        private final List<Binding> bindings = new ArrayList<>();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   420
        private final Type declaringType;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   421
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   422
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   423
         * Creates an object for registering {@link InvocationPlugin}s for methods declared by a
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   424
         * given class.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   425
         *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   426
         * @param plugins where to register the plugins
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   427
         * @param declaringType the class declaring the methods for which plugins will be registered
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   428
         *            via this object
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   429
         */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   430
        public LateRegistration(InvocationPlugins plugins, Type declaringType) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   431
            this.plugins = plugins;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   432
            this.declaringType = declaringType;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   433
        }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   434
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   435
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   436
         * Registers an invocation plugin for a given method. There must be no plugin currently
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   437
         * registered for {@code method}.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   438
         *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   439
         * @param argumentTypes the argument types of the method. Element 0 of this array must be
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   440
         *            the {@link Class} value for {@link InvocationPlugin.Receiver} iff the method
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   441
         *            is non-static. Upon returning, element 0 will have been rewritten to
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   442
         *            {@code declaringClass}
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   443
         */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   444
        public void register(InvocationPlugin plugin, String name, Type... argumentTypes) {
51436
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   445
            assert plugins != null : String.format("Late registrations of invocation plugins for %s is already closed", declaringType);
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   446
            boolean isStatic = argumentTypes.length == 0 || argumentTypes[0] != InvocationPlugin.Receiver.class;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   447
            if (!isStatic) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   448
                argumentTypes[0] = declaringType;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   449
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   450
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   451
            assert isStatic || argumentTypes[0] == declaringType;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   452
            Binding binding = new Binding(plugin, isStatic, name, argumentTypes);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   453
            bindings.add(binding);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   454
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   455
            assert Checks.check(this.plugins, declaringType, binding);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   456
            assert Checks.checkResolvable(false, declaringType, binding);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   457
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   458
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   459
        @Override
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   460
        public void close() {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   461
            assert plugins != null : String.format("Late registrations of invocation plugins for %s is already closed", declaringType);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   462
            plugins.registerLate(declaringType, bindings);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   463
            plugins = null;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   464
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   465
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   466
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   467
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   468
     * Associates an {@link InvocationPlugin} with the details of a method it substitutes.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   469
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   470
    public static class Binding {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   471
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   472
         * The plugin this binding is for.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   473
         */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   474
        public final InvocationPlugin plugin;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   475
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   476
        /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   477
         * Specifies if the associated method is static.
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   478
         */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   479
        public final boolean isStatic;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   480
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   481
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   482
         * The name of the associated method.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   483
         */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   484
        public final String name;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   485
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   486
        /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   487
         * A partial
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   488
         * <a href="http://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.3.3">method
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   489
         * descriptor</a> for the associated method. The descriptor includes enclosing {@code '('}
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   490
         * and {@code ')'} characters but omits the return type suffix.
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   491
         */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   492
        public final String argumentsDescriptor;
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   493
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   494
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   495
         * Link in a list of bindings.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   496
         */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   497
        private Binding next;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   498
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   499
        Binding(InvocationPlugin data, boolean isStatic, String name, Type... argumentTypes) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   500
            this.plugin = data;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   501
            this.isStatic = isStatic;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   502
            this.name = name;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   503
            StringBuilder buf = new StringBuilder();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   504
            buf.append('(');
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   505
            for (int i = isStatic ? 0 : 1; i < argumentTypes.length; i++) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   506
                buf.append(MetaUtil.toInternalName(argumentTypes[i].getTypeName()));
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   507
            }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   508
            buf.append(')');
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   509
            this.argumentsDescriptor = buf.toString();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   510
            assert !name.equals("<init>") || !isStatic : this;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   511
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   512
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   513
        Binding(ResolvedJavaMethod resolved, InvocationPlugin data) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   514
            this.plugin = data;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   515
            this.isStatic = resolved.isStatic();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   516
            this.name = resolved.getName();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   517
            Signature sig = resolved.getSignature();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   518
            String desc = sig.toMethodDescriptor();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   519
            assert desc.indexOf(')') != -1 : desc;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   520
            this.argumentsDescriptor = desc.substring(0, desc.indexOf(')') + 1);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   521
            assert !name.equals("<init>") || !isStatic : this;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   522
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   523
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   524
        @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   525
        public String toString() {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   526
            return name + argumentsDescriptor;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   527
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   528
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   529
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   530
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   531
     * Plugin registrations for already resolved methods. If non-null, then {@link #registrations}
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   532
     * is null and no further registrations can be made.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   533
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   534
    private final UnmodifiableEconomicMap<ResolvedJavaMethod, InvocationPlugin> resolvedRegistrations;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   535
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   536
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   537
     * Map from class names in {@linkplain MetaUtil#toInternalName(String) internal} form to the
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   538
     * invocation plugin bindings for the class. Tf non-null, then {@link #resolvedRegistrations}
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   539
     * will be null.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   540
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   541
    private final EconomicMap<String, ClassPlugins> registrations;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   542
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   543
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   544
     * Deferred registrations as well as the guard for delimiting the initial registration phase.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   545
     * The guard uses double-checked locking which is why this field is {@code volatile}.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   546
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   547
    private volatile List<Runnable> deferredRegistrations = new ArrayList<>();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   548
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   549
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   550
     * Adds a {@link Runnable} for doing registration deferred until the first time
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   551
     * {@link #get(ResolvedJavaMethod)} or {@link #closeRegistration()} is called on this object.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   552
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   553
    public void defer(Runnable deferrable) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   554
        assert deferredRegistrations != null : "registration is closed";
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   555
        deferredRegistrations.add(deferrable);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   556
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   557
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   558
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   559
     * Support for registering plugins once this object may be accessed by multiple threads.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   560
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   561
    private volatile LateClassPlugins lateRegistrations;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   562
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   563
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   564
     * Per-class bindings.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   565
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   566
    static class ClassPlugins {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   567
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   568
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   569
         * Maps method names to binding lists.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   570
         */
51436
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   571
        final EconomicMap<String, Binding> bindings = EconomicMap.create(Equivalence.DEFAULT);
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   572
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   573
        /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   574
         * Gets the invocation plugin for a given method.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   575
         *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   576
         * @return the invocation plugin for {@code method} or {@code null}
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   577
         */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   578
        InvocationPlugin get(ResolvedJavaMethod method) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   579
            assert !method.isBridge();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   580
            Binding binding = bindings.get(method.getName());
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   581
            while (binding != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   582
                if (method.isStatic() == binding.isStatic) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   583
                    if (method.getSignature().toMethodDescriptor().startsWith(binding.argumentsDescriptor)) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   584
                        return binding.plugin;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   585
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   586
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   587
                binding = binding.next;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   588
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   589
            return null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   590
        }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   591
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   592
        public void register(Binding binding, boolean allowOverwrite) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   593
            if (allowOverwrite) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   594
                if (lookup(binding) != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   595
                    register(binding);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   596
                    return;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   597
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   598
            } else {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   599
                assert lookup(binding) == null : "a value is already registered for " + binding;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   600
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   601
            register(binding);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   602
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   603
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   604
        InvocationPlugin lookup(Binding binding) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   605
            Binding b = bindings.get(binding.name);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   606
            while (b != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   607
                if (b.isStatic == binding.isStatic && b.argumentsDescriptor.equals(binding.argumentsDescriptor)) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   608
                    return b.plugin;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   609
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   610
                b = b.next;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   611
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   612
            return null;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   613
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   614
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   615
        /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   616
         * Registers {@code binding}.
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   617
         */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   618
        void register(Binding binding) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   619
            Binding head = bindings.get(binding.name);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   620
            assert binding.next == null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   621
            binding.next = head;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   622
            bindings.put(binding.name, binding);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   623
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   624
    }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   625
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   626
    static class LateClassPlugins extends ClassPlugins {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   627
        static final String CLOSED_LATE_CLASS_PLUGIN = "-----";
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   628
        private final String className;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   629
        private final LateClassPlugins next;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   630
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   631
        LateClassPlugins(LateClassPlugins next, String className) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   632
            assert next == null || next.className != CLOSED_LATE_CLASS_PLUGIN : "Late registration of invocation plugins is closed";
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   633
            this.next = next;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   634
            this.className = className;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   635
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   636
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   637
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   638
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   639
     * Registers a binding of a method to an invocation plugin.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   640
     *
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   641
     * @param plugin invocation plugin to be associated with the specified method
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   642
     * @param isStatic specifies if the method is static
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   643
     * @param declaringClass the class declaring the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   644
     * @param name the name of the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   645
     * @param argumentTypes the argument types of the method. Element 0 of this array must be
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   646
     *            {@code declaringClass} iff the method is non-static.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   647
     * @return an object representing the method
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   648
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   649
    Binding put(InvocationPlugin plugin, boolean isStatic, boolean allowOverwrite, Type declaringClass, String name, Type... argumentTypes) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   650
        assert resolvedRegistrations == null : "registration is closed";
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   651
        String internalName = MetaUtil.toInternalName(declaringClass.getTypeName());
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   652
        assert isStatic || argumentTypes[0] == declaringClass;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   653
        assert deferredRegistrations != null : "initial registration is closed - use " + LateRegistration.class.getName() + " for late registrations";
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   654
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   655
        ClassPlugins classPlugins = registrations.get(internalName);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   656
        if (classPlugins == null) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   657
            classPlugins = new ClassPlugins();
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   658
            registrations.put(internalName, classPlugins);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   659
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   660
        Binding binding = new Binding(plugin, isStatic, name, argumentTypes);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   661
        classPlugins.register(binding, allowOverwrite);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   662
        return binding;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   663
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   664
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   665
    InvocationPlugin get(ResolvedJavaMethod method) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   666
        if (resolvedRegistrations != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   667
            return resolvedRegistrations.get(method);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   668
        } else {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   669
            if (!method.isBridge()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   670
                ResolvedJavaType declaringClass = method.getDeclaringClass();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   671
                flushDeferrables();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   672
                String internalName = declaringClass.getName();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   673
                ClassPlugins classPlugins = registrations.get(internalName);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   674
                InvocationPlugin res = null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   675
                if (classPlugins != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   676
                    res = classPlugins.get(method);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   677
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   678
                if (res == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   679
                    LateClassPlugins lcp = findLateClassPlugins(internalName);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   680
                    if (lcp != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   681
                        res = lcp.get(method);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   682
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   683
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   684
                if (res != null) {
50330
2cbc42a5764b 8202670: Update Graal
dlong
parents: 49873
diff changeset
   685
                    // A decorator plugin is trusted since it does not replace
2cbc42a5764b 8202670: Update Graal
dlong
parents: 49873
diff changeset
   686
                    // the method it intrinsifies.
2cbc42a5764b 8202670: Update Graal
dlong
parents: 49873
diff changeset
   687
                    if (res.isDecorator() || canBeIntrinsified(declaringClass)) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   688
                        return res;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   689
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   690
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   691
                if (testExtensions != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   692
                    // Avoid the synchronization in the common case that there
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   693
                    // are no test extensions.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   694
                    synchronized (this) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   695
                        if (testExtensions != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   696
                            List<Binding> bindings = testExtensions.get(internalName);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   697
                            if (bindings != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   698
                                String name = method.getName();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   699
                                String descriptor = method.getSignature().toMethodDescriptor();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   700
                                for (Binding b : bindings) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   701
                                    if (b.isStatic == method.isStatic() &&
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   702
                                                    b.name.equals(name) &&
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   703
                                                    descriptor.startsWith(b.argumentsDescriptor)) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   704
                                        return b.plugin;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   705
                                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   706
                                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   707
                            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   708
                        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   709
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   710
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   711
            } else {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   712
                // Supporting plugins for bridge methods would require including
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   713
                // the return type in the registered signature. Until needed,
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   714
                // this extra complexity is best avoided.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   715
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   716
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   717
        return null;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   718
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   719
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   720
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   721
     * Determines if methods in a given class can have invocation plugins.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   722
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   723
     * @param declaringClass the class to test
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   724
     */
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
   725
    public boolean canBeIntrinsified(ResolvedJavaType declaringClass) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   726
        return true;
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   727
    }
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   728
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   729
    LateClassPlugins findLateClassPlugins(String internalClassName) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   730
        for (LateClassPlugins lcp = lateRegistrations; lcp != null; lcp = lcp.next) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   731
            if (lcp.className.equals(internalClassName)) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   732
                return lcp;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   733
            }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   734
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   735
        return null;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   736
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   737
46509
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   738
    @SuppressWarnings("serial")
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   739
    static class InvocationPlugRegistrationError extends GraalError {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   740
        InvocationPlugRegistrationError(Throwable cause) {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   741
            super(cause);
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   742
        }
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   743
    }
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   744
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   745
    private void flushDeferrables() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   746
        if (deferredRegistrations != null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   747
            synchronized (this) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   748
                if (deferredRegistrations != null) {
46509
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   749
                    try {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   750
                        for (Runnable deferrable : deferredRegistrations) {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   751
                            deferrable.run();
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   752
                        }
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   753
                        deferredRegistrations = null;
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   754
                    } catch (InvocationPlugRegistrationError t) {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   755
                        throw t;
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   756
                    } catch (Throwable t) {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   757
                        /*
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   758
                         * Something went wrong during registration but it's possible we'll end up
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   759
                         * coming back into this code. nulling out deferredRegistrations would just
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   760
                         * cause other things to break and rerunning them would cause errors about
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   761
                         * already registered plugins, so rethrow the original exception during
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   762
                         * later invocations.
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   763
                         */
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   764
                        deferredRegistrations.clear();
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   765
                        Runnable rethrow = new Runnable() {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   766
                            @Override
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   767
                            public void run() {
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   768
                                throw new InvocationPlugRegistrationError(t);
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   769
                            }
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   770
                        };
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   771
                        deferredRegistrations.add(rethrow);
b32d3928ad6a 8181369: Update Graal
iveresov
parents: 46459
diff changeset
   772
                        rethrow.run();
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   773
                    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   774
                }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   775
            }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   776
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   777
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   778
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   779
    private volatile EconomicMap<String, List<Binding>> testExtensions;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   780
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   781
    private static int findBinding(List<Binding> list, Binding key) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   782
        for (int i = 0; i < list.size(); i++) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   783
            Binding b = list.get(i);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   784
            if (b.isStatic == key.isStatic && b.name.equals(key.name) && b.argumentsDescriptor.equals(key.argumentsDescriptor)) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   785
                return i;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   786
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   787
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   788
        return -1;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   789
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   790
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   791
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   792
     * Extends the plugins in this object with those from {@code other}. The added plugins should be
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   793
     * {@linkplain #removeTestPlugins(InvocationPlugins) removed} after the test.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   794
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   795
     * This extension mechanism exists only for tests that want to add extra invocation plugins
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   796
     * after the compiler has been initialized.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   797
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   798
     * @param ignored if non-null, the bindings from {@code other} already in this object prior to
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   799
     *            calling this method are added to this list. These bindings are not added to this
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   800
     *            object.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   801
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   802
    public synchronized void addTestPlugins(InvocationPlugins other, List<Pair<String, Binding>> ignored) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   803
        assert resolvedRegistrations == null : "registration is closed";
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   804
        EconomicMap<String, List<Binding>> otherBindings = other.getBindings(true, false);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   805
        if (otherBindings.isEmpty()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   806
            return;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   807
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   808
        if (testExtensions == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   809
            testExtensions = EconomicMap.create();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   810
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   811
        MapCursor<String, List<Binding>> c = otherBindings.getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   812
        while (c.advance()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   813
            String declaringClass = c.getKey();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   814
            List<Binding> bindings = testExtensions.get(declaringClass);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   815
            if (bindings == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   816
                bindings = new ArrayList<>();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   817
                testExtensions.put(declaringClass, bindings);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   818
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   819
            for (Binding b : c.getValue()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   820
                int index = findBinding(bindings, b);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   821
                if (index != -1) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   822
                    if (ignored != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   823
                        ignored.add(Pair.create(declaringClass, b));
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   824
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   825
                } else {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   826
                    bindings.add(b);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   827
                }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   828
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   829
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   830
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   831
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   832
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   833
     * Removes the plugins from {@code other} in this object that were added by
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   834
     * {@link #addTestPlugins}.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   835
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   836
    public synchronized void removeTestPlugins(InvocationPlugins other) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   837
        assert resolvedRegistrations == null : "registration is closed";
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   838
        if (testExtensions != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   839
            MapCursor<String, List<Binding>> c = other.getBindings(false).getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   840
            while (c.advance()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   841
                String declaringClass = c.getKey();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   842
                List<Binding> bindings = testExtensions.get(declaringClass);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   843
                if (bindings != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   844
                    for (Binding b : c.getValue()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   845
                        int index = findBinding(bindings, b);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   846
                        if (index != -1) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   847
                            bindings.remove(index);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   848
                        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   849
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   850
                    if (bindings.isEmpty()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   851
                        testExtensions.removeKey(declaringClass);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   852
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   853
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   854
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   855
            if (testExtensions.isEmpty()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   856
                testExtensions = null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   857
            }
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   858
        }
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   859
    }
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
   860
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   861
    synchronized void registerLate(Type declaringType, List<Binding> bindings) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   862
        String internalName = MetaUtil.toInternalName(declaringType.getTypeName());
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   863
        assert findLateClassPlugins(internalName) == null : "Cannot have more than one late registration of invocation plugins for " + internalName;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   864
        LateClassPlugins lateClassPlugins = new LateClassPlugins(lateRegistrations, internalName);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   865
        for (Binding b : bindings) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   866
            lateClassPlugins.register(b);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   867
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   868
        lateRegistrations = lateClassPlugins;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   869
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   870
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 48861
diff changeset
   871
    @SuppressFBWarnings(value = "ES_COMPARING_STRINGS_WITH_EQ", justification = "string literal object identity used as sentinel")
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   872
    private synchronized boolean closeLateRegistrations() {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   873
        if (lateRegistrations == null || lateRegistrations.className != CLOSED_LATE_CLASS_PLUGIN) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   874
            lateRegistrations = new LateClassPlugins(lateRegistrations, CLOSED_LATE_CLASS_PLUGIN);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   875
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   876
        return true;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   877
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   878
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   879
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   880
     * Processes deferred registrations and then closes this object for future registration.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   881
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   882
    public void closeRegistration() {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   883
        assert closeLateRegistrations();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   884
        flushDeferrables();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   885
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   886
51436
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   887
    /**
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   888
     * Determines if this object currently contains any plugins (in any state of registration). If
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   889
     * this object has any {@link #defer(Runnable) deferred registrations}, it is assumed that
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   890
     * executing them will result in at least one plugin being registered.
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   891
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   892
    public boolean isEmpty() {
51436
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   893
        if (parent != null && !parent.isEmpty()) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   894
            return false;
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   895
        }
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   896
        UnmodifiableEconomicMap<ResolvedJavaMethod, InvocationPlugin> resolvedRegs = resolvedRegistrations;
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   897
        if (resolvedRegs != null) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   898
            if (!resolvedRegs.isEmpty()) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   899
                return false;
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   900
            }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   901
        }
51436
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   902
        List<Runnable> deferred = deferredRegistrations;
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   903
        if (deferred != null) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   904
            if (!deferred.isEmpty()) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   905
                return false;
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   906
            }
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   907
        }
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   908
        for (LateClassPlugins late = lateRegistrations; late != null; late = late.next) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   909
            if (!late.bindings.isEmpty()) {
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   910
                return false;
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   911
            }
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   912
        }
091c0d22e735 8206992: Update Graal
iveresov
parents: 50858
diff changeset
   913
        return registrations.size() == 0;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   914
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   915
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   916
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   917
     * The plugins {@linkplain #lookupInvocation(ResolvedJavaMethod) searched} before searching in
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   918
     * this object.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   919
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   920
    protected final InvocationPlugins parent;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   921
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   922
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   923
     * Creates a set of invocation plugins with no parent.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   924
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   925
    public InvocationPlugins() {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   926
        this(null);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   927
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   928
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   929
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   930
     * Creates a set of invocation plugins.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   931
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   932
     * @param parent if non-null, this object will be searched first when looking up plugins
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   933
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   934
    public InvocationPlugins(InvocationPlugins parent) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   935
        InvocationPlugins p = parent;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   936
        this.parent = p;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   937
        this.registrations = EconomicMap.create();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   938
        this.resolvedRegistrations = null;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   939
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   940
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   941
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   942
     * Creates a closed set of invocation plugins for a set of resolved methods. Such an object
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   943
     * cannot have further plugins registered.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   944
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   945
    public InvocationPlugins(Map<ResolvedJavaMethod, InvocationPlugin> plugins, InvocationPlugins parent) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   946
        this.parent = parent;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   947
        this.registrations = null;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   948
        this.deferredRegistrations = null;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   949
        EconomicMap<ResolvedJavaMethod, InvocationPlugin> map = EconomicMap.create(plugins.size());
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   950
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   951
        for (Map.Entry<ResolvedJavaMethod, InvocationPlugin> entry : plugins.entrySet()) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   952
            map.put(entry.getKey(), entry.getValue());
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   953
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   954
        this.resolvedRegistrations = map;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   955
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   956
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   957
    protected void register(InvocationPlugin plugin, boolean isOptional, boolean allowOverwrite, Type declaringClass, String name, Type... argumentTypes) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   958
        boolean isStatic = argumentTypes.length == 0 || argumentTypes[0] != InvocationPlugin.Receiver.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   959
        if (!isStatic) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   960
            argumentTypes[0] = declaringClass;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   961
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   962
        Binding binding = put(plugin, isStatic, allowOverwrite, declaringClass, name, argumentTypes);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   963
        assert Checks.check(this, declaringClass, binding);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   964
        assert Checks.checkResolvable(isOptional, declaringClass, binding);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   965
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   966
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   967
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   968
     * Registers an invocation plugin for a given method. There must be no plugin currently
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   969
     * registered for {@code method}.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   970
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   971
     * @param argumentTypes the argument types of the method. Element 0 of this array must be the
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   972
     *            {@link Class} value for {@link InvocationPlugin.Receiver} iff the method is
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   973
     *            non-static. Upon returning, element 0 will have been rewritten to
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   974
     *            {@code declaringClass}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   975
     */
50609
bf414874c28f 8204231: Update Graal
dlong
parents: 50330
diff changeset
   976
    public final void register(InvocationPlugin plugin, Type declaringClass, String name, Type... argumentTypes) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   977
        register(plugin, false, false, declaringClass, name, argumentTypes);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   978
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   979
50609
bf414874c28f 8204231: Update Graal
dlong
parents: 50330
diff changeset
   980
    public final void register(InvocationPlugin plugin, String declaringClass, String name, Type... argumentTypes) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   981
        register(plugin, false, false, new OptionalLazySymbol(declaringClass), name, argumentTypes);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   982
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   983
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   984
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   985
     * Registers an invocation plugin for a given, optional method. There must be no plugin
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   986
     * currently registered for {@code method}.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   987
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   988
     * @param argumentTypes the argument types of the method. Element 0 of this array must be the
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   989
     *            {@link Class} value for {@link InvocationPlugin.Receiver} iff the method is
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   990
     *            non-static. Upon returning, element 0 will have been rewritten to
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   991
     *            {@code declaringClass}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   992
     */
50609
bf414874c28f 8204231: Update Graal
dlong
parents: 50330
diff changeset
   993
    public final void registerOptional(InvocationPlugin plugin, Type declaringClass, String name, Type... argumentTypes) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   994
        register(plugin, true, false, declaringClass, name, argumentTypes);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   995
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   996
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   997
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   998
     * Gets the plugin for a given method.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   999
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1000
     * @param method the method to lookup
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1001
     * @return the plugin associated with {@code method} or {@code null} if none exists
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1002
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1003
    public InvocationPlugin lookupInvocation(ResolvedJavaMethod method) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1004
        if (parent != null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1005
            InvocationPlugin plugin = parent.lookupInvocation(method);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1006
            if (plugin != null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1007
                return plugin;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1008
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1009
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1010
        return get(method);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1011
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1012
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1013
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1014
     * Gets the set of registered invocation plugins.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1015
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1016
     * @return a map from class names in {@linkplain MetaUtil#toInternalName(String) internal} form
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1017
     *         to the invocation plugin bindings for methods in the class
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1018
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1019
    public EconomicMap<String, List<Binding>> getBindings(boolean includeParents) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1020
        return getBindings(includeParents, true);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1021
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1022
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1023
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1024
     * Gets the set of registered invocation plugins.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1025
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1026
     * @return a map from class names in {@linkplain MetaUtil#toInternalName(String) internal} form
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1027
     *         to the invocation plugin bindings for methods in the class
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1028
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1029
    private EconomicMap<String, List<Binding>> getBindings(boolean includeParents, boolean flushDeferrables) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1030
        EconomicMap<String, List<Binding>> res = EconomicMap.create(Equivalence.DEFAULT);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1031
        if (parent != null && includeParents) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1032
            res.putAll(parent.getBindings(true, flushDeferrables));
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1033
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1034
        if (resolvedRegistrations != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1035
            UnmodifiableMapCursor<ResolvedJavaMethod, InvocationPlugin> cursor = resolvedRegistrations.getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1036
            while (cursor.advance()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1037
                ResolvedJavaMethod method = cursor.getKey();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1038
                InvocationPlugin plugin = cursor.getValue();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1039
                String type = method.getDeclaringClass().getName();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1040
                List<Binding> bindings = res.get(type);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1041
                if (bindings == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1042
                    bindings = new ArrayList<>();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1043
                    res.put(type, bindings);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1044
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1045
                bindings.add(new Binding(method, plugin));
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1046
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1047
        } else {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1048
            if (flushDeferrables) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1049
                flushDeferrables();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1050
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1051
            MapCursor<String, ClassPlugins> classes = registrations.getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1052
            while (classes.advance()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1053
                String type = classes.getKey();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1054
                ClassPlugins cp = classes.getValue();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1055
                collectBindingsTo(res, type, cp);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1056
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1057
            for (LateClassPlugins lcp = lateRegistrations; lcp != null; lcp = lcp.next) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1058
                String type = lcp.className;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1059
                collectBindingsTo(res, type, lcp);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1060
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1061
            if (testExtensions != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1062
                // Avoid the synchronization in the common case that there
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1063
                // are no test extensions.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1064
                synchronized (this) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1065
                    if (testExtensions != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1066
                        MapCursor<String, List<Binding>> c = testExtensions.getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1067
                        while (c.advance()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1068
                            String name = c.getKey();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1069
                            List<Binding> bindings = res.get(name);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1070
                            if (bindings == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1071
                                bindings = new ArrayList<>();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1072
                                res.put(name, bindings);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1073
                            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1074
                            bindings.addAll(c.getValue());
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1075
                        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1076
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1077
                }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1078
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1079
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1080
        return res;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1081
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1082
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1083
    private static void collectBindingsTo(EconomicMap<String, List<Binding>> res, String type, ClassPlugins cp) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1084
        MapCursor<String, Binding> methods = cp.bindings.getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1085
        while (methods.advance()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1086
            List<Binding> bindings = res.get(type);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1087
            if (bindings == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1088
                bindings = new ArrayList<>();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1089
                res.put(type, bindings);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1090
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1091
            for (Binding b = methods.getValue(); b != null; b = b.next) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1092
                bindings.add(b);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1093
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1094
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1095
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1096
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1097
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1098
     * Gets the invocation plugins {@linkplain #lookupInvocation(ResolvedJavaMethod) searched}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1099
     * before searching in this object.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1100
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1101
    public InvocationPlugins getParent() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1102
        return parent;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1103
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1104
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1105
    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1106
    public String toString() {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1107
        UnmodifiableMapCursor<String, List<Binding>> entries = getBindings(false, false).getEntries();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1108
        List<String> all = new ArrayList<>();
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents: 43972
diff changeset
  1109
        while (entries.advance()) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1110
            String c = MetaUtil.internalNameToJava(entries.getKey(), true, false);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1111
            for (Binding b : entries.getValue()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1112
                all.add(c + '.' + b);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1113
            }
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
  1114
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1115
        Collections.sort(all);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1116
        StringBuilder buf = new StringBuilder();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1117
        String nl = String.format("%n");
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1118
        for (String s : all) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1119
            if (buf.length() != 0) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1120
                buf.append(nl);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1121
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1122
            buf.append(s);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1123
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1124
        if (parent != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1125
            if (buf.length() != 0) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1126
                buf.append(nl);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1127
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1128
            buf.append("// parent").append(nl).append(parent);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1129
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1130
        return buf.toString();
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1131
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1132
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1133
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1134
     * Code only used in assertions. Putting this in a separate class reduces class load time.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1135
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1136
    private static class Checks {
44836
45196b7066e3 8177845: Need a mechanism to load Graal
dnsimon
parents: 43972
diff changeset
  1137
        private static final int MAX_ARITY = 7;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1138
        /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1139
         * The set of all {@link InvocationPlugin#apply} method signatures.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1140
         */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1141
        static final Class<?>[][] SIGS;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1142
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1143
        static {
46762
f7defa99f173 8185829: Update Graal
dlong
parents: 46509
diff changeset
  1144
            if (!Assertions.assertionsEnabled()) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1145
                throw new GraalError("%s must only be used in assertions", Checks.class.getName());
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1146
            }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1147
            ArrayList<Class<?>[]> sigs = new ArrayList<>(MAX_ARITY);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1148
            for (Method method : InvocationPlugin.class.getDeclaredMethods()) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1149
                if (!Modifier.isStatic(method.getModifiers()) && method.getName().equals("apply")) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1150
                    Class<?>[] sig = method.getParameterTypes();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1151
                    assert sig[0] == GraphBuilderContext.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1152
                    assert sig[1] == ResolvedJavaMethod.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1153
                    assert sig[2] == InvocationPlugin.Receiver.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1154
                    assert Arrays.asList(sig).subList(3, sig.length).stream().allMatch(c -> c == ValueNode.class);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1155
                    while (sigs.size() < sig.length - 2) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1156
                        sigs.add(null);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1157
                    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1158
                    sigs.set(sig.length - 3, sig);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1159
                }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1160
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1161
            assert sigs.indexOf(null) == -1 : format("need to add an apply() method to %s that takes %d %s arguments ", InvocationPlugin.class.getName(), sigs.indexOf(null),
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1162
                            ValueNode.class.getSimpleName());
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1163
            SIGS = sigs.toArray(new Class<?>[sigs.size()][]);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1164
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1165
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1166
        static boolean containsBinding(InvocationPlugins p, Type declaringType, Binding key) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1167
            String internalName = MetaUtil.toInternalName(declaringType.getTypeName());
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1168
            ClassPlugins classPlugins = p.registrations.get(internalName);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1169
            return classPlugins != null && classPlugins.lookup(key) != null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1170
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1171
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1172
        public static boolean check(InvocationPlugins plugins, Type declaringType, Binding binding) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1173
            InvocationPlugin plugin = binding.plugin;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1174
            InvocationPlugins p = plugins.parent;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1175
            while (p != null) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1176
                assert !containsBinding(p, declaringType, binding) : "a plugin is already registered for " + binding;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1177
                p = p.parent;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1178
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1179
            if (plugin instanceof ForeignCallPlugin || plugin instanceof GeneratedInvocationPlugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1180
                return true;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1181
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1182
            if (plugin instanceof MethodSubstitutionPlugin) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1183
                MethodSubstitutionPlugin msplugin = (MethodSubstitutionPlugin) plugin;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1184
                Method substitute = msplugin.getJavaSubstitute();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1185
                assert substitute.getAnnotation(MethodSubstitution.class) != null : format("Substitute method must be annotated with @%s: %s", MethodSubstitution.class.getSimpleName(), substitute);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1186
                return true;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1187
            }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1188
            int arguments = parseParameters(binding.argumentsDescriptor).size();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1189
            assert arguments < SIGS.length : format("need to extend %s to support method with %d arguments: %s", InvocationPlugin.class.getSimpleName(), arguments, binding);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1190
            for (Method m : plugin.getClass().getDeclaredMethods()) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1191
                if (m.getName().equals("apply")) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1192
                    Class<?>[] parameterTypes = m.getParameterTypes();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1193
                    if (Arrays.equals(SIGS[arguments], parameterTypes)) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1194
                        return true;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1195
                    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1196
                }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1197
            }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1198
            throw new AssertionError(format("graph builder plugin for %s not found", binding));
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1199
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1200
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1201
        static boolean checkResolvable(boolean isOptional, Type declaringType, Binding binding) {
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1202
            if (declaringType instanceof ResolvedJavaSymbol) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1203
                return checkResolvable(isOptional, ((ResolvedJavaSymbol) declaringType).getResolved(), binding);
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1204
            }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1205
            Class<?> declaringClass = InvocationPlugins.resolveType(declaringType, isOptional);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1206
            if (declaringClass == null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1207
                return true;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1208
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1209
            if (binding.name.equals("<init>")) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1210
                if (resolveConstructor(declaringClass, binding) == null && !isOptional) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1211
                    throw new AssertionError(String.format("Constructor not found: %s%s", declaringClass.getName(), binding.argumentsDescriptor));
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1212
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1213
            } else {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1214
                if (resolveMethod(declaringClass, binding) == null && !isOptional) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1215
                    throw new AssertionError(String.format("Method not found: %s.%s%s", declaringClass.getName(), binding.name, binding.argumentsDescriptor));
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1216
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1217
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1218
            return true;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1219
        }
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1220
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1221
        private static boolean checkResolvable(boolean isOptional, ResolvedJavaType declaringType, Binding binding) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1222
            if (resolveJavaMethod(declaringType, binding) == null && !isOptional) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1223
                throw new AssertionError(String.format("Method not found: %s.%s%s", declaringType.toJavaName(), binding.name, binding.argumentsDescriptor));
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1224
            }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1225
            return true;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1226
        }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1227
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1228
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1229
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1230
     * Checks a set of nodes added to the graph by an {@link InvocationPlugin}.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1231
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1232
     * @param b the graph builder that applied the plugin
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1233
     * @param plugin a plugin that was just applied
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1234
     * @param newNodes the nodes added to the graph by {@code plugin}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1235
     * @throws AssertionError if any check fail
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1236
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1237
    public void checkNewNodes(GraphBuilderContext b, InvocationPlugin plugin, NodeIterable<Node> newNodes) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1238
        if (parent != null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1239
            parent.checkNewNodes(b, plugin, newNodes);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1240
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1241
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1242
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1243
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1244
     * Resolves a name to a class.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1245
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1246
     * @param className the name of the class to resolve
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1247
     * @param optional if true, resolution failure returns null
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1248
     * @return the resolved class or null if resolution fails and {@code optional} is true
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1249
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1250
    public static Class<?> resolveClass(String className, boolean optional) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1251
        try {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1252
            // Need to use the system class loader to handle classes
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1253
            // loaded by the application class loader which is not
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1254
            // delegated to by the JVMCI class loader.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1255
            ClassLoader cl = ClassLoader.getSystemClassLoader();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1256
            return Class.forName(className, false, cl);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1257
        } catch (ClassNotFoundException e) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1258
            if (optional) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1259
                return null;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1260
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1261
            throw new GraalError("Could not resolve type " + className);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1262
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1263
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1264
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1265
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1266
     * Resolves a {@link Type} to a {@link Class}.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1267
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1268
     * @param type the type to resolve
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1269
     * @param optional if true, resolution failure returns null
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1270
     * @return the resolved class or null if resolution fails and {@code optional} is true
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1271
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1272
    public static Class<?> resolveType(Type type, boolean optional) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1273
        if (type instanceof Class) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1274
            return (Class<?>) type;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1275
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1276
        if (type instanceof OptionalLazySymbol) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1277
            return ((OptionalLazySymbol) type).resolve();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1278
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1279
        return resolveClass(type.getTypeName(), optional);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1280
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1281
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1282
    private static List<String> toInternalTypeNames(Class<?>[] types) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1283
        String[] res = new String[types.length];
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1284
        for (int i = 0; i < types.length; i++) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1285
            res[i] = MetaUtil.toInternalName(types[i].getTypeName());
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1286
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1287
        return Arrays.asList(res);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1288
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1289
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1290
    /**
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1291
     * Resolves a given binding to a method in a given class. If more than one method with the
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1292
     * parameter types matching {@code binding} is found and the return types of all the matching
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1293
     * methods form an inheritance chain, the one with the most specific type is returned; otherwise
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1294
     * {@link NoSuchMethodError} is thrown.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1295
     *
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1296
     * @param declaringClass the class to search for a method matching {@code binding}
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1297
     * @return the method (if any) in {@code declaringClass} matching {@code binding}
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1298
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1299
    public static Method resolveMethod(Class<?> declaringClass, Binding binding) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1300
        if (binding.name.equals("<init>")) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1301
            return null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1302
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1303
        Method[] methods = declaringClass.getDeclaredMethods();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1304
        List<String> parameterTypeNames = parseParameters(binding.argumentsDescriptor);
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1305
        Method match = null;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1306
        for (int i = 0; i < methods.length; ++i) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1307
            Method m = methods[i];
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1308
            if (binding.isStatic == Modifier.isStatic(m.getModifiers()) &&
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1309
                            m.getName().equals(binding.name) &&
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1310
                            parameterTypeNames.equals(toInternalTypeNames(m.getParameterTypes()))) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1311
                if (match == null) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1312
                    match = m;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1313
                } else if (match.getReturnType().isAssignableFrom(m.getReturnType())) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1314
                    // `m` has a more specific return type - choose it
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1315
                    // (`match` is most likely a bridge method)
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1316
                    match = m;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1317
                } else {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1318
                    if (!m.getReturnType().isAssignableFrom(match.getReturnType())) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1319
                        throw new NoSuchMethodError(String.format(
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1320
                                        "Found 2 methods with same name and parameter types but unrelated return types:%n %s%n %s", match, m));
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1321
                    }
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1322
                }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1323
            }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1324
        }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1325
        return match;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1326
    }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1327
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1328
    /**
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1329
     * Same as {@link #resolveMethod(Class, Binding)} and
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1330
     * {@link #resolveConstructor(Class, Binding)} except in terms of {@link ResolvedJavaType} and
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1331
     * {@link ResolvedJavaMethod}.
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1332
     */
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1333
    public static ResolvedJavaMethod resolveJavaMethod(ResolvedJavaType declaringClass, Binding binding) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1334
        ResolvedJavaMethod[] methods = declaringClass.getDeclaredMethods();
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1335
        if (binding.name.equals("<init>")) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1336
            for (ResolvedJavaMethod m : methods) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1337
                if (m.getName().equals("<init>") && m.getSignature().toMethodDescriptor().startsWith(binding.argumentsDescriptor)) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1338
                    return m;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1339
                }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1340
            }
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1341
            return null;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1342
        }
48861
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1343
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1344
        ResolvedJavaMethod match = null;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1345
        for (int i = 0; i < methods.length; ++i) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1346
            ResolvedJavaMethod m = methods[i];
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1347
            if (binding.isStatic == m.isStatic() &&
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1348
                            m.getName().equals(binding.name) &&
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1349
                            m.getSignature().toMethodDescriptor().startsWith(binding.argumentsDescriptor)) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1350
                if (match == null) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1351
                    match = m;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1352
                } else {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1353
                    final ResolvedJavaType matchReturnType = (ResolvedJavaType) match.getSignature().getReturnType(declaringClass);
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1354
                    final ResolvedJavaType mReturnType = (ResolvedJavaType) m.getSignature().getReturnType(declaringClass);
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1355
                    if (matchReturnType.isAssignableFrom(mReturnType)) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1356
                        // `m` has a more specific return type - choose it
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1357
                        // (`match` is most likely a bridge method)
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1358
                        match = m;
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1359
                    } else {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1360
                        if (!mReturnType.isAssignableFrom(matchReturnType)) {
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1361
                            throw new NoSuchMethodError(String.format(
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1362
                                            "Found 2 methods with same name and parameter types but unrelated return types:%n %s%n %s", match, m));
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1363
                        }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1364
                    }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1365
                }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1366
            }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1367
        }
47f19ff9903c 8194819: Update Graal
iveresov
parents: 47216
diff changeset
  1368
        return match;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1369
    }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1370
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1371
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1372
     * Resolves a given binding to a constructor in a given class.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1373
     *
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1374
     * @param declaringClass the class to search for a constructor matching {@code binding}
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1375
     * @return the constructor (if any) in {@code declaringClass} matching binding
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1376
     */
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1377
    public static Constructor<?> resolveConstructor(Class<?> declaringClass, Binding binding) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1378
        if (!binding.name.equals("<init>")) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1379
            return null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1380
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1381
        Constructor<?>[] constructors = declaringClass.getDeclaredConstructors();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1382
        List<String> parameterTypeNames = parseParameters(binding.argumentsDescriptor);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1383
        for (int i = 0; i < constructors.length; ++i) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1384
            Constructor<?> c = constructors[i];
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1385
            if (parameterTypeNames.equals(toInternalTypeNames(c.getParameterTypes()))) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1386
                return c;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1387
            }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1388
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1389
        return null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1390
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1391
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1392
    private static List<String> parseParameters(String argumentsDescriptor) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1393
        assert argumentsDescriptor.startsWith("(") && argumentsDescriptor.endsWith(")") : argumentsDescriptor;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1394
        List<String> res = new ArrayList<>();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1395
        int cur = 1;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1396
        int end = argumentsDescriptor.length() - 1;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1397
        while (cur != end) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1398
            char first;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1399
            int start = cur;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1400
            do {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1401
                first = argumentsDescriptor.charAt(cur++);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1402
            } while (first == '[');
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1403
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1404
            switch (first) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1405
                case 'L':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1406
                    int endObject = argumentsDescriptor.indexOf(';', cur);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1407
                    if (endObject == -1) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1408
                        throw new GraalError("Invalid object type at index %d in signature: %s", cur, argumentsDescriptor);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1409
                    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1410
                    cur = endObject + 1;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1411
                    break;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1412
                case 'V':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1413
                case 'I':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1414
                case 'B':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1415
                case 'C':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1416
                case 'D':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1417
                case 'F':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1418
                case 'J':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1419
                case 'S':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1420
                case 'Z':
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1421
                    break;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1422
                default:
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1423
                    throw new GraalError("Invalid character at index %d in signature: %s", cur, argumentsDescriptor);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1424
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1425
            res.add(argumentsDescriptor.substring(start, cur));
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1426
        }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
  1427
        return res;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1428
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
  1429
}