SSAUtil.java revision 12657:6ef01bd40ce2
1/*
2 * Copyright (c) 2015, 2015, 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.lir.ssa;
24
25import java.util.Arrays;
26import java.util.List;
27
28import org.graalvm.compiler.core.common.LIRKind;
29import org.graalvm.compiler.core.common.cfg.AbstractBlockBase;
30import org.graalvm.compiler.lir.LIR;
31import org.graalvm.compiler.lir.LIRInstruction;
32import org.graalvm.compiler.lir.LIRInstruction.OperandMode;
33import org.graalvm.compiler.lir.StandardOp.BlockEndOp;
34import org.graalvm.compiler.lir.StandardOp.JumpOp;
35import org.graalvm.compiler.lir.StandardOp.LabelOp;
36import org.graalvm.compiler.lir.ValueConsumer;
37
38import jdk.vm.ci.meta.Value;
39
40/**
41 * Utilities for working with Static-Single-Assignment LIR form.
42 *
43 * <h2>Representation of <code>PHI</code>s</h2>
44 *
45 * There is no explicit <code>PHI</code> {@linkplain LIRInstruction}. Instead, they are implemented
46 * as parallel copy that span across a control-flow edge.
47 *
48 * The variables introduced by <code>PHI</code>s of a specific {@linkplain AbstractBlockBase merge
49 * block} are {@linkplain LabelOp#setIncomingValues attached} to the {@linkplain LabelOp} of the
50 * block. The outgoing values from the predecessor are {@link JumpOp#setOutgoingValues input} to the
51 * {@linkplain BlockEndOp} of the predecessor. Because there are no critical edges we know that the
52 * {@link BlockEndOp} of the predecessor has to be a {@link JumpOp}.
53 *
54 * <h3>Example:</h3>
55 *
56 * <pre>
57 * B0 -> B1
58 *   ...
59 *   v0|i = ...
60 *   JUMP ~[v0|i, int[0|0x0]] destination: B0 -> B1
61 * ________________________________________________
62 *
63 * B2 -> B1
64 *   ...
65 *   v1|i = ...
66 *   v2|i = ...
67 *   JUMP ~[v1|i, v2|i] destination: B2 -> B1
68 * ________________________________________________
69 *
70 * B1 <- B0,B2
71 *   [v3|i, v4|i] = LABEL
72 *   ...
73 * </pre>
74 */
75public final class SSAUtil {
76
77    public interface PhiValueVisitor {
78        /**
79         * @param phiIn the incoming value at the merge block
80         * @param phiOut the outgoing value from the predecessor block
81         */
82        void visit(Value phiIn, Value phiOut);
83    }
84
85    /**
86     * Visits each phi value pair of an edge, i.e. the outgoing value from the predecessor and the
87     * incoming value to the merge block.
88     */
89    public static void forEachPhiValuePair(LIR lir, AbstractBlockBase<?> merge, AbstractBlockBase<?> pred, PhiValueVisitor visitor) {
90        if (merge.getPredecessorCount() < 2) {
91            return;
92        }
93        assert Arrays.asList(merge.getPredecessors()).contains(pred) : String.format("%s not in predecessor list: %s", pred, Arrays.toString(merge.getPredecessors()));
94        assert pred.getSuccessorCount() == 1 : String.format("Merge predecessor block %s has more than one successor? %s", pred, Arrays.toString(pred.getSuccessors()));
95        assert pred.getSuccessors()[0] == merge : String.format("Predecessor block %s has wrong successor: %s, should be: %s", pred, pred.getSuccessors()[0], merge);
96
97        JumpOp jump = phiOut(lir, pred);
98        LabelOp label = phiIn(lir, merge);
99
100        assert label.getIncomingSize() == jump.getOutgoingSize() : String.format("Phi In/Out size mismatch: in=%d vs. out=%d", label.getIncomingSize(), jump.getOutgoingSize());
101        assert label.getPhiSize() == jump.getPhiSize() : String.format("Phi In/Out size mismatch: in=%d vs. out=%d", label.getPhiSize(), jump.getPhiSize());
102
103        for (int i = 0; i < label.getPhiSize(); i++) {
104            visitor.visit(label.getIncomingValue(i), jump.getOutgoingValue(i));
105        }
106    }
107
108    private static JumpOp phiOut(LIR lir, AbstractBlockBase<?> block) {
109        assert block.getSuccessorCount() == 1;
110        List<LIRInstruction> instructions = lir.getLIRforBlock(block);
111        int index = instructions.size() - 1;
112        LIRInstruction op = instructions.get(index);
113        return (JumpOp) op;
114    }
115
116    public static int phiOutIndex(LIR lir, AbstractBlockBase<?> block) {
117        assert block.getSuccessorCount() == 1;
118        List<LIRInstruction> instructions = lir.getLIRforBlock(block);
119        int index = instructions.size() - 1;
120        assert instructions.get(index) instanceof JumpOp;
121        return index;
122    }
123
124    private static LabelOp phiIn(LIR lir, AbstractBlockBase<?> block) {
125        assert block.getPredecessorCount() > 1;
126        LabelOp label = (LabelOp) lir.getLIRforBlock(block).get(0);
127        return label;
128    }
129
130    public static void removePhiOut(LIR lir, AbstractBlockBase<?> block) {
131        JumpOp jump = phiOut(lir, block);
132        jump.clearOutgoingValues();
133    }
134
135    public static void removePhiIn(LIR lir, AbstractBlockBase<?> block) {
136        LabelOp label = phiIn(lir, block);
137        label.clearIncomingValues();
138    }
139
140    public static boolean verifySSAForm(LIR lir) {
141        return new SSAVerifier(lir).verify();
142    }
143
144    public static void verifyPhi(LIR lir, AbstractBlockBase<?> merge) {
145        assert merge.getPredecessorCount() > 1;
146        for (AbstractBlockBase<?> pred : merge.getPredecessors()) {
147            forEachPhiValuePair(lir, merge, pred, (phiIn, phiOut) -> {
148                assert phiIn.getValueKind().equals(phiOut.getValueKind()) ||
149                                (phiIn.getPlatformKind().equals(phiOut.getPlatformKind()) && LIRKind.isUnknownReference(phiIn) && LIRKind.isValue(phiOut));
150            });
151        }
152    }
153
154    public static void forEachPhiRegisterHint(LIR lir, AbstractBlockBase<?> block, LabelOp label, Value targetValue, OperandMode mode, ValueConsumer valueConsumer) {
155        assert mode == OperandMode.DEF : "Wrong operand mode: " + mode;
156        assert lir.getLIRforBlock(block).get(0).equals(label) : String.format("Block %s and Label %s do not match!", block, label);
157
158        if (!label.isPhiIn()) {
159            return;
160        }
161        int idx = indexOfValue(label, targetValue);
162        assert idx >= 0 : String.format("Value %s not in label %s", targetValue, label);
163
164        for (AbstractBlockBase<?> pred : block.getPredecessors()) {
165            JumpOp jump = phiOut(lir, pred);
166            Value sourceValue = jump.getOutgoingValue(idx);
167            valueConsumer.visitValue(jump, sourceValue, null, null);
168        }
169
170    }
171
172    private static int indexOfValue(LabelOp label, Value value) {
173        for (int i = 0; i < label.getIncomingSize(); i++) {
174            if (label.getIncomingValue(i).equals(value)) {
175                return i;
176            }
177        }
178        return -1;
179    }
180
181}
182