src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/MethodSubstitutionPlugin.java
author dlong
Thu, 31 Oct 2019 16:54:16 -0700
changeset 58877 aec7bf35d6f5
parent 58299 6df94ce3ab2f
permissions -rw-r--r--
8233273: 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
/*
58299
6df94ce3ab2f 8229201: Update Graal
dlong
parents: 55509
diff changeset
     2
 * Copyright (c) 2015, 2019, Oracle and/or its affiliates. All rights reserved.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     4
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     7
 * published by the Free Software Foundation.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     8
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    13
 * accompanied this code).
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    14
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    18
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    21
 * questions.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    22
 */
50858
2d3e99a72541 8205824: Update Graal
never
parents: 47216
diff changeset
    23
2d3e99a72541 8205824: Update Graal
never
parents: 47216
diff changeset
    24
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    25
package org.graalvm.compiler.nodes.graphbuilderconf;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    26
54084
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 52910
diff changeset
    27
import static jdk.vm.ci.services.Services.IS_IN_NATIVE_IMAGE;
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
    28
import static org.graalvm.compiler.core.common.GraalOptions.UseEncodedGraphs;
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
    29
import static org.graalvm.compiler.nodes.graphbuilderconf.IntrinsicContext.CompilationContext.INLINE_AFTER_PARSING;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    30
import static org.graalvm.compiler.nodes.graphbuilderconf.InvocationPlugins.resolveType;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    31
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    32
import java.lang.reflect.Method;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    33
import java.lang.reflect.Modifier;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    34
import java.lang.reflect.Type;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    35
import java.util.Arrays;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    36
import java.util.stream.Collectors;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    37
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    38
import org.graalvm.compiler.bytecode.BytecodeProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    39
import org.graalvm.compiler.debug.GraalError;
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
    40
import org.graalvm.compiler.nodes.StructuredGraph;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    41
import org.graalvm.compiler.nodes.ValueNode;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    42
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    43
import jdk.vm.ci.meta.MetaAccessProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    44
import jdk.vm.ci.meta.ResolvedJavaMethod;
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    45
import jdk.vm.ci.meta.ResolvedJavaType;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    46
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    47
/**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    48
 * An {@link InvocationPlugin} for a method where the implementation of the method is provided by a
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    49
 * {@linkplain #getSubstitute(MetaAccessProvider) substitute} method. A substitute method must be
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    50
 * static even if the substituted method is not.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    51
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    52
 * While performing intrinsification with method substitutions is simpler than writing an
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    53
 * {@link InvocationPlugin} that does manual graph weaving, it has a higher compile time cost than
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    54
 * the latter; parsing bytecodes to create nodes is slower than simply creating nodes. As such, the
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    55
 * recommended practice is to use {@link MethodSubstitutionPlugin} only for complex
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    56
 * intrinsifications which is typically those using non-straight-line control flow.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    57
 */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    58
public final class MethodSubstitutionPlugin implements InvocationPlugin {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    59
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    60
    private InvocationPlugins.Registration registration;
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    61
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    62
    private ResolvedJavaMethod cachedSubstitute;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    63
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    64
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    65
     * The class in which the substitute method is declared.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    66
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    67
    private final Class<?> declaringClass;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    68
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    69
    /**
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    70
     * The name of the substitute method.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    71
     */
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    72
    private final String substituteName;
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    73
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    74
    /**
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    75
     * The name of the original method.
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    76
     */
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    77
    private final String originalName;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    78
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    79
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    80
     * The parameter types of the substitute method.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    81
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    82
    private final Type[] parameters;
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
    private final boolean originalIsStatic;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    85
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    86
    private final BytecodeProvider bytecodeProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    87
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    88
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    89
     * Creates a method substitution plugin.
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
     * @param bytecodeProvider used to get the bytecodes to parse for the substitute method
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    92
     * @param originalName the name of the original method
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    93
     * @param declaringClass the class in which the substitute method is declared
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    94
     * @param substituteName the name of the substitute method
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    95
     * @param parameters the parameter types of the substitute method. If the original method is not
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    96
     *            static, then {@code parameters[0]} must be the {@link Class} value denoting
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    97
     *            {@link InvocationPlugin.Receiver}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    98
     */
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    99
    public MethodSubstitutionPlugin(InvocationPlugins.Registration registration, BytecodeProvider bytecodeProvider, String originalName, Class<?> declaringClass, String substituteName,
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   100
                    Type... parameters) {
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   101
        assert bytecodeProvider != null : "Requires a non-null methodSubstitutionBytecodeProvider";
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   102
        this.registration = registration;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   103
        this.bytecodeProvider = bytecodeProvider;
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   104
        this.originalName = originalName;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   105
        this.declaringClass = declaringClass;
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   106
        this.substituteName = substituteName;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   107
        this.parameters = parameters;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   108
        this.originalIsStatic = parameters.length == 0 || parameters[0] != InvocationPlugin.Receiver.class;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   109
    }
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
    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   112
    public boolean inlineOnly() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   113
        // Conservatively assume MacroNodes may be used in a substitution
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   114
        return true;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   115
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   116
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   117
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   118
     * Gets the substitute method, resolving it first if necessary.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   119
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   120
    public ResolvedJavaMethod getSubstitute(MetaAccessProvider metaAccess) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   121
        if (cachedSubstitute == null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   122
            cachedSubstitute = metaAccess.lookupJavaMethod(getJavaSubstitute());
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   123
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   124
        return cachedSubstitute;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   125
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   126
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   127
    /**
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   128
     * Gets the object used to access the bytecodes of the substitute method.
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   129
     */
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   130
    public BytecodeProvider getBytecodeProvider() {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   131
        return bytecodeProvider;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   132
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   133
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   134
    /**
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   135
     * Gets the class in which the substitute method is declared.
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   136
     */
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   137
    public Class<?> getDeclaringClass() {
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   138
        return declaringClass;
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   139
    }
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   140
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   141
    /**
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   142
     * Gets the reflection API version of the substitution method.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   143
     */
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   144
    public Method getJavaSubstitute() throws GraalError {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   145
        Method substituteMethod = lookupSubstitute();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   146
        int modifiers = substituteMethod.getModifiers();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   147
        if (Modifier.isAbstract(modifiers) || Modifier.isNative(modifiers)) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   148
            throw new GraalError("Substitution method must not be abstract or native: " + substituteMethod);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   149
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   150
        if (!Modifier.isStatic(modifiers)) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   151
            throw new GraalError("Substitution method must be static: " + substituteMethod);
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
        return substituteMethod;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   154
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   155
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   156
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   157
     * Determines if a given method is the substitute method of this plugin.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   158
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   159
    private boolean isSubstitute(Method m) {
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   160
        if (Modifier.isStatic(m.getModifiers()) && m.getName().equals(substituteName)) {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   161
            if (parameters.length == m.getParameterCount()) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   162
                Class<?>[] mparams = m.getParameterTypes();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   163
                int start = 0;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   164
                if (!originalIsStatic) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   165
                    start = 1;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   166
                    if (!mparams[0].isAssignableFrom(resolveType(parameters[0], false))) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   167
                        return false;
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
                for (int i = start; i < mparams.length; i++) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   171
                    if (mparams[i] != resolveType(parameters[i], false)) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   172
                        return false;
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
                }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   175
                return true;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   176
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   177
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   178
        return false;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   179
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   180
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   181
    private Method lookupSubstitute(Method excluding) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   182
        for (Method m : declaringClass.getDeclaredMethods()) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   183
            if (!m.equals(excluding) && isSubstitute(m)) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   184
                return m;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   185
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   186
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   187
        return null;
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   188
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   189
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   190
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   191
     * Gets the substitute method of this plugin.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   192
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   193
    private Method lookupSubstitute() {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   194
        Method m = lookupSubstitute(null);
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   195
        if (m != null) {
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   196
            assert lookupSubstitute(m) == null : String.format("multiple matches found for %s:%n%s%n%s", this, m, lookupSubstitute(m));
7d4e637d3f21 8180267: Update Graal
kvn
parents: 43972
diff changeset
   197
            return m;
43972
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
        throw new GraalError("No method found specified by %s", this);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   200
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   201
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   202
    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   203
    public boolean execute(GraphBuilderContext b, ResolvedJavaMethod targetMethod, InvocationPlugin.Receiver receiver, ValueNode[] argsIncludingReceiver) {
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   204
        if (IS_IN_NATIVE_IMAGE || (UseEncodedGraphs.getValue(b.getOptions()) && !b.parsingIntrinsic())) {
55509
d58442b8abc1 8225497: Update Graal
jwilhelm
parents: 54601
diff changeset
   205
            StructuredGraph subst = b.getReplacements().getMethodSubstitution(this,
d58442b8abc1 8225497: Update Graal
jwilhelm
parents: 54601
diff changeset
   206
                            targetMethod,
d58442b8abc1 8225497: Update Graal
jwilhelm
parents: 54601
diff changeset
   207
                            INLINE_AFTER_PARSING,
d58442b8abc1 8225497: Update Graal
jwilhelm
parents: 54601
diff changeset
   208
                            StructuredGraph.AllowAssumptions.ifNonNull(b.getAssumptions()),
d58442b8abc1 8225497: Update Graal
jwilhelm
parents: 54601
diff changeset
   209
                            null /* cancellable */,
d58442b8abc1 8225497: Update Graal
jwilhelm
parents: 54601
diff changeset
   210
                            b.getOptions());
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   211
            if (subst == null) {
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   212
                throw new GraalError("No graphs found for substitution %s", this);
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   213
            }
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   214
            return b.intrinsify(targetMethod, subst, receiver, argsIncludingReceiver);
54084
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 52910
diff changeset
   215
        }
54601
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   216
        ResolvedJavaMethod substitute = getSubstitute(b.getMetaAccess());
c40b2a190173 8221598: Update Graal
jwilhelm
parents: 54084
diff changeset
   217
        return b.intrinsify(bytecodeProvider, targetMethod, substitute, receiver, argsIncludingReceiver);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   218
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   219
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   220
    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   221
    public StackTraceElement getApplySourceLocation(MetaAccessProvider metaAccess) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   222
        Class<?> c = getClass();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   223
        for (Method m : c.getDeclaredMethods()) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   224
            if (m.getName().equals("execute")) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   225
                return metaAccess.lookupJavaMethod(m).asStackTraceElement(0);
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   226
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   227
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   228
        throw new GraalError("could not find method named \"execute\" in " + c.getName());
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   229
    }
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
    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   232
    public String toString() {
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   233
        return String.format("%s[%s.%s(%s)]", getClass().getSimpleName(), declaringClass.getName(), substituteName,
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   234
                        Arrays.asList(parameters).stream().map(c -> c.getTypeName()).collect(Collectors.joining(", ")));
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   235
    }
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   236
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   237
    public String originalMethodAsString() {
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   238
        return String.format("%s.%s(%s)", declaringClass.getName(), substituteName, Arrays.asList(parameters).stream().map(c -> c.getTypeName()).collect(Collectors.joining(", ")));
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   239
    }
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   240
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   241
    public ResolvedJavaMethod getOriginalMethod(MetaAccessProvider metaAccess) {
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   242
        Class<?> clazz = resolveType(registration.getDeclaringType(), false);
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   243
        if (clazz == null) {
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   244
            throw new GraalError("Can't find original class for " + this + " with class " + registration.getDeclaringType());
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   245
        }
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   246
        ResolvedJavaType type = metaAccess.lookupJavaType(clazz);
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   247
        String argumentsDescriptor = InvocationPlugins.toArgumentDescriptor(originalIsStatic, this.parameters);
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   248
        for (ResolvedJavaMethod declared : type.getDeclaredMethods()) {
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   249
            if (declared.getName().equals(originalName) && declared.isStatic() == originalIsStatic &&
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   250
                            declared.getSignature().toMethodDescriptor().startsWith(argumentsDescriptor)) {
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   251
                return declared;
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   252
            }
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   253
        }
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   254
        throw new GraalError("Can't find original method for " + this + " with class " + registration.getDeclaringType());
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   255
    }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   256
}