blob: 57b194362ff95e1c4e2f69fbbbb7f1afaf98890e [file] [log] [blame]
Andreas Gampeaf13ab92017-01-11 20:57:40 -08001/* Copyright (C) 2017 The Android Open Source Project
2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
3 *
4 * This file implements interfaces from the file jvmti.h. This implementation
5 * is licensed under the same terms as the file jvmti.h. The
6 * copyright and license information for the file jvmti.h follows.
7 *
8 * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved.
9 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
10 *
11 * This code is free software; you can redistribute it and/or modify it
12 * under the terms of the GNU General Public License version 2 only, as
13 * published by the Free Software Foundation. Oracle designates this
14 * particular file as subject to the "Classpath" exception as provided
15 * by Oracle in the LICENSE file that accompanied this code.
16 *
17 * This code is distributed in the hope that it will be useful, but WITHOUT
18 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
20 * version 2 for more details (a copy is included in the LICENSE file that
21 * accompanied this code).
22 *
23 * You should have received a copy of the GNU General Public License version
24 * 2 along with this work; if not, write to the Free Software Foundation,
25 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
26 *
27 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
28 * or visit www.oracle.com if you need additional information or have any
29 * questions.
30 */
31
Andreas Gampe06c42a52017-07-26 14:17:14 -070032#ifndef ART_OPENJDKJVMTI_TI_THREAD_H_
33#define ART_OPENJDKJVMTI_TI_THREAD_H_
Andreas Gampeaf13ab92017-01-11 20:57:40 -080034
35#include "jni.h"
36#include "jvmti.h"
37
Alex Lightbebd7bd2017-07-25 14:05:52 -070038#include "base/macros.h"
Alex Light88fd7202017-06-30 08:31:59 -070039#include "base/mutex.h"
40
Andreas Gampedb6c2ab2017-03-28 17:28:32 -070041namespace art {
42class ArtField;
Alex Lightbebd7bd2017-07-25 14:05:52 -070043class ScopedObjectAccessAlreadyRunnable;
Alex Light88fd7202017-06-30 08:31:59 -070044class Thread;
Andreas Gampedeae7db2017-05-30 09:56:41 -070045} // namespace art
Andreas Gampedb6c2ab2017-03-28 17:28:32 -070046
Andreas Gampeaf13ab92017-01-11 20:57:40 -080047namespace openjdkjvmti {
48
Andreas Gampeeafaf572017-01-20 12:34:15 -080049class EventHandler;
50
Andreas Gampeaf13ab92017-01-11 20:57:40 -080051class ThreadUtil {
52 public:
Andreas Gampeeafaf572017-01-20 12:34:15 -080053 static void Register(EventHandler* event_handler);
54 static void Unregister();
55
Alex Light1d8a9742017-08-17 11:12:06 -070056 // To be called when it is safe to cache data. This means that we have at least entered the
57 // RuntimePhase::kInit but we might or might not have already called VMInit event.
Andreas Gampedb6c2ab2017-03-28 17:28:32 -070058 static void CacheData();
59
Alex Light1d8a9742017-08-17 11:12:06 -070060 // Called just after we have sent the VMInit callback so that ThreadUtil can do final setup. This
61 // ensures that there are no timing issues between the two callbacks.
62 static void VMInitEventSent() REQUIRES_SHARED(art::Locks::mutator_lock_);
63
Alex Light092a4042017-07-12 08:46:44 -070064 // Handle a jvmtiEnv going away.
65 static void RemoveEnvironment(jvmtiEnv* env);
66
Andreas Gampe85807442017-01-13 14:40:58 -080067 static jvmtiError GetAllThreads(jvmtiEnv* env, jint* threads_count_ptr, jthread** threads_ptr);
68
Andreas Gampeaf13ab92017-01-11 20:57:40 -080069 static jvmtiError GetCurrentThread(jvmtiEnv* env, jthread* thread_ptr);
70
71 static jvmtiError GetThreadInfo(jvmtiEnv* env, jthread thread, jvmtiThreadInfo* info_ptr);
Andreas Gampe72c19832017-01-12 13:22:16 -080072
73 static jvmtiError GetThreadState(jvmtiEnv* env, jthread thread, jint* thread_state_ptr);
Andreas Gampef26bf2d2017-01-13 16:47:14 -080074
75 static jvmtiError SetThreadLocalStorage(jvmtiEnv* env, jthread thread, const void* data);
76 static jvmtiError GetThreadLocalStorage(jvmtiEnv* env, jthread thread, void** data_ptr);
Andreas Gampe732b0ac2017-01-18 15:23:39 -080077
78 static jvmtiError RunAgentThread(jvmtiEnv* env,
79 jthread thread,
80 jvmtiStartFunction proc,
81 const void* arg,
82 jint priority);
Andreas Gampedb6c2ab2017-03-28 17:28:32 -070083
Alex Light88fd7202017-06-30 08:31:59 -070084 static jvmtiError SuspendThread(jvmtiEnv* env, jthread thread);
85 static jvmtiError ResumeThread(jvmtiEnv* env, jthread thread);
86
87 static jvmtiError SuspendThreadList(jvmtiEnv* env,
88 jint request_count,
89 const jthread* threads,
90 jvmtiError* results);
91 static jvmtiError ResumeThreadList(jvmtiEnv* env,
92 jint request_count,
93 const jthread* threads,
94 jvmtiError* results);
95
Alex Lightbebd7bd2017-07-25 14:05:52 -070096 static art::Thread* GetNativeThread(jthread thread,
97 const art::ScopedObjectAccessAlreadyRunnable& soa)
Alex Light3ae82532017-07-26 13:59:07 -070098 REQUIRES_SHARED(art::Locks::mutator_lock_)
99 REQUIRES(art::Locks::thread_list_lock_);
Alex Lightbebd7bd2017-07-25 14:05:52 -0700100
Alex Light23aa7482017-08-16 10:01:13 -0700101 // Go to sleep if this thread is suspended.
102 static void SuspendCheck(art::Thread* self)
103 REQUIRES(!art::Locks::mutator_lock_, !art::Locks::user_code_suspension_lock_);
104
105 // Returns true if the thread would be suspended if it locks the mutator-lock or calls
106 // SuspendCheck. This function is called with the user_code_suspension_lock already held.
107 static bool WouldSuspendForUserCodeLocked(art::Thread* self)
108 REQUIRES(art::Locks::user_code_suspension_lock_,
109 !art::Locks::thread_suspend_count_lock_);
110
111 // Returns true if this thread would go to sleep if it locks the mutator-lock or calls
112 // SuspendCheck.
113 static bool WouldSuspendForUserCode(art::Thread* self)
114 REQUIRES(!art::Locks::user_code_suspension_lock_,
115 !art::Locks::thread_suspend_count_lock_);
116
Andreas Gampedb6c2ab2017-03-28 17:28:32 -0700117 private:
Alex Light88fd7202017-06-30 08:31:59 -0700118 // We need to make sure only one thread tries to suspend threads at a time so we can get the
119 // 'suspend-only-once' behavior the spec requires. Internally, ART considers suspension to be a
120 // counted state, allowing a single thread to be suspended multiple times by different users. This
121 // makes mapping into the JVMTI idea of thread suspension difficult. We have decided to split the
122 // difference and ensure that JVMTI tries to treat suspension as the boolean flag as much as
123 // possible with the suspend/resume methods but only do best effort. On the other hand
124 // GetThreadState will be totally accurate as much as possible. This means that calling
125 // ResumeThread on a thread that has state JVMTI_THREAD_STATE_SUSPENDED will not necessarily
126 // cause the thread to wake up if the thread is suspended for the debugger or gc or something.
127 static jvmtiError SuspendSelf(art::Thread* self)
128 REQUIRES(!art::Locks::mutator_lock_, !art::Locks::user_code_suspension_lock_);
Alex Light3ae82532017-07-26 13:59:07 -0700129 static jvmtiError SuspendOther(art::Thread* self, jthread target_jthread)
Alex Light88fd7202017-06-30 08:31:59 -0700130 REQUIRES(!art::Locks::mutator_lock_, !art::Locks::user_code_suspension_lock_);
131
Andreas Gampedb6c2ab2017-03-28 17:28:32 -0700132 static art::ArtField* context_class_loader_;
Andreas Gampeaf13ab92017-01-11 20:57:40 -0800133};
134
135} // namespace openjdkjvmti
136
Andreas Gampe06c42a52017-07-26 14:17:14 -0700137#endif // ART_OPENJDKJVMTI_TI_THREAD_H_