blob: ae3eaf2e228bbc13d22c7606a9d39ccf043e10b0 [file] [log] [blame]
Ian Rogers00f7d0e2012-07-19 15:28:27 -07001/*
2 * Copyright (C) 2012 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
Brian Carlstromfc0e3212013-07-17 14:40:12 -070017#ifndef ART_RUNTIME_SCOPED_THREAD_STATE_CHANGE_H_
18#define ART_RUNTIME_SCOPED_THREAD_STATE_CHANGE_H_
Ian Rogers00f7d0e2012-07-19 15:28:27 -070019
Elliott Hughes1aa246d2012-12-13 09:29:36 -080020#include "base/casts.h"
Ian Rogers68d8b422014-07-17 11:09:10 -070021#include "java_vm_ext.h"
22#include "jni_env_ext-inl.h"
Hiroshi Yamauchi94f7b492014-07-22 18:08:23 -070023#include "read_barrier.h"
Ian Rogers693ff612013-02-01 10:56:12 -080024#include "thread-inl.h"
Mathieu Chartier4e305412014-02-19 10:54:44 -080025#include "verify_object.h"
Ian Rogers00f7d0e2012-07-19 15:28:27 -070026
27namespace art {
28
29// Scoped change into and out of a particular state. Handles Runnable transitions that require
30// more complicated suspension checking. The subclasses ScopedObjectAccessUnchecked and
Mathieu Chartiereb8167a2014-05-07 15:43:14 -070031// ScopedObjectAccess are used to handle the change into Runnable to Get direct access to objects,
Ian Rogers00f7d0e2012-07-19 15:28:27 -070032// the unchecked variant doesn't aid annotalysis.
33class ScopedThreadStateChange {
34 public:
35 ScopedThreadStateChange(Thread* self, ThreadState new_thread_state)
Ian Rogers1ffa32f2013-02-05 18:29:08 -080036 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_) ALWAYS_INLINE
Ian Rogers00f7d0e2012-07-19 15:28:27 -070037 : self_(self), thread_state_(new_thread_state), expected_has_no_thread_(false) {
Ian Rogersc0fa3ad2013-02-05 00:11:55 -080038 if (UNLIKELY(self_ == NULL)) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -070039 // Value chosen arbitrarily and won't be used in the destructor since thread_ == NULL.
40 old_thread_state_ = kTerminated;
Ian Rogers120f1c72012-09-28 17:17:10 -070041 Runtime* runtime = Runtime::Current();
Mathieu Chartier590fee92013-09-13 13:46:47 -070042 CHECK(runtime == NULL || !runtime->IsStarted() || runtime->IsShuttingDown(self_));
Ian Rogers00f7d0e2012-07-19 15:28:27 -070043 } else {
Ian Rogers22f454c2012-09-08 11:06:29 -070044 DCHECK_EQ(self, Thread::Current());
45 // Read state without locks, ok as state is effectively thread local and we're not interested
46 // in the suspend count (this will be handled in the runnable transitions).
Ian Rogers474b6da2012-09-25 00:20:38 -070047 old_thread_state_ = self->GetState();
Mathieu Chartier92b78892014-04-24 16:14:43 -070048 if (old_thread_state_ != new_thread_state) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -070049 if (new_thread_state == kRunnable) {
50 self_->TransitionFromSuspendedToRunnable();
Mathieu Chartier92b78892014-04-24 16:14:43 -070051 } else if (old_thread_state_ == kRunnable) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -070052 self_->TransitionFromRunnableToSuspended(new_thread_state);
Mathieu Chartier92b78892014-04-24 16:14:43 -070053 } else {
54 // A suspended transition to another effectively suspended transition, ok to use Unsafe.
55 self_->SetState(new_thread_state);
Ian Rogers00f7d0e2012-07-19 15:28:27 -070056 }
57 }
58 }
59 }
60
Ian Rogers1ffa32f2013-02-05 18:29:08 -080061 ~ScopedThreadStateChange() LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_) ALWAYS_INLINE {
Ian Rogersc0fa3ad2013-02-05 00:11:55 -080062 if (UNLIKELY(self_ == NULL)) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -070063 if (!expected_has_no_thread_) {
Ian Rogers120f1c72012-09-28 17:17:10 -070064 Runtime* runtime = Runtime::Current();
Mathieu Chartier590fee92013-09-13 13:46:47 -070065 bool shutting_down = (runtime == NULL) || runtime->IsShuttingDown(nullptr);
Ian Rogers120f1c72012-09-28 17:17:10 -070066 CHECK(shutting_down);
Ian Rogers00f7d0e2012-07-19 15:28:27 -070067 }
68 } else {
69 if (old_thread_state_ != thread_state_) {
70 if (old_thread_state_ == kRunnable) {
71 self_->TransitionFromSuspendedToRunnable();
72 } else if (thread_state_ == kRunnable) {
73 self_->TransitionFromRunnableToSuspended(old_thread_state_);
74 } else {
Ian Rogers22f454c2012-09-08 11:06:29 -070075 // A suspended transition to another effectively suspended transition, ok to use Unsafe.
Ian Rogers474b6da2012-09-25 00:20:38 -070076 self_->SetState(old_thread_state_);
Ian Rogers00f7d0e2012-07-19 15:28:27 -070077 }
78 }
79 }
80 }
81
82 Thread* Self() const {
83 return self_;
84 }
85
86 protected:
87 // Constructor used by ScopedJniThreadState for an unattached thread that has access to the VM*.
88 ScopedThreadStateChange()
89 : self_(NULL), thread_state_(kTerminated), old_thread_state_(kTerminated),
90 expected_has_no_thread_(true) {}
91
92 Thread* const self_;
93 const ThreadState thread_state_;
94
95 private:
96 ThreadState old_thread_state_;
97 const bool expected_has_no_thread_;
98
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -070099 friend class ScopedObjectAccessUnchecked;
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700100 DISALLOW_COPY_AND_ASSIGN(ScopedThreadStateChange);
101};
102
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700103// Assumes we are already runnable.
104class ScopedObjectAccessAlreadyRunnable {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700105 public:
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700106 Thread* Self() const {
107 return self_;
Ian Rogersc0fa3ad2013-02-05 00:11:55 -0800108 }
109
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700110 JNIEnvExt* Env() const {
111 return env_;
112 }
113
114 JavaVMExt* Vm() const {
115 return vm_;
116 }
117
Ian Rogers68d8b422014-07-17 11:09:10 -0700118 bool ForceCopy() const {
119 return vm_->ForceCopy();
120 }
121
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700122 /*
123 * Add a local reference for an object to the indirect reference table associated with the
124 * current stack frame. When the native function returns, the reference will be discarded.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700125 *
Elliott Hughes9dcd45c2013-07-29 14:40:52 -0700126 * We need to allow the same reference to be added multiple times, and cope with NULL.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700127 *
Elliott Hughes9dcd45c2013-07-29 14:40:52 -0700128 * This will be called on otherwise unreferenced objects. We cannot do GC allocations here, and
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700129 * it's best if we don't grab a mutex.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700130 */
131 template<typename T>
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800132 T AddLocalReference(mirror::Object* obj) const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700133 Locks::mutator_lock_->AssertSharedHeld(Self());
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700134 DCHECK(IsRunnable()); // Don't work with raw objects in non-runnable states.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700135 if (obj == NULL) {
136 return NULL;
137 }
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700138 DCHECK_NE((reinterpret_cast<uintptr_t>(obj) & 0xffff0000), 0xebad0000);
Ian Rogers987560f2014-04-22 11:42:59 -0700139 return Env()->AddLocalReference<T>(obj);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700140 }
141
142 template<typename T>
143 T Decode(jobject obj) const
Ian Rogersb726dcb2012-09-05 08:57:23 -0700144 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers81d425b2012-09-27 16:03:43 -0700145 Locks::mutator_lock_->AssertSharedHeld(Self());
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700146 DCHECK(IsRunnable()); // Don't work with raw objects in non-runnable states.
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700147 return down_cast<T>(Self()->DecodeJObject(obj));
148 }
149
Brian Carlstromea46f952013-07-30 01:26:50 -0700150 mirror::ArtField* DecodeField(jfieldID fid) const
Ian Rogersb726dcb2012-09-05 08:57:23 -0700151 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers81d425b2012-09-27 16:03:43 -0700152 Locks::mutator_lock_->AssertSharedHeld(Self());
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700153 DCHECK(IsRunnable()); // Don't work with raw objects in non-runnable states.
Mathieu Chartier590fee92013-09-13 13:46:47 -0700154 CHECK(!kMovingFields);
Hiroshi Yamauchiea2e1bd2014-06-18 13:47:35 -0700155 mirror::ArtField* field = reinterpret_cast<mirror::ArtField*>(fid);
156 return ReadBarrier::BarrierForRoot<mirror::ArtField, kWithReadBarrier>(&field);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700157 }
158
Brian Carlstromea46f952013-07-30 01:26:50 -0700159 jfieldID EncodeField(mirror::ArtField* field) const
Ian Rogersb726dcb2012-09-05 08:57:23 -0700160 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers81d425b2012-09-27 16:03:43 -0700161 Locks::mutator_lock_->AssertSharedHeld(Self());
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700162 DCHECK(IsRunnable()); // Don't work with raw objects in non-runnable states.
Mathieu Chartier590fee92013-09-13 13:46:47 -0700163 CHECK(!kMovingFields);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700164 return reinterpret_cast<jfieldID>(field);
165 }
166
Brian Carlstromea46f952013-07-30 01:26:50 -0700167 mirror::ArtMethod* DecodeMethod(jmethodID mid) const
Ian Rogersb726dcb2012-09-05 08:57:23 -0700168 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers81d425b2012-09-27 16:03:43 -0700169 Locks::mutator_lock_->AssertSharedHeld(Self());
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700170 DCHECK(IsRunnable()); // Don't work with raw objects in non-runnable states.
Mathieu Chartier590fee92013-09-13 13:46:47 -0700171 CHECK(!kMovingMethods);
Hiroshi Yamauchiea2e1bd2014-06-18 13:47:35 -0700172 mirror::ArtMethod* method = reinterpret_cast<mirror::ArtMethod*>(mid);
173 return ReadBarrier::BarrierForRoot<mirror::ArtMethod, kWithReadBarrier>(&method);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700174 }
175
Brian Carlstromea46f952013-07-30 01:26:50 -0700176 jmethodID EncodeMethod(mirror::ArtMethod* method) const
Ian Rogersb726dcb2012-09-05 08:57:23 -0700177 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers81d425b2012-09-27 16:03:43 -0700178 Locks::mutator_lock_->AssertSharedHeld(Self());
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700179 DCHECK(IsRunnable()); // Don't work with raw objects in non-runnable states.
Mathieu Chartier590fee92013-09-13 13:46:47 -0700180 CHECK(!kMovingMethods);
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700181 return reinterpret_cast<jmethodID>(method);
182 }
183
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700184 bool IsRunnable() const {
185 return self_->GetState() == kRunnable;
186 }
187
188 protected:
189 explicit ScopedObjectAccessAlreadyRunnable(JNIEnv* env)
190 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_) ALWAYS_INLINE
191 : self_(ThreadForEnv(env)), env_(down_cast<JNIEnvExt*>(env)), vm_(env_->vm) {
192 }
193
194 explicit ScopedObjectAccessAlreadyRunnable(Thread* self)
195 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_) ALWAYS_INLINE
196 : self_(self), env_(down_cast<JNIEnvExt*>(self->GetJniEnv())),
197 vm_(env_ != nullptr ? env_->vm : nullptr) {
198 }
199
200 // Used when we want a scoped JNI thread state but have no thread/JNIEnv. Consequently doesn't
201 // change into Runnable or acquire a share on the mutator_lock_.
202 explicit ScopedObjectAccessAlreadyRunnable(JavaVM* vm)
203 : self_(nullptr), env_(nullptr), vm_(down_cast<JavaVMExt*>(vm)) {}
204
205 // Here purely to force inlining.
206 ~ScopedObjectAccessAlreadyRunnable() ALWAYS_INLINE {
207 }
208
209 // Self thread, can be null.
210 Thread* const self_;
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700211 // The full JNIEnv.
212 JNIEnvExt* const env_;
213 // The full JavaVM.
214 JavaVMExt* const vm_;
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700215};
216
217// Entry/exit processing for transitions from Native to Runnable (ie within JNI functions).
218//
219// This class performs the necessary thread state switching to and from Runnable and lets us
220// amortize the cost of working out the current thread. Additionally it lets us check (and repair)
221// apps that are using a JNIEnv on the wrong thread. The class also decodes and encodes Objects
222// into jobjects via methods of this class. Performing this here enforces the Runnable thread state
223// for use of Object, thereby inhibiting the Object being modified by GC whilst native or VM code
224// is also manipulating the Object.
225//
226// The destructor transitions back to the previous thread state, typically Native. In this state
227// GC and thread suspension may occur.
228//
229// For annotalysis the subclass ScopedObjectAccess (below) makes it explicit that a shared of
230// the mutator_lock_ will be acquired on construction.
231class ScopedObjectAccessUnchecked : public ScopedObjectAccessAlreadyRunnable {
232 public:
233 explicit ScopedObjectAccessUnchecked(JNIEnv* env)
234 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_) ALWAYS_INLINE
235 : ScopedObjectAccessAlreadyRunnable(env), tsc_(Self(), kRunnable) {
236 Self()->VerifyStack();
237 Locks::mutator_lock_->AssertSharedHeld(Self());
238 }
239
240 explicit ScopedObjectAccessUnchecked(Thread* self)
241 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_) ALWAYS_INLINE
242 : ScopedObjectAccessAlreadyRunnable(self), tsc_(self, kRunnable) {
243 Self()->VerifyStack();
244 Locks::mutator_lock_->AssertSharedHeld(Self());
245 }
246
247 // Used when we want a scoped JNI thread state but have no thread/JNIEnv. Consequently doesn't
248 // change into Runnable or acquire a share on the mutator_lock_.
249 explicit ScopedObjectAccessUnchecked(JavaVM* vm) ALWAYS_INLINE
250 : ScopedObjectAccessAlreadyRunnable(vm), tsc_() {}
251
252 private:
253 // The scoped thread state change makes sure that we are runnable and restores the thread state
254 // in the destructor.
255 const ScopedThreadStateChange tsc_;
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700256
257 DISALLOW_COPY_AND_ASSIGN(ScopedObjectAccessUnchecked);
258};
259
260// Annotalysis helping variant of the above.
261class ScopedObjectAccess : public ScopedObjectAccessUnchecked {
262 public:
263 explicit ScopedObjectAccess(JNIEnv* env)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700264 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_)
Ian Rogers1ffa32f2013-02-05 18:29:08 -0800265 SHARED_LOCK_FUNCTION(Locks::mutator_lock_) ALWAYS_INLINE
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700266 : ScopedObjectAccessUnchecked(env) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700267 }
268
269 explicit ScopedObjectAccess(Thread* self)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700270 LOCKS_EXCLUDED(Locks::thread_suspend_count_lock_)
Mathieu Chartier2b7c4d12014-05-19 10:52:16 -0700271 SHARED_LOCK_FUNCTION(Locks::mutator_lock_) ALWAYS_INLINE
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700272 : ScopedObjectAccessUnchecked(self) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700273 }
274
Ian Rogers1ffa32f2013-02-05 18:29:08 -0800275 ~ScopedObjectAccess() UNLOCK_FUNCTION(Locks::mutator_lock_) ALWAYS_INLINE {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700276 // Base class will release share of lock. Invoked after this destructor.
277 }
278
279 private:
280 // TODO: remove this constructor. It is used by check JNI's ScopedCheck to make it believe that
281 // routines operating with just a VM are sound, they are not, but when you have just a VM
282 // you cannot call the unsound routines.
283 explicit ScopedObjectAccess(JavaVM* vm)
Ian Rogersb726dcb2012-09-05 08:57:23 -0700284 SHARED_LOCK_FUNCTION(Locks::mutator_lock_)
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700285 : ScopedObjectAccessUnchecked(vm) {}
286
287 friend class ScopedCheck;
288 DISALLOW_COPY_AND_ASSIGN(ScopedObjectAccess);
289};
290
291} // namespace art
292
Brian Carlstromfc0e3212013-07-17 14:40:12 -0700293#endif // ART_RUNTIME_SCOPED_THREAD_STATE_CHANGE_H_