File: StackWalkNativeToJava.java

package info (click to toggle)
openjdk-21 21.0.10~4ea-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 823,260 kB
  • sloc: java: 5,638,490; xml: 1,643,607; cpp: 1,298,236; ansic: 421,374; asm: 404,850; objc: 21,018; sh: 15,085; javascript: 11,217; python: 6,895; makefile: 2,368; perl: 357; awk: 351; sed: 172; jsp: 24
file content (134 lines) | stat: -rw-r--r-- 5,975 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
/*
 * Copyright (c) 2023, 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.
 *
 */

import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
import jdk.test.lib.Utils;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

/*
 * @test StackWalkNativeToJava
 * @bug 8316309
 * @summary Check that walking the stack works fine when going from C++ frame to Java frame.
 * @requires os.arch=="amd64" | os.arch=="x86_64" | os.arch=="aarch64"
 * @requires os.family != "windows"
 * @requires vm.flagless
 * @library /test/lib
 * @run driver StackWalkNativeToJava
 */

public class StackWalkNativeToJava {

    public static void main(String[] args) throws Exception {
        // Check stack walking works fine when sender of C++ frame
        // is a Java native method.
        testStackWalkNativeToJavaNative("-Xint");
        testStackWalkNativeToJavaNative("-Xcomp", "-XX:CompileCommand=dontinline,StackWalkNativeToJava$TestNativeToJavaNative::*");

        // Check stack walking works fine when sender of C++ frame
        // is a runtime stub or interpreted Java method (VM call from Java).
        testStackWalkNativeToJava("-Xint");
        testStackWalkNativeToJava("-Xcomp", "-XX:TieredStopAtLevel=3",
                                  "-XX:CompileCommand=dontinline,StackWalkNativeToJava$TestNativeToJava::*");
    }

    public static void testStackWalkNativeToJavaNative(String... extraFlags) throws Exception {
        List<String> commands = new ArrayList<>();
        commands.add("-Xbootclasspath/a:.");
        commands.add("-XX:-CreateCoredumpOnCrash");
        commands.add("-XX:+UnlockDiagnosticVMOptions");
        commands.add("-XX:AbortVMOnException=java.lang.IllegalMonitorStateException");
        commands.add("-XX:+ErrorFileToStdout");
        commands.addAll(Arrays.asList(extraFlags));
        commands.add("StackWalkNativeToJava$TestNativeToJavaNative");

        ProcessBuilder pb = ProcessTools.createLimitedTestJavaProcessBuilder(commands);
        OutputAnalyzer output = new OutputAnalyzer(pb.start());
        output.shouldNotContain("java.lang.RuntimeException: Reached statement after obj.wait()");
        output.shouldNotContain("[error occurred during error reporting (printing native stack");
        String[] res = output.getOutput().split("StackWalkNativeToJava\\$TestNativeToJavaNative\\.callNativeMethod\\(\\)V");
        assertTrue(res.length - 1 == 2, res.length - 1);
        output.shouldNotHaveExitValue(0);
    }

    public static class TestNativeToJavaNative {
        public static void main(String[] args) throws Exception {
            TestNativeToJavaNative test = new TestNativeToJavaNative();
            test.callNativeMethod();
        }

        public void callNativeMethod() throws Exception {
            Object obj = new Object();
            // Trigger a fatal exit due to IllegalMonitorStateException during
            // a call to the VM from a Java native method.
            obj.wait();
            throw new RuntimeException("Reached statement after obj.wait()");
        }
    }

    public static void testStackWalkNativeToJava(String... extraFlags) throws Exception {
        List<String> commands = new ArrayList<>();
        commands.add("-Xbootclasspath/a:.");
        commands.add("-XX:-CreateCoredumpOnCrash");
        commands.add("-XX:+UnlockDiagnosticVMOptions");
        commands.add("-XX:DiagnoseSyncOnValueBasedClasses=1");
        commands.add("-XX:+ErrorFileToStdout");
        commands.addAll(Arrays.asList(extraFlags));
        commands.add("StackWalkNativeToJava$TestNativeToJava");

        ProcessBuilder pb = ProcessTools.createLimitedTestJavaProcessBuilder(commands);
        OutputAnalyzer output = new OutputAnalyzer(pb.start());
        output.shouldNotContain("java.lang.RuntimeException: Reached statement after synchronized");
        output.shouldNotContain("[error occurred during error reporting (printing native stack");
        String[] res = output.getOutput().split("StackWalkNativeToJava\\$TestNativeToJava\\.callVMMethod\\(\\)V");
        assertTrue(res.length - 1 == 2, res.length - 1);
        output.shouldNotHaveExitValue(0);
    }

    public static class TestNativeToJava {
        static Integer counter = 0;

        public static void main(String[] args) throws Exception {
            TestNativeToJava test = new TestNativeToJava();
            test.callVMMethod();
        }

        public void callVMMethod() throws Exception {
            // Trigger a fatal exit for trying to synchronize on a value based class
            // during a call to the VM from a Java method.
            synchronized (counter) {
                counter++;
            }
            throw new RuntimeException("Reached statement after synchronized");
        }
    }

    private static void assertTrue(boolean condition, int count) {
        if (!condition) {
            throw new RuntimeException("Count error: count was " + count);
        }
    }
}