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