test/jdk/com/sun/jdi/JdbMethodExitTest.java
author amenkov
Fri, 12 Oct 2018 10:08:11 -0700
changeset 52110 65efb9c57fef
parent 51596 b8eea2a7569a
permissions -rw-r--r--
8195703: BasicJDWPConnectionTest.java: 'App exited unexpectedly with 2' Reviewed-by: sspitsyn, jcbeyler

/*
 * Copyright (c) 2004, 2018, 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 6202891
 * @summary TTY: Add support for method exit event return values to jdb
 * @comment converted from test/jdk/com/sun/jdi/JdbMethodExitTest.sh
 *
 * @library /test/lib
 * @compile -g JdbMethodExitTest.java
 * @run main/othervm JdbMethodExitTest
 */

import jdk.test.lib.process.OutputAnalyzer;
import lib.jdb.JdbCommand;
import lib.jdb.JdbTest;

import java.util.*;
import java.net.URLClassLoader;
import java.net.URL;
import java.util.stream.Collectors;

/*
 * This tests the jdb trace command
 */

class JdbMethodExitTestTarg {
    // These are the values that will be returned by the methods
    static URL[] urls = new URL[1];
    public static byte      byteValue = 89;
    public static char      charValue = 'x';
    public static double    doubleValue = 2.2;
    public static float     floatValue = 3.3f;
    public static int       intValue = 1;
    public static short     shortValue = 8;
    public static boolean   booleanValue = false;

    public static Class       classValue = Object.class;
    public static ClassLoader classLoaderValue;
    {
        try {
            urls[0] = new URL("file:/foo");
        } catch (java.net.MalformedURLException ee) {
        }
        classLoaderValue = new URLClassLoader(urls);
    }

    public static Thread      threadValue;
    public static ThreadGroup threadGroupValue;
    public static String      stringValue = "abc";
    public static int[]       intArrayValue = new int[] {1, 2, 3};

    public static JdbMethodExitTestTarg  objectValue =
        new JdbMethodExitTestTarg();
    public String ivar = stringValue;

    // These are the instance methods
    public byte i_bytef()            { return byteValue; }
    public char i_charf()            { return charValue; }
    public double i_doublef()        { return doubleValue; }
    public float i_floatf()          { return floatValue; }
    public int i_intf()              { return intValue; }
    public short i_shortf()          { return shortValue; }
    public boolean i_booleanf()      { return booleanValue; }
    public String i_stringf()        { return stringValue; }
    public Class i_classf()          { return classValue; }
    public ClassLoader i_classLoaderf()
                                     { return classLoaderValue; }
    public Thread i_threadf()        { return threadValue = Thread.currentThread(); }
    public ThreadGroup i_threadGroupf()
                                     { return threadGroupValue = threadValue.getThreadGroup(); }
    public int[] i_intArrayf()       { return intArrayValue; }
    public Object i_nullObjectf()    { return null; }
    public Object i_objectf()        { return objectValue; }
    public void i_voidf()            {}

    static void doit(JdbMethodExitTestTarg xx) {

        xx.i_bytef();
        xx.i_charf();
        xx.i_doublef();
        xx.i_floatf();
        xx.i_intf();
        xx.i_shortf();
        xx.i_booleanf();
        xx.i_stringf();
        xx.i_intArrayf();
        xx.i_classf();
        xx.i_classLoaderf();
        xx.i_threadf();
        xx.i_threadGroupf();
        xx.i_nullObjectf();
        xx.i_objectf();
        xx.i_voidf();

        // Prove it works for native methods too
        StrictMath.sin(doubleValue);
        stringValue.intern();
    }

    public static void bkpt() {
       int i = 0;     //@1 breakpoint
    }

    public static String traceMethods() {
        return "traceMethods";
    }

    public static String traceMethods1() {
        return "traceMethods1";
    }

    public static String traceExits() {
        return "traceExits";
    }

    public static String traceExits1() {
        return "traceExits1";
    }

    public static String traceExit() {
        return "traceExit";
    }

    public static String traceExit1() {
        return "traceExit1";
    }

    public static void main(String[] args) {
        // The debugger will stop at the start of main,
        // enable method exit events, and then do
        // a resume.

        JdbMethodExitTestTarg xx = new JdbMethodExitTestTarg();
        System.out.println("threadid="+Thread.currentThread().getId());
        bkpt();

        // test all possible return types
        doit(xx);
        bkpt();

       // test trace methods
       traceMethods();

       // test trace go methods
       traceMethods1();
       bkpt();

       // test trace method exits
       traceExits();

       // test trace method exits
       traceExits1();
       bkpt();

       // test trace method exit
       traceExit();

       // test trace method exit
       traceExit1();
       bkpt();

    }
}

public class JdbMethodExitTest extends JdbTest {
    public static void main(String argv[]) {
        new JdbMethodExitTest().run();
    }

    private JdbMethodExitTest() {
        super(DEBUGGEE_CLASS);
    }

    private static final String DEBUGGEE_CLASS = JdbMethodExitTestTarg.class.getName();

    @Override
    protected void runCases() {
        setBreakpointsFromTestSource("JdbMethodExitTest.java", 1);

        // test all possible return types
        execCommand(JdbCommand.run())
                .shouldContain("Breakpoint hit");
        Integer threadId = Integer.parseInt(
                new OutputAnalyzer(getDebuggeeOutput())
                        .firstMatch("^threadid=(.*)$", 1));
        jdb.command(JdbCommand.untrace());

        jdb.command(JdbCommand.traceMethods(false, null));
        execCommand(JdbCommand.trace())
                .shouldContain("trace methods in effect");

        jdb.command(JdbCommand.traceMethods(true, null));
        execCommand(JdbCommand.trace())
                .shouldContain("trace go methods in effect");

        jdb.command(JdbCommand.traceMethodExits(false, null));
        execCommand(JdbCommand.trace())
                .shouldContain("trace method exits in effect");

        jdb.command(JdbCommand.traceMethodExits(true, null));
        execCommand(JdbCommand.trace())
                .shouldContain("trace go method exits in effect");

        jdb.command(JdbCommand.traceMethodExit(false, null));
        execCommand(JdbCommand.trace())
                .shouldContain("trace method exit in effect for JdbMethodExitTestTarg.bkpt");

        jdb.command(JdbCommand.traceMethodExit(true, null));
        execCommand(JdbCommand.trace())
                .shouldContain("trace go method exit in effect for JdbMethodExitTestTarg.bkpt");


        // trace exit of methods with all the return values
        // (but just check a couple of them)
        jdb.command(JdbCommand.traceMethodExits(true, threadId));
        execCommand(JdbCommand.cont())
                .shouldContain("instance of JdbMethodExitTestTarg")
                .shouldContain("return value = 8");

        // Get out of bkpt back to the call to traceMethods
        jdb.command(JdbCommand.stepUp());


        jdb.command(JdbCommand.traceMethods(false, threadId));
        execCommand(JdbCommand.cont())
                .shouldContain("Method entered:");
        execCommand(JdbCommand.cont())
                .shouldContain("Method exited: return value = \"traceMethods\"");
        jdb.command(JdbCommand.stepUp());


        List<String> reply = new LinkedList<>();
        reply.addAll(jdb.command(JdbCommand.traceMethods(true, threadId)));
        reply.addAll(jdb.command(JdbCommand.cont()));
        reply.addAll(jdb.command(JdbCommand.cont()));
        reply.addAll(jdb.command(JdbCommand.cont()));
        new OutputAnalyzer(reply.stream().collect(Collectors.joining(lineSeparator)))
                .shouldContain("Method entered: \"thread=main\", JdbMethodExitTestTarg.traceMethods1")
                .shouldMatch("Method exited: .* JdbMethodExitTestTarg.traceMethods1");
        jdb.command(JdbCommand.untrace());
        jdb.command(JdbCommand.stepUp());


        reply.clear();
        reply.addAll(jdb.command(JdbCommand.traceMethodExits(false, threadId)));
        reply.addAll(jdb.command(JdbCommand.cont()));
        new OutputAnalyzer(reply.stream().collect(Collectors.joining(lineSeparator)))
                .shouldContain("Method exited: return value = \"traceExits\"");
        jdb.command(JdbCommand.untrace());
        jdb.command(JdbCommand.stepUp());


        reply.clear();
        reply.addAll(jdb.command(JdbCommand.traceMethodExits(true, threadId)));
        reply.addAll(jdb.command(JdbCommand.cont()));
        new OutputAnalyzer(reply.stream().collect(Collectors.joining(lineSeparator)))
                .shouldMatch("Method exited: .* JdbMethodExitTestTarg.traceExits1");
        jdb.command(JdbCommand.untrace());
        jdb.command(JdbCommand.stepUp());


        reply.clear();
        reply.addAll(jdb.command(JdbCommand.step()));   // step into traceExit()
        reply.addAll(jdb.command(JdbCommand.traceMethodExit(false, threadId)));
        reply.addAll(jdb.command(JdbCommand.cont()));
        new OutputAnalyzer(reply.stream().collect(Collectors.joining(lineSeparator)))
                .shouldContain("Method exited: return value = \"traceExit\"");
        jdb.command(JdbCommand.untrace());
        jdb.command(JdbCommand.stepUp());


        reply.clear();
        reply.addAll(jdb.command(JdbCommand.step()));
        reply.addAll(jdb.command(JdbCommand.step()));   // skip over setting return value in caller :-(
        reply.addAll(jdb.command(JdbCommand.traceMethodExit(true, threadId)));
        reply.addAll(jdb.command(JdbCommand.cont()));
        new OutputAnalyzer(reply.stream().collect(Collectors.joining(lineSeparator)))
                .shouldMatch("Method exited: .*JdbMethodExitTestTarg.traceExit1");

        new OutputAnalyzer(getJdbOutput())
                .shouldContain("Breakpoint hit");
    }
}