src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.test/src/org/graalvm/compiler/replacements/test/IntegerExactFoldTest.java
author never
Wed, 27 Jun 2018 17:02:41 -0700
changeset 50858 2d3e99a72541
parent 48190 25cfedf27edc
child 52910 583fd71c47d6
permissions -rw-r--r--
8205824: Update Graal Reviewed-by: iveresov, kvn
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     1
/*
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
     2
 * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
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.StructuredGraph.GuardsStage;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    44
import org.graalvm.compiler.nodes.ValueNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    45
import org.graalvm.compiler.nodes.spi.LoweringTool;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    46
import org.graalvm.compiler.phases.common.CanonicalizerPhase;
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;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    49
import org.graalvm.compiler.phases.tiers.PhaseContext;
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
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    97
        new CanonicalizerPhase().apply(graph, getDefaultHighTierContext());
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    98
        ValueNode node = findNode(graph);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
    99
        boolean overflowExpected = node instanceof IntegerExactArithmeticNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   100
48190
25cfedf27edc 8192814: Update Graal
dlong
parents: 47216
diff changeset
   101
        IntegerStamp resultStamp = (IntegerStamp) node.stamp(NodeView.DEFAULT);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   102
        operation.verifyOverflow(lowerBoundA, upperBoundA, lowerBoundB, upperBoundB, bits, overflowExpected, resultStamp);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   103
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   104
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   105
    @Test
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   106
    public void testFoldingAfterLowering() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   107
        StructuredGraph graph = prepareGraph();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   108
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   109
        Node originalNode = graph.getNodes().filter(x -> x instanceof IntegerExactArithmeticNode).first();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   110
        assertNotNull("original node must be in the graph", originalNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   111
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   112
        graph.setGuardsStage(GuardsStage.FIXED_DEOPTS);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   113
        CanonicalizerPhase canonicalizer = new CanonicalizerPhase();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   114
        PhaseContext context = new PhaseContext(getProviders());
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   115
        new LoweringPhase(canonicalizer, LoweringTool.StandardLoweringStage.HIGH_TIER).apply(graph, context);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   116
        IntegerExactArithmeticSplitNode loweredNode = graph.getNodes().filter(IntegerExactArithmeticSplitNode.class).first();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   117
        assertNotNull("the lowered node must be in the graph", loweredNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   118
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   119
        loweredNode.getX().setStamp(StampFactory.forInteger(bits, lowerBoundA, upperBoundA));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   120
        loweredNode.getY().setStamp(StampFactory.forInteger(bits, lowerBoundB, upperBoundB));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   121
        new CanonicalizerPhase().apply(graph, context);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   122
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   123
        ValueNode node = findNode(graph);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   124
        boolean overflowExpected = node instanceof IntegerExactArithmeticSplitNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   125
48190
25cfedf27edc 8192814: Update Graal
dlong
parents: 47216
diff changeset
   126
        IntegerStamp resultStamp = (IntegerStamp) node.stamp(NodeView.DEFAULT);
46344
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   127
        operation.verifyOverflow(lowerBoundA, upperBoundA, lowerBoundB, upperBoundB, bits, overflowExpected, resultStamp);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   128
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   129
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   130
    private static boolean isInteger(long value) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   131
        return value >= Integer.MIN_VALUE && value <= Integer.MAX_VALUE;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   132
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   133
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   134
    private static ValueNode findNode(StructuredGraph graph) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   135
        ValueNode resultNode = graph.getNodes().filter(ReturnNode.class).first().result();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   136
        assertNotNull("some node must be the returned value", resultNode);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   137
        return resultNode;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   138
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   139
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   140
    protected StructuredGraph prepareGraph() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   141
        String snippet = "snippetInt" + bits;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   142
        StructuredGraph graph = parseEager(getResolvedJavaMethod(operation.getClass(), snippet), AllowAssumptions.NO);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   143
        HighTierContext context = getDefaultHighTierContext();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   144
        new CanonicalizerPhase().apply(graph, context);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   145
        return graph;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   146
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   147
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   148
    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
   149
        tests.add(new Object[]{lowerBound1, upperBound1, lowerBound2, upperBound2, bits, operation});
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   150
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   151
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   152
    @Parameters(name = "a[{0} / {1}], b[{2} / {3}], bits={4}, operation={5}")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   153
    public static Collection<Object[]> data() {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   154
        ArrayList<Object[]> tests = new ArrayList<>();
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   155
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   156
        Operation[] operations = new Operation[]{new AddOperation(), new SubOperation(), new MulOperation()};
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   157
        for (Operation operation : operations) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   158
            for (int bits : new int[]{32, 64}) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   159
                // zero related
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   160
                addTest(tests, 0, 0, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   161
                addTest(tests, 1, 1, 0, 0, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   162
                addTest(tests, -1, 1, 0, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   163
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   164
                // bounds
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   165
                addTest(tests, -2, 2, 3, 3, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   166
                addTest(tests, -1, 1, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   167
                addTest(tests, -1, 1, -1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   168
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   169
                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
   170
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE + 0xF, -1, -1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   171
                addTest(tests, Integer.MAX_VALUE, Integer.MAX_VALUE, -1, -1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   172
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE, -1, -1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   173
                addTest(tests, Integer.MAX_VALUE, Integer.MAX_VALUE, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   174
                addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE, 1, 1, bits, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   175
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   176
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   177
            // bit-specific test cases
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   178
            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
   179
            addTest(tests, Integer.MIN_VALUE, Integer.MIN_VALUE + 0xF, -1, -1, 64, operation);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   180
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   181
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   182
        return tests;
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
    private abstract static class Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   186
        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
   187
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   188
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   189
    private static final class AddOperation extends Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   190
        @Override
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   191
        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
   192
            try {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   193
                long res = addExact(lowerBoundA, lowerBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   194
                resultStamp.contains(res);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   195
                res = addExact(upperBoundA, upperBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   196
                resultStamp.contains(res);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   197
                Assert.assertFalse(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   198
            } catch (ArithmeticException e) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   199
                Assert.assertTrue(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   200
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   201
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   202
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   203
        private static long addExact(long x, long y, int bits) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   204
            if (bits == 32) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   205
                return Math.addExact((int) x, (int) y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   206
            } else {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   207
                return Math.addExact(x, y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   208
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   209
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   210
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   211
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   212
        public static int snippetInt32(int a, int b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   213
            return Math.addExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   214
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   215
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   216
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   217
        public static long snippetInt64(long a, long b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   218
            return Math.addExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   219
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   220
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   221
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   222
    private static final class SubOperation extends Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   223
        @Override
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   224
        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
   225
            try {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   226
                long res = subExact(lowerBoundA, upperBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   227
                Assert.assertTrue(resultStamp.contains(res));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   228
                res = subExact(upperBoundA, lowerBoundB, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   229
                Assert.assertTrue(resultStamp.contains(res));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   230
                Assert.assertFalse(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   231
            } catch (ArithmeticException e) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   232
                Assert.assertTrue(overflowExpected);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   233
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   234
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   235
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   236
        private static long subExact(long x, long y, int bits) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   237
            if (bits == 32) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   238
                return Math.subtractExact((int) x, (int) y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   239
            } else {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   240
                return Math.subtractExact(x, y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   241
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   242
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   243
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   244
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   245
        public static int snippetInt32(int a, int b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   246
            return Math.subtractExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   247
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   248
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   249
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   250
        public static long snippetInt64(long a, long b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   251
            return Math.subtractExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   252
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   253
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   254
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   255
    private static final class MulOperation extends Operation {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   256
        @Override
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   257
        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
   258
            // now check for all values in the stamp whether their products overflow overflow
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   259
            boolean overflowOccurred = false;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   260
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   261
            for (long l1 = lowerBoundA; l1 <= upperBoundA; l1++) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   262
                for (long l2 = lowerBoundB; l2 <= upperBoundB; l2++) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   263
                    try {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   264
                        long res = mulExact(l1, l2, bits);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   265
                        Assert.assertTrue(resultStamp.contains(res));
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   266
                    } catch (ArithmeticException e) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   267
                        overflowOccurred = true;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   268
                    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   269
                    if (l2 == Long.MAX_VALUE) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   270
                        // do not want to overflow the check loop
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   271
                        break;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   272
                    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   273
                }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   274
                if (l1 == Long.MAX_VALUE) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   275
                    // do not want to overflow the check loop
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   276
                    break;
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   277
                }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   278
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   279
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   280
            Assert.assertEquals(overflowExpected, overflowOccurred);
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
        private static long mulExact(long x, long y, int bits) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   284
            if (bits == 32) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   285
                return Math.multiplyExact((int) x, (int) y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   286
            } else {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   287
                return Math.multiplyExact(x, y);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   288
            }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   289
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   290
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   291
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   292
        public static int snippetInt32(int a, int b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   293
            return Math.multiplyExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   294
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   295
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   296
        @SuppressWarnings("unused")
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   297
        public static long snippetInt64(long a, long b) {
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   298
            return Math.multiplyExact(a, b);
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   299
        }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   300
    }
694c102fd8ed 8177046: Update Graal
iveresov
parents:
diff changeset
   301
}