| /* |
| * Copyright (C) 2008 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "debugger.h" |
| #include "jni_internal.h" |
| #include "object.h" |
| #include "scoped_thread_list_lock.h" |
| #include "ScopedUtfChars.h" |
| #include "thread.h" |
| #include "thread_list.h" |
| |
| #include "JniConstants.h" // Last to avoid problems with LOG redefinition. |
| |
| namespace art { |
| |
| static jobject Thread_currentThread(JNIEnv* env, jclass) { |
| return AddLocalReference<jobject>(env, Thread::Current()->GetPeer()); |
| } |
| |
| static jboolean Thread_interrupted(JNIEnv*, jclass) { |
| return Thread::Current()->Interrupted(); |
| } |
| |
| static jboolean Thread_isInterrupted(JNIEnv* env, jobject javaThread) { |
| ScopedThreadListLock thread_list_lock; |
| Thread* thread = Thread::FromManagedThread(env, javaThread); |
| return (thread != NULL) ? thread->IsInterrupted() : JNI_FALSE; |
| } |
| |
| static void Thread_nativeCreate(JNIEnv* env, jclass, jobject javaThread, jlong stackSize) { |
| Object* managedThread = Decode<Object*>(env, javaThread); |
| Thread::Create(managedThread, stackSize); |
| } |
| |
| static jint Thread_nativeGetStatus(JNIEnv* env, jobject javaThread, jboolean hasBeenStarted) { |
| // Ordinals from Java's Thread.State. |
| const jint kJavaNew = 0; |
| const jint kJavaRunnable = 1; |
| const jint kJavaBlocked = 2; |
| const jint kJavaWaiting = 3; |
| const jint kJavaTimedWaiting = 4; |
| const jint kJavaTerminated = 5; |
| |
| Thread::State internal_thread_state = (hasBeenStarted ? Thread::kTerminated : Thread::kStarting); |
| ScopedThreadListLock thread_list_lock; |
| Thread* thread = Thread::FromManagedThread(env, javaThread); |
| if (thread != NULL) { |
| internal_thread_state = thread->GetState(); |
| } |
| switch (internal_thread_state) { |
| case Thread::kTerminated: return kJavaTerminated; |
| case Thread::kRunnable: return kJavaRunnable; |
| case Thread::kTimedWaiting: return kJavaTimedWaiting; |
| case Thread::kBlocked: return kJavaBlocked; |
| case Thread::kWaiting: return kJavaWaiting; |
| case Thread::kStarting: return kJavaNew; |
| case Thread::kNative: return kJavaRunnable; |
| case Thread::kVmWait: return kJavaWaiting; |
| case Thread::kSuspended: return kJavaRunnable; |
| // Don't add a 'default' here so the compiler can spot incompatible enum changes. |
| } |
| return -1; // Unreachable. |
| } |
| |
| static jboolean Thread_nativeHoldsLock(JNIEnv* env, jobject javaThread, jobject javaObject) { |
| Object* object = Decode<Object*>(env, javaObject); |
| if (object == NULL) { |
| ScopedThreadStateChange tsc(Thread::Current(), Thread::kRunnable); |
| Thread::Current()->ThrowNewException("Ljava/lang/NullPointerException;", "object == null"); |
| return JNI_FALSE; |
| } |
| ScopedThreadListLock thread_list_lock; |
| Thread* thread = Thread::FromManagedThread(env, javaThread); |
| return thread->HoldsLock(object); |
| } |
| |
| static void Thread_nativeInterrupt(JNIEnv* env, jobject javaThread) { |
| ScopedThreadListLock thread_list_lock; |
| Thread* thread = Thread::FromManagedThread(env, javaThread); |
| if (thread != NULL) { |
| thread->Interrupt(); |
| } |
| } |
| |
| static void Thread_nativeSetName(JNIEnv* env, jobject javaThread, jstring javaName) { |
| ScopedThreadListLock thread_list_lock; |
| Thread* thread = Thread::FromManagedThread(env, javaThread); |
| if (thread == NULL) { |
| return; |
| } |
| ScopedUtfChars name(env, javaName); |
| if (name.c_str() == NULL) { |
| return; |
| } |
| thread->SetThreadName(name.c_str()); |
| } |
| |
| /* |
| * Alter the priority of the specified thread. "newPriority" will range |
| * from Thread.MIN_PRIORITY to Thread.MAX_PRIORITY (1-10), with "normal" |
| * threads at Thread.NORM_PRIORITY (5). |
| */ |
| static void Thread_nativeSetPriority(JNIEnv* env, jobject javaThread, jint newPriority) { |
| ScopedThreadListLock thread_list_lock; |
| Thread* thread = Thread::FromManagedThread(env, javaThread); |
| if (thread != NULL) { |
| thread->SetNativePriority(newPriority); |
| } |
| } |
| |
| /* |
| * Causes the thread to temporarily pause and allow other threads to execute. |
| * |
| * The exact behavior is poorly defined. Some discussion here: |
| * http://www.cs.umd.edu/~pugh/java/memoryModel/archive/0944.html |
| */ |
| static void Thread_yield(JNIEnv*, jobject) { |
| sched_yield(); |
| } |
| |
| static JNINativeMethod gMethods[] = { |
| NATIVE_METHOD(Thread, currentThread, "()Ljava/lang/Thread;"), |
| NATIVE_METHOD(Thread, interrupted, "()Z"), |
| NATIVE_METHOD(Thread, isInterrupted, "()Z"), |
| NATIVE_METHOD(Thread, nativeCreate, "(Ljava/lang/Thread;J)V"), |
| NATIVE_METHOD(Thread, nativeGetStatus, "(Z)I"), |
| NATIVE_METHOD(Thread, nativeHoldsLock, "(Ljava/lang/Object;)Z"), |
| NATIVE_METHOD(Thread, nativeInterrupt, "()V"), |
| NATIVE_METHOD(Thread, nativeSetName, "(Ljava/lang/String;)V"), |
| NATIVE_METHOD(Thread, nativeSetPriority, "(I)V"), |
| NATIVE_METHOD(Thread, yield, "()V"), |
| }; |
| |
| void register_java_lang_Thread(JNIEnv* env) { |
| jniRegisterNativeMethods(env, "java/lang/Thread", gMethods, NELEM(gMethods)); |
| } |
| |
| } // namespace art |