1999-04-07 14:42:40 +00:00
|
|
|
|
// posix-threads.cc - interface between libjava and POSIX threads.
|
|
|
|
|
|
2006-06-19 14:10:02 +00:00
|
|
|
|
/* Copyright (C) 1998, 1999, 2000, 2001, 2004, 2006 Free Software Foundation
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
|
|
|
|
This file is part of libgcj.
|
|
|
|
|
|
|
|
|
|
This software is copyrighted work licensed under the terms of the
|
|
|
|
|
Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
|
|
|
|
|
details. */
|
|
|
|
|
|
|
|
|
|
// TO DO:
|
|
|
|
|
// * Document signal handling limitations
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
2006-09-14 09:55:47 +00:00
|
|
|
|
#include "posix.h"
|
2007-05-12 17:37:55 +00:00
|
|
|
|
#include "posix-threads.h"
|
2006-09-14 09:55:47 +00:00
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
// If we're using the Boehm GC, then we need to override some of the
|
|
|
|
|
// thread primitives. This is fairly gross.
|
|
|
|
|
#ifdef HAVE_BOEHM_GC
|
|
|
|
|
#include <gc.h>
|
|
|
|
|
#endif /* HAVE_BOEHM_GC */
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
#include <signal.h>
|
1999-08-25 00:46:23 +00:00
|
|
|
|
#include <errno.h>
|
1999-12-24 01:00:46 +00:00
|
|
|
|
#include <limits.h>
|
2003-03-02 01:24:40 +00:00
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
#include <unistd.h> // To test for _POSIX_THREAD_PRIORITY_SCHEDULING
|
|
|
|
|
#endif
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
1999-09-10 22:03:10 +00:00
|
|
|
|
#include <gcj/cni.h>
|
1999-04-07 14:42:40 +00:00
|
|
|
|
#include <jvm.h>
|
|
|
|
|
#include <java/lang/Thread.h>
|
|
|
|
|
#include <java/lang/System.h>
|
1999-12-24 01:00:46 +00:00
|
|
|
|
#include <java/lang/Long.h>
|
|
|
|
|
#include <java/lang/OutOfMemoryError.h>
|
2004-08-12 16:20:11 +00:00
|
|
|
|
#include <java/lang/InternalError.h>
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
|
|
|
|
// This is used to implement thread startup.
|
|
|
|
|
struct starter
|
|
|
|
|
{
|
|
|
|
|
_Jv_ThreadStartFunc *method;
|
|
|
|
|
_Jv_Thread_t *data;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// This is the key used to map from the POSIX thread value back to the
|
|
|
|
|
// Java object representing the thread. The key is global to all
|
|
|
|
|
// threads, so it is ok to make it a global here.
|
|
|
|
|
pthread_key_t _Jv_ThreadKey;
|
|
|
|
|
|
1999-09-08 00:43:06 +00:00
|
|
|
|
// This is the key used to map from the POSIX thread value back to the
|
|
|
|
|
// _Jv_Thread_t* representing the thread.
|
|
|
|
|
pthread_key_t _Jv_ThreadDataKey;
|
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
// We keep a count of all non-daemon threads which are running. When
|
|
|
|
|
// this reaches zero, _Jv_ThreadWait returns.
|
|
|
|
|
static pthread_mutex_t daemon_mutex;
|
|
|
|
|
static pthread_cond_t daemon_cond;
|
|
|
|
|
static int non_daemon_count;
|
|
|
|
|
|
|
|
|
|
// The signal to use when interrupting a thread.
|
2001-10-24 21:57:08 +00:00
|
|
|
|
#if defined(LINUX_THREADS) || defined(FREEBSD_THREADS)
|
1999-12-24 01:00:46 +00:00
|
|
|
|
// LinuxThreads (prior to glibc 2.1) usurps both SIGUSR1 and SIGUSR2.
|
2001-10-24 21:57:08 +00:00
|
|
|
|
// GC on FreeBSD uses both SIGUSR1 and SIGUSR2.
|
1999-04-07 14:42:40 +00:00
|
|
|
|
# define INTR SIGHUP
|
|
|
|
|
#else /* LINUX_THREADS */
|
|
|
|
|
# define INTR SIGUSR2
|
|
|
|
|
#endif /* LINUX_THREADS */
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// These are the flags that can appear in _Jv_Thread_t.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
// Thread started.
|
|
|
|
|
#define FLAG_START 0x01
|
|
|
|
|
// Thread is daemon.
|
|
|
|
|
#define FLAG_DAEMON 0x02
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
int
|
|
|
|
|
_Jv_MutexLock (_Jv_Mutex_t *mu)
|
|
|
|
|
{
|
|
|
|
|
pthread_t self = pthread_self ();
|
|
|
|
|
if (mu->owner == self)
|
|
|
|
|
{
|
|
|
|
|
mu->count++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
JvSetThreadState holder (_Jv_ThreadCurrent(), JV_BLOCKED);
|
|
|
|
|
|
|
|
|
|
# ifdef LOCK_DEBUG
|
|
|
|
|
int result = pthread_mutex_lock (&mu->mutex);
|
|
|
|
|
if (0 != result)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Pthread_mutex_lock returned %d\n", result);
|
|
|
|
|
for (;;) {}
|
|
|
|
|
}
|
|
|
|
|
# else
|
|
|
|
|
pthread_mutex_lock (&mu->mutex);
|
|
|
|
|
# endif
|
|
|
|
|
mu->count = 1;
|
|
|
|
|
mu->owner = self;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
// Wait for the condition variable "CV" to be notified.
|
|
|
|
|
// Return values:
|
|
|
|
|
// 0: the condition was notified, or the timeout expired.
|
|
|
|
|
// _JV_NOT_OWNER: the thread does not own the mutex "MU".
|
|
|
|
|
// _JV_INTERRUPTED: the thread was interrupted. Its interrupted flag is set.
|
1999-04-07 14:42:40 +00:00
|
|
|
|
int
|
|
|
|
|
_Jv_CondWait (_Jv_ConditionVariable_t *cv, _Jv_Mutex_t *mu,
|
|
|
|
|
jlong millis, jint nanos)
|
|
|
|
|
{
|
2000-03-28 02:22:24 +00:00
|
|
|
|
pthread_t self = pthread_self();
|
|
|
|
|
if (mu->owner != self)
|
|
|
|
|
return _JV_NOT_OWNER;
|
1999-09-01 18:29:39 +00:00
|
|
|
|
|
1999-12-24 01:00:46 +00:00
|
|
|
|
struct timespec ts;
|
1999-09-01 18:29:39 +00:00
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
JvThreadState new_state = JV_WAITING;
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (millis > 0 || nanos > 0)
|
1999-04-07 14:42:40 +00:00
|
|
|
|
{
|
2005-11-25 16:18:17 +00:00
|
|
|
|
// Calculate the abstime corresponding to the timeout.
|
2006-09-14 01:17:31 +00:00
|
|
|
|
unsigned long long seconds;
|
|
|
|
|
unsigned long usec;
|
|
|
|
|
|
|
|
|
|
// For better accuracy, should use pthread_condattr_setclock
|
|
|
|
|
// and clock_gettime.
|
|
|
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
|
|
|
timeval tv;
|
|
|
|
|
gettimeofday (&tv, NULL);
|
|
|
|
|
usec = tv.tv_usec;
|
|
|
|
|
seconds = tv.tv_sec;
|
|
|
|
|
#else
|
|
|
|
|
unsigned long long startTime = java::lang::System::currentTimeMillis();
|
|
|
|
|
seconds = startTime / 1000;
|
|
|
|
|
/* Assume we're about half-way through this millisecond. */
|
|
|
|
|
usec = (startTime % 1000) * 1000 + 500;
|
|
|
|
|
#endif
|
|
|
|
|
/* These next two statements cannot overflow. */
|
|
|
|
|
usec += nanos / 1000;
|
|
|
|
|
usec += (millis % 1000) * 1000;
|
|
|
|
|
/* These two statements could overflow only if tv.tv_sec was
|
|
|
|
|
insanely large. */
|
|
|
|
|
seconds += millis / 1000;
|
|
|
|
|
seconds += usec / 1000000;
|
2005-11-25 16:18:17 +00:00
|
|
|
|
|
|
|
|
|
ts.tv_sec = seconds;
|
|
|
|
|
if (ts.tv_sec < 0 || (unsigned long long)ts.tv_sec != seconds)
|
|
|
|
|
{
|
|
|
|
|
// We treat a timeout that won't fit into a struct timespec
|
|
|
|
|
// as a wait forever.
|
|
|
|
|
millis = nanos = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
2006-09-14 01:17:31 +00:00
|
|
|
|
/* This next statement also cannot overflow. */
|
|
|
|
|
ts.tv_nsec = (usec % 1000000) * 1000 + (nanos % 1000);
|
1999-12-24 01:00:46 +00:00
|
|
|
|
}
|
1999-09-08 00:43:06 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
_Jv_Thread_t *current = _Jv_ThreadCurrentData ();
|
|
|
|
|
java::lang::Thread *current_obj = _Jv_ThreadCurrent ();
|
|
|
|
|
|
2000-09-30 10:01:04 +00:00
|
|
|
|
pthread_mutex_lock (¤t->wait_mutex);
|
|
|
|
|
|
|
|
|
|
// Now that we hold the wait mutex, check if this thread has been
|
|
|
|
|
// interrupted already.
|
|
|
|
|
if (current_obj->interrupt_flag)
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_unlock (¤t->wait_mutex);
|
|
|
|
|
return _JV_INTERRUPTED;
|
|
|
|
|
}
|
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
// Set the thread's state.
|
|
|
|
|
JvSetThreadState holder (current_obj, new_state);
|
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
// Add this thread to the cv's wait set.
|
|
|
|
|
current->next = NULL;
|
1999-09-08 00:43:06 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (cv->first == NULL)
|
|
|
|
|
cv->first = current;
|
|
|
|
|
else
|
|
|
|
|
for (_Jv_Thread_t *t = cv->first;; t = t->next)
|
|
|
|
|
{
|
|
|
|
|
if (t->next == NULL)
|
|
|
|
|
{
|
|
|
|
|
t->next = current;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Record the current lock depth, so it can be restored when we re-aquire it.
|
|
|
|
|
int count = mu->count;
|
|
|
|
|
|
|
|
|
|
// Release the monitor mutex.
|
|
|
|
|
mu->count = 0;
|
|
|
|
|
mu->owner = 0;
|
|
|
|
|
pthread_mutex_unlock (&mu->mutex);
|
|
|
|
|
|
|
|
|
|
int r = 0;
|
|
|
|
|
bool done_sleeping = false;
|
|
|
|
|
|
|
|
|
|
while (! done_sleeping)
|
|
|
|
|
{
|
|
|
|
|
if (millis == 0 && nanos == 0)
|
|
|
|
|
r = pthread_cond_wait (¤t->wait_cond, ¤t->wait_mutex);
|
|
|
|
|
else
|
|
|
|
|
r = pthread_cond_timedwait (¤t->wait_cond, ¤t->wait_mutex,
|
|
|
|
|
&ts);
|
2000-12-30 12:18:39 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
// In older glibc's (prior to 2.1.3), the cond_wait functions may
|
|
|
|
|
// spuriously wake up on a signal. Catch that here.
|
|
|
|
|
if (r != EINTR)
|
|
|
|
|
done_sleeping = true;
|
|
|
|
|
}
|
|
|
|
|
|
2000-09-30 10:01:04 +00:00
|
|
|
|
// Check for an interrupt *before* releasing the wait mutex.
|
2000-03-28 02:22:24 +00:00
|
|
|
|
jboolean interrupted = current_obj->interrupt_flag;
|
|
|
|
|
|
|
|
|
|
pthread_mutex_unlock (¤t->wait_mutex);
|
|
|
|
|
|
|
|
|
|
// Reaquire the monitor mutex, and restore the lock count.
|
|
|
|
|
pthread_mutex_lock (&mu->mutex);
|
|
|
|
|
mu->owner = self;
|
|
|
|
|
mu->count = count;
|
|
|
|
|
|
HACKING, [...]: Fix spelling errors.
* HACKING, gnu/gcj/xlib/Pixmap.java, gnu/gcj/xlib/XException.java,
gnu/java/rmi/rmic/RMIC.java, java/awt/Window.java,
java/awt/AWTEvent.java, java/io/ByteArrayOutputStream.java,
java/io/CharConversionException.java,
java/io/PipedInputStream.java, java/io/PipedReader.java,
java/io/PrintWriter.java, java/io/WriteAbortedException.java,
java/io/natFileWin32.cc, java/lang/Class.h,
java/lang/natClassLoader.cc, java/lang/natObject.cc,
java/lang/Package.java, java/net/BindException.java,
java/net/ConnectException.java, java/net/ProtocolException.java,
java/net/SocketException.java,
java/net/UnknownServiceException.java,
java/security/cert/X509Certificate.java,
java/security/interfaces/DSAKey.java,
java/security/SecureRandom.java, java/security/SignedObject.java,
java/sql/DatabaseMetaData.java,
java/text/DecimalFormatSymbols.java,
java/util/jar/Attributes.java, java/util/jar/JarEntry.java,
java/util/jar/JarInputStream.java,
java/util/jar/JarOutputStream.java, java/util/Calendar.java,
java/util/Collections.java, java/util/GregorianCalendar.java,
java/util/HashMap.java, java/util/List.java,
java/util/Properties.java, java/util/Timer.java,
java/util/Vector.java, java/util/WeakHashMap.java,
javax/naming/NamingException.java,
testsuite/libjava.lang/Thread_Wait.java,
org/xml/sax/helpers/DefaultHandler.java,
org/xml/sax/HandlerBase.java, org/xml/sax/SAXParseException.java,
ChangeLog, acinclude.m4, aclocal.m4, posix-threads.cc: Fix
spelling errors.
* configure: Regenerate.
From-SVN: r46665
2001-10-31 00:48:17 +00:00
|
|
|
|
// If we were interrupted, or if a timeout occurred, remove ourself from
|
2000-03-28 02:22:24 +00:00
|
|
|
|
// the cv wait list now. (If we were notified normally, notify() will have
|
|
|
|
|
// already taken care of this)
|
|
|
|
|
if (r == ETIMEDOUT || interrupted)
|
|
|
|
|
{
|
|
|
|
|
_Jv_Thread_t *prev = NULL;
|
|
|
|
|
for (_Jv_Thread_t *t = cv->first; t != NULL; t = t->next)
|
1999-12-24 01:00:46 +00:00
|
|
|
|
{
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (t == current)
|
1999-09-08 00:43:06 +00:00
|
|
|
|
{
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (prev != NULL)
|
|
|
|
|
prev->next = t->next;
|
|
|
|
|
else
|
|
|
|
|
cv->first = t->next;
|
|
|
|
|
t->next = NULL;
|
|
|
|
|
break;
|
1999-09-08 00:43:06 +00:00
|
|
|
|
}
|
2000-03-28 02:22:24 +00:00
|
|
|
|
prev = t;
|
1999-09-08 00:43:06 +00:00
|
|
|
|
}
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (interrupted)
|
|
|
|
|
return _JV_INTERRUPTED;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
2000-03-28 02:22:24 +00:00
|
|
|
|
|
|
|
|
|
return 0;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
int
|
|
|
|
|
_Jv_CondNotify (_Jv_ConditionVariable_t *cv, _Jv_Mutex_t *mu)
|
1999-04-07 14:42:40 +00:00
|
|
|
|
{
|
2003-10-21 04:46:19 +00:00
|
|
|
|
if (_Jv_MutexCheckMonitor (mu))
|
2000-03-28 02:22:24 +00:00
|
|
|
|
return _JV_NOT_OWNER;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
_Jv_Thread_t *target;
|
|
|
|
|
_Jv_Thread_t *prev = NULL;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
for (target = cv->first; target != NULL; target = target->next)
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_lock (&target->wait_mutex);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (target->thread_obj->interrupt_flag)
|
|
|
|
|
{
|
|
|
|
|
// Don't notify a thread that has already been interrupted.
|
|
|
|
|
pthread_mutex_unlock (&target->wait_mutex);
|
|
|
|
|
prev = target;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
pthread_cond_signal (&target->wait_cond);
|
|
|
|
|
pthread_mutex_unlock (&target->wait_mutex);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-09-07 12:09:41 +00:00
|
|
|
|
// Two concurrent notify() calls must not be delivered to the same
|
|
|
|
|
// thread, so remove the target thread from the cv wait list now.
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (prev == NULL)
|
|
|
|
|
cv->first = target->next;
|
|
|
|
|
else
|
|
|
|
|
prev->next = target->next;
|
|
|
|
|
|
|
|
|
|
target->next = NULL;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
return 0;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2000-03-28 02:22:24 +00:00
|
|
|
|
_Jv_CondNotifyAll (_Jv_ConditionVariable_t *cv, _Jv_Mutex_t *mu)
|
1999-04-07 14:42:40 +00:00
|
|
|
|
{
|
2003-10-21 04:46:19 +00:00
|
|
|
|
if (_Jv_MutexCheckMonitor (mu))
|
2000-03-28 02:22:24 +00:00
|
|
|
|
return _JV_NOT_OWNER;
|
|
|
|
|
|
|
|
|
|
_Jv_Thread_t *target;
|
|
|
|
|
_Jv_Thread_t *prev = NULL;
|
|
|
|
|
|
|
|
|
|
for (target = cv->first; target != NULL; target = target->next)
|
1999-04-07 14:42:40 +00:00
|
|
|
|
{
|
2000-03-28 02:22:24 +00:00
|
|
|
|
pthread_mutex_lock (&target->wait_mutex);
|
|
|
|
|
pthread_cond_signal (&target->wait_cond);
|
|
|
|
|
pthread_mutex_unlock (&target->wait_mutex);
|
|
|
|
|
|
|
|
|
|
if (prev != NULL)
|
|
|
|
|
prev->next = NULL;
|
|
|
|
|
prev = target;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
2000-03-28 02:22:24 +00:00
|
|
|
|
if (prev != NULL)
|
|
|
|
|
prev->next = NULL;
|
|
|
|
|
|
|
|
|
|
cv->first = NULL;
|
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadInterrupt (_Jv_Thread_t *data)
|
1999-04-07 14:42:40 +00:00
|
|
|
|
{
|
2000-03-28 02:22:24 +00:00
|
|
|
|
pthread_mutex_lock (&data->wait_mutex);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-03-28 02:22:24 +00:00
|
|
|
|
// Set the thread's interrupted flag *after* aquiring its wait_mutex. This
|
|
|
|
|
// ensures that there are no races with the interrupt flag being set after
|
|
|
|
|
// the waiting thread checks it and before pthread_cond_wait is entered.
|
|
|
|
|
data->thread_obj->interrupt_flag = true;
|
|
|
|
|
|
|
|
|
|
// Interrupt blocking system calls using a signal.
|
2001-09-21 04:23:31 +00:00
|
|
|
|
pthread_kill (data->thread, INTR);
|
2000-03-28 02:22:24 +00:00
|
|
|
|
|
|
|
|
|
pthread_cond_signal (&data->wait_cond);
|
|
|
|
|
|
|
|
|
|
pthread_mutex_unlock (&data->wait_mutex);
|
|
|
|
|
}
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
/**
|
|
|
|
|
* Releases the block on a thread created by _Jv_ThreadPark(). This
|
|
|
|
|
* method can also be used to terminate a blockage caused by a prior
|
|
|
|
|
* call to park. This operation is unsafe, as the thread must be
|
|
|
|
|
* guaranteed to be live.
|
|
|
|
|
*
|
|
|
|
|
* @param thread the thread to unblock.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ParkHelper::unpark ()
|
|
|
|
|
{
|
|
|
|
|
using namespace ::java::lang;
|
|
|
|
|
volatile obj_addr_t *ptr = &permit;
|
|
|
|
|
|
|
|
|
|
/* If this thread is in state RUNNING, give it a permit and return
|
|
|
|
|
immediately. */
|
|
|
|
|
if (compare_and_swap
|
|
|
|
|
(ptr, Thread::THREAD_PARK_RUNNING, Thread::THREAD_PARK_PERMIT))
|
|
|
|
|
return;
|
2009-11-17 18:05:00 +00:00
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
/* If this thread is parked, put it into state RUNNING and send it a
|
|
|
|
|
signal. */
|
2009-11-17 18:05:00 +00:00
|
|
|
|
if (compare_and_swap
|
2007-01-09 19:58:05 +00:00
|
|
|
|
(ptr, Thread::THREAD_PARK_PARKED, Thread::THREAD_PARK_RUNNING))
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_lock (&mutex);
|
2009-11-17 18:05:00 +00:00
|
|
|
|
int result = pthread_cond_signal (&cond);
|
2007-01-09 19:58:05 +00:00
|
|
|
|
pthread_mutex_unlock (&mutex);
|
2009-11-17 18:05:00 +00:00
|
|
|
|
JvAssert (result == 0);
|
2007-01-09 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Sets our state to dead.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ParkHelper::deactivate ()
|
|
|
|
|
{
|
|
|
|
|
permit = ::java::lang::Thread::THREAD_PARK_DEAD;
|
|
|
|
|
}
|
|
|
|
|
|
2009-11-17 18:05:00 +00:00
|
|
|
|
void
|
|
|
|
|
ParkHelper::init ()
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_init (&mutex, NULL);
|
|
|
|
|
pthread_cond_init (&cond, NULL);
|
|
|
|
|
permit = ::java::lang::Thread::THREAD_PARK_RUNNING;
|
|
|
|
|
}
|
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
/**
|
|
|
|
|
* Blocks the thread until a matching _Jv_ThreadUnpark() occurs, the
|
|
|
|
|
* thread is interrupted or the optional timeout expires. If an
|
|
|
|
|
* unpark call has already occurred, this also counts. A timeout
|
|
|
|
|
* value of zero is defined as no timeout. When isAbsolute is true,
|
|
|
|
|
* the timeout is in milliseconds relative to the epoch. Otherwise,
|
|
|
|
|
* the value is the number of nanoseconds which must occur before
|
|
|
|
|
* timeout. This call may also return spuriously (i.e. for no
|
|
|
|
|
* apparent reason).
|
|
|
|
|
*
|
|
|
|
|
* @param isAbsolute true if the timeout is specified in milliseconds from
|
|
|
|
|
* the epoch.
|
|
|
|
|
* @param time either the number of nanoseconds to wait, or a time in
|
|
|
|
|
* milliseconds from the epoch to wait for.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ParkHelper::park (jboolean isAbsolute, jlong time)
|
|
|
|
|
{
|
|
|
|
|
using namespace ::java::lang;
|
|
|
|
|
volatile obj_addr_t *ptr = &permit;
|
|
|
|
|
|
|
|
|
|
/* If we have a permit, return immediately. */
|
|
|
|
|
if (compare_and_swap
|
|
|
|
|
(ptr, Thread::THREAD_PARK_PERMIT, Thread::THREAD_PARK_RUNNING))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
struct timespec ts;
|
|
|
|
|
|
|
|
|
|
if (time)
|
|
|
|
|
{
|
2009-11-17 18:05:00 +00:00
|
|
|
|
unsigned long long seconds;
|
|
|
|
|
unsigned long usec;
|
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
if (isAbsolute)
|
|
|
|
|
{
|
2009-11-17 18:05:00 +00:00
|
|
|
|
ts.tv_sec = time / 1000;
|
|
|
|
|
ts.tv_nsec = (time % 1000) * 1000 * 1000;
|
2007-01-09 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Calculate the abstime corresponding to the timeout.
|
2009-11-17 18:05:00 +00:00
|
|
|
|
jlong nanos = time;
|
|
|
|
|
jlong millis = 0;
|
2007-01-09 19:58:05 +00:00
|
|
|
|
|
2009-11-17 18:05:00 +00:00
|
|
|
|
// For better accuracy, should use pthread_condattr_setclock
|
|
|
|
|
// and clock_gettime.
|
|
|
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
|
|
|
timeval tv;
|
|
|
|
|
gettimeofday (&tv, NULL);
|
|
|
|
|
usec = tv.tv_usec;
|
|
|
|
|
seconds = tv.tv_sec;
|
|
|
|
|
#else
|
|
|
|
|
unsigned long long startTime
|
|
|
|
|
= java::lang::System::currentTimeMillis();
|
|
|
|
|
seconds = startTime / 1000;
|
|
|
|
|
/* Assume we're about half-way through this millisecond. */
|
|
|
|
|
usec = (startTime % 1000) * 1000 + 500;
|
|
|
|
|
#endif
|
|
|
|
|
/* These next two statements cannot overflow. */
|
|
|
|
|
usec += nanos / 1000;
|
|
|
|
|
usec += (millis % 1000) * 1000;
|
|
|
|
|
/* These two statements could overflow only if tv.tv_sec was
|
|
|
|
|
insanely large. */
|
|
|
|
|
seconds += millis / 1000;
|
|
|
|
|
seconds += usec / 1000000;
|
2007-01-09 19:58:05 +00:00
|
|
|
|
|
|
|
|
|
ts.tv_sec = seconds;
|
|
|
|
|
if (ts.tv_sec < 0 || (unsigned long long)ts.tv_sec != seconds)
|
|
|
|
|
{
|
|
|
|
|
// We treat a timeout that won't fit into a struct timespec
|
|
|
|
|
// as a wait forever.
|
|
|
|
|
millis = nanos = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
2009-11-17 18:05:00 +00:00
|
|
|
|
/* This next statement also cannot overflow. */
|
|
|
|
|
ts.tv_nsec = (usec % 1000000) * 1000 + (nanos % 1000);
|
2007-01-09 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2009-11-17 18:05:00 +00:00
|
|
|
|
|
|
|
|
|
pthread_mutex_lock (&mutex);
|
2007-01-09 19:58:05 +00:00
|
|
|
|
if (compare_and_swap
|
|
|
|
|
(ptr, Thread::THREAD_PARK_RUNNING, Thread::THREAD_PARK_PARKED))
|
|
|
|
|
{
|
2009-11-17 18:05:00 +00:00
|
|
|
|
int result = 0;
|
|
|
|
|
|
|
|
|
|
if (! time)
|
|
|
|
|
result = pthread_cond_wait (&cond, &mutex);
|
2007-01-09 19:58:05 +00:00
|
|
|
|
else
|
2009-11-17 18:05:00 +00:00
|
|
|
|
result = pthread_cond_timedwait (&cond, &mutex, &ts);
|
|
|
|
|
|
|
|
|
|
JvAssert (result == 0 || result == ETIMEDOUT);
|
|
|
|
|
|
2007-01-09 19:58:05 +00:00
|
|
|
|
/* If we were unparked by some other thread, this will already
|
2009-11-17 18:05:00 +00:00
|
|
|
|
be in state THREAD_PARK_RUNNING. If we timed out or were
|
|
|
|
|
interrupted, we have to do it ourself. */
|
|
|
|
|
permit = Thread::THREAD_PARK_RUNNING;
|
2007-01-09 19:58:05 +00:00
|
|
|
|
}
|
2009-11-17 18:05:00 +00:00
|
|
|
|
pthread_mutex_unlock (&mutex);
|
2007-01-09 19:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
static void
|
|
|
|
|
handle_intr (int)
|
|
|
|
|
{
|
|
|
|
|
// Do nothing.
|
|
|
|
|
}
|
|
|
|
|
|
2007-05-12 17:37:55 +00:00
|
|
|
|
void
|
|
|
|
|
_Jv_BlockSigchld()
|
2005-02-16 04:16:06 +00:00
|
|
|
|
{
|
|
|
|
|
sigset_t mask;
|
|
|
|
|
sigemptyset (&mask);
|
|
|
|
|
sigaddset (&mask, SIGCHLD);
|
|
|
|
|
int c = pthread_sigmask (SIG_BLOCK, &mask, NULL);
|
|
|
|
|
if (c != 0)
|
|
|
|
|
JvFail (strerror (c));
|
|
|
|
|
}
|
|
|
|
|
|
2007-05-12 17:37:55 +00:00
|
|
|
|
void
|
|
|
|
|
_Jv_UnBlockSigchld()
|
|
|
|
|
{
|
|
|
|
|
sigset_t mask;
|
|
|
|
|
sigemptyset (&mask);
|
|
|
|
|
sigaddset (&mask, SIGCHLD);
|
|
|
|
|
int c = pthread_sigmask (SIG_UNBLOCK, &mask, NULL);
|
|
|
|
|
if (c != 0)
|
|
|
|
|
JvFail (strerror (c));
|
|
|
|
|
}
|
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
void
|
|
|
|
|
_Jv_InitThreads (void)
|
|
|
|
|
{
|
|
|
|
|
pthread_key_create (&_Jv_ThreadKey, NULL);
|
1999-09-08 00:43:06 +00:00
|
|
|
|
pthread_key_create (&_Jv_ThreadDataKey, NULL);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
pthread_mutex_init (&daemon_mutex, NULL);
|
|
|
|
|
pthread_cond_init (&daemon_cond, 0);
|
|
|
|
|
non_daemon_count = 0;
|
|
|
|
|
|
|
|
|
|
// Arrange for the interrupt signal to interrupt system calls.
|
|
|
|
|
struct sigaction act;
|
|
|
|
|
act.sa_handler = handle_intr;
|
|
|
|
|
sigemptyset (&act.sa_mask);
|
|
|
|
|
act.sa_flags = 0;
|
|
|
|
|
sigaction (INTR, &act, NULL);
|
2005-02-16 04:16:06 +00:00
|
|
|
|
|
|
|
|
|
// Block SIGCHLD here to ensure that any non-Java threads inherit the new
|
|
|
|
|
// signal mask.
|
2007-05-12 17:37:55 +00:00
|
|
|
|
_Jv_BlockSigchld();
|
2005-11-17 20:25:57 +00:00
|
|
|
|
|
|
|
|
|
// Check/set the thread stack size.
|
|
|
|
|
size_t min_ss = 32 * 1024;
|
|
|
|
|
|
|
|
|
|
if (sizeof (void *) == 8)
|
|
|
|
|
// Bigger default on 64-bit systems.
|
|
|
|
|
min_ss *= 2;
|
|
|
|
|
|
2006-01-04 22:11:51 +01:00
|
|
|
|
#ifdef PTHREAD_STACK_MIN
|
2005-11-17 20:25:57 +00:00
|
|
|
|
if (min_ss < PTHREAD_STACK_MIN)
|
|
|
|
|
min_ss = PTHREAD_STACK_MIN;
|
2006-01-04 22:11:51 +01:00
|
|
|
|
#endif
|
2005-11-17 20:25:57 +00:00
|
|
|
|
|
|
|
|
|
if (gcj::stack_size > 0 && gcj::stack_size < min_ss)
|
|
|
|
|
gcj::stack_size = min_ss;
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-12-30 12:18:39 +00:00
|
|
|
|
_Jv_Thread_t *
|
|
|
|
|
_Jv_ThreadInitData (java::lang::Thread *obj)
|
1999-04-07 14:42:40 +00:00
|
|
|
|
{
|
2001-04-28 00:04:55 +00:00
|
|
|
|
_Jv_Thread_t *data = (_Jv_Thread_t *) _Jv_Malloc (sizeof (_Jv_Thread_t));
|
2000-12-30 12:18:39 +00:00
|
|
|
|
data->flags = 0;
|
|
|
|
|
data->thread_obj = obj;
|
2000-03-28 02:22:24 +00:00
|
|
|
|
|
2000-12-30 12:18:39 +00:00
|
|
|
|
pthread_mutex_init (&data->wait_mutex, NULL);
|
|
|
|
|
pthread_cond_init (&data->wait_cond, NULL);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-12-30 12:18:39 +00:00
|
|
|
|
return data;
|
|
|
|
|
}
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
2000-12-30 12:18:39 +00:00
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadDestroyData (_Jv_Thread_t *data)
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_destroy (&data->wait_mutex);
|
|
|
|
|
pthread_cond_destroy (&data->wait_cond);
|
2001-04-28 00:04:55 +00:00
|
|
|
|
_Jv_Free ((void *)data);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadSetPriority (_Jv_Thread_t *data, jint prio)
|
|
|
|
|
{
|
2003-03-02 01:24:40 +00:00
|
|
|
|
#ifdef _POSIX_THREAD_PRIORITY_SCHEDULING
|
1999-04-07 14:42:40 +00:00
|
|
|
|
if (data->flags & FLAG_START)
|
|
|
|
|
{
|
|
|
|
|
struct sched_param param;
|
|
|
|
|
|
|
|
|
|
param.sched_priority = prio;
|
2005-07-08 14:04:22 +00:00
|
|
|
|
pthread_setschedparam (data->thread, SCHED_OTHER, ¶m);
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
2003-03-02 01:24:40 +00:00
|
|
|
|
#endif
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2001-05-21 23:47:48 -07:00
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadRegister (_Jv_Thread_t *data)
|
|
|
|
|
{
|
|
|
|
|
pthread_setspecific (_Jv_ThreadKey, data->thread_obj);
|
|
|
|
|
pthread_setspecific (_Jv_ThreadDataKey, data);
|
|
|
|
|
|
|
|
|
|
// glibc 2.1.3 doesn't set the value of `thread' until after start_routine
|
|
|
|
|
// is called. Since it may need to be accessed from the new thread, work
|
|
|
|
|
// around the potential race here by explicitly setting it again.
|
|
|
|
|
data->thread = pthread_self ();
|
2001-05-24 05:40:37 +00:00
|
|
|
|
|
|
|
|
|
# ifdef SLOW_PTHREAD_SELF
|
|
|
|
|
// Clear all self cache slots that might be needed by this thread.
|
|
|
|
|
int dummy;
|
|
|
|
|
int low_index = SC_INDEX(&dummy) + SC_CLEAR_MIN;
|
|
|
|
|
int high_index = SC_INDEX(&dummy) + SC_CLEAR_MAX;
|
|
|
|
|
for (int i = low_index; i <= high_index; ++i)
|
|
|
|
|
{
|
|
|
|
|
int current_index = i;
|
|
|
|
|
if (current_index < 0)
|
|
|
|
|
current_index += SELF_CACHE_SIZE;
|
|
|
|
|
if (current_index >= SELF_CACHE_SIZE)
|
|
|
|
|
current_index -= SELF_CACHE_SIZE;
|
|
|
|
|
_Jv_self_cache[current_index].high_sp_bits = BAD_HIGH_SP_VALUE;
|
|
|
|
|
}
|
|
|
|
|
# endif
|
2004-08-12 16:20:11 +00:00
|
|
|
|
// Block SIGCHLD which is used in natPosixProcess.cc.
|
2007-05-12 17:37:55 +00:00
|
|
|
|
_Jv_BlockSigchld();
|
2001-05-21 23:47:48 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadUnRegister ()
|
|
|
|
|
{
|
|
|
|
|
pthread_setspecific (_Jv_ThreadKey, NULL);
|
|
|
|
|
pthread_setspecific (_Jv_ThreadDataKey, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
// This function is called when a thread is started. We don't arrange
|
|
|
|
|
// to call the `run' method directly, because this function must
|
|
|
|
|
// return a value.
|
|
|
|
|
static void *
|
|
|
|
|
really_start (void *x)
|
|
|
|
|
{
|
|
|
|
|
struct starter *info = (struct starter *) x;
|
|
|
|
|
|
2001-05-21 23:47:48 -07:00
|
|
|
|
_Jv_ThreadRegister (info->data);
|
2000-03-28 02:22:24 +00:00
|
|
|
|
|
|
|
|
|
info->method (info->data->thread_obj);
|
2001-05-24 05:40:37 +00:00
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
if (! (info->data->flags & FLAG_DAEMON))
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_lock (&daemon_mutex);
|
|
|
|
|
--non_daemon_count;
|
|
|
|
|
if (! non_daemon_count)
|
|
|
|
|
pthread_cond_signal (&daemon_cond);
|
|
|
|
|
pthread_mutex_unlock (&daemon_mutex);
|
|
|
|
|
}
|
2001-05-24 05:40:37 +00:00
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadStart (java::lang::Thread *thread, _Jv_Thread_t *data,
|
|
|
|
|
_Jv_ThreadStartFunc *meth)
|
|
|
|
|
{
|
|
|
|
|
struct sched_param param;
|
|
|
|
|
pthread_attr_t attr;
|
|
|
|
|
struct starter *info;
|
|
|
|
|
|
|
|
|
|
if (data->flags & FLAG_START)
|
|
|
|
|
return;
|
|
|
|
|
data->flags |= FLAG_START;
|
|
|
|
|
|
2004-08-12 16:20:11 +00:00
|
|
|
|
// Block SIGCHLD which is used in natPosixProcess.cc.
|
|
|
|
|
// The current mask is inherited by the child thread.
|
2007-05-12 17:37:55 +00:00
|
|
|
|
_Jv_BlockSigchld();
|
2004-08-12 16:20:11 +00:00
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
param.sched_priority = thread->getPriority();
|
|
|
|
|
|
|
|
|
|
pthread_attr_init (&attr);
|
|
|
|
|
pthread_attr_setschedparam (&attr, ¶m);
|
2000-12-30 12:18:39 +00:00
|
|
|
|
pthread_attr_setdetachstate (&attr, PTHREAD_CREATE_DETACHED);
|
2005-11-17 20:25:57 +00:00
|
|
|
|
|
|
|
|
|
// Set stack size if -Xss option was given.
|
|
|
|
|
if (gcj::stack_size > 0)
|
|
|
|
|
{
|
|
|
|
|
int e = pthread_attr_setstacksize (&attr, gcj::stack_size);
|
|
|
|
|
if (e != 0)
|
|
|
|
|
JvFail (strerror (e));
|
|
|
|
|
}
|
1999-04-07 14:42:40 +00:00
|
|
|
|
|
|
|
|
|
info = (struct starter *) _Jv_AllocBytes (sizeof (struct starter));
|
|
|
|
|
info->method = meth;
|
|
|
|
|
info->data = data;
|
|
|
|
|
|
|
|
|
|
if (! thread->isDaemon())
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_lock (&daemon_mutex);
|
|
|
|
|
++non_daemon_count;
|
|
|
|
|
pthread_mutex_unlock (&daemon_mutex);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
data->flags |= FLAG_DAEMON;
|
1999-12-24 01:00:46 +00:00
|
|
|
|
int r = pthread_create (&data->thread, &attr, really_start, (void *) info);
|
|
|
|
|
|
1999-04-07 14:42:40 +00:00
|
|
|
|
pthread_attr_destroy (&attr);
|
1999-12-24 01:00:46 +00:00
|
|
|
|
|
|
|
|
|
if (r)
|
|
|
|
|
{
|
|
|
|
|
const char* msg = "Cannot create additional threads";
|
exception.cc (java_eh_info): Make value type jthrowable.
* exception.cc (java_eh_info): Make value type jthrowable.
(_Jv_type_matcher): Remove now unneeded cast.
(_Jv_Throw): Make argument type jthrowable. Munge name
for SJLJ_EXCEPTIONS here ...
* gcj/cni.h: ... not here.
(JvThrow): Remove.
* gcj/javaprims.h (_Jv_Throw, _Jv_Sjlj_Throw): Update declarations.
* defineclass.cc, interpret.cc, jni.cc, posix-threads.cc,
prims.cc, resolve.cc, gnu/gcj/runtime/natFirstThread.cc,
gnu/gcj/xlib/natDrawable.cc, gnu/gcj/xlib/natFont.cc,
gnu/gcj/xlib/natWMSizeHints.cc, gnu/gcj/xlib/natWindowAttributes.cc,
gnu/gcj/xlib/natXImage.cc, java/io/natFile.cc,
java/io/natFileDescriptorEcos.cc, java/io/natFileDescriptorPosix.cc,
java/io/natFileDescriptorWin32.cc, java/io/natFileWin32.cc,
java/lang/natClass.cc, java/lang/natClassLoader.cc,
java/lang/natDouble.cc, java/lang/natObject.cc,
java/lang/natPosixProcess.cc, java/lang/natRuntime.cc,
java/lang/natString.cc, java/lang/natSystem.cc,
java/lang/natThread.cc, java/lang/reflect/natArray.cc,
java/lang/reflect/natConstructor.cc, java/lang/reflect/natField.cc,
java/lang/reflect/natMethod.cc, java/util/zip/natDeflater.cc,
java/util/zip/natInflater.cc:
Use throw, not JvThrow or _Jv_Throw.
From-SVN: r40838
2001-03-25 23:05:32 -08:00
|
|
|
|
throw new java::lang::OutOfMemoryError (JvNewStringUTF (msg));
|
1999-12-24 01:00:46 +00:00
|
|
|
|
}
|
1999-04-07 14:42:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
_Jv_ThreadWait (void)
|
|
|
|
|
{
|
|
|
|
|
pthread_mutex_lock (&daemon_mutex);
|
|
|
|
|
if (non_daemon_count)
|
|
|
|
|
pthread_cond_wait (&daemon_cond, &daemon_mutex);
|
|
|
|
|
pthread_mutex_unlock (&daemon_mutex);
|
|
|
|
|
}
|
2001-05-24 05:40:37 +00:00
|
|
|
|
|
|
|
|
|
#if defined(SLOW_PTHREAD_SELF)
|
|
|
|
|
|
2002-03-10 03:53:16 +00:00
|
|
|
|
#include "sysdep/locks.h"
|
2001-05-24 05:40:37 +00:00
|
|
|
|
|
2002-03-10 03:53:16 +00:00
|
|
|
|
// Support for pthread_self() lookup cache.
|
2001-05-24 05:40:37 +00:00
|
|
|
|
volatile self_cache_entry _Jv_self_cache[SELF_CACHE_SIZE];
|
|
|
|
|
|
|
|
|
|
_Jv_ThreadId_t
|
|
|
|
|
_Jv_ThreadSelf_out_of_line(volatile self_cache_entry *sce, size_t high_sp_bits)
|
|
|
|
|
{
|
|
|
|
|
pthread_t self = pthread_self();
|
|
|
|
|
sce -> high_sp_bits = high_sp_bits;
|
2002-03-21 00:26:46 +00:00
|
|
|
|
write_barrier();
|
|
|
|
|
sce -> self = self;
|
2001-05-24 05:40:37 +00:00
|
|
|
|
return self;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* SLOW_PTHREAD_SELF */
|