src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.test/src/org/graalvm/compiler/replacements/test/IntegerExactFoldTest.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:
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     1
/*
58299
6df94ce3ab2f 8229201: Update Graal
dlong
parents: 54084
diff changeset
     2
 * Copyright (c) 2017, 2019, Oracle and/or its affiliates. All rights reserved.
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     4
 *
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     7
 * published by the Free Software Foundation.
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     8
 *
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    13
 * accompanied this code).
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    14
 *
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    18
 *
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    21
 * questions.
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    22
 */
50858
2d3e99a72541 8205824: Update Graal
never
parents: 48190
diff changeset
    23
2d3e99a72541 8205824: Update Graal
never
parents: 48190
diff changeset
    24
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    25
package org.graalvm.compiler.replacements.test;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    26
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    27
import static org.junit.Assert.assertNotNull;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    28
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    29
import java.util.ArrayList;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    30
import java.util.Collection;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    31
import java.util.List;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    32
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    33
import org.graalvm.compiler.core.common.type.IntegerStamp;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    34
import org.graalvm.compiler.core.common.type.StampFactory;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    35
import org.graalvm.compiler.core.test.GraalCompilerTest;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    36
import org.graalvm.compiler.graph.Node;
48190
25cfedf27edc 8192814: Update Graal
dlong
parents: 47216
diff changeset
    37
import org.graalvm.compiler.nodes.NodeView;
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    38
import org.graalvm.compiler.nodes.ParameterNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    39
import org.graalvm.compiler.nodes.PiNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    40
import org.graalvm.compiler.nodes.ReturnNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    41
import org.graalvm.compiler.nodes.StructuredGraph;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    42
import org.graalvm.compiler.nodes.StructuredGraph.AllowAssumptions;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    43
import org.graalvm.compiler.nodes.ValueNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    44
import org.graalvm.compiler.nodes.spi.LoweringTool;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    45
import org.graalvm.compiler.phases.common.CanonicalizerPhase;
54084
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
    46
import org.graalvm.compiler.phases.common.GuardLoweringPhase;
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    47
import org.graalvm.compiler.phases.common.LoweringPhase;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    48
import org.graalvm.compiler.phases.tiers.HighTierContext;
54084
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
    49
import org.graalvm.compiler.phases.tiers.MidTierContext;
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    50
import org.graalvm.compiler.replacements.nodes.arithmetic.IntegerExactArithmeticNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    51
import org.graalvm.compiler.replacements.nodes.arithmetic.IntegerExactArithmeticSplitNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    52
import org.junit.Assert;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    53
import org.junit.Test;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    54
import org.junit.runner.RunWith;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    55
import org.junit.runners.Parameterized;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    56
import org.junit.runners.Parameterized.Parameters;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    57
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    58
@RunWith(Parameterized.class)
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    59
public class IntegerExactFoldTest extends GraalCompilerTest {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    60
    private final long lowerBoundA;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    61
    private final long upperBoundA;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    62
    private final long lowerBoundB;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    63
    private final long upperBoundB;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    64
    private final int bits;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    65
    private final Operation operation;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    66
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    67
    public IntegerExactFoldTest(long lowerBoundA, long upperBoundA, long lowerBoundB, long upperBoundB, int bits, Operation operation) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    68
        this.lowerBoundA = lowerBoundA;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    69
        this.upperBoundA = upperBoundA;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    70
        this.lowerBoundB = lowerBoundB;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    71
        this.upperBoundB = upperBoundB;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    72
        this.bits = bits;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    73
        this.operation = operation;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    74
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    75
        assert bits == 32 || bits == 64;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    76
        assert lowerBoundA <= upperBoundA;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    77
        assert lowerBoundB <= upperBoundB;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    78
        assert bits == 64 || isInteger(lowerBoundA);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    79
        assert bits == 64 || isInteger(upperBoundA);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    80
        assert bits == 64 || isInteger(lowerBoundB);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    81
        assert bits == 64 || isInteger(upperBoundB);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    82
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    83
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    84
    @Test
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    85
    public void testFolding() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    86
        StructuredGraph graph = prepareGraph();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    87
        IntegerStamp a = StampFactory.forInteger(bits, lowerBoundA, upperBoundA);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    88
        IntegerStamp b = StampFactory.forInteger(bits, lowerBoundB, upperBoundB);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    89
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    90
        List<ParameterNode> params = graph.getNodes(ParameterNode.TYPE).snapshot();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    91
        params.get(0).replaceAtMatchingUsages(graph.addOrUnique(new PiNode(params.get(0), a)), x -> x instanceof IntegerExactArithmeticNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    92
        params.get(1).replaceAtMatchingUsages(graph.addOrUnique(new PiNode(params.get(1), b)), x -> x instanceof IntegerExactArithmeticNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    93
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    94
        Node originalNode = graph.getNodes().filter(x -> x instanceof IntegerExactArithmeticNode).first();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    95
        assertNotNull("original node must be in the graph", originalNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    96
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
    97
        createCanonicalizerPhase().apply(graph, getDefaultHighTierContext());
54084
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
    98
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    99
        ValueNode node = findNode(graph);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   100
        boolean overflowExpected = node instanceof IntegerExactArithmeticNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   101
48190
25cfedf27edc 8192814: Update Graal
dlong
parents: 47216
diff changeset
   102
        IntegerStamp resultStamp = (IntegerStamp) node.stamp(NodeView.DEFAULT);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   103
        operation.verifyOverflow(lowerBoundA, upperBoundA, lowerBoundB, upperBoundB, bits, overflowExpected, resultStamp);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   104
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   105
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   106
    @Test
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   107
    public void testFoldingAfterLowering() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   108
        StructuredGraph graph = prepareGraph();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   109
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   110
        Node originalNode = graph.getNodes().filter(x -> x instanceof IntegerExactArithmeticNode).first();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   111
        assertNotNull("original node must be in the graph", originalNode);
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   112
        CanonicalizerPhase canonicalizer = createCanonicalizerPhase();
54084
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
   113
        HighTierContext highTierContext = getDefaultHighTierContext();
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
   114
        new LoweringPhase(canonicalizer, LoweringTool.StandardLoweringStage.HIGH_TIER).apply(graph, highTierContext);
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
   115
        MidTierContext midTierContext = getDefaultMidTierContext();
84f10bbf993f 8218074: Update Graal
jwilhelm
parents: 53309
diff changeset
   116
        new GuardLoweringPhase().apply(graph, midTierContext);
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   117
        createCanonicalizerPhase().apply(graph, midTierContext);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   118
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   119
        IntegerExactArithmeticSplitNode loweredNode = graph.getNodes().filter(IntegerExactArithmeticSplitNode.class).first();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   120
        assertNotNull("the lowered node must be in the graph", loweredNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   121
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   122
        loweredNode.getX().setStamp(StampFactory.forInteger(bits, lowerBoundA, upperBoundA));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   123
        loweredNode.getY().setStamp(StampFactory.forInteger(bits, lowerBoundB, upperBoundB));
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   124
        createCanonicalizerPhase().apply(graph, midTierContext);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   125
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   126
        ValueNode node = findNode(graph);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   127
        boolean overflowExpected = node instanceof IntegerExactArithmeticSplitNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   128
48190
25cfedf27edc 8192814: Update Graal
dlong
parents: 47216
diff changeset
   129
        IntegerStamp resultStamp = (IntegerStamp) node.stamp(NodeView.DEFAULT);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   130
        operation.verifyOverflow(lowerBoundA, upperBoundA, lowerBoundB, upperBoundB, bits, overflowExpected, resultStamp);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   131
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   132
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   133
    private static boolean isInteger(long value) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   134
        return value >= Integer.MIN_VALUE && value <= Integer.MAX_VALUE;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   135
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   136
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   137
    private static ValueNode findNode(StructuredGraph graph) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   138
        ValueNode resultNode = graph.getNodes().filter(ReturnNode.class).first().result();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   139
        assertNotNull("some node must be the returned value", resultNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   140
        return resultNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   141
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   142
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   143
    protected StructuredGraph prepareGraph() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   144
        String snippet = "snippetInt" + bits;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   145
        StructuredGraph graph = parseEager(getResolvedJavaMethod(operation.getClass(), snippet), AllowAssumptions.NO);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   146
        HighTierContext context = getDefaultHighTierContext();
58877
aec7bf35d6f5 8233273: Update Graal
dlong
parents: 58299
diff changeset
   147
        createCanonicalizerPhase().apply(graph, context);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   148
        return graph;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   149
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   150
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   151
    private static void addTest(ArrayList<Object[]> tests, long lowerBound1, long upperBound1, long lowerBound2, long upperBound2, int bits, Operation operation) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   152
        tests.add(new Object[]{lowerBound1, upperBound1, lowerBound2, upperBound2, bits, operation});
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   153
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   154
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   155
    @Parameters(name = "a[{0} / {1}], b[{2} / {3}], bits={4}, operation={5}")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   156
    public static Collection<Object[]> data() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   157
        ArrayList<Object[]> tests = new ArrayList<>();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   158
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   159
        Operation[] operations = new Operation[]{new AddOperation(), new SubOperation(), new MulOperation()};
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   160
        for (Operation operation : operations) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   161
            for (int bits : new int[]{32, 64}) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   162
                // zero related
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   163
                addTest(tests, 0, 0, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   164
                addTest(tests, 1, 1, 0, 0, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   165
                addTest(tests, -1, 1, 0, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   166
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   167
                // bounds
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   168
                addTest(tests, -2, 2, 3, 3, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   169
                addTest(tests, -1, 1, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   170
                addTest(tests, -1, 1, -1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   171
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   172
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE + 0xF, Integer.MAX_VALUE - 0xF, Integer.MAX_VALUE, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   173
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE + 0xF, -1, -1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   174
                addTest(tests, Integer.MAX_VALUE, Integer.MAX_VALUE, -1, -1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   175
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE, -1, -1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   176
                addTest(tests, Integer.MAX_VALUE, Integer.MAX_VALUE, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   177
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   178
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   179
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   180
            // bit-specific test cases
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   181
            addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE + 0xF, Integer.MAX_VALUE - 0xF, Integer.MAX_VALUE, 64, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   182
            addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE + 0xF, -1, -1, 64, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   183
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   184
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   185
        return tests;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   186
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   187
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   188
    private abstract static class Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   189
        abstract void verifyOverflow(long lowerBoundA, long upperBoundA, long lowerBoundB, long upperBoundB, int bits, boolean overflowExpected, IntegerStamp resultStamp);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   190
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   191
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   192
    private static final class AddOperation extends Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   193
        @Override
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   194
        public void verifyOverflow(long lowerBoundA, long upperBoundA, long lowerBoundB, long upperBoundB, int bits, boolean overflowExpected, IntegerStamp resultStamp) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   195
            try {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   196
                long res = addExact(lowerBoundA, lowerBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   197
                resultStamp.contains(res);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   198
                res = addExact(upperBoundA, upperBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   199
                resultStamp.contains(res);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   200
                Assert.assertFalse(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   201
            } catch (ArithmeticException e) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   202
                Assert.assertTrue(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   203
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   204
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   205
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   206
        private static long addExact(long x, long y, int bits) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   207
            if (bits == 32) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   208
                return Math.addExact((int) x, (int) y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   209
            } else {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   210
                return Math.addExact(x, y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   211
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   212
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   213
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   214
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   215
        public static int snippetInt32(int a, int b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   216
            return Math.addExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   217
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   218
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   219
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   220
        public static long snippetInt64(long a, long b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   221
            return Math.addExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   222
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   223
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   224
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   225
    private static final class SubOperation extends Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   226
        @Override
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   227
        public void verifyOverflow(long lowerBoundA, long upperBoundA, long lowerBoundB, long upperBoundB, int bits, boolean overflowExpected, IntegerStamp resultStamp) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   228
            try {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   229
                long res = subExact(lowerBoundA, upperBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   230
                Assert.assertTrue(resultStamp.contains(res));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   231
                res = subExact(upperBoundA, lowerBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   232
                Assert.assertTrue(resultStamp.contains(res));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   233
                Assert.assertFalse(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   234
            } catch (ArithmeticException e) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   235
                Assert.assertTrue(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   236
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   237
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   238
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   239
        private static long subExact(long x, long y, int bits) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   240
            if (bits == 32) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   241
                return Math.subtractExact((int) x, (int) y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   242
            } else {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   243
                return Math.subtractExact(x, y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   244
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   245
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   246
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   247
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   248
        public static int snippetInt32(int a, int b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   249
            return Math.subtractExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   250
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   251
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   252
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   253
        public static long snippetInt64(long a, long b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   254
            return Math.subtractExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   255
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   256
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   257
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   258
    private static final class MulOperation extends Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   259
        @Override
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   260
        public void verifyOverflow(long lowerBoundA, long upperBoundA, long lowerBoundB, long upperBoundB, int bits, boolean overflowExpected, IntegerStamp resultStamp) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   261
            // now check for all values in the stamp whether their products overflow overflow
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   262
            boolean overflowOccurred = false;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   263
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   264
            for (long l1 = lowerBoundA; l1 <= upperBoundA; l1++) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   265
                for (long l2 = lowerBoundB; l2 <= upperBoundB; l2++) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   266
                    try {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   267
                        long res = mulExact(l1, l2, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   268
                        Assert.assertTrue(resultStamp.contains(res));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   269
                    } catch (ArithmeticException e) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   270
                        overflowOccurred = true;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   271
                    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   272
                    if (l2 == Long.MAX_VALUE) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   273
                        // do not want to overflow the check loop
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   274
                        break;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   275
                    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   276
                }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   277
                if (l1 == Long.MAX_VALUE) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   278
                    // do not want to overflow the check loop
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   279
                    break;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   280
                }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   281
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   282
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   283
            Assert.assertEquals(overflowExpected, overflowOccurred);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   284
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   285
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   286
        private static long mulExact(long x, long y, int bits) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   287
            if (bits == 32) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   288
                return Math.multiplyExact((int) x, (int) y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   289
            } else {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   290
                return Math.multiplyExact(x, y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   291
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   292
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   293
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   294
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   295
        public static int snippetInt32(int a, int b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   296
            return Math.multiplyExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   297
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   298
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   299
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   300
        public static long snippetInt64(long a, long b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   301
            return Math.multiplyExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   302
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   303
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   304
}