/* * 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)); } } } }