mirror of
https://github.com/openjdk/jdk.git
synced 2025-09-18 01:54:47 +02:00
8205878: pthread_getcpuclockid is expected to return 0 code
Reviewed-by: cjplummer, amenkov, coleenp
This commit is contained in:
parent
84234d1abb
commit
e9ad46f26c
6 changed files with 259 additions and 20 deletions
|
@ -836,6 +836,10 @@ BUILD_HOTSPOT_JTREG_LIBRARIES_CFLAGS_libVirtualMachine09agent00 := $(NSK_AOD_INC
|
||||||
|
|
||||||
################################################################################
|
################################################################################
|
||||||
|
|
||||||
|
ifeq ($(TOOLCHAIN_TYPE), solstudio)
|
||||||
|
BUILD_HOTSPOT_JTREG_LIBRARIES_CFLAGS_libji06t001 += -erroff=E_END_OF_LOOP_CODE_NOT_REACHED
|
||||||
|
endif
|
||||||
|
|
||||||
# Platform specific setup
|
# Platform specific setup
|
||||||
ifneq ($(OPENJDK_TARGET_OS)-$(OPENJDK_TARGET_CPU_ARCH), solaris-sparc)
|
ifneq ($(OPENJDK_TARGET_OS)-$(OPENJDK_TARGET_CPU_ARCH), solaris-sparc)
|
||||||
BUILD_HOTSPOT_JTREG_EXCLUDE += liboverflow.c exeThreadSignalMask.c
|
BUILD_HOTSPOT_JTREG_EXCLUDE += liboverflow.c exeThreadSignalMask.c
|
||||||
|
@ -858,9 +862,13 @@ endif
|
||||||
|
|
||||||
BUILD_HOTSPOT_JTREG_EXECUTABLES_LIBS_exesigtest := -ljvm
|
BUILD_HOTSPOT_JTREG_EXECUTABLES_LIBS_exesigtest := -ljvm
|
||||||
|
|
||||||
|
ifeq ($(OPENJDK_TARGET_OS), solaris)
|
||||||
|
BUILD_HOTSPOT_JTREG_EXCLUDE += libterminatedThread.c
|
||||||
|
endif
|
||||||
|
|
||||||
ifeq ($(OPENJDK_TARGET_OS), windows)
|
ifeq ($(OPENJDK_TARGET_OS), windows)
|
||||||
BUILD_HOTSPOT_JTREG_EXECUTABLES_CFLAGS_exeFPRegs := -MT
|
BUILD_HOTSPOT_JTREG_EXECUTABLES_CFLAGS_exeFPRegs := -MT
|
||||||
BUILD_HOTSPOT_JTREG_EXCLUDE += exesigtest.c
|
BUILD_HOTSPOT_JTREG_EXCLUDE += exesigtest.c libterminatedThread.c
|
||||||
|
|
||||||
else
|
else
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libbootclssearch_agent += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libbootclssearch_agent += -lpthread
|
||||||
|
@ -1494,6 +1502,7 @@ else
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libvmdeath001 += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libvmdeath001 += -lpthread
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libgetphase001 += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libgetphase001 += -lpthread
|
||||||
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libgetphase002 += -lpthread
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libgetphase002 += -lpthread
|
||||||
|
BUILD_HOTSPOT_JTREG_LIBRARIES_LIBS_libterminatedThread += -lpthread
|
||||||
endif
|
endif
|
||||||
|
|
||||||
$(eval $(call SetupTestFilesCompilation, BUILD_HOTSPOT_JTREG_LIBRARIES, \
|
$(eval $(call SetupTestFilesCompilation, BUILD_HOTSPOT_JTREG_LIBRARIES, \
|
||||||
|
|
|
@ -5555,14 +5555,18 @@ bool os::pd_unmap_memory(char* addr, size_t bytes) {
|
||||||
|
|
||||||
static jlong slow_thread_cpu_time(Thread *thread, bool user_sys_cpu_time);
|
static jlong slow_thread_cpu_time(Thread *thread, bool user_sys_cpu_time);
|
||||||
|
|
||||||
static clockid_t thread_cpu_clockid(Thread* thread) {
|
static jlong fast_cpu_time(Thread *thread) {
|
||||||
pthread_t tid = thread->osthread()->pthread_id();
|
|
||||||
clockid_t clockid;
|
clockid_t clockid;
|
||||||
|
int rc = os::Linux::pthread_getcpuclockid(thread->osthread()->pthread_id(),
|
||||||
// Get thread clockid
|
&clockid);
|
||||||
int rc = os::Linux::pthread_getcpuclockid(tid, &clockid);
|
if (rc == 0) {
|
||||||
assert(rc == 0, "pthread_getcpuclockid is expected to return 0 code");
|
return os::Linux::fast_thread_cpu_time(clockid);
|
||||||
return clockid;
|
} else {
|
||||||
|
// It's possible to encounter a terminated native thread that failed
|
||||||
|
// to detach itself from the VM - which should result in ESRCH.
|
||||||
|
assert_status(rc == ESRCH, rc, "pthread_getcpuclockid failed");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// current_thread_cpu_time(bool) and thread_cpu_time(Thread*, bool)
|
// current_thread_cpu_time(bool) and thread_cpu_time(Thread*, bool)
|
||||||
|
@ -5584,7 +5588,7 @@ jlong os::current_thread_cpu_time() {
|
||||||
jlong os::thread_cpu_time(Thread* thread) {
|
jlong os::thread_cpu_time(Thread* thread) {
|
||||||
// consistent with what current_thread_cpu_time() returns
|
// consistent with what current_thread_cpu_time() returns
|
||||||
if (os::Linux::supports_fast_thread_cpu_time()) {
|
if (os::Linux::supports_fast_thread_cpu_time()) {
|
||||||
return os::Linux::fast_thread_cpu_time(thread_cpu_clockid(thread));
|
return fast_cpu_time(thread);
|
||||||
} else {
|
} else {
|
||||||
return slow_thread_cpu_time(thread, true /* user + sys */);
|
return slow_thread_cpu_time(thread, true /* user + sys */);
|
||||||
}
|
}
|
||||||
|
@ -5600,7 +5604,7 @@ jlong os::current_thread_cpu_time(bool user_sys_cpu_time) {
|
||||||
|
|
||||||
jlong os::thread_cpu_time(Thread *thread, bool user_sys_cpu_time) {
|
jlong os::thread_cpu_time(Thread *thread, bool user_sys_cpu_time) {
|
||||||
if (user_sys_cpu_time && os::Linux::supports_fast_thread_cpu_time()) {
|
if (user_sys_cpu_time && os::Linux::supports_fast_thread_cpu_time()) {
|
||||||
return os::Linux::fast_thread_cpu_time(thread_cpu_clockid(thread));
|
return fast_cpu_time(thread);
|
||||||
} else {
|
} else {
|
||||||
return slow_thread_cpu_time(thread, user_sys_cpu_time);
|
return slow_thread_cpu_time(thread, user_sys_cpu_time);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,112 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 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.
|
||||||
|
*/
|
||||||
|
import java.lang.management.*;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8205878
|
||||||
|
* @requires os.family != "windows" & os.family != "solaris"
|
||||||
|
* @summary Basic test of Thread and ThreadMXBean queries on a natively
|
||||||
|
* attached thread that has failed to detach before terminating.
|
||||||
|
* @comment The native code only supports POSIX so no windows testing; also
|
||||||
|
* we have to skip solaris as a terminating thread that fails to
|
||||||
|
* detach will hit an infinite loop due to TLS destructor issues - see
|
||||||
|
* comments in JDK-8156708
|
||||||
|
* @run main/native TestTerminatedThread
|
||||||
|
*/
|
||||||
|
|
||||||
|
public class TestTerminatedThread {
|
||||||
|
|
||||||
|
static native Thread createTerminatedThread();
|
||||||
|
|
||||||
|
static {
|
||||||
|
System.loadLibrary("terminatedThread");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ThreadMXBean mbean = ManagementFactory.getThreadMXBean();
|
||||||
|
|
||||||
|
public static void main(String[] args) throws Throwable {
|
||||||
|
|
||||||
|
Thread t = createTerminatedThread();
|
||||||
|
|
||||||
|
if (!t.isAlive())
|
||||||
|
throw new Error("Thread is only supposed to terminate at native layer!");
|
||||||
|
|
||||||
|
// Now invoke the various functions on this thread to
|
||||||
|
// make sure the VM handles it okay. The focus is on
|
||||||
|
// functions with an underlying native OS implementation.
|
||||||
|
// Generally as long as we don't crash or throw unexpected
|
||||||
|
// exceptions then the test passes. In some cases we know exactly
|
||||||
|
// what a function should return and so can check that.
|
||||||
|
|
||||||
|
System.out.println("Working with thread: " + t +
|
||||||
|
", in state: " + t.getState());
|
||||||
|
|
||||||
|
System.out.println("Calling suspend ...");
|
||||||
|
t.suspend();
|
||||||
|
System.out.println("Calling resume ...");
|
||||||
|
t.resume();
|
||||||
|
System.out.println("Calling getStackTrace ...");
|
||||||
|
StackTraceElement[] stack = t.getStackTrace();
|
||||||
|
System.out.println(java.util.Arrays.toString(stack));
|
||||||
|
if (stack.length != 0)
|
||||||
|
throw new Error("Terminated thread should have empty java stack trace");
|
||||||
|
System.out.println("Calling setName(\"NewName\") ...");
|
||||||
|
t.setName("NewName");
|
||||||
|
System.out.println("Calling interrupt ...");
|
||||||
|
t.interrupt();
|
||||||
|
System.out.println("Calling stop ...");
|
||||||
|
t.stop();
|
||||||
|
|
||||||
|
// Now the ThreadMXBean functions
|
||||||
|
|
||||||
|
if (mbean.isThreadCpuTimeSupported() &&
|
||||||
|
mbean.isThreadCpuTimeEnabled() ) {
|
||||||
|
System.out.println("Calling getThreadCpuTime ...");
|
||||||
|
long t1 = mbean.getThreadCpuTime(t.getId());
|
||||||
|
if (t1 != -1) {
|
||||||
|
throw new RuntimeException("Invalid ThreadCpuTime returned = " +
|
||||||
|
t1 + " expected = -1");
|
||||||
|
}
|
||||||
|
System.out.println("Okay: getThreadCpuTime() reported -1 as expected");
|
||||||
|
} else {
|
||||||
|
System.out.println("Skipping Thread CPU time test as it's not supported");
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.println("Calling getThreadUserTime ...");
|
||||||
|
long t1 = mbean.getThreadUserTime(t.getId());
|
||||||
|
if (t1 != -1) {
|
||||||
|
throw new RuntimeException("Invalid ThreadUserTime returned = " +
|
||||||
|
t1 + " expected = -1");
|
||||||
|
}
|
||||||
|
System.out.println("Okay: getThreadUserTime() reported -1 as expected");
|
||||||
|
|
||||||
|
System.out.println("Calling getThreadInfo ...");
|
||||||
|
ThreadInfo info = mbean.getThreadInfo(t.getId());
|
||||||
|
System.out.println(info);
|
||||||
|
|
||||||
|
System.out.println("Calling getThreadInfo with stack ...");
|
||||||
|
info = mbean.getThreadInfo(t.getId(), Integer.MAX_VALUE);
|
||||||
|
System.out.println(info);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,96 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "jni.h"
|
||||||
|
#include "jni_util.h"
|
||||||
|
|
||||||
|
|
||||||
|
JavaVM* jvm;
|
||||||
|
jobject nativeThread;
|
||||||
|
|
||||||
|
static void * thread_start(void* unused) {
|
||||||
|
JNIEnv *env;
|
||||||
|
jclass class_id;
|
||||||
|
jmethodID method_id;
|
||||||
|
int res;
|
||||||
|
|
||||||
|
printf("Native thread is running and attaching as daemon ...\n");
|
||||||
|
|
||||||
|
res = (*jvm)->AttachCurrentThreadAsDaemon(jvm, (void **)&env, NULL);
|
||||||
|
if (res != JNI_OK) {
|
||||||
|
fprintf(stderr, "Test ERROR. Can't attach current thread: %d\n", res);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
class_id = (*env)->FindClass (env, "java/lang/Thread");
|
||||||
|
if (class_id == NULL) {
|
||||||
|
fprintf(stderr, "Test ERROR. Can't load class Thread\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
method_id = (*env)->GetStaticMethodID(env, class_id, "currentThread",
|
||||||
|
"()Ljava/lang/Thread;");
|
||||||
|
if (method_id == NULL) {
|
||||||
|
fprintf(stderr, "Test ERROR. Can't find method currentThread\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
nativeThread = (*env)->CallStaticObjectMethod(env, class_id, method_id, NULL);
|
||||||
|
|
||||||
|
if ((*env)->ExceptionOccurred(env) != NULL) {
|
||||||
|
(*env)->ExceptionDescribe(env);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
printf("Native thread terminating\n");
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT jobject JNICALL
|
||||||
|
Java_TestTerminatedThread_createTerminatedThread
|
||||||
|
(JNIEnv *env, jclass cls) {
|
||||||
|
pthread_t thread;
|
||||||
|
int res = (*env)->GetJavaVM(env, &jvm);
|
||||||
|
if (res != JNI_OK) {
|
||||||
|
fprintf(stderr, "Test ERROR. Can't extract JavaVM: %d\n", res);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((res = pthread_create(&thread, NULL, thread_start, NULL)) != 0) {
|
||||||
|
fprintf(stderr, "TEST ERROR: pthread_create failed: %s (%d)\n", strerror(res), res);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((res = pthread_join(thread, NULL)) != 0) {
|
||||||
|
fprintf(stderr, "TEST ERROR: pthread_join failed: %s (%d)\n", strerror(res), res);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return nativeThread;
|
||||||
|
}
|
|
@ -333,8 +333,11 @@ static int agentA(void *context) {
|
||||||
checkIntercept(1, 0, 1); /* expected interceptions: 1 */
|
checkIntercept(1, 0, 1); /* expected interceptions: 1 */
|
||||||
NSK_DISPLAY0("\n<<< TEST CASE #4) done\n");
|
NSK_DISPLAY0("\n<<< TEST CASE #4) done\n");
|
||||||
|
|
||||||
NSK_DISPLAY1("\nagent A: returning exit code %d\n",
|
NSK_DISPLAY1("\nagent A: detaching and returning exit code %d\n",
|
||||||
exitCode);
|
exitCode);
|
||||||
|
if ((res = JNI_ENV_PTR(vm)->DetachCurrentThread(JNI_ENV_ARG1(vm))) != 0) {
|
||||||
|
NSK_COMPLAIN1("TEST WARNING: agent A: DetachCurrentThread() returns: %d\n", res);
|
||||||
|
}
|
||||||
return exitCode;
|
return exitCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -393,8 +396,11 @@ static int agentB(void *context) {
|
||||||
|
|
||||||
redir[1] = 1;
|
redir[1] = 1;
|
||||||
|
|
||||||
NSK_DISPLAY1("\nagent B: returning exit code %d\n",
|
NSK_DISPLAY1("\nagent B: detaching and returning exit code %d\n",
|
||||||
exitCode);
|
exitCode);
|
||||||
|
if ((res = JNI_ENV_PTR(vm)->DetachCurrentThread(JNI_ENV_ARG1(vm))) != 0) {
|
||||||
|
NSK_COMPLAIN1("TEST WARNING: agent B: DetachCurrentThread() returns: %d\n", res);
|
||||||
|
}
|
||||||
return exitCode;
|
return exitCode;
|
||||||
}
|
}
|
||||||
/*********************/
|
/*********************/
|
||||||
|
|
|
@ -62,6 +62,18 @@ extern "C" {
|
||||||
#define TRIES 30
|
#define TRIES 30
|
||||||
#define MAX_THREADS 5
|
#define MAX_THREADS 5
|
||||||
|
|
||||||
|
// Helper for thread detach and terminate
|
||||||
|
#define THREAD_return(status) \
|
||||||
|
do { \
|
||||||
|
int res = JNI_ENV_PTR(vm)->DetachCurrentThread(JNI_ENV_ARG1(vm)); \
|
||||||
|
if (res != 0) \
|
||||||
|
NSK_COMPLAIN1("TEST WARNING: DetachCurrentThread() returns: %d\n", res); \
|
||||||
|
else \
|
||||||
|
NSK_DISPLAY0("Detaching thread ...\n"); \
|
||||||
|
return status; \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
|
||||||
static const char *javaField = "_ji06t001a";
|
static const char *javaField = "_ji06t001a";
|
||||||
static const char *classSig =
|
static const char *classSig =
|
||||||
"Lnsk/jvmti/scenarios/jni_interception/JI06/ji06t001a;";
|
"Lnsk/jvmti/scenarios/jni_interception/JI06/ji06t001a;";
|
||||||
|
@ -225,16 +237,16 @@ static int waitingThread(void *context) {
|
||||||
thrStarted[indx-1] = 1; /* the thread is started */
|
thrStarted[indx-1] = 1; /* the thread is started */
|
||||||
|
|
||||||
if (enterMonitor(env, "waitingThread") == STATUS_FAILED)
|
if (enterMonitor(env, "waitingThread") == STATUS_FAILED)
|
||||||
return STATUS_FAILED;
|
THREAD_return(STATUS_FAILED);
|
||||||
if (verbose)
|
if (verbose)
|
||||||
printf("waitingThread: thread #%d entered the monitor\n",
|
printf("waitingThread: thread #%d entered the monitor\n",
|
||||||
indx);
|
indx);
|
||||||
if (exitMonitor(env, "waitingThread") == STATUS_FAILED)
|
if (exitMonitor(env, "waitingThread") == STATUS_FAILED)
|
||||||
return STATUS_FAILED;
|
THREAD_return(STATUS_FAILED);
|
||||||
|
|
||||||
NSK_DISPLAY2("waitingThread: thread #%d exits the monitor\n\treturning %d\n",
|
NSK_DISPLAY2("waitingThread: thread #%d exits the monitor\n\treturning %d\n",
|
||||||
indx, exitCode);
|
indx, exitCode);
|
||||||
return exitCode;
|
THREAD_return(exitCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ownerThread(void *context) {
|
static int ownerThread(void *context) {
|
||||||
|
@ -254,7 +266,7 @@ static int ownerThread(void *context) {
|
||||||
|
|
||||||
NSK_DISPLAY0("ownerThread: trying to enter the monitor ...\n");
|
NSK_DISPLAY0("ownerThread: trying to enter the monitor ...\n");
|
||||||
if (enterMonitor(env, "ownerThread") == STATUS_FAILED)
|
if (enterMonitor(env, "ownerThread") == STATUS_FAILED)
|
||||||
return STATUS_FAILED;
|
THREAD_return(STATUS_FAILED);
|
||||||
|
|
||||||
monEntered = 1; /* the monitor has been entered */
|
monEntered = 1; /* the monitor has been entered */
|
||||||
NSK_DISPLAY1("ownerThread: entered the monitor: monEntered=%d\n\
|
NSK_DISPLAY1("ownerThread: entered the monitor: monEntered=%d\n\
|
||||||
|
@ -272,12 +284,12 @@ static int ownerThread(void *context) {
|
||||||
} while(releaseMon != 1);
|
} while(releaseMon != 1);
|
||||||
|
|
||||||
if (exitMonitor(env, "ownerThread") == STATUS_FAILED)
|
if (exitMonitor(env, "ownerThread") == STATUS_FAILED)
|
||||||
return STATUS_FAILED;
|
THREAD_return(STATUS_FAILED);
|
||||||
|
|
||||||
NSK_DISPLAY1("ownerThread: exits the monitor\n\treturning %d\n",
|
NSK_DISPLAY1("ownerThread: exits the monitor\n\treturning %d\n",
|
||||||
exitCode);
|
exitCode);
|
||||||
|
|
||||||
return exitCode;
|
THREAD_return(exitCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int redirectorThread(void *context) {
|
static int redirectorThread(void *context) {
|
||||||
|
@ -301,7 +313,7 @@ static int redirectorThread(void *context) {
|
||||||
NSK_DISPLAY1("redirectorThread: the MonitorEnter() redirected\n\treturning %d\n",
|
NSK_DISPLAY1("redirectorThread: the MonitorEnter() redirected\n\treturning %d\n",
|
||||||
exitCode);
|
exitCode);
|
||||||
|
|
||||||
return exitCode;
|
THREAD_return(exitCode);
|
||||||
}
|
}
|
||||||
/*********************/
|
/*********************/
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue