hotspot/test/compiler/arraycopy/TestEliminatedArrayCopyDeopt.java
author thartmann
Thu, 04 Aug 2016 10:34:34 +0200
changeset 40365 3791c1b34e36
parent 40072 08c515bbbdea
permissions -rw-r--r--
8136818: Test compiler/arraycopy/TestEliminatedArrayCopyDeopt.java fails with "m1 failed" Summary: Added missing Op_StoreB check to ArrayCopyNode::may_modify_helper(). Reviewed-by: kvn

/*
 * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

/*
 * @test
 * @bug 8130847 8156760
 * @summary Eliminated instance/array written to by an array copy variant must be correctly initialized when reallocated at a deopt
 * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement
 *                   compiler.arraycopy.TestEliminatedArrayCopyDeopt
 * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement
 *                   -XX:+IgnoreUnrecognizedVMOptions -XX:-ReduceInitialCardMarks
 *                   compiler.arraycopy.TestEliminatedArrayCopyDeopt
 */

// Test that if an ArrayCopy node is eliminated because it doesn't
// escape, then the correct field/array element values are captured so
// on a deoptimization, when the object/array is reallocated, it is
// correctly initialized

package compiler.arraycopy;

public class TestEliminatedArrayCopyDeopt {

    static class A implements Cloneable {
        int f0;
        int f1;
        int f2;
        int f3;
        int f4;
        int f5;
        int f6;
        int f7;
        int f8;
        int f9;
        int f10;
        int f11;
        int f12;
        int f13;
        int f14;
        int f15;

        public Object clone() throws CloneNotSupportedException {
            return super.clone();
        }
    }

    // Clone
    static boolean m1(A a, boolean flag) throws CloneNotSupportedException {
        A c = (A)a.clone();
        if (flag) {
            // never taken branch that causes the deoptimization
            if (c.f0 != 0x42) {
                return false;
            }
        }
        return true;
    }

    // Array clone
    static int[] m2_src = null;
    static boolean m2(boolean flag) throws CloneNotSupportedException {
        int[] src  = new int[10];
        m2_src = src;
        for (int i = 0; i < src.length; i++) {
            src[i] = 0x42+i;
        }
        int[] c = (int[])src.clone();
        if (flag) {
            for (int i = 0; i < c.length; i++) {
                if (c[i] != src[i]) {
                    return false;
                }
            }
        }
        return true;
    }

    // Array copy
    static boolean m3(int[] src, boolean flag) {
        int[] dst = new int[10];
        System.arraycopy(src, 0, dst, 0, 10);
        if (flag) {
            for (int i = 0; i < dst.length; i++) {
                if (dst[i] != src[i]) {
                    return false;
                }
            }
        }
        return true;
    }

    // Array copy of subrange
    static boolean m4(int[] src, boolean flag) {
        int[] dst = new int[10];
        dst[0] = 0x42;
        dst[1] = 0x42 - 1;
        dst[2] = 0x42 - 2;
        dst[8] = 0x42 - 8;
        dst[9] = 0x42 - 9;
        int src_off = 2;
        int dst_off = 3;
        int len = 5;
        System.arraycopy(src, src_off, dst, dst_off, len);
        if (flag) {
            for (int i = 0; i < dst.length; i++) {
                if (i >= dst_off &&  i < dst_off + len) {
                    if (dst[i] != src[i - dst_off + src_off]) {
                        return false;
                    }
                } else {
                    if (dst[i] != 0x42-i) {
                        return false;
                    }
                }
            }
        }
        return true;
    }

    // Array copy with Phi
    static boolean m5(int[] src, boolean flag1, boolean flag2) {
        int[] dst = new int[10];
        if (flag1) {
            System.arraycopy(src, 0, dst, 0, 10);
        }
        if (flag2) {
            for (int i = 0; i < dst.length; i++) {
                if (dst[i] != src[i]) {
                    return false;
                }
            }
        }
        return true;
    }

    static public void main(String[] args) throws Exception {
        boolean success = true;
        A a = new A();
        a.f0 = 0x42;
        for (int i = 0; i < 20000; i++) {
            m1(a, false);
        }
        if (!m1(a, true)) {
            System.out.println("m1 failed");
            success = false;
        }

        for (int i = 0; i < 20000; i++) {
            m2(false);
        }
        if (!m2(true)) {
            System.out.println("m2 failed");
            success = false;
        }

        int[] src = new int[10];
        for (int i = 0; i < src.length; i++) {
            src[i] = 0x42+i;
        }

        for (int i = 0; i < 20000; i++) {
            m3(src, false);
        }
        if (!m3(src, true)) {
            System.out.println("m3 failed");
            success = false;
        }

        for (int i = 0; i < 20000; i++) {
            m4(src, false);
        }
        if (!m4(src, true)) {
            System.out.println("m4 failed");
            success = false;
        }

        for (int i = 0; i < 20000; i++) {
            m5(src, i%2 == 0, false);
        }
        if (!m5(src, true, true)) {
            System.out.println("m4 failed");
            success = false;
        }

        if (!success) {
            throw new RuntimeException("Test failed");
        }
    }
}