test/jdk/com/sun/jdi/SuspendThreadTest.java
author iignatyev
Fri, 14 Sep 2018 14:02:57 -0700
changeset 51754 594919232b8f
parent 47216 71c04702a3d5
permissions -rw-r--r--
8210732: remove jdk.testlibrary.Utils Reviewed-by: alanb, jcbeyler

/*
 * Copyright (c) 2006, 2015, 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 6485605
 * @summary com.sun.jdi.InternalException: Inconsistent suspend policy in internal event handler
 * @author jjh
 *
 * @run build TestScaffold VMConnection TargetListener TargetAdapter
 * @run compile -g SuspendThreadTest.java
 * @run driver SuspendThreadTest
 */
import com.sun.jdi.*;
import com.sun.jdi.event.*;
import com.sun.jdi.request.*;


    /********** target program **********/

class SuspendThreadTarg {
    public static long count;
    public static boolean active = true;

    public static void bkpt() {
        count++;
    }

    public static void main(String[] args){
        System.out.println("Howdy!");

        // We need this to be running so the bkpt
        // can be hit immediately when it is enabled
        // in the back-end.
        while(active) {
            bkpt();
        }
        System.out.println("Goodbye from SuspendThreadTarg, count = " + count);
    }
}

    /********** test program **********/

public class SuspendThreadTest extends TestScaffold {
    ClassType targetClass;
    ThreadReference mainThread;

    SuspendThreadTest (String args[]) {
        super(args);
    }

    public static void main(String[] args)      throws Exception {
        new SuspendThreadTest(args).startTests();
    }

    /********** event handlers **********/

    // 1000 makes the test take over 2 mins on win32
    static int maxBkpts = 200;
    volatile int bkptCount;
    // to guard against spurious wakeups from bkptSignal.wait()
    boolean signalSent;
    // signal that a breakpoint has happened
    final private Object bkptSignal = new Object() {};
    BreakpointRequest bkptRequest;
    Field debuggeeCountField, debuggeeActiveField;

    // When we get a bkpt we want to disable the request,
    // resume the debuggee, and then re-enable the request
    public void breakpointReached(BreakpointEvent event) {
        System.out.println("Got BreakpointEvent: " + bkptCount +
                           ", debuggeeCount = " +
                           ((LongValue)targetClass.
                            getValue(debuggeeCountField)).value()
                           );
        bkptRequest.disable();
    }

    public void eventSetComplete(EventSet set) {
        set.resume();

        // The main thread watchs the bkptCount to
        // see if bkpts stop coming in.  The
        // test _should_ fail well before maxBkpts bkpts.
        synchronized (bkptSignal) {
            if (bkptCount++ < maxBkpts) {
                bkptRequest.enable();
            }
            signalSent = true;
            bkptSignal.notifyAll();
        }
    }

    public void vmDisconnected(VMDisconnectEvent event) {
        println("Got VMDisconnectEvent");
    }

    /********** test core **********/

    protected void runTests() throws Exception {
        try {
            /*
             * Get to the top of main()
             * to determine targetClass and mainThread
             */
            BreakpointEvent bpe = startToMain("SuspendThreadTarg");
            targetClass = (ClassType)bpe.location().declaringType();
            mainThread = bpe.thread();
            EventRequestManager erm = vm().eventRequestManager();

            Location loc1 = findMethod(targetClass, "bkpt", "()V").location();

            bkptRequest = erm.createBreakpointRequest(loc1);

            // Without this, it is a SUSPEND_ALL bkpt and the test will pass
            bkptRequest.setSuspendPolicy(EventRequest.SUSPEND_EVENT_THREAD);
            bkptRequest.enable();

            debuggeeCountField = targetClass.fieldByName("count");
            debuggeeActiveField = targetClass.fieldByName("active");
            try {
                addListener (this);
            } catch (Exception ex){
                ex.printStackTrace();
                failure("failure: Could not add listener");
                throw new Exception("SuspendThreadTest: failed", ex);
            }

            int prevBkptCount;
            vm().resume();
            synchronized (bkptSignal) {
                while (bkptCount < maxBkpts) {
                    prevBkptCount = bkptCount;
                    // If we don't get a bkpt within 5 secs,
                    // the test fails
                    signalSent = false;
                    do {
                        try {
                            bkptSignal.wait(5000);
                        } catch (InterruptedException ee) {
                        }
                    } while (signalSent == false);
                    if (prevBkptCount == bkptCount) {
                        failure("failure: test hung");
                        break;
                    }
                }
            }
            println("done with loop");
            bkptRequest.disable();
            removeListener(this);

            /*
             * deal with results of test
             * if anything has called failure("foo") testFailed will be true
             */
            if (!testFailed) {
                println("SuspendThreadTest: passed");
            } else {
                throw new Exception("SuspendThreadTest: failed");
            }
        } finally {
            if (targetClass != null && debuggeeActiveField != null) {
                targetClass.setValue(debuggeeActiveField, vm().mirrorOf(false));
            }
        }
    }
}