/* * Copyright (C) 2012 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 "callee_save_frame.h" #include "jit/jit.h" #include "runtime.h" #include "thread-inl.h" namespace art HIDDEN { extern "C" void artDeoptimizeIfNeeded(Thread* self, uintptr_t result, bool is_ref) REQUIRES_SHARED(Locks::mutator_lock_) { instrumentation::Instrumentation* instr = Runtime::Current()->GetInstrumentation(); DCHECK(!self->IsExceptionPending()); ArtMethod** sp = self->GetManagedStack()->GetTopQuickFrame(); DCHECK(sp != nullptr && (*sp)->IsRuntimeMethod()); DeoptimizationMethodType type = instr->GetDeoptimizationMethodType(*sp); JValue jvalue; jvalue.SetJ(result); instr->DeoptimizeIfNeeded(self, sp, type, jvalue, is_ref); } extern "C" void artTestSuspendFromCode(Thread* self) REQUIRES_SHARED(Locks::mutator_lock_) { // Called when there is a pending checkpoint or suspend request. ScopedQuickEntrypointChecks sqec(self); self->CheckSuspend(); // We could have other dex instructions at the same dex pc as suspend and we need to execute // those instructions. So we should start executing from the current dex pc. ArtMethod** sp = self->GetManagedStack()->GetTopQuickFrame(); JValue result; result.SetJ(0); Runtime::Current()->GetInstrumentation()->DeoptimizeIfNeeded( self, sp, DeoptimizationMethodType::kKeepDexPc, result, /* is_ref= */ false); } extern "C" void artImplicitSuspendFromCode(Thread* self) REQUIRES_SHARED(Locks::mutator_lock_) { // Called when there is a pending checkpoint or suspend request. ScopedQuickEntrypointChecks sqec(self); self->CheckSuspend(/*implicit=*/ true); // We could have other dex instructions at the same dex pc as suspend and we need to execute // those instructions. So we should start executing from the current dex pc. ArtMethod** sp = self->GetManagedStack()->GetTopQuickFrame(); JValue result; result.SetJ(0); Runtime::Current()->GetInstrumentation()->DeoptimizeIfNeeded( self, sp, DeoptimizationMethodType::kKeepDexPc, result, /* is_ref= */ false); } extern "C" void artCompileOptimized(ArtMethod* method, Thread* self) REQUIRES_SHARED(Locks::mutator_lock_) { ScopedQuickEntrypointChecks sqec(self); // It is important this method is not suspended due to: // * It is called on entry, and object parameters are in locations that are // not marked in the stack map. // * Async deoptimization does not expect runtime methods other than the // suspend entrypoint before executing the first instruction of a Java // method. ScopedAssertNoThreadSuspension sants("Enqueuing optimized compilation"); Runtime::Current()->GetJit()->EnqueueOptimizedCompilation(method, self); } } // namespace art