2009-06-03 16:17:55 -06:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 11:06:52 -07:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-07-04 16:27:08 -06:00
|
|
|
package java.lang;
|
|
|
|
|
2007-07-21 14:44:39 -06:00
|
|
|
import java.util.Map;
|
|
|
|
import java.util.WeakHashMap;
|
|
|
|
|
2007-07-04 16:27:08 -06:00
|
|
|
public class Thread implements Runnable {
|
2007-07-28 10:55:24 -06:00
|
|
|
private long peer;
|
2009-06-03 16:17:55 -06:00
|
|
|
private boolean interrupted;
|
|
|
|
private boolean daemon;
|
|
|
|
private byte state;
|
|
|
|
private byte priority;
|
2007-07-04 16:27:08 -06:00
|
|
|
private final Runnable task;
|
2007-07-21 14:44:39 -06:00
|
|
|
private Map<ThreadLocal, Object> locals;
|
2007-07-28 15:28:25 -06:00
|
|
|
private Object sleepLock;
|
2007-07-31 18:08:20 -06:00
|
|
|
private ClassLoader classLoader;
|
2009-06-03 16:17:55 -06:00
|
|
|
private UncaughtExceptionHandler exceptionHandler;
|
2007-07-04 16:27:08 -06:00
|
|
|
|
2009-06-04 18:03:09 -06:00
|
|
|
// package private for GNU Classpath, which inexplicably bypasses
|
|
|
|
// the accessor methods:
|
2009-06-03 16:17:55 -06:00
|
|
|
String name;
|
|
|
|
ThreadGroup group;
|
|
|
|
|
|
|
|
private static UncaughtExceptionHandler defaultExceptionHandler;
|
|
|
|
|
|
|
|
public static final int MIN_PRIORITY = 1;
|
|
|
|
public static final int NORM_PRIORITY = 5;
|
|
|
|
public static final int MAX_PRIORITY = 10;
|
|
|
|
|
|
|
|
public Thread(ThreadGroup group, Runnable task, String name, long stackSize)
|
|
|
|
{
|
2009-08-10 07:46:59 -06:00
|
|
|
this.group = (group == null ? Thread.currentThread().group : group);
|
2008-07-13 18:34:59 -06:00
|
|
|
this.task = task;
|
2008-07-13 18:21:04 -06:00
|
|
|
this.name = name;
|
2007-07-28 15:28:25 -06:00
|
|
|
|
2007-07-31 18:08:20 -06:00
|
|
|
Thread current = currentThread();
|
|
|
|
|
|
|
|
Map<ThreadLocal, Object> map = current.locals;
|
2007-07-21 16:36:51 -06:00
|
|
|
if (map != null) {
|
|
|
|
for (Map.Entry<ThreadLocal, Object> e: map.entrySet()) {
|
|
|
|
if (e.getKey() instanceof InheritableThreadLocal) {
|
2007-07-28 10:10:13 -06:00
|
|
|
InheritableThreadLocal itl = (InheritableThreadLocal) e.getKey();
|
|
|
|
locals().put(itl, itl.childValue(e.getValue()));
|
2007-07-21 16:36:51 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-07-31 18:08:20 -06:00
|
|
|
|
|
|
|
classLoader = current.classLoader;
|
2007-07-29 19:27:42 -06:00
|
|
|
}
|
2007-07-21 16:36:51 -06:00
|
|
|
|
2009-06-03 16:17:55 -06:00
|
|
|
public Thread(ThreadGroup group, Runnable task, String name) {
|
|
|
|
this(group, task, name, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Thread(ThreadGroup group, String name) {
|
|
|
|
this(null, null, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Thread(Runnable task, String name) {
|
|
|
|
this(null, task, name);
|
|
|
|
}
|
|
|
|
|
2008-07-13 18:21:04 -06:00
|
|
|
public Thread(Runnable task) {
|
2009-06-03 16:17:55 -06:00
|
|
|
this(null, task, "Thread["+task+"]");
|
|
|
|
}
|
|
|
|
|
|
|
|
public Thread(String name) {
|
|
|
|
this(null, null, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Thread() {
|
|
|
|
this((Runnable) null);
|
2007-11-06 17:41:53 -07:00
|
|
|
}
|
|
|
|
|
2007-07-29 19:27:42 -06:00
|
|
|
public synchronized void start() {
|
|
|
|
if (peer != 0) {
|
|
|
|
throw new IllegalStateException("thread already started");
|
|
|
|
}
|
|
|
|
|
2009-08-10 17:36:11 -06:00
|
|
|
state = (byte) State.RUNNABLE.ordinal();
|
|
|
|
|
2007-07-29 19:27:42 -06:00
|
|
|
peer = doStart();
|
|
|
|
if (peer == 0) {
|
2009-08-10 17:36:11 -06:00
|
|
|
state = (byte) State.NEW.ordinal();
|
2007-07-29 19:27:42 -06:00
|
|
|
throw new RuntimeException("unable to start native thread");
|
|
|
|
}
|
2007-07-21 16:36:51 -06:00
|
|
|
}
|
|
|
|
|
2007-07-29 19:27:42 -06:00
|
|
|
private native long doStart();
|
2007-07-04 16:27:08 -06:00
|
|
|
|
2009-06-03 16:17:55 -06:00
|
|
|
private static void run(Thread t) throws Throwable {
|
|
|
|
try {
|
|
|
|
t.run();
|
|
|
|
} catch (Throwable e) {
|
|
|
|
UncaughtExceptionHandler eh = t.exceptionHandler;
|
|
|
|
UncaughtExceptionHandler deh = defaultExceptionHandler;
|
|
|
|
if (eh != null) {
|
|
|
|
eh.uncaughtException(t, e);
|
|
|
|
} else if (deh != null) {
|
|
|
|
deh.uncaughtException(t, e);
|
|
|
|
} else {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
synchronized (t) {
|
|
|
|
t.state = (byte) State.TERMINATED.ordinal();
|
|
|
|
t.notifyAll();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-04 16:27:08 -06:00
|
|
|
public void run() {
|
|
|
|
if (task != null) {
|
|
|
|
task.run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-31 18:08:20 -06:00
|
|
|
public ClassLoader getContextClassLoader() {
|
|
|
|
return classLoader;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setContextClassLoader(ClassLoader v) {
|
|
|
|
classLoader = v;
|
|
|
|
}
|
|
|
|
|
2007-07-21 14:44:39 -06:00
|
|
|
public Map<ThreadLocal, Object> locals() {
|
|
|
|
if (locals == null) {
|
|
|
|
locals = new WeakHashMap();
|
|
|
|
}
|
|
|
|
return locals;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static native Thread currentThread();
|
|
|
|
|
2007-07-28 15:28:25 -06:00
|
|
|
private static native void interrupt(long peer);
|
|
|
|
|
|
|
|
public synchronized void interrupt() {
|
|
|
|
if (peer != 0) {
|
|
|
|
interrupt(peer);
|
|
|
|
} else {
|
|
|
|
interrupted = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean interrupted() {
|
|
|
|
Thread t = currentThread();
|
|
|
|
|
|
|
|
synchronized (t) {
|
|
|
|
boolean v = t.interrupted;
|
|
|
|
t.interrupted = false;
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-03 16:17:55 -06:00
|
|
|
public static boolean isInterrupted() {
|
|
|
|
return currentThread().interrupted;
|
|
|
|
}
|
|
|
|
|
2007-07-28 10:10:13 -06:00
|
|
|
public static void sleep(long milliseconds) throws InterruptedException {
|
|
|
|
Thread t = currentThread();
|
|
|
|
if (t.sleepLock == null) {
|
|
|
|
t.sleepLock = new Object();
|
|
|
|
}
|
|
|
|
synchronized (t.sleepLock) {
|
|
|
|
t.sleepLock.wait(milliseconds);
|
|
|
|
}
|
|
|
|
}
|
2008-04-11 16:48:39 -06:00
|
|
|
|
2009-06-03 16:17:55 -06:00
|
|
|
public static void sleep(long milliseconds, int nanoseconds)
|
|
|
|
throws InterruptedException
|
|
|
|
{
|
|
|
|
if (nanoseconds > 0) {
|
|
|
|
++ milliseconds;
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep(milliseconds);
|
|
|
|
}
|
|
|
|
|
2008-04-11 16:48:39 -06:00
|
|
|
public StackTraceElement[] getStackTrace() {
|
|
|
|
return Throwable.resolveTrace(getStackTrace(peer));
|
|
|
|
}
|
|
|
|
|
|
|
|
private static native Object getStackTrace(long peer);
|
|
|
|
|
|
|
|
public static native int activeCount();
|
|
|
|
|
|
|
|
public static native int enumerate(Thread[] array);
|
2008-07-13 18:21:04 -06:00
|
|
|
|
|
|
|
public String getName() {
|
|
|
|
return name;
|
|
|
|
}
|
2009-06-03 16:17:55 -06:00
|
|
|
|
|
|
|
public void setName(String name) {
|
|
|
|
this.name = name;
|
|
|
|
}
|
|
|
|
|
|
|
|
public UncaughtExceptionHandler getUncaughtExceptionHandler() {
|
|
|
|
UncaughtExceptionHandler eh = exceptionHandler;
|
|
|
|
return (eh == null ? group : eh);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static UncaughtExceptionHandler getDefaultUncaughtExceptionHandler() {
|
|
|
|
return defaultExceptionHandler;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setUncaughtExceptionHandler(UncaughtExceptionHandler h) {
|
|
|
|
exceptionHandler = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void setDefaultUncaughtExceptionHandler
|
|
|
|
(UncaughtExceptionHandler h)
|
|
|
|
{
|
|
|
|
defaultExceptionHandler = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
public State getState() {
|
|
|
|
return State.values()[state];
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isAlive() {
|
|
|
|
switch (getState()) {
|
|
|
|
case NEW:
|
|
|
|
case TERMINATED:
|
|
|
|
return false;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getPriority() {
|
|
|
|
return priority;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setPriority(int priority) {
|
|
|
|
if (priority < MIN_PRIORITY || priority > MAX_PRIORITY) {
|
|
|
|
throw new IllegalArgumentException();
|
|
|
|
}
|
|
|
|
this.priority = (byte) priority;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isDaemon() {
|
|
|
|
return daemon;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setDaemon(boolean v) {
|
|
|
|
daemon = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static native void yield();
|
|
|
|
|
|
|
|
public synchronized void join() throws InterruptedException {
|
|
|
|
while (getState() != State.TERMINATED) {
|
|
|
|
wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public synchronized void join(long milliseconds) throws InterruptedException
|
|
|
|
{
|
|
|
|
long then = System.currentTimeMillis();
|
2009-06-04 18:07:00 -06:00
|
|
|
long remaining = milliseconds;
|
2009-06-04 18:08:51 -06:00
|
|
|
while (remaining > 0 && getState() != State.TERMINATED) {
|
2009-06-04 18:07:00 -06:00
|
|
|
wait(remaining);
|
2009-06-03 16:17:55 -06:00
|
|
|
|
2009-06-04 18:07:00 -06:00
|
|
|
remaining = milliseconds - (System.currentTimeMillis() - then);
|
2009-06-03 16:17:55 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void join(long milliseconds, int nanoseconds)
|
|
|
|
throws InterruptedException
|
|
|
|
{
|
|
|
|
if (nanoseconds > 0) {
|
|
|
|
++ milliseconds;
|
|
|
|
}
|
|
|
|
|
|
|
|
join(milliseconds);
|
|
|
|
}
|
|
|
|
|
|
|
|
public ThreadGroup getThreadGroup() {
|
|
|
|
return group;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static native boolean holdsLock(Object o);
|
|
|
|
|
|
|
|
public long getId() {
|
|
|
|
return peer;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void stop() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void stop(Throwable t) {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void suspend() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void resume() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public interface UncaughtExceptionHandler {
|
|
|
|
public void uncaughtException(Thread t, Throwable e);
|
|
|
|
}
|
|
|
|
|
|
|
|
public enum State {
|
|
|
|
NEW, RUNNABLE, BLOCKED, WAITING, TIMED_WAITING, TERMINATED
|
|
|
|
}
|
2008-07-13 18:21:04 -06:00
|
|
|
|
2007-07-04 16:27:08 -06:00
|
|
|
}
|