GraphBuilderContext.java revision 12651:6ef01bd40ce2
1/*
2 * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 */
23package org.graalvm.compiler.nodes.graphbuilderconf;
24
25import static org.graalvm.compiler.core.common.type.StampFactory.objectNonNull;
26import static jdk.vm.ci.meta.DeoptimizationAction.InvalidateReprofile;
27import static jdk.vm.ci.meta.DeoptimizationReason.NullCheckException;
28
29import org.graalvm.compiler.bytecode.Bytecode;
30import org.graalvm.compiler.bytecode.BytecodeProvider;
31import org.graalvm.compiler.core.common.type.ObjectStamp;
32import org.graalvm.compiler.core.common.type.Stamp;
33import org.graalvm.compiler.core.common.type.StampFactory;
34import org.graalvm.compiler.core.common.type.StampPair;
35import org.graalvm.compiler.nodes.CallTargetNode.InvokeKind;
36import org.graalvm.compiler.nodes.FixedGuardNode;
37import org.graalvm.compiler.nodes.LogicNode;
38import org.graalvm.compiler.nodes.PiNode;
39import org.graalvm.compiler.nodes.StateSplit;
40import org.graalvm.compiler.nodes.ValueNode;
41import org.graalvm.compiler.nodes.calc.IsNullNode;
42import org.graalvm.compiler.nodes.type.StampTool;
43
44import jdk.vm.ci.code.BailoutException;
45import jdk.vm.ci.meta.Assumptions;
46import jdk.vm.ci.meta.JavaKind;
47import jdk.vm.ci.meta.JavaType;
48import jdk.vm.ci.meta.ResolvedJavaMethod;
49
50/**
51 * Used by a {@link GraphBuilderPlugin} to interface with an object that parses the bytecode of a
52 * single {@linkplain #getMethod() method} as part of building a {@linkplain #getGraph() graph} .
53 */
54public interface GraphBuilderContext extends GraphBuilderTool {
55
56    /**
57     * Pushes a given value to the frame state stack using an explicit kind. This should be used
58     * when {@code value.getJavaKind()} is different from the kind that the bytecode instruction
59     * currently being parsed pushes to the stack.
60     *
61     * @param kind the kind to use when type checking this operation
62     * @param value the value to push to the stack. The value must already have been
63     *            {@linkplain #append(ValueNode) appended}.
64     */
65    void push(JavaKind kind, ValueNode value);
66
67    /**
68     * Adds a node to the graph. If the returned node is a {@link StateSplit} with a null
69     * {@linkplain StateSplit#stateAfter() frame state}, the frame state is initialized.
70     *
71     * @param value the value to add to the graph and push to the stack. The
72     *            {@code value.getJavaKind()} kind is used when type checking this operation.
73     * @return a node equivalent to {@code value} in the graph
74     */
75    default <T extends ValueNode> T add(T value) {
76        if (value.graph() != null) {
77            assert !(value instanceof StateSplit) || ((StateSplit) value).stateAfter() != null;
78            return value;
79        }
80        T equivalentValue = append(value);
81        if (equivalentValue instanceof StateSplit) {
82            StateSplit stateSplit = (StateSplit) equivalentValue;
83            if (stateSplit.stateAfter() == null && stateSplit.hasSideEffect()) {
84                setStateAfter(stateSplit);
85            }
86        }
87        return equivalentValue;
88    }
89
90    /**
91     * Adds a node with a non-void kind to the graph, pushes it to the stack. If the returned node
92     * is a {@link StateSplit} with a null {@linkplain StateSplit#stateAfter() frame state}, the
93     * frame state is initialized.
94     *
95     * @param kind the kind to use when type checking this operation
96     * @param value the value to add to the graph and push to the stack
97     * @return a node equivalent to {@code value} in the graph
98     */
99    default <T extends ValueNode> T addPush(JavaKind kind, T value) {
100        T equivalentValue = value.graph() != null ? value : append(value);
101        push(kind, equivalentValue);
102        if (equivalentValue instanceof StateSplit) {
103            StateSplit stateSplit = (StateSplit) equivalentValue;
104            if (stateSplit.stateAfter() == null && stateSplit.hasSideEffect()) {
105                setStateAfter(stateSplit);
106            }
107        }
108        return equivalentValue;
109    }
110
111    /**
112     * Handles an invocation that a plugin determines can replace the original invocation (i.e., the
113     * one for which the plugin was applied). This applies all standard graph builder processing to
114     * the replaced invocation including applying any relevant plugins.
115     *
116     * @param invokeKind the kind of the replacement invocation
117     * @param targetMethod the target of the replacement invocation
118     * @param args the arguments to the replacement invocation
119     * @param forceInlineEverything specifies if all invocations encountered in the scope of
120     *            handling the replaced invoke are to be force inlined
121     */
122    void handleReplacedInvoke(InvokeKind invokeKind, ResolvedJavaMethod targetMethod, ValueNode[] args, boolean forceInlineEverything);
123
124    /**
125     * Intrinsifies an invocation of a given method by inlining the bytecodes of a given
126     * substitution method.
127     *
128     * @param bytecodeProvider used to get the bytecodes to parse for the substitution method
129     * @param targetMethod the method being intrinsified
130     * @param substitute the intrinsic implementation
131     * @param receiver the receiver, or null for static methods
132     * @param argsIncludingReceiver the arguments with which to inline the invocation
133     *
134     * @return whether the intrinsification was successful
135     */
136    boolean intrinsify(BytecodeProvider bytecodeProvider, ResolvedJavaMethod targetMethod, ResolvedJavaMethod substitute, InvocationPlugin.Receiver receiver, ValueNode[] argsIncludingReceiver);
137
138    /**
139     * Creates a snap shot of the current frame state with the BCI of the instruction after the one
140     * currently being parsed and assigns it to a given {@linkplain StateSplit#hasSideEffect() side
141     * effect} node.
142     *
143     * @param sideEffect a side effect node just appended to the graph
144     */
145    void setStateAfter(StateSplit sideEffect);
146
147    /**
148     * Gets the parsing context for the method that inlines the method being parsed by this context.
149     */
150    GraphBuilderContext getParent();
151
152    /**
153     * Gets the first ancestor parsing context that is not parsing a {@linkplain #parsingIntrinsic()
154     * intrinsic}.
155     */
156    default GraphBuilderContext getNonIntrinsicAncestor() {
157        GraphBuilderContext ancestor = getParent();
158        while (ancestor != null && ancestor.parsingIntrinsic()) {
159            ancestor = ancestor.getParent();
160        }
161        return ancestor;
162    }
163
164    /**
165     * Gets the code being parsed.
166     */
167    Bytecode getCode();
168
169    /**
170     * Gets the method being parsed by this context.
171     */
172    ResolvedJavaMethod getMethod();
173
174    /**
175     * Gets the index of the bytecode instruction currently being parsed.
176     */
177    int bci();
178
179    /**
180     * Gets the kind of invocation currently being parsed.
181     */
182    InvokeKind getInvokeKind();
183
184    /**
185     * Gets the return type of the invocation currently being parsed.
186     */
187    JavaType getInvokeReturnType();
188
189    default StampPair getInvokeReturnStamp(Assumptions assumptions) {
190        JavaType returnType = getInvokeReturnType();
191        return StampFactory.forDeclaredType(assumptions, returnType, false);
192    }
193
194    /**
195     * Gets the inline depth of this context. A return value of 0 implies that this is the context
196     * for the parse root.
197     */
198    default int getDepth() {
199        GraphBuilderContext parent = getParent();
200        return parent == null ? 0 : 1 + parent.getDepth();
201    }
202
203    /**
204     * Determines if this parsing context is within the bytecode of an intrinsic or a method inlined
205     * by an intrinsic.
206     */
207    @Override
208    default boolean parsingIntrinsic() {
209        return getIntrinsic() != null;
210    }
211
212    /**
213     * Gets the intrinsic of the current parsing context or {@code null} if not
214     * {@link #parsingIntrinsic() parsing an intrinsic}.
215     */
216    IntrinsicContext getIntrinsic();
217
218    BailoutException bailout(String string);
219
220    /**
221     * Gets a version of a given value that has a {@linkplain StampTool#isPointerNonNull(ValueNode)
222     * non-null} stamp.
223     */
224    default ValueNode nullCheckedValue(ValueNode value) {
225        if (!StampTool.isPointerNonNull(value.stamp())) {
226            LogicNode condition = getGraph().unique(IsNullNode.create(value));
227            ObjectStamp receiverStamp = (ObjectStamp) value.stamp();
228            Stamp stamp = receiverStamp.join(objectNonNull());
229            FixedGuardNode fixedGuard = append(new FixedGuardNode(condition, NullCheckException, InvalidateReprofile, true));
230            PiNode nonNullReceiver = getGraph().unique(new PiNode(value, stamp, fixedGuard));
231            // TODO: Propogating the non-null into the frame state would
232            // remove subsequent null-checks on the same value. However,
233            // it currently causes an assertion failure when merging states.
234            //
235            // frameState.replace(value, nonNullReceiver);
236            return nonNullReceiver;
237        }
238        return value;
239    }
240}
241