blob: 53e14d0d7a59e1b8f4a825bffaacae7b74fa5739 [file] [log] [blame]
/*
* 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