| Ian Rogers | 57b86d4 | 2012-03-27 16:05:41 -0700 | [diff] [blame] | 1 | /* |
| Elliott Hughes | 0f3c553 | 2012-03-30 14:51:51 -0700 | [diff] [blame] | 2 | * Copyright (C) 2012 The Android Open Source Project |
| Ian Rogers | 57b86d4 | 2012-03-27 16:05:41 -0700 | [diff] [blame] | 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 | |
| Andreas Gampe | 5794381 | 2017-12-06 21:39:13 -0800 | [diff] [blame] | 17 | #include <android-base/logging.h> |
| 18 | |
| Mathieu Chartier | e401d14 | 2015-04-22 13:56:20 -0700 | [diff] [blame] | 19 | #include "art_method-inl.h" |
| Andreas Gampe | e03662b | 2016-10-13 17:12:56 -0700 | [diff] [blame] | 20 | #include "base/casts.h" |
| Mathieu Chartier | 7643327 | 2014-09-26 14:32:37 -0700 | [diff] [blame] | 21 | #include "entrypoints/entrypoint_utils-inl.h" |
| Andreas Gampe | e03662b | 2016-10-13 17:12:56 -0700 | [diff] [blame] | 22 | #include "indirect_reference_table.h" |
| Ian Rogers | 2dd0e2c | 2013-01-24 12:42:14 -0800 | [diff] [blame] | 23 | #include "mirror/object-inl.h" |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 24 | #include "palette/palette.h" |
| Ian Rogers | 7b078e8 | 2014-09-10 14:44:24 -0700 | [diff] [blame] | 25 | #include "thread-inl.h" |
| Andreas Gampe | 90b936d | 2017-01-31 08:58:55 -0800 | [diff] [blame] | 26 | #include "verify_object.h" |
| Ian Rogers | 57b86d4 | 2012-03-27 16:05:41 -0700 | [diff] [blame] | 27 | |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 28 | // For methods that monitor JNI invocations and report their begin/end to |
| 29 | // palette hooks. |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 30 | #define MONITOR_JNI(kind) \ |
| 31 | { \ |
| 32 | bool should_report = false; \ |
| 33 | PaletteShouldReportJniInvocations(&should_report); \ |
| 34 | if (should_report) { \ |
| 35 | kind(self->GetJniEnv()); \ |
| 36 | } \ |
| 37 | } |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 38 | |
| Ian Rogers | 57b86d4 | 2012-03-27 16:05:41 -0700 | [diff] [blame] | 39 | namespace art { |
| 40 | |
| Andreas Gampe | e03662b | 2016-10-13 17:12:56 -0700 | [diff] [blame] | 41 | static_assert(sizeof(IRTSegmentState) == sizeof(uint32_t), "IRTSegmentState size unexpected"); |
| 42 | static_assert(std::is_trivial<IRTSegmentState>::value, "IRTSegmentState not trivial"); |
| 43 | |
| Vladimir Marko | 2ca0900 | 2021-10-19 16:19:34 +0000 | [diff] [blame^] | 44 | static inline void GoToRunnableFast(Thread* self) REQUIRES_SHARED(Locks::mutator_lock_); |
| 45 | |
| Vladimir Marko | cedec9d | 2021-02-08 16:16:13 +0000 | [diff] [blame] | 46 | extern void ReadBarrierJni(mirror::CompressedReference<mirror::Class>* declaring_class, |
| Hiroshi Yamauchi | 1cc71eb | 2015-05-07 10:47:27 -0700 | [diff] [blame] | 47 | Thread* self ATTRIBUTE_UNUSED) { |
| Hiroshi Yamauchi | 043eb9a | 2016-10-14 11:21:38 -0700 | [diff] [blame] | 48 | DCHECK(kUseReadBarrier); |
| 49 | if (kUseBakerReadBarrier) { |
| Vladimir Marko | cedec9d | 2021-02-08 16:16:13 +0000 | [diff] [blame] | 50 | DCHECK(declaring_class->AsMirrorPtr() != nullptr) |
| Hiroshi Yamauchi | 043eb9a | 2016-10-14 11:21:38 -0700 | [diff] [blame] | 51 | << "The class of a static jni call must not be null"; |
| 52 | // Check the mark bit and return early if it's already marked. |
| Vladimir Marko | cedec9d | 2021-02-08 16:16:13 +0000 | [diff] [blame] | 53 | if (LIKELY(declaring_class->AsMirrorPtr()->GetMarkBit() != 0)) { |
| Hiroshi Yamauchi | 043eb9a | 2016-10-14 11:21:38 -0700 | [diff] [blame] | 54 | return; |
| 55 | } |
| 56 | } |
| Hiroshi Yamauchi | 1cc71eb | 2015-05-07 10:47:27 -0700 | [diff] [blame] | 57 | // Call the read barrier and update the handle. |
| Vladimir Marko | cedec9d | 2021-02-08 16:16:13 +0000 | [diff] [blame] | 58 | mirror::Class* to_ref = ReadBarrier::BarrierForRoot(declaring_class); |
| 59 | declaring_class->Assign(to_ref); |
| Hiroshi Yamauchi | 1cc71eb | 2015-05-07 10:47:27 -0700 | [diff] [blame] | 60 | } |
| 61 | |
| Vladimir Marko | 2ca0900 | 2021-10-19 16:19:34 +0000 | [diff] [blame^] | 62 | // Called on entry to fast JNI, push a new local reference table only. |
| 63 | extern void JniMethodFastStart(Thread* self) { |
| 64 | if (kIsDebugBuild) { |
| 65 | ArtMethod* native_method = *self->GetManagedStack()->GetTopQuickFrame(); |
| 66 | CHECK(native_method->IsFastNative()) << native_method->PrettyMethod(); |
| 67 | } |
| 68 | } |
| 69 | |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 70 | // Called on entry to JNI, transition out of Runnable and release share of mutator_lock_. |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 71 | extern void JniMethodStart(Thread* self) { |
| Vladimir Marko | 68e8a7c | 2021-03-05 08:40:22 +0000 | [diff] [blame] | 72 | if (kIsDebugBuild) { |
| 73 | ArtMethod* native_method = *self->GetManagedStack()->GetTopQuickFrame(); |
| 74 | CHECK(!native_method->IsFastNative()) << native_method->PrettyMethod(); |
| Ian Rogers | 1eb512d | 2013-10-18 15:42:20 -0700 | [diff] [blame] | 75 | } |
| Vladimir Marko | 68e8a7c | 2021-03-05 08:40:22 +0000 | [diff] [blame] | 76 | |
| 77 | // Transition out of runnable. |
| 78 | self->TransitionFromRunnableToSuspended(kNative); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 79 | } |
| Elliott Hughes | b264f08 | 2012-04-06 17:10:10 -0700 | [diff] [blame] | 80 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 81 | extern void JniMethodStartSynchronized(jobject to_lock, Thread* self) { |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 82 | self->DecodeJObject(to_lock)->MonitorEnter(self); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 83 | JniMethodStart(self); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 84 | } |
| 85 | |
| Ian Rogers | 1eb512d | 2013-10-18 15:42:20 -0700 | [diff] [blame] | 86 | // TODO: NO_THREAD_SAFETY_ANALYSIS due to different control paths depending on fast JNI. |
| 87 | static void GoToRunnable(Thread* self) NO_THREAD_SAFETY_ANALYSIS { |
| Vladimir Marko | 34ce1b8 | 2021-03-11 12:11:31 +0000 | [diff] [blame] | 88 | if (kIsDebugBuild) { |
| 89 | ArtMethod* native_method = *self->GetManagedStack()->GetTopQuickFrame(); |
| 90 | CHECK(!native_method->IsFastNative()) << native_method->PrettyMethod(); |
| Igor Murashkin | af1e299 | 2016-10-12 17:44:50 -0700 | [diff] [blame] | 91 | } |
| Vladimir Marko | 34ce1b8 | 2021-03-11 12:11:31 +0000 | [diff] [blame] | 92 | |
| 93 | self->TransitionFromSuspendedToRunnable(); |
| Igor Murashkin | af1e299 | 2016-10-12 17:44:50 -0700 | [diff] [blame] | 94 | } |
| 95 | |
| Vladimir Marko | 2ca0900 | 2021-10-19 16:19:34 +0000 | [diff] [blame^] | 96 | ALWAYS_INLINE static inline void GoToRunnableFast(Thread* self) { |
| 97 | if (kIsDebugBuild) { |
| 98 | // Should only enter here if the method is @FastNative. |
| 99 | ArtMethod* native_method = *self->GetManagedStack()->GetTopQuickFrame(); |
| 100 | CHECK(native_method->IsFastNative()) << native_method->PrettyMethod(); |
| 101 | } |
| 102 | |
| 103 | // When we are in @FastNative, we are already Runnable. |
| 104 | // Only do a suspend check on the way out of JNI. |
| 105 | if (UNLIKELY(self->TestAllFlags())) { |
| 106 | // In fast JNI mode we never transitioned out of runnable. Perform a suspend check if there |
| 107 | // is a flag raised. |
| 108 | DCHECK(Locks::mutator_lock_->IsSharedHeld(self)); |
| 109 | self->CheckSuspend(); |
| 110 | } |
| 111 | } |
| 112 | |
| Yevgeny Rouban | 35aef2c | 2014-05-19 16:19:36 +0700 | [diff] [blame] | 113 | static void PopLocalReferences(uint32_t saved_local_ref_cookie, Thread* self) |
| Andreas Gampe | bdf7f1c | 2016-08-30 16:38:47 -0700 | [diff] [blame] | 114 | REQUIRES_SHARED(Locks::mutator_lock_) { |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 115 | JNIEnvExt* env = self->GetJniEnv(); |
| Ian Rogers | 55256cb | 2017-12-21 17:07:11 -0800 | [diff] [blame] | 116 | if (UNLIKELY(env->IsCheckJniEnabled())) { |
| Andreas Gampe | 5f4a09a | 2015-09-28 13:16:33 -0700 | [diff] [blame] | 117 | env->CheckNoHeldMonitors(); |
| 118 | } |
| Ian Rogers | 55256cb | 2017-12-21 17:07:11 -0800 | [diff] [blame] | 119 | env->SetLocalSegmentState(env->GetLocalRefCookie()); |
| 120 | env->SetLocalRefCookie(bit_cast<IRTSegmentState>(saved_local_ref_cookie)); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 121 | } |
| 122 | |
| Vladimir Marko | 5c33d35 | 2020-02-10 12:29:13 +0000 | [diff] [blame] | 123 | // TODO: annotalysis disabled as monitor semantics are maintained in Java code. |
| 124 | static inline void UnlockJniSynchronizedMethod(jobject locked, Thread* self) |
| 125 | NO_THREAD_SAFETY_ANALYSIS REQUIRES(!Roles::uninterruptible_) { |
| 126 | // Save any pending exception over monitor exit call. |
| 127 | ObjPtr<mirror::Throwable> saved_exception = nullptr; |
| 128 | if (UNLIKELY(self->IsExceptionPending())) { |
| 129 | saved_exception = self->GetException(); |
| 130 | self->ClearException(); |
| 131 | } |
| 132 | // Decode locked object and unlock, before popping local references. |
| 133 | self->DecodeJObject(locked)->MonitorExit(self); |
| 134 | if (UNLIKELY(self->IsExceptionPending())) { |
| 135 | LOG(FATAL) << "Synchronized JNI code returning with an exception:\n" |
| 136 | << saved_exception->Dump() |
| 137 | << "\nEncountered second exception during implicit MonitorExit:\n" |
| 138 | << self->GetException()->Dump(); |
| 139 | } |
| 140 | // Restore pending exception. |
| 141 | if (saved_exception != nullptr) { |
| 142 | self->SetException(saved_exception); |
| 143 | } |
| 144 | } |
| 145 | |
| Igor Murashkin | 9d4b6da | 2016-07-29 09:51:58 -0700 | [diff] [blame] | 146 | // TODO: These should probably be templatized or macro-ized. |
| 147 | // Otherwise there's just too much repetitive boilerplate. |
| 148 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 149 | extern void JniMethodEnd(Thread* self) { |
| Ian Rogers | 1eb512d | 2013-10-18 15:42:20 -0700 | [diff] [blame] | 150 | GoToRunnable(self); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 151 | } |
| 152 | |
| Vladimir Marko | 2ca0900 | 2021-10-19 16:19:34 +0000 | [diff] [blame^] | 153 | extern void JniMethodFastEnd(Thread* self) { |
| 154 | GoToRunnableFast(self); |
| 155 | } |
| 156 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 157 | extern void JniMethodEndSynchronized(jobject locked, Thread* self) { |
| Ian Rogers | 1eb512d | 2013-10-18 15:42:20 -0700 | [diff] [blame] | 158 | GoToRunnable(self); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 159 | UnlockJniSynchronizedMethod(locked, self); // Must decode before pop. |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 160 | } |
| 161 | |
| Andreas Gampe | 48ee356 | 2015-04-10 19:57:29 -0700 | [diff] [blame] | 162 | // Common result handling for EndWithReference. |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 163 | static mirror::Object* JniMethodEndWithReferenceHandleResult(jobject result, Thread* self) |
| Andreas Gampe | 48ee356 | 2015-04-10 19:57:29 -0700 | [diff] [blame] | 164 | NO_THREAD_SAFETY_ANALYSIS { |
| 165 | // Must decode before pop. The 'result' may not be valid in case of an exception, though. |
| Mathieu Chartier | c4f3925 | 2016-10-05 18:32:08 -0700 | [diff] [blame] | 166 | ObjPtr<mirror::Object> o; |
| 167 | if (!self->IsExceptionPending()) { |
| 168 | o = self->DecodeJObject(result); |
| 169 | } |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 170 | // Process result. |
| Ian Rogers | 55256cb | 2017-12-21 17:07:11 -0800 | [diff] [blame] | 171 | if (UNLIKELY(self->GetJniEnv()->IsCheckJniEnabled())) { |
| Mathieu Chartier | be08cf5 | 2016-09-13 13:41:24 -0700 | [diff] [blame] | 172 | // CheckReferenceResult can resolve types. |
| 173 | StackHandleScope<1> hs(self); |
| Mathieu Chartier | c4f3925 | 2016-10-05 18:32:08 -0700 | [diff] [blame] | 174 | HandleWrapperObjPtr<mirror::Object> h_obj(hs.NewHandleWrapper(&o)); |
| Mathieu Chartier | be08cf5 | 2016-09-13 13:41:24 -0700 | [diff] [blame] | 175 | CheckReferenceResult(h_obj, self); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 176 | } |
| Mathieu Chartier | 9d156d5 | 2016-10-06 17:44:26 -0700 | [diff] [blame] | 177 | VerifyObject(o); |
| Mathieu Chartier | c4f3925 | 2016-10-05 18:32:08 -0700 | [diff] [blame] | 178 | return o.Ptr(); |
| Ian Rogers | 00f7d0e | 2012-07-19 15:28:27 -0700 | [diff] [blame] | 179 | } |
| 180 | |
| Vladimir Marko | 2ca0900 | 2021-10-19 16:19:34 +0000 | [diff] [blame^] | 181 | extern mirror::Object* JniMethodFastEndWithReference(jobject result, Thread* self) { |
| 182 | GoToRunnableFast(self); |
| 183 | return JniMethodEndWithReferenceHandleResult(result, self); |
| 184 | } |
| 185 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 186 | extern mirror::Object* JniMethodEndWithReference(jobject result, Thread* self) { |
| Andreas Gampe | 48ee356 | 2015-04-10 19:57:29 -0700 | [diff] [blame] | 187 | GoToRunnable(self); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 188 | return JniMethodEndWithReferenceHandleResult(result, self); |
| Andreas Gampe | 48ee356 | 2015-04-10 19:57:29 -0700 | [diff] [blame] | 189 | } |
| 190 | |
| Ian Rogers | 2dd0e2c | 2013-01-24 12:42:14 -0800 | [diff] [blame] | 191 | extern mirror::Object* JniMethodEndWithReferenceSynchronized(jobject result, |
| Mathieu Chartier | be08cf5 | 2016-09-13 13:41:24 -0700 | [diff] [blame] | 192 | jobject locked, |
| 193 | Thread* self) { |
| Ian Rogers | 1eb512d | 2013-10-18 15:42:20 -0700 | [diff] [blame] | 194 | GoToRunnable(self); |
| Andreas Gampe | 48ee356 | 2015-04-10 19:57:29 -0700 | [diff] [blame] | 195 | UnlockJniSynchronizedMethod(locked, self); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 196 | return JniMethodEndWithReferenceHandleResult(result, self); |
| Ian Rogers | 57b86d4 | 2012-03-27 16:05:41 -0700 | [diff] [blame] | 197 | } |
| 198 | |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 199 | extern uint64_t GenericJniMethodEnd(Thread* self, |
| 200 | uint32_t saved_local_ref_cookie, |
| 201 | jvalue result, |
| 202 | uint64_t result_f, |
| Vladimir Marko | 6e043bb | 2020-02-10 16:56:54 +0000 | [diff] [blame] | 203 | ArtMethod* called) |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 204 | // TODO: NO_THREAD_SAFETY_ANALYSIS as GoToRunnable() is NO_THREAD_SAFETY_ANALYSIS |
| 205 | NO_THREAD_SAFETY_ANALYSIS { |
| Vladimir Marko | b0a6aee | 2017-10-27 10:34:04 +0100 | [diff] [blame] | 206 | bool critical_native = called->IsCriticalNative(); |
| 207 | bool fast_native = called->IsFastNative(); |
| Igor Murashkin | 06a04e0 | 2016-09-13 15:57:37 -0700 | [diff] [blame] | 208 | bool normal_native = !critical_native && !fast_native; |
| 209 | |
| Vladimir Marko | 34ce1b8 | 2021-03-11 12:11:31 +0000 | [diff] [blame] | 210 | // @CriticalNative does not do a state transition. @FastNative usually does not do a state |
| 211 | // transition either but it performs a suspend check that may do state transitions. |
| Igor Murashkin | 06a04e0 | 2016-09-13 15:57:37 -0700 | [diff] [blame] | 212 | if (LIKELY(normal_native)) { |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 213 | MONITOR_JNI(PaletteNotifyEndJniInvocation); |
| Igor Murashkin | 06a04e0 | 2016-09-13 15:57:37 -0700 | [diff] [blame] | 214 | GoToRunnable(self); |
| Vladimir Marko | 34ce1b8 | 2021-03-11 12:11:31 +0000 | [diff] [blame] | 215 | } else if (fast_native) { |
| Vladimir Marko | 2ca0900 | 2021-10-19 16:19:34 +0000 | [diff] [blame^] | 216 | GoToRunnableFast(self); |
| Igor Murashkin | 06a04e0 | 2016-09-13 15:57:37 -0700 | [diff] [blame] | 217 | } |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 218 | // We need the mutator lock (i.e., calling GoToRunnable()) before accessing the shorty or the |
| 219 | // locked object. |
| Vladimir Marko | 5c33d35 | 2020-02-10 12:29:13 +0000 | [diff] [blame] | 220 | if (called->IsSynchronized()) { |
| 221 | DCHECK(normal_native) << "@FastNative/@CriticalNative and synchronize is not supported"; |
| Vladimir Marko | cedec9d | 2021-02-08 16:16:13 +0000 | [diff] [blame] | 222 | jobject lock = GetGenericJniSynchronizationObject(self, called); |
| Vladimir Marko | 5c33d35 | 2020-02-10 12:29:13 +0000 | [diff] [blame] | 223 | DCHECK(lock != nullptr); |
| 224 | UnlockJniSynchronizedMethod(lock, self); |
| 225 | } |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 226 | char return_shorty_char = called->GetShorty()[0]; |
| 227 | if (return_shorty_char == 'L') { |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 228 | uint64_t ret = |
| 229 | reinterpret_cast<uint64_t>(JniMethodEndWithReferenceHandleResult(result.l, self)); |
| 230 | PopLocalReferences(saved_local_ref_cookie, self); |
| 231 | return ret; |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 232 | } else { |
| Igor Murashkin | 06a04e0 | 2016-09-13 15:57:37 -0700 | [diff] [blame] | 233 | if (LIKELY(!critical_native)) { |
| Vladimir Marko | cedec9d | 2021-02-08 16:16:13 +0000 | [diff] [blame] | 234 | PopLocalReferences(saved_local_ref_cookie, self); |
| Igor Murashkin | 06a04e0 | 2016-09-13 15:57:37 -0700 | [diff] [blame] | 235 | } |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 236 | switch (return_shorty_char) { |
| 237 | case 'F': { |
| Vladimir Marko | 33bff25 | 2017-11-01 14:35:42 +0000 | [diff] [blame] | 238 | if (kRuntimeISA == InstructionSet::kX86) { |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 239 | // Convert back the result to float. |
| 240 | double d = bit_cast<double, uint64_t>(result_f); |
| 241 | return bit_cast<uint32_t, float>(static_cast<float>(d)); |
| 242 | } else { |
| 243 | return result_f; |
| 244 | } |
| 245 | } |
| 246 | case 'D': |
| 247 | return result_f; |
| 248 | case 'Z': |
| 249 | return result.z; |
| 250 | case 'B': |
| 251 | return result.b; |
| 252 | case 'C': |
| 253 | return result.c; |
| 254 | case 'S': |
| 255 | return result.s; |
| 256 | case 'I': |
| 257 | return result.i; |
| 258 | case 'J': |
| 259 | return result.j; |
| 260 | case 'V': |
| 261 | return 0; |
| 262 | default: |
| 263 | LOG(FATAL) << "Unexpected return shorty character " << return_shorty_char; |
| Elliott Hughes | c1896c9 | 2018-11-29 11:33:18 -0800 | [diff] [blame] | 264 | UNREACHABLE(); |
| Hiroshi Yamauchi | a23b468 | 2015-09-28 17:47:32 -0700 | [diff] [blame] | 265 | } |
| 266 | } |
| 267 | } |
| 268 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 269 | extern void JniMonitoredMethodStart(Thread* self) { |
| 270 | JniMethodStart(self); |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 271 | MONITOR_JNI(PaletteNotifyBeginJniInvocation); |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 272 | } |
| 273 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 274 | extern void JniMonitoredMethodStartSynchronized(jobject to_lock, Thread* self) { |
| 275 | JniMethodStartSynchronized(to_lock, self); |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 276 | MONITOR_JNI(PaletteNotifyBeginJniInvocation); |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 277 | } |
| 278 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 279 | extern void JniMonitoredMethodEnd(Thread* self) { |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 280 | MONITOR_JNI(PaletteNotifyEndJniInvocation); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 281 | JniMethodEnd(self); |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 282 | } |
| 283 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 284 | extern void JniMonitoredMethodEndSynchronized(jobject locked, Thread* self) { |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 285 | MONITOR_JNI(PaletteNotifyEndJniInvocation); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 286 | JniMethodEndSynchronized(locked, self); |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 287 | } |
| 288 | |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 289 | extern mirror::Object* JniMonitoredMethodEndWithReference(jobject result, Thread* self) { |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 290 | MONITOR_JNI(PaletteNotifyEndJniInvocation); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 291 | return JniMethodEndWithReference(result, self); |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 292 | } |
| 293 | |
| 294 | extern mirror::Object* JniMonitoredMethodEndWithReferenceSynchronized( |
| 295 | jobject result, |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 296 | jobject locked, |
| 297 | Thread* self) { |
| Nicolas Geoffray | 4f6bb44 | 2021-06-02 18:05:51 +0100 | [diff] [blame] | 298 | MONITOR_JNI(PaletteNotifyEndJniInvocation); |
| Vladimir Marko | d95a1f2 | 2021-03-23 16:32:52 +0000 | [diff] [blame] | 299 | return JniMethodEndWithReferenceSynchronized(result, locked, self); |
| Nicolas Geoffray | c3db254 | 2021-04-20 15:50:04 +0000 | [diff] [blame] | 300 | } |
| 301 | |
| Ian Rogers | 57b86d4 | 2012-03-27 16:05:41 -0700 | [diff] [blame] | 302 | } // namespace art |