File: TestNestedRelockAtDeopt.java

package info (click to toggle)
openjdk-25 25.0.1%2B8-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 825,408 kB
  • sloc: java: 5,585,680; cpp: 1,333,948; xml: 1,321,242; ansic: 488,034; asm: 404,003; objc: 21,088; sh: 15,106; javascript: 13,265; python: 8,319; makefile: 2,518; perl: 357; awk: 351; pascal: 103; exp: 83; sed: 72; jsp: 24
file content (146 lines) | stat: -rw-r--r-- 5,436 bytes parent folder | download | duplicates (6)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
/*
 * Copyright (c) 2024, 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 8324174
 * @summary During deoptimization locking and unlocking for nested locks are executed in incorrect order.
 * @requires vm.compMode != "Xint"
 * @run main/othervm -XX:-TieredCompilation -XX:-BackgroundCompilation -Xmx128M
 *                   -XX:CompileCommand=exclude,TestNestedRelockAtDeopt::main TestNestedRelockAtDeopt
 */

import java.util.ArrayList;
public class TestNestedRelockAtDeopt {

    static final int CHUNK = 1000;
    static ArrayList<Object> arr = null;

    public static void main(String[] args) {
        arr = new ArrayList<>();
        try {
            while (true) {
                test1();
            }
        } catch (OutOfMemoryError oom) {
            arr = null; // Free memory
            System.out.println("OOM caught in test1");
        }
        arr = new ArrayList<>();
        try {
            while (true) {
                test2();
            }
        } catch (OutOfMemoryError oom) {
            arr = null; // Free memory
            System.out.println("OOM caught in test2");
        }
        arr = new ArrayList<>();
        TestNestedRelockAtDeopt obj = new TestNestedRelockAtDeopt();
        try {
            while (true) {
                test3(obj);
            }
        } catch (OutOfMemoryError oom) {
            arr = null; // Free memory
            System.out.println("OOM caught in test3");
        }
        arr = new ArrayList<>();
        try {
            while (true) {
                test4(obj);
            }
        } catch (OutOfMemoryError oom) {
            arr = null; // Free memory
            System.out.println("OOM caught in test4");
        }
    }

    // Nested locks in one method
    static void test1() { // Nested lock in one method
        synchronized (TestNestedRelockAtDeopt.class) {
            synchronized (new TestNestedRelockAtDeopt()) { // lock eliminated - not escaped allocation
                synchronized (TestNestedRelockAtDeopt.class) { // nested lock eliminated
                    synchronized (new TestNestedRelockAtDeopt()) { // lock eliminated - not escaped allocation
                        synchronized (TestNestedRelockAtDeopt.class) { // nested lock eliminated
                            arr.add(new byte[CHUNK]);
                        }
                    }
                }
            }
        }
    }

    // Nested locks in inlined method
    static void foo() {
        synchronized (new TestNestedRelockAtDeopt()) {  // lock eliminated - not escaped allocation
            synchronized (TestNestedRelockAtDeopt.class) {  // nested lock eliminated when inlined
                arr.add(new byte[CHUNK]);
            }
        }
    }

    static void test2() {
        synchronized (TestNestedRelockAtDeopt.class) {
            synchronized (new TestNestedRelockAtDeopt()) {  // lock eliminated - not escaped allocation
                synchronized (TestNestedRelockAtDeopt.class) { // nested lock eliminated
                    foo(); // Inline
                }
            }
        }
    }

    // Nested locks in one method
    static void test3(TestNestedRelockAtDeopt obj) {
        synchronized (TestNestedRelockAtDeopt.class) {
            synchronized (obj) { // lock not eliminated - external object
                synchronized (TestNestedRelockAtDeopt.class) { // nested lock eliminated
                    synchronized (obj) { // nested lock eliminated
                        synchronized (TestNestedRelockAtDeopt.class) { // nested lock eliminated
                            arr.add(new byte[CHUNK]);
                        }
                    }
                }
            }
        }
    }

    // Nested locks with different objects in inlined method
    static void bar(TestNestedRelockAtDeopt obj) {
        synchronized (obj) {  // nested lock eliminated when inlined
            synchronized (TestNestedRelockAtDeopt.class) {  // nested lock eliminated when inlined
                arr.add(new byte[CHUNK]);
            }
        }
    }

    static void test4(TestNestedRelockAtDeopt obj) {
        synchronized (TestNestedRelockAtDeopt.class) {
            synchronized (obj) {  // lock not eliminated - external object
                synchronized (TestNestedRelockAtDeopt.class) { // nested lock eliminated
                    bar(obj); // Inline
                }
            }
        }
    }
}