blob: 568a23dfe42a1962d289876f668424777d414d02 [file] [log] [blame]
// Copyright 2014 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "src/execution/execution.h"
#include "src/api/api-inl.h"
#include "src/debug/debug.h"
#include "src/execution/frames.h"
#include "src/execution/isolate-inl.h"
#include "src/execution/vm-state-inl.h"
#include "src/logging/runtime-call-stats-scope.h"
#if V8_ENABLE_WEBASSEMBLY
#include "src/compiler/wasm-compiler.h" // Only for static asserts.
#include "src/wasm/code-space-access.h"
#include "src/wasm/wasm-engine.h"
#endif // V8_ENABLE_WEBASSEMBLY
namespace v8 {
namespace internal {
namespace {
DirectHandle<Object> NormalizeReceiver(Isolate* isolate,
DirectHandle<Object> receiver) {
// Convert calls on global objects to be calls on the global
// receiver instead to avoid having a 'this' pointer which refers
// directly to a global object.
if (IsJSGlobalObject(*receiver)) {
return direct_handle(Cast<JSGlobalObject>(receiver)->global_proxy(),
isolate);
}
return receiver;
}
struct InvokeParams {
static InvokeParams SetUpForNew(
Isolate* isolate, DirectHandle<Object> constructor,
DirectHandle<Object> new_target,
base::Vector<const DirectHandle<Object>> args);
static InvokeParams SetUpForCall(
Isolate* isolate, DirectHandle<Object> callable,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args);
static InvokeParams SetUpForTryCall(
Isolate* isolate, DirectHandle<Object> callable,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args,
Execution::MessageHandling message_handling,
MaybeDirectHandle<Object>* exception_out);
static InvokeParams SetUpForRunMicrotasks(Isolate* isolate,
MicrotaskQueue* microtask_queue);
bool IsScript() const {
if (!IsJSFunction(*target)) return false;
auto function = Cast<JSFunction>(target);
return function->shared()->is_script();
}
DirectHandle<FixedArray> GetAndResetHostDefinedOptions() {
DCHECK(IsScript());
DCHECK_EQ(args.size(), 1);
auto options = Cast<FixedArray>(args[0]);
args = {};
return options;
}
DirectHandle<Object> target;
DirectHandle<Object> receiver;
base::Vector<const DirectHandle<Object>> args;
DirectHandle<Object> new_target;
MicrotaskQueue* microtask_queue;
Execution::MessageHandling message_handling;
MaybeDirectHandle<Object>* exception_out;
bool is_construct;
Execution::Target execution_target;
};
// static
InvokeParams InvokeParams::SetUpForNew(
Isolate* isolate, DirectHandle<Object> constructor,
DirectHandle<Object> new_target,
base::Vector<const DirectHandle<Object>> args) {
InvokeParams params;
params.target = constructor;
params.receiver = isolate->factory()->undefined_value();
DCHECK(!params.IsScript());
params.args = args;
params.new_target = new_target;
params.microtask_queue = nullptr;
params.message_handling = Execution::MessageHandling::kReport;
params.exception_out = nullptr;
params.is_construct = true;
params.execution_target = Execution::Target::kCallable;
return params;
}
// static
InvokeParams InvokeParams::SetUpForCall(
Isolate* isolate, DirectHandle<Object> callable,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args) {
InvokeParams params;
params.target = callable;
params.receiver = NormalizeReceiver(isolate, receiver);
// Check for host-defined options argument for scripts.
DCHECK_IMPLIES(params.IsScript(), args.size() == 1);
DCHECK_IMPLIES(params.IsScript(), IsFixedArray(*args[0]));
params.args = args;
params.new_target = isolate->factory()->undefined_value();
params.microtask_queue = nullptr;
params.message_handling = Execution::MessageHandling::kReport;
params.exception_out = nullptr;
params.is_construct = false;
params.execution_target = Execution::Target::kCallable;
return params;
}
// static
InvokeParams InvokeParams::SetUpForTryCall(
Isolate* isolate, DirectHandle<Object> callable,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args,
Execution::MessageHandling message_handling,
MaybeDirectHandle<Object>* exception_out) {
InvokeParams params;
params.target = callable;
params.receiver = NormalizeReceiver(isolate, receiver);
// Check for host-defined options argument for scripts.
DCHECK_IMPLIES(params.IsScript(), args.size() == 1);
DCHECK_IMPLIES(params.IsScript(), IsFixedArray(*args[0]));
params.args = args;
params.new_target = isolate->factory()->undefined_value();
params.microtask_queue = nullptr;
params.message_handling = message_handling;
params.exception_out = exception_out;
params.is_construct = false;
params.execution_target = Execution::Target::kCallable;
return params;
}
// static
InvokeParams InvokeParams::SetUpForRunMicrotasks(
Isolate* isolate, MicrotaskQueue* microtask_queue) {
auto undefined = isolate->factory()->undefined_value();
InvokeParams params;
params.target = undefined;
params.receiver = undefined;
params.args = {};
params.new_target = undefined;
params.microtask_queue = microtask_queue;
params.message_handling = Execution::MessageHandling::kReport;
params.exception_out = nullptr;
params.is_construct = false;
params.execution_target = Execution::Target::kRunMicrotasks;
return params;
}
DirectHandle<Code> JSEntry(Isolate* isolate, Execution::Target execution_target,
bool is_construct) {
if (is_construct) {
DCHECK_EQ(Execution::Target::kCallable, execution_target);
return BUILTIN_CODE(isolate, JSConstructEntry);
} else if (execution_target == Execution::Target::kCallable) {
DCHECK(!is_construct);
return BUILTIN_CODE(isolate, JSEntry);
} else if (execution_target == Execution::Target::kRunMicrotasks) {
DCHECK(!is_construct);
return BUILTIN_CODE(isolate, JSRunMicrotasksEntry);
}
UNREACHABLE();
}
MaybeDirectHandle<Context> NewScriptContext(
Isolate* isolate, DirectHandle<JSFunction> function,
DirectHandle<FixedArray> host_defined_options) {
// TODO(cbruni, 1244145): Use passed in host_defined_options.
// Creating a script context is a side effect, so abort if that's not
// allowed.
if (isolate->should_check_side_effects()) {
isolate->Throw(*isolate->factory()->NewEvalError(
MessageTemplate::kNoSideEffectDebugEvaluate));
return MaybeDirectHandle<Context>();
}
SaveAndSwitchContext save(isolate, function->context());
Tagged<SharedFunctionInfo> sfi = function->shared();
Handle<Script> script(Cast<Script>(sfi->script()), isolate);
DirectHandle<ScopeInfo> scope_info(sfi->scope_info(), isolate);
DirectHandle<NativeContext> native_context(
Cast<NativeContext>(function->context()), isolate);
DirectHandle<JSGlobalObject> global_object(native_context->global_object(),
isolate);
Handle<ScriptContextTable> script_context(
native_context->script_context_table(), isolate);
// Find name clashes.
for (auto name_it : ScopeInfo::IterateLocalNames(scope_info)) {
Handle<String> name(name_it->name(), isolate);
VariableMode mode = scope_info->ContextLocalMode(name_it->index());
VariableLookupResult lookup;
if (script_context->Lookup(name, &lookup)) {
if (IsLexicalVariableMode(mode) || IsLexicalVariableMode(lookup.mode)) {
DirectHandle<Context> context(script_context->get(lookup.context_index),
isolate);
// If we are trying to redeclare a REPL-mode let as a let, REPL-mode
// const as a const, REPL-mode using as a using and REPL-mode await
// using as an await using allow it.
if (!((mode == lookup.mode && IsLexicalVariableMode(mode)) &&
scope_info->IsReplModeScope() &&
context->scope_info()->IsReplModeScope())) {
// ES#sec-globaldeclarationinstantiation 5.b:
// If envRec.HasLexicalDeclaration(name) is true, throw a SyntaxError
// exception.
MessageLocation location(script, 0, 1);
isolate->ThrowAt(isolate->factory()->NewSyntaxError(
MessageTemplate::kVarRedeclaration, name),
&location);
return MaybeDirectHandle<Context>();
}
}
}
if (IsLexicalVariableMode(mode)) {
LookupIterator lookup_it(isolate, global_object, name, global_object,
LookupIterator::OWN_SKIP_INTERCEPTOR);
Maybe<PropertyAttributes> maybe =
JSReceiver::GetPropertyAttributes(&lookup_it);
// Can't fail since the we looking up own properties on the global object
// skipping interceptors.
CHECK(!maybe.IsNothing());
if ((maybe.FromJust() & DONT_DELETE) != 0) {
// ES#sec-globaldeclarationinstantiation 5.a:
// If envRec.HasVarDeclaration(name) is true, throw a SyntaxError
// exception.
// ES#sec-globaldeclarationinstantiation 5.d:
// If hasRestrictedGlobal is true, throw a SyntaxError exception.
MessageLocation location(script, 0, 1);
isolate->ThrowAt(isolate->factory()->NewSyntaxError(
MessageTemplate::kVarRedeclaration, name),
&location);
return MaybeDirectHandle<Context>();
}
JSGlobalObject::InvalidatePropertyCell(global_object, name);
}
}
DirectHandle<Context> result =
isolate->factory()->NewScriptContext(native_context, scope_info);
result->Initialize(isolate);
// In REPL mode, we are allowed to add/modify let/const/using/await using
// variables. We use the previous defined script context for those.
const bool ignore_duplicates = scope_info->IsReplModeScope();
DirectHandle<ScriptContextTable> new_script_context_table =
ScriptContextTable::Add(isolate, script_context, result,
ignore_duplicates);
native_context->synchronized_set_script_context_table(
*new_script_context_table);
return result;
}
V8_WARN_UNUSED_RESULT MaybeHandle<Object> Invoke(Isolate* isolate,
const InvokeParams& params) {
RCS_SCOPE(isolate, RuntimeCallCounterId::kInvoke);
DCHECK(!IsJSGlobalObject(*params.receiver));
DCHECK_LE(params.args.size(), FixedArray::kMaxLength);
DCHECK(!isolate->has_exception());
#if V8_ENABLE_WEBASSEMBLY
// If we have PKU support for Wasm, ensure that code is currently write
// protected for this thread.
DCHECK_IMPLIES(wasm::GetWasmCodeManager()->HasMemoryProtectionKeySupport(),
!wasm::GetWasmCodeManager()->MemoryProtectionKeyWritable());
#endif // V8_ENABLE_WEBASSEMBLY
#ifdef USE_SIMULATOR
// Simulators use separate stacks for C++ and JS. JS stack overflow checks
// are performed whenever a JS function is called. However, it can be the case
// that the C++ stack grows faster than the JS stack, resulting in an overflow
// there. Add a check here to make that less likely.
StackLimitCheck check(isolate);
if (check.HasOverflowed()) {
isolate->StackOverflow();
isolate->ReportPendingMessages(params.message_handling ==
Execution::MessageHandling::kReport);
return MaybeHandle<Object>();
}
#endif
// api callbacks can be called directly, unless we want to take the detour
// through JS to set up a frame for break-at-entry.
if (IsJSFunction(*params.target)) {
auto function = Cast<JSFunction>(params.target);
if ((!params.is_construct || IsConstructor(*function)) &&
function->shared()->IsApiFunction() &&
!function->shared()->BreakAtEntry(isolate)) {
SaveAndSwitchContext save(isolate, function->context());
DCHECK(IsJSGlobalObject(function->context()->global_object()));
DirectHandle<Object> receiver = params.is_construct
? isolate->factory()->the_hole_value()
: params.receiver;
DirectHandle<FunctionTemplateInfo> fun_data(
function->shared()->api_func_data(), isolate);
auto value = Builtins::InvokeApiFunction(
isolate, params.is_construct, fun_data, receiver, params.args,
Cast<HeapObject>(params.new_target));
bool has_exception = value.is_null();
DCHECK_EQ(has_exception, isolate->has_exception());
if (has_exception) {
isolate->ReportPendingMessages(params.message_handling ==
Execution::MessageHandling::kReport);
return MaybeHandle<Object>();
} else {
isolate->clear_pending_message();
}
return value;
}
#ifdef DEBUG
if (function->shared()->is_script()) {
DCHECK(params.IsScript());
DCHECK(IsJSGlobalProxy(*params.receiver));
DCHECK_EQ(params.args.size(), 1);
DCHECK(IsFixedArray(*params.args[0]));
} else {
DCHECK(!params.IsScript());
}
#endif
// Set up a ScriptContext when running scripts that need it.
if (function->shared()->needs_script_context()) {
DirectHandle<Context> context;
DirectHandle<FixedArray> host_defined_options =
const_cast<InvokeParams&>(params).GetAndResetHostDefinedOptions();
if (!NewScriptContext(isolate, function, host_defined_options)
.ToHandle(&context)) {
isolate->ReportPendingMessages(params.message_handling ==
Execution::MessageHandling::kReport);
return MaybeHandle<Object>();
}
// We mutate the context if we allocate a script context. This is
// guaranteed to only happen once in a native context since scripts will
// always produce name clashes with themselves.
function->set_context(*context);
}
}
// Entering JavaScript.
VMState<JS> state(isolate);
if (!AllowJavascriptExecution::IsAllowed(isolate)) {
GRACEFUL_FATAL("Invoke in DisallowJavascriptExecutionScope");
}
if (!ThrowOnJavascriptExecution::IsAllowed(isolate)) {
isolate->ThrowIllegalOperation();
isolate->ReportPendingMessages(params.message_handling ==
Execution::MessageHandling::kReport);
return MaybeHandle<Object>();
}
if (!DumpOnJavascriptExecution::IsAllowed(isolate)) {
V8::GetCurrentPlatform()->DumpWithoutCrashing();
return isolate->factory()->undefined_value();
}
isolate->IncrementJavascriptExecutionCounter();
if (params.execution_target == Execution::Target::kCallable) {
DirectHandle<NativeContext> context = isolate->native_context();
if (!IsUndefined(context->script_execution_callback(), isolate)) {
v8::Context::AbortScriptExecutionCallback callback =
v8::ToCData<v8::Context::AbortScriptExecutionCallback,
kApiAbortScriptExecutionCallbackTag>(
isolate, context->script_execution_callback());
v8::Isolate* api_isolate = reinterpret_cast<v8::Isolate*>(isolate);
v8::Local<v8::Context> api_context = v8::Utils::ToLocal(context);
callback(api_isolate, api_context);
DCHECK(!isolate->has_exception());
// Always throw an exception to abort execution, if callback exists.
isolate->ThrowIllegalOperation();
return MaybeHandle<Object>();
}
}
// Placeholder for return value.
Tagged<Object> value;
DirectHandle<Code> code =
JSEntry(isolate, params.execution_target, params.is_construct);
{
// Save and restore context around invocation.
SaveContext save(isolate);
if (v8_flags.clear_exceptions_on_js_entry) isolate->clear_exception();
if (params.execution_target == Execution::Target::kCallable) {
// clang-format off
// {new_target}, {target}, {receiver}, return value: tagged pointers
// {argv}: pointer to array of tagged pointers
using JSEntryFunction = GeneratedCode<Address(
Address root_register_value, Address new_target, Address target,
Address receiver, intptr_t argc, Address** argv)>;
// clang-format on
JSEntryFunction stub_entry =
JSEntryFunction::FromAddress(isolate, code->instruction_start());
Address orig_func = (*params.new_target).ptr();
Address func = (*params.target).ptr();
Address recv = (*params.receiver).ptr();
int argc = static_cast<int>(params.args.size());
#ifdef V8_ENABLE_DIRECT_HANDLE
// TODO(42203211): Store the arguments to indirect handles because
// generated code still expects them in indirect handles. A fresh handle
// scope is introduced for these handles, which is then sealed. When this
// is eventually removed, sealing can go back together with saving the
// context for both branches.
HandleScope scope_for_conversion(isolate);
std::vector<IndirectHandle<Object>> args(argc);
for (int i = 0; i < argc; ++i)
args[i] = indirect_handle(params.args[i], isolate);
Address** argv = reinterpret_cast<Address**>(args.data());
#else
Address** argv = const_cast<Address**>(
reinterpret_cast<Address* const*>(params.args.data()));
#endif
// Block the allocation of handles without explicit handle scopes.
SealHandleScope shs(isolate);
RCS_SCOPE(isolate, RuntimeCallCounterId::kJS_Execution);
value = Tagged<Object>(
stub_entry.Call(isolate->isolate_data()->isolate_root(), orig_func,
func, recv, JSParameterCount(argc), argv));
} else {
DCHECK_EQ(Execution::Target::kRunMicrotasks, params.execution_target);
// clang-format off
// return value: tagged pointers
// {microtask_queue}: pointer to a C++ object
using JSEntryFunction = GeneratedCode<Address(
Address root_register_value, MicrotaskQueue* microtask_queue)>;
// clang-format on
JSEntryFunction stub_entry =
JSEntryFunction::FromAddress(isolate, code->instruction_start());
// Block the allocation of handles without explicit handle scopes.
SealHandleScope shs(isolate);
RCS_SCOPE(isolate, RuntimeCallCounterId::kJS_Execution);
value = Tagged<Object>(stub_entry.Call(
isolate->isolate_data()->isolate_root(), params.microtask_queue));
}
}
#ifdef VERIFY_HEAP
if (v8_flags.verify_heap) {
Object::ObjectVerify(value, isolate);
}
#endif
// Update the pending exception flag and return the value.
bool has_exception = IsException(value, isolate);
DCHECK_EQ(has_exception, isolate->has_exception());
if (has_exception) {
isolate->ReportPendingMessages(params.message_handling ==
Execution::MessageHandling::kReport);
return MaybeHandle<Object>();
} else {
isolate->clear_pending_message();
}
return Handle<Object>(value, isolate);
}
MaybeDirectHandle<Object> InvokeWithTryCatch(Isolate* isolate,
const InvokeParams& params) {
DCHECK_IMPLIES(v8_flags.strict_termination_checks,
!isolate->is_execution_terminating());
MaybeDirectHandle<Object> maybe_result;
if (params.exception_out != nullptr) {
*params.exception_out = {};
}
// Enter a try-block while executing the JavaScript code. To avoid
// duplicate error printing it must be non-verbose. Also, to avoid
// creating message objects during stack overflow we shouldn't
// capture messages.
v8::TryCatch catcher(reinterpret_cast<v8::Isolate*>(isolate));
catcher.SetVerbose(false);
catcher.SetCaptureMessage(false);
maybe_result = Invoke(isolate, params);
if (V8_LIKELY(!maybe_result.is_null())) {
DCHECK(!isolate->has_exception());
return maybe_result;
}
DCHECK(isolate->has_exception());
if (isolate->is_execution_terminating()) {
return maybe_result;
}
if (params.exception_out != nullptr) {
DCHECK(catcher.HasCaught());
*params.exception_out = v8::Utils::OpenDirectHandle(*catcher.Exception());
}
return maybe_result;
}
} // namespace
// static
MaybeHandle<Object> Execution::Call(
Isolate* isolate, DirectHandle<Object> callable,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args) {
// Use Execution::CallScript instead for scripts:
DCHECK_IMPLIES(IsJSFunction(*callable),
!Cast<JSFunction>(*callable)->shared()->is_script());
return Invoke(isolate,
InvokeParams::SetUpForCall(isolate, callable, receiver, args));
}
// static
MaybeHandle<Object> Execution::CallScript(
Isolate* isolate, DirectHandle<JSFunction> script_function,
DirectHandle<Object> receiver, DirectHandle<Object> host_defined_options) {
DCHECK(script_function->shared()->is_script());
DCHECK(IsJSGlobalProxy(*receiver) || IsJSGlobalObject(*receiver));
return Invoke(isolate,
InvokeParams::SetUpForCall(isolate, script_function, receiver,
{&host_defined_options, 1}));
}
MaybeHandle<Object> Execution::CallBuiltin(
Isolate* isolate, DirectHandle<JSFunction> builtin,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args) {
DCHECK(builtin->code(isolate)->is_builtin());
DisableBreak no_break(isolate->debug());
return Invoke(isolate,
InvokeParams::SetUpForCall(isolate, builtin, receiver, args));
}
// static
MaybeDirectHandle<JSReceiver> Execution::New(
Isolate* isolate, DirectHandle<Object> constructor,
base::Vector<const DirectHandle<Object>> args) {
return New(isolate, constructor, constructor, args);
}
// static
MaybeDirectHandle<JSReceiver> Execution::New(
Isolate* isolate, DirectHandle<Object> constructor,
DirectHandle<Object> new_target,
base::Vector<const DirectHandle<Object>> args) {
return Cast<JSReceiver>(Invoke(
isolate,
InvokeParams::SetUpForNew(isolate, constructor, new_target, args)));
}
// static
MaybeDirectHandle<Object> Execution::TryCallScript(
Isolate* isolate, DirectHandle<JSFunction> script_function,
DirectHandle<Object> receiver,
DirectHandle<FixedArray> host_defined_options) {
DCHECK(script_function->shared()->is_script());
DCHECK(IsJSGlobalProxy(*receiver) || IsJSGlobalObject(*receiver));
DirectHandle<Object> argument = host_defined_options;
return InvokeWithTryCatch(
isolate, InvokeParams::SetUpForTryCall(
isolate, script_function, receiver, {&argument, 1},
MessageHandling::kKeepPending, nullptr));
}
// static
MaybeDirectHandle<Object> Execution::TryCall(
Isolate* isolate, DirectHandle<Object> callable,
DirectHandle<Object> receiver,
base::Vector<const DirectHandle<Object>> args,
MessageHandling message_handling,
MaybeDirectHandle<Object>* exception_out) {
// Use Execution::TryCallScript instead for scripts:
DCHECK_IMPLIES(IsJSFunction(*callable),
!Cast<JSFunction>(*callable)->shared()->is_script());
return InvokeWithTryCatch(
isolate, InvokeParams::SetUpForTryCall(isolate, callable, receiver, args,
message_handling, exception_out));
}
// static
MaybeDirectHandle<Object> Execution::TryRunMicrotasks(
Isolate* isolate, MicrotaskQueue* microtask_queue) {
return InvokeWithTryCatch(
isolate, InvokeParams::SetUpForRunMicrotasks(isolate, microtask_queue));
}
struct StackHandlerMarker {
Address next;
Address padding;
};
static_assert(offsetof(StackHandlerMarker, next) ==
StackHandlerConstants::kNextOffset);
static_assert(offsetof(StackHandlerMarker, padding) ==
StackHandlerConstants::kPaddingOffset);
static_assert(sizeof(StackHandlerMarker) == StackHandlerConstants::kSize);
#if V8_ENABLE_WEBASSEMBLY
void Execution::CallWasm(Isolate* isolate, DirectHandle<Code> wrapper_code,
WasmCodePointer wasm_call_target,
DirectHandle<Object> object_ref, Address packed_args) {
using WasmEntryStub = GeneratedCode<Address(
Address target, Address object_ref, Address argv, Address c_entry_fp)>;
WasmEntryStub stub_entry =
WasmEntryStub::FromAddress(isolate, wrapper_code->instruction_start());
// Save and restore context around invocation and block the
// allocation of handles without explicit handle scopes.
SaveContext save(isolate);
SealHandleScope shs(isolate);
Address saved_c_entry_fp = *isolate->c_entry_fp_address();
Address saved_js_entry_sp = *isolate->js_entry_sp_address();
if (saved_js_entry_sp == kNullAddress) {
*isolate->js_entry_sp_address() = GetCurrentStackPosition();
}
StackHandlerMarker stack_handler;
stack_handler.next = isolate->thread_local_top()->handler_;
#ifdef V8_USE_ADDRESS_SANITIZER
stack_handler.padding = GetCurrentStackPosition();
#else
stack_handler.padding = 0;
#endif
isolate->thread_local_top()->handler_ =
reinterpret_cast<Address>(&stack_handler);
trap_handler::SetThreadInWasm();
{
RCS_SCOPE(isolate, RuntimeCallCounterId::kJS_Execution);
static_assert(compiler::CWasmEntryParameters::kCodeEntry == 0);
static_assert(compiler::CWasmEntryParameters::kObjectRef == 1);
static_assert(compiler::CWasmEntryParameters::kArgumentsBuffer == 2);
static_assert(compiler::CWasmEntryParameters::kCEntryFp == 3);
Address result =
stub_entry.Call(wasm_call_target.value(), (*object_ref).ptr(),
packed_args, saved_c_entry_fp);
if (result != kNullAddress) isolate->set_exception(Tagged<Object>(result));
}
// If there was an exception, then the thread-in-wasm flag is cleared
// already.
if (trap_handler::IsThreadInWasm()) {
trap_handler::ClearThreadInWasm();
}
isolate->thread_local_top()->handler_ = stack_handler.next;
if (saved_js_entry_sp == kNullAddress) {
*isolate->js_entry_sp_address() = saved_js_entry_sp;
}
*isolate->c_entry_fp_address() = saved_c_entry_fp;
}
#endif // V8_ENABLE_WEBASSEMBLY
} // namespace internal
} // namespace v8