e48cfc709d
Reviewed-by: erikj, kvn, sspitsyn
285 lines
9.5 KiB
Java
285 lines
9.5 KiB
Java
/*
|
|
* Copyright (c) 2001, 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.
|
|
*/
|
|
|
|
package nsk.jdi.ThreadReference.interrupt;
|
|
|
|
import nsk.share.jpda.*;
|
|
import nsk.share.jdi.*;
|
|
|
|
/**
|
|
* This class is used as debuggee application for the interrupt001 JDI test.
|
|
*/
|
|
|
|
public class interrupt001a {
|
|
|
|
//----------------------------------------------------- templete section
|
|
|
|
static final int PASSED = 0;
|
|
|
|
static final int FAILED = 2;
|
|
|
|
static final int PASS_BASE = 95;
|
|
|
|
//-------------------------------------------------- log procedures
|
|
|
|
static boolean verbMode = false;
|
|
|
|
public static void log1(String message) {
|
|
if (verbMode)
|
|
System.err.println("**> mainThread: " + message);
|
|
}
|
|
|
|
public static void log2(String message) {
|
|
if (verbMode)
|
|
System.err.println("**> " + message);
|
|
}
|
|
|
|
private static void logErr(String message) {
|
|
if (verbMode)
|
|
System.err.println("ERROR: " + message);
|
|
}
|
|
|
|
//====================================================== test program
|
|
|
|
// scaffold objects
|
|
private static volatile ArgumentHandler argumentHandler = null;
|
|
|
|
private static interrupt001aThread thread2 = null;
|
|
|
|
private static interrupt001aThread thread3 = null;
|
|
|
|
private static IOPipe pipe;
|
|
|
|
//---------------------------------------------------- main method
|
|
|
|
public static void main(String argv[]) {
|
|
|
|
int exitCode = PASSED;
|
|
|
|
argumentHandler = new ArgumentHandler(argv);
|
|
|
|
for (int i = 0; i < argv.length; i++) {
|
|
if (argv[i].equals("-vbs") || argv[i].equals("-verbose")) {
|
|
verbMode = true;
|
|
break;
|
|
}
|
|
}
|
|
log1("debuggee started!");
|
|
|
|
// informing a debugger of readyness
|
|
pipe = argumentHandler.createDebugeeIOPipe();
|
|
pipe.println("ready");
|
|
|
|
label0: for (int i = 0;; i++) {
|
|
|
|
log1("waiting for an instruction from the debugger ...");
|
|
String instruction = pipe.readln();
|
|
|
|
if (instruction.equals("quit")) {
|
|
log1("'quit' recieved");
|
|
break;
|
|
|
|
} else if (instruction.equals("newcheck")) {
|
|
|
|
switch (i) {
|
|
case 0:
|
|
synchronized (interrupt001aThread.lockingObject) {
|
|
thread2 = threadStart("Thread02");
|
|
thread3 = threadStart("Thread03");
|
|
|
|
pipe.println("checkready");
|
|
if (checkInterruptStatus() == FAILED) {
|
|
exitCode = FAILED;
|
|
break label0;
|
|
}
|
|
}
|
|
log1("mainThread is out of: synchronized (lockingObject)");
|
|
|
|
if (waitThreadJoin(thread2) == FAILED) {
|
|
exitCode = FAILED;
|
|
}
|
|
if (waitThreadJoin(thread3) == FAILED) {
|
|
exitCode = FAILED;
|
|
}
|
|
|
|
instruction = pipe.readln();
|
|
if (!instruction.equals("continue")) {
|
|
logErr("Unexpected instruction #1: " + instruction);
|
|
exitCode = FAILED;
|
|
break label0;
|
|
}
|
|
pipe.println("docontinue");
|
|
break;
|
|
|
|
case 1:
|
|
synchronized (interrupt001aThread.lockingObject) {
|
|
thread2 = threadStart("Thread12");
|
|
thread3 = threadStart("Thread13");
|
|
|
|
log1("suspending Thread2");
|
|
thread2.suspend();
|
|
|
|
log1("suspending Thread3");
|
|
thread3.suspend();
|
|
|
|
log1("interrupting the Thread3");
|
|
thread3.interrupt();
|
|
|
|
pipe.println("checkready");
|
|
if (checkInterruptStatus() == FAILED) {
|
|
exitCode = FAILED;
|
|
break label0;
|
|
}
|
|
}
|
|
log1("mainThread is out of: synchronized (lockingObject)");
|
|
|
|
log1("resuming Thread2");
|
|
thread2.resume();
|
|
if (waitThreadJoin(thread2) == FAILED) {
|
|
exitCode = FAILED;
|
|
}
|
|
log1("resuming Thread3");
|
|
thread3.resume();
|
|
if (waitThreadJoin(thread3) == FAILED) {
|
|
exitCode = FAILED;
|
|
}
|
|
|
|
instruction = pipe.readln();
|
|
if (!instruction.equals("continue")) {
|
|
logErr("Unexpected instruction #2: " + instruction);
|
|
exitCode = FAILED;
|
|
break label0;
|
|
}
|
|
pipe.println("docontinue");
|
|
break;
|
|
|
|
//------------------------------------------------- standard end section
|
|
}
|
|
|
|
} else {
|
|
logErr("Unexpected instruction #0: " + instruction);
|
|
exitCode = FAILED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
System.exit(exitCode + PASS_BASE);
|
|
}
|
|
|
|
private static interrupt001aThread threadStart(String threadName) {
|
|
interrupt001aThread resultThread = new interrupt001aThread(threadName);
|
|
synchronized (resultThread.waitnotifyObj) {
|
|
resultThread.start();
|
|
try {
|
|
log1(" before: waitnotifyObj.wait();");
|
|
while (!resultThread.ready)
|
|
resultThread.waitnotifyObj.wait();
|
|
log1(" after: waitnotifyObj.wait();");
|
|
} catch (InterruptedException e) {
|
|
logErr("Unexpected InterruptedException while waiting for start of : " + threadName);
|
|
}
|
|
}
|
|
return resultThread;
|
|
}
|
|
|
|
private static int waitThreadJoin(Thread testedThread) {
|
|
long timeout = argumentHandler.getWaitTime() * 60 * 1000; // milliseconds
|
|
|
|
if (testedThread.isAlive()) {
|
|
// wait for thread finished in a waittime interval
|
|
log1("Waiting for tested thread " + testedThread.getName() + " finished for timeout: " + timeout);
|
|
try {
|
|
testedThread.join(timeout);
|
|
} catch (InterruptedException e) {
|
|
logErr("Interruption while waiting for tested thread " + testedThread.getName() + " finished:\n\t" + e);
|
|
return interrupt001a.FAILED;
|
|
}
|
|
}
|
|
return interrupt001a.PASSED;
|
|
}
|
|
|
|
private static int checkInterruptStatus() {
|
|
log1("waiting for 'check_interruption'");
|
|
String instruction = pipe.readln();
|
|
if (!instruction.equals("check_interruption")) {
|
|
logErr("Unexpected instruction #3: " + instruction);
|
|
return FAILED;
|
|
}
|
|
|
|
log1("interrupting thread3");
|
|
thread3.interrupt();
|
|
|
|
log1("checking up statuses of (checked) Thread2 and Thread3");
|
|
if (thread2.isInterrupted()) {
|
|
if (thread3.isInterrupted()) {
|
|
pipe.println(interrupt001.EQUALS_INTERRUPTED);
|
|
log1(" " + interrupt001.EQUALS_INTERRUPTED);
|
|
} else {
|
|
pipe.println(interrupt001.ONLY_CHECKED_INTERRUPTED);
|
|
log1(" " + interrupt001.ONLY_CHECKED_INTERRUPTED);
|
|
}
|
|
} else if (thread3.isInterrupted()) {
|
|
pipe.println(interrupt001.CHECKED_NOT_INTERRUPTED);
|
|
log1(" " + interrupt001.CHECKED_NOT_INTERRUPTED);
|
|
} else {
|
|
pipe.println(interrupt001.EQUALS_NOT_INTERRUPTED);
|
|
log1(" " + interrupt001.EQUALS_NOT_INTERRUPTED);
|
|
}
|
|
return PASSED;
|
|
}
|
|
}
|
|
|
|
class interrupt001aThread extends Thread {
|
|
|
|
public interrupt001aThread(String threadName) {
|
|
super(threadName);
|
|
}
|
|
|
|
public boolean ready;
|
|
|
|
public Object waitnotifyObj = new Object();
|
|
|
|
public static Object lockingObject = new Object();
|
|
|
|
public void run() {
|
|
log("started");
|
|
synchronized (waitnotifyObj) {
|
|
log("entered into block: synchronized (waitnotifyObj)");
|
|
ready = true;
|
|
waitnotifyObj.notifyAll();
|
|
}
|
|
|
|
synchronized (lockingObject) {
|
|
log("entered into block: synchronized (lockingObject)");
|
|
}
|
|
log("exited from block: synchronized (lockingObject)");
|
|
|
|
log("exited");
|
|
return;
|
|
}
|
|
|
|
void log(String str) {
|
|
interrupt001a.log2(Thread.currentThread().getName() + " : " + str);
|
|
}
|
|
}
|