The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2008 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | /* |
| 17 | * VM thread support. |
| 18 | */ |
| 19 | #ifndef _DALVIK_THREAD |
| 20 | #define _DALVIK_THREAD |
| 21 | |
| 22 | #include "jni.h" |
| 23 | |
| 24 | #if defined(CHECK_MUTEX) && !defined(__USE_UNIX98) |
| 25 | /* Linux lacks this unless you #define __USE_UNIX98 */ |
| 26 | int pthread_mutexattr_settype(pthread_mutexattr_t *attr, int type); |
| 27 | enum { PTHREAD_MUTEX_ERRORCHECK = PTHREAD_MUTEX_ERRORCHECK_NP }; |
| 28 | #endif |
| 29 | |
| 30 | #ifdef WITH_MONITOR_TRACKING |
| 31 | struct LockedObjectData; |
| 32 | #endif |
| 33 | |
| 34 | /* |
| 35 | * Current status; these map to JDWP constants, so don't rearrange them. |
| 36 | * (If you do alter this, update the strings in dvmDumpThread and the |
| 37 | * conversion table in VMThread.java.) |
| 38 | * |
| 39 | * Note that "suspended" is orthogonal to these values (so says JDWP). |
| 40 | */ |
| 41 | typedef enum ThreadStatus { |
| 42 | /* these match up with JDWP values */ |
| 43 | THREAD_ZOMBIE = 0, /* TERMINATED */ |
| 44 | THREAD_RUNNING = 1, /* RUNNABLE or running now */ |
| 45 | THREAD_TIMED_WAIT = 2, /* TIMED_WAITING in Object.wait() */ |
| 46 | THREAD_MONITOR = 3, /* BLOCKED on a monitor */ |
| 47 | THREAD_WAIT = 4, /* WAITING in Object.wait() */ |
| 48 | /* non-JDWP states */ |
| 49 | THREAD_INITIALIZING = 5, /* allocated, not yet running */ |
| 50 | THREAD_STARTING = 6, /* started, not yet on thread list */ |
| 51 | THREAD_NATIVE = 7, /* off in a JNI native method */ |
| 52 | THREAD_VMWAIT = 8, /* waiting on a VM resource */ |
| 53 | } ThreadStatus; |
| 54 | |
| 55 | /* thread priorities, from java.lang.Thread */ |
| 56 | enum { |
| 57 | THREAD_MIN_PRIORITY = 1, |
| 58 | THREAD_NORM_PRIORITY = 5, |
| 59 | THREAD_MAX_PRIORITY = 10, |
| 60 | }; |
| 61 | |
| 62 | |
| 63 | /* initialization */ |
| 64 | bool dvmThreadStartup(void); |
| 65 | bool dvmThreadObjStartup(void); |
| 66 | void dvmThreadShutdown(void); |
| 67 | void dvmSlayDaemons(void); |
| 68 | |
| 69 | |
| 70 | #define kJniLocalRefMax 512 /* arbitrary; should be plenty */ |
| 71 | #define kInternalRefDefault 32 /* equally arbitrary */ |
| 72 | #define kInternalRefMax 4096 /* mainly a sanity check */ |
| 73 | |
| 74 | #define kMinStackSize (512 + STACK_OVERFLOW_RESERVE) |
Andy McFadden | 39b99b0 | 2009-05-11 14:39:13 -0700 | [diff] [blame] | 75 | #define kDefaultStackSize (12*1024) /* three 4K pages */ |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 76 | #define kMaxStackSize (256*1024 + STACK_OVERFLOW_RESERVE) |
| 77 | |
| 78 | /* |
| 79 | * Our per-thread data. |
| 80 | * |
| 81 | * These are allocated on the system heap. |
| 82 | */ |
| 83 | typedef struct Thread { |
| 84 | /* small unique integer; useful for "thin" locks and debug messages */ |
| 85 | u4 threadId; |
| 86 | |
| 87 | /* |
| 88 | * Thread's current status. Can only be changed by the thread itself |
| 89 | * (i.e. don't mess with this from other threads). |
| 90 | */ |
| 91 | ThreadStatus status; |
| 92 | |
| 93 | /* |
| 94 | * This is the number of times the thread has been suspended. When the |
| 95 | * count drops to zero, the thread resumes. |
| 96 | * |
| 97 | * "dbgSuspendCount" is the portion of the suspend count that the |
| 98 | * debugger is responsible for. This has to be tracked separately so |
| 99 | * that we can recover correctly if the debugger abruptly disconnects |
| 100 | * (suspendCount -= dbgSuspendCount). The debugger should not be able |
| 101 | * to resume GC-suspended threads, because we ignore the debugger while |
| 102 | * a GC is in progress. |
| 103 | * |
| 104 | * Both of these are guarded by gDvm.threadSuspendCountLock. |
| 105 | * |
| 106 | * (We could store both of these in the same 32-bit, using 16-bit |
| 107 | * halves, to make atomic ops possible. In practice, you only need |
| 108 | * to read suspendCount, and we need to hold a mutex when making |
| 109 | * changes, so there's no need to merge them. Note the non-debug |
| 110 | * component will rarely be other than 1 or 0 -- not sure it's even |
| 111 | * possible with the way mutexes are currently used.) |
| 112 | */ |
| 113 | int suspendCount; |
| 114 | int dbgSuspendCount; |
| 115 | |
| 116 | /* |
| 117 | * Set to true when the thread suspends itself, false when it wakes up. |
| 118 | * This is only expected to be set when status==THREAD_RUNNING. |
| 119 | */ |
| 120 | bool isSuspended; |
| 121 | |
| 122 | /* thread handle, as reported by pthread_self() */ |
| 123 | pthread_t handle; |
| 124 | |
| 125 | /* thread ID, only useful under Linux */ |
| 126 | pid_t systemTid; |
| 127 | |
| 128 | /* start (high addr) of interp stack (subtract size to get malloc addr) */ |
| 129 | u1* interpStackStart; |
| 130 | |
| 131 | /* current limit of stack; flexes for StackOverflowError */ |
| 132 | const u1* interpStackEnd; |
| 133 | |
| 134 | /* interpreter stack size; our stacks are fixed-length */ |
| 135 | int interpStackSize; |
| 136 | bool stackOverflowed; |
| 137 | |
| 138 | /* FP of bottom-most (currently executing) stack frame on interp stack */ |
| 139 | void* curFrame; |
| 140 | |
| 141 | /* current exception, or NULL if nothing pending */ |
| 142 | Object* exception; |
| 143 | |
| 144 | /* the java/lang/Thread that we are associated with */ |
| 145 | Object* threadObj; |
| 146 | |
| 147 | /* the JNIEnv pointer associated with this thread */ |
| 148 | JNIEnv* jniEnv; |
| 149 | |
| 150 | /* internal reference tracking */ |
| 151 | ReferenceTable internalLocalRefTable; |
| 152 | |
| 153 | /* JNI local reference tracking */ |
Andy McFadden | 734155e | 2009-07-16 18:11:22 -0700 | [diff] [blame] | 154 | // TODO: move this to JNIEnvExt to avoid an indirection? |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 155 | ReferenceTable jniLocalRefTable; |
| 156 | |
| 157 | /* JNI native monitor reference tracking (initialized on first use) */ |
| 158 | ReferenceTable jniMonitorRefTable; |
| 159 | |
| 160 | /* hack to make JNI_OnLoad work right */ |
| 161 | Object* classLoaderOverride; |
| 162 | |
| 163 | /* pointer to the monitor lock we're currently waiting on */ |
| 164 | /* (do not set or clear unless the Monitor itself is held) */ |
| 165 | /* TODO: consider changing this to Object* for better JDWP interaction */ |
| 166 | Monitor* waitMonitor; |
| 167 | /* set when we confirm that the thread must be interrupted from a wait */ |
| 168 | bool interruptingWait; |
| 169 | /* thread "interrupted" status; stays raised until queried or thrown */ |
| 170 | bool interrupted; |
| 171 | |
| 172 | /* |
| 173 | * Set to true when the thread is in the process of throwing an |
| 174 | * OutOfMemoryError. |
| 175 | */ |
| 176 | bool throwingOOME; |
| 177 | |
| 178 | /* links to rest of thread list; grab global lock before traversing */ |
| 179 | struct Thread* prev; |
| 180 | struct Thread* next; |
| 181 | |
| 182 | /* JDWP invoke-during-breakpoint support */ |
| 183 | DebugInvokeReq invokeReq; |
| 184 | |
| 185 | #ifdef WITH_MONITOR_TRACKING |
| 186 | /* objects locked by this thread; most recent is at head of list */ |
| 187 | struct LockedObjectData* pLockedObjects; |
| 188 | #endif |
| 189 | |
| 190 | #ifdef WITH_ALLOC_LIMITS |
| 191 | /* allocation limit, for Debug.setAllocationLimit() regression testing */ |
| 192 | int allocLimit; |
| 193 | #endif |
| 194 | |
| 195 | #ifdef WITH_PROFILER |
| 196 | /* base time for per-thread CPU timing */ |
| 197 | bool cpuClockBaseSet; |
| 198 | u8 cpuClockBase; |
| 199 | |
| 200 | /* memory allocation profiling state */ |
| 201 | AllocProfState allocProf; |
| 202 | #endif |
| 203 | |
| 204 | #ifdef WITH_JNI_STACK_CHECK |
| 205 | u4 stackCrc; |
| 206 | #endif |
The Android Open Source Project | 9940988 | 2009-03-18 22:20:24 -0700 | [diff] [blame] | 207 | |
| 208 | #if WITH_EXTRA_GC_CHECKS > 1 |
| 209 | /* PC, saved on every instruction; redundant with StackSaveArea */ |
| 210 | const u2* currentPc2; |
| 211 | #endif |
Jeff Hao | 97319a8 | 2009-08-12 16:57:15 -0700 | [diff] [blame^] | 212 | |
| 213 | #if defined(WITH_SELF_VERIFICATION) |
| 214 | /* Buffer for register state during self verification */ |
| 215 | struct ShadowSpace* shadowSpace; |
| 216 | #endif |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 217 | } Thread; |
| 218 | |
| 219 | /* start point for an internal thread; mimics pthread args */ |
| 220 | typedef void* (*InternalThreadStart)(void* arg); |
| 221 | |
| 222 | /* args for internal thread creation */ |
| 223 | typedef struct InternalStartArgs { |
| 224 | /* inputs */ |
| 225 | InternalThreadStart func; |
| 226 | void* funcArg; |
| 227 | char* name; |
| 228 | Object* group; |
| 229 | bool isDaemon; |
| 230 | /* result */ |
| 231 | volatile Thread** pThread; |
| 232 | volatile int* pCreateStatus; |
| 233 | } InternalStartArgs; |
| 234 | |
| 235 | /* finish init */ |
| 236 | bool dvmPrepMainForJni(JNIEnv* pEnv); |
| 237 | bool dvmPrepMainThread(void); |
| 238 | |
| 239 | /* utility function to get the tid */ |
| 240 | pid_t dvmGetSysThreadId(void); |
| 241 | |
| 242 | /* |
| 243 | * Get our Thread* from TLS. |
| 244 | * |
| 245 | * Returns NULL if this isn't a thread that the VM is aware of. |
| 246 | */ |
| 247 | Thread* dvmThreadSelf(void); |
| 248 | |
| 249 | /* grab the thread list global lock */ |
| 250 | void dvmLockThreadList(Thread* self); |
| 251 | /* release the thread list global lock */ |
| 252 | void dvmUnlockThreadList(void); |
| 253 | |
| 254 | /* |
| 255 | * Thread suspend/resume, used by the GC and debugger. |
| 256 | */ |
| 257 | typedef enum SuspendCause { |
| 258 | SUSPEND_NOT = 0, |
| 259 | SUSPEND_FOR_GC, |
| 260 | SUSPEND_FOR_DEBUG, |
| 261 | SUSPEND_FOR_DEBUG_EVENT, |
| 262 | SUSPEND_FOR_STACK_DUMP, |
| 263 | SUSPEND_FOR_DEX_OPT, |
Bill Buzbee | 2717622 | 2009-06-09 09:20:16 -0700 | [diff] [blame] | 264 | #if defined(WITH_JIT) |
| 265 | SUSPEND_FOR_JIT, |
| 266 | #endif |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 267 | } SuspendCause; |
| 268 | void dvmSuspendThread(Thread* thread); |
| 269 | void dvmSuspendSelf(bool jdwpActivity); |
| 270 | void dvmResumeThread(Thread* thread); |
| 271 | void dvmSuspendAllThreads(SuspendCause why); |
| 272 | void dvmResumeAllThreads(SuspendCause why); |
| 273 | void dvmUndoDebuggerSuspensions(void); |
| 274 | |
| 275 | /* |
| 276 | * Check suspend state. Grab threadListLock before calling. |
| 277 | */ |
| 278 | bool dvmIsSuspended(Thread* thread); |
| 279 | |
| 280 | /* |
| 281 | * Wait until a thread has suspended. (Used by debugger support.) |
| 282 | */ |
| 283 | void dvmWaitForSuspend(Thread* thread); |
| 284 | |
| 285 | /* |
| 286 | * Check to see if we should be suspended now. If so, suspend ourselves |
| 287 | * by sleeping on a condition variable. |
| 288 | * |
| 289 | * If "self" is NULL, this will use dvmThreadSelf(). |
| 290 | */ |
| 291 | bool dvmCheckSuspendPending(Thread* self); |
| 292 | |
| 293 | /* |
The Android Open Source Project | 9940988 | 2009-03-18 22:20:24 -0700 | [diff] [blame] | 294 | * Fast test for use in the interpreter. Returns "true" if our suspend |
| 295 | * count is nonzero. |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 296 | */ |
The Android Open Source Project | 9940988 | 2009-03-18 22:20:24 -0700 | [diff] [blame] | 297 | INLINE bool dvmCheckSuspendQuick(Thread* self) { |
| 298 | return (self->suspendCount != 0); |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 299 | } |
| 300 | |
| 301 | /* |
| 302 | * Used when changing thread state. Threads may only change their own. |
| 303 | * The "self" argument, which may be NULL, is accepted as an optimization. |
| 304 | * |
| 305 | * If you're calling this before waiting on a resource (e.g. THREAD_WAIT |
| 306 | * or THREAD_MONITOR), do so in the same function as the wait -- this records |
| 307 | * the current stack depth for the GC. |
| 308 | * |
| 309 | * If you're changing to THREAD_RUNNING, this will check for suspension. |
| 310 | * |
| 311 | * Returns the old status. |
| 312 | */ |
| 313 | ThreadStatus dvmChangeStatus(Thread* self, ThreadStatus newStatus); |
| 314 | |
| 315 | /* |
| 316 | * Initialize a mutex. |
| 317 | */ |
| 318 | INLINE void dvmInitMutex(pthread_mutex_t* pMutex) |
| 319 | { |
| 320 | #ifdef CHECK_MUTEX |
| 321 | pthread_mutexattr_t attr; |
| 322 | int cc; |
| 323 | |
| 324 | pthread_mutexattr_init(&attr); |
| 325 | cc = pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK_NP); |
| 326 | assert(cc == 0); |
| 327 | pthread_mutex_init(pMutex, &attr); |
| 328 | pthread_mutexattr_destroy(&attr); |
| 329 | #else |
| 330 | pthread_mutex_init(pMutex, NULL); // default=PTHREAD_MUTEX_FAST_NP |
| 331 | #endif |
| 332 | } |
| 333 | |
| 334 | /* |
| 335 | * Grab a plain mutex. |
| 336 | */ |
| 337 | INLINE void dvmLockMutex(pthread_mutex_t* pMutex) |
| 338 | { |
| 339 | int cc = pthread_mutex_lock(pMutex); |
| 340 | assert(cc == 0); |
| 341 | } |
| 342 | |
| 343 | /* |
| 344 | * Unlock pthread mutex. |
| 345 | */ |
| 346 | INLINE void dvmUnlockMutex(pthread_mutex_t* pMutex) |
| 347 | { |
| 348 | int cc = pthread_mutex_unlock(pMutex); |
| 349 | assert(cc == 0); |
| 350 | } |
| 351 | |
| 352 | /* |
| 353 | * Destroy a mutex. |
| 354 | */ |
| 355 | INLINE void dvmDestroyMutex(pthread_mutex_t* pMutex) |
| 356 | { |
| 357 | int cc = pthread_mutex_destroy(pMutex); |
| 358 | assert(cc == 0); |
| 359 | } |
| 360 | |
| 361 | /* |
| 362 | * Create a thread as a result of java.lang.Thread.start(). |
| 363 | */ |
| 364 | bool dvmCreateInterpThread(Object* threadObj, int reqStackSize); |
| 365 | |
| 366 | /* |
| 367 | * Create a thread internal to the VM. It's visible to interpreted code, |
| 368 | * but found in the "system" thread group rather than "main". |
| 369 | */ |
| 370 | bool dvmCreateInternalThread(pthread_t* pHandle, const char* name, |
| 371 | InternalThreadStart func, void* funcArg); |
| 372 | |
| 373 | /* |
| 374 | * Attach or detach the current thread from the VM. |
| 375 | */ |
| 376 | bool dvmAttachCurrentThread(const JavaVMAttachArgs* pArgs, bool isDaemon); |
| 377 | void dvmDetachCurrentThread(void); |
| 378 | |
| 379 | /* |
| 380 | * Get the "main" or "system" thread group. |
| 381 | */ |
| 382 | Object* dvmGetMainThreadGroup(void); |
| 383 | Object* dvmGetSystemThreadGroup(void); |
| 384 | |
| 385 | /* |
| 386 | * Given a java/lang/VMThread object, return our Thread. |
| 387 | */ |
| 388 | Thread* dvmGetThreadFromThreadObject(Object* vmThreadObj); |
| 389 | |
| 390 | /* |
| 391 | * Sleep in a thread. Returns when the sleep timer returns or the thread |
| 392 | * is interrupted. |
| 393 | */ |
| 394 | void dvmThreadSleep(u8 msec, u4 nsec); |
| 395 | |
| 396 | /* |
| 397 | * Get the name of a thread. (For safety, hold the thread list lock.) |
| 398 | */ |
| 399 | char* dvmGetThreadName(Thread* thread); |
| 400 | |
| 401 | /* |
| 402 | * Return true if a thread is on the internal list. If it is, the |
| 403 | * thread is part of the GC's root set. |
| 404 | */ |
| 405 | bool dvmIsOnThreadList(const Thread* thread); |
Jeff Hao | 97319a8 | 2009-08-12 16:57:15 -0700 | [diff] [blame^] | 406 | |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 407 | /* |
| 408 | * Get/set the JNIEnv field. |
| 409 | */ |
| 410 | INLINE JNIEnv* dvmGetThreadJNIEnv(Thread* self) { return self->jniEnv; } |
| 411 | INLINE void dvmSetThreadJNIEnv(Thread* self, JNIEnv* env) { self->jniEnv = env;} |
| 412 | |
| 413 | /* |
San Mehat | 256fc15 | 2009-04-21 14:03:06 -0700 | [diff] [blame] | 414 | * Change the scheduler group of the current process |
| 415 | */ |
| 416 | int dvmChangeThreadSchedulerGroup(const char *group); |
| 417 | |
| 418 | /* |
The Android Open Source Project | f6c3871 | 2009-03-03 19:28:47 -0800 | [diff] [blame] | 419 | * Update the priority value of the underlying pthread. |
| 420 | */ |
| 421 | void dvmChangeThreadPriority(Thread* thread, int newPriority); |
| 422 | |
| 423 | |
| 424 | /* |
| 425 | * Debug: dump information about a single thread. |
| 426 | */ |
| 427 | void dvmDumpThread(Thread* thread, bool isRunning); |
| 428 | void dvmDumpThreadEx(const DebugOutputTarget* target, Thread* thread, |
| 429 | bool isRunning); |
| 430 | |
| 431 | /* |
| 432 | * Debug: dump information about all threads. |
| 433 | */ |
| 434 | void dvmDumpAllThreads(bool grabLock); |
| 435 | void dvmDumpAllThreadsEx(const DebugOutputTarget* target, bool grabLock); |
| 436 | |
| 437 | |
| 438 | #ifdef WITH_MONITOR_TRACKING |
| 439 | /* |
| 440 | * Track locks held by the current thread, along with the stack trace at |
| 441 | * the point the lock was acquired. |
| 442 | * |
| 443 | * At any given time the number of locks held across the VM should be |
| 444 | * fairly small, so there's no reason not to generate and store the entire |
| 445 | * stack trace. |
| 446 | */ |
| 447 | typedef struct LockedObjectData { |
| 448 | /* the locked object */ |
| 449 | struct Object* obj; |
| 450 | |
| 451 | /* number of times it has been locked recursively (zero-based ref count) */ |
| 452 | int recursionCount; |
| 453 | |
| 454 | /* stack trace at point of initial acquire */ |
| 455 | u4 stackDepth; |
| 456 | int* rawStackTrace; |
| 457 | |
| 458 | struct LockedObjectData* next; |
| 459 | } LockedObjectData; |
| 460 | |
| 461 | /* |
| 462 | * Add/remove/find objects from the thread's monitor list. |
| 463 | */ |
| 464 | void dvmAddToMonitorList(Thread* self, Object* obj, bool withTrace); |
| 465 | void dvmRemoveFromMonitorList(Thread* self, Object* obj); |
| 466 | LockedObjectData* dvmFindInMonitorList(const Thread* self, const Object* obj); |
| 467 | #endif |
| 468 | |
| 469 | #endif /*_DALVIK_THREAD*/ |