blob: e9d0c36f0b2d745dce26d6199b3025658dae0d52 [file] [log] [blame]
// Copyright 2016 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/codegen/optimized-compilation-info.h"
#include "src/api/api.h"
#include "src/builtins/builtins.h"
#include "src/codegen/source-position.h"
#include "src/debug/debug.h"
#include "src/execution/isolate.h"
#include "src/objects/objects-inl.h"
#include "src/objects/shared-function-info.h"
#include "src/tracing/trace-event.h"
#include "src/tracing/traced-value.h"
#if V8_ENABLE_WEBASSEMBLY
#include "src/wasm/function-compiler.h"
#endif // V8_ENABLE_WEBASSEMBLY
namespace v8 {
namespace internal {
OptimizedCompilationInfo::OptimizedCompilationInfo(
Zone* zone, Isolate* isolate, IndirectHandle<SharedFunctionInfo> shared,
IndirectHandle<JSFunction> closure, CodeKind code_kind,
BytecodeOffset osr_offset)
: isolate_unsafe_(isolate),
code_kind_(code_kind),
osr_offset_(osr_offset),
zone_(zone),
optimization_id_(isolate->NextOptimizationId()) {
DCHECK_EQ(*shared, closure->shared());
DCHECK(shared->is_compiled());
DCHECK_IMPLIES(is_osr(), IsOptimizing());
bytecode_array_ = handle(shared->GetBytecodeArray(isolate), isolate);
shared_info_ = shared;
closure_ = closure;
canonical_handles_ = std::make_unique<CanonicalHandlesMap>(
isolate->heap(), ZoneAllocationPolicy(zone));
// Collect source positions for optimized code when profiling or if debugger
// is active, to be able to get more precise source positions at the price of
// more memory consumption.
if (isolate->NeedsDetailedOptimizedCodeLineInfo()) {
// We might not have source positions if collection fails (e.g. because we
// run out of stack space).
if (bytecode_array_->HasSourcePositionTable()) {
set_source_positions();
}
}
SetTracingFlags(shared->PassesFilter(v8_flags.trace_turbo_filter));
ConfigureFlags();
if (isolate->node_observer()) {
SetNodeObserver(isolate->node_observer());
}
}
OptimizedCompilationInfo::OptimizedCompilationInfo(
base::Vector<const char> debug_name, Zone* zone, CodeKind code_kind,
Builtin builtin)
: isolate_unsafe_(nullptr),
code_kind_(code_kind),
builtin_(builtin),
zone_(zone),
optimization_id_(kNoOptimizationId),
debug_name_(debug_name) {
DCHECK_IMPLIES(builtin_ != Builtin::kNoBuiltinId,
(code_kind_ == CodeKind::BUILTIN ||
code_kind_ == CodeKind::BYTECODE_HANDLER));
SetTracingFlags(
PassesFilter(debug_name, base::CStrVector(v8_flags.trace_turbo_filter)));
ConfigureFlags();
DCHECK(!has_shared_info());
}
void OptimizedCompilationInfo::ConfigureFlags() {
if (v8_flags.turbo_inline_js_wasm_calls) set_inline_js_wasm_calls();
if (v8_flags.cet_compatible) {
set_shadow_stack_compliant_lazy_deopt();
}
switch (code_kind_) {
case CodeKind::TURBOFAN_JS:
set_called_with_code_start_register();
set_switch_jump_table();
if (v8_flags.analyze_environment_liveness) {
set_analyze_environment_liveness();
}
if (v8_flags.turbo_splitting) set_splitting();
break;
case CodeKind::BYTECODE_HANDLER:
set_called_with_code_start_register();
#ifdef V8_ENABLE_BUILTIN_JUMP_TABLE_SWITCH
set_switch_jump_table();
#endif // V8_ENABLE_BUILTIN_JUMP_TABLE_SWITCH
if (v8_flags.turbo_splitting) set_splitting();
if (v8_flags.enable_allocation_folding) set_allocation_folding();
break;
case CodeKind::BUILTIN:
#ifdef V8_ENABLE_BUILTIN_JUMP_TABLE_SWITCH
set_switch_jump_table();
#endif // V8_ENABLE_BUILTIN_JUMP_TABLE_SWITCH
[[fallthrough]];
case CodeKind::FOR_TESTING:
case CodeKind::FOR_TESTING_JS:
if (v8_flags.turbo_splitting) set_splitting();
if (v8_flags.enable_allocation_folding) set_allocation_folding();
#if ENABLE_GDB_JIT_INTERFACE && DEBUG
set_source_positions();
#endif // ENABLE_GDB_JIT_INTERFACE && DEBUG
break;
case CodeKind::WASM_FUNCTION:
case CodeKind::WASM_TO_CAPI_FUNCTION:
set_switch_jump_table();
break;
case CodeKind::C_WASM_ENTRY:
case CodeKind::JS_TO_WASM_FUNCTION:
case CodeKind::WASM_TO_JS_FUNCTION:
case CodeKind::WASM_STACK_ENTRY:
break;
case CodeKind::BASELINE:
case CodeKind::MAGLEV:
case CodeKind::INTERPRETED_FUNCTION:
case CodeKind::REGEXP:
UNREACHABLE();
}
}
OptimizedCompilationInfo::~OptimizedCompilationInfo() {
if (disable_future_optimization() && has_shared_info()) {
DCHECK_NOT_NULL(isolate_unsafe_);
shared_info()->DisableOptimization(isolate_unsafe_, bailout_reason());
}
}
void OptimizedCompilationInfo::ReopenAndCanonicalizeHandlesInNewScope(
Isolate* isolate) {
if (!shared_info_.is_null()) {
shared_info_ = CanonicalHandle(*shared_info_, isolate);
}
if (!bytecode_array_.is_null()) {
bytecode_array_ = CanonicalHandle(*bytecode_array_, isolate);
}
if (!closure_.is_null()) {
closure_ = CanonicalHandle(*closure_, isolate);
}
DCHECK(code_.is_null());
}
void OptimizedCompilationInfo::AbortOptimization(BailoutReason reason) {
DCHECK_NE(reason, BailoutReason::kNoReason);
if (bailout_reason_ == BailoutReason::kNoReason) {
bailout_reason_ = reason;
}
if (IsTerminalBailoutReasonForTurbofan(reason)) {
set_disable_future_optimization();
}
}
void OptimizedCompilationInfo::RetryOptimization(BailoutReason reason) {
DCHECK_NE(reason, BailoutReason::kNoReason);
if (disable_future_optimization()) return;
bailout_reason_ = reason;
}
std::unique_ptr<char[]> OptimizedCompilationInfo::GetDebugName() const {
if (!shared_info().is_null()) {
return shared_info()->DebugNameCStr();
}
base::Vector<const char> name_vec = debug_name_;
if (name_vec.empty()) name_vec = base::ArrayVector("unknown");
std::unique_ptr<char[]> name(new char[name_vec.length() + 1]);
memcpy(name.get(), name_vec.begin(), name_vec.length());
name[name_vec.length()] = '\0';
return name;
}
StackFrame::Type OptimizedCompilationInfo::GetOutputStackFrameType() const {
switch (code_kind()) {
case CodeKind::FOR_TESTING:
case CodeKind::FOR_TESTING_JS:
case CodeKind::BYTECODE_HANDLER:
case CodeKind::BUILTIN:
return StackFrame::STUB;
#if V8_ENABLE_WEBASSEMBLY
case CodeKind::WASM_FUNCTION:
return StackFrame::WASM;
case CodeKind::WASM_TO_CAPI_FUNCTION:
return StackFrame::WASM_EXIT;
case CodeKind::JS_TO_WASM_FUNCTION:
return StackFrame::JS_TO_WASM;
case CodeKind::WASM_TO_JS_FUNCTION:
return StackFrame::WASM_TO_JS;
case CodeKind::C_WASM_ENTRY:
return StackFrame::C_WASM_ENTRY;
case CodeKind::WASM_STACK_ENTRY:
return StackFrame::WASM_STACK_ENTRY;
#else
case CodeKind::WASM_FUNCTION:
case CodeKind::WASM_TO_CAPI_FUNCTION:
case CodeKind::JS_TO_WASM_FUNCTION:
case CodeKind::WASM_TO_JS_FUNCTION:
case CodeKind::C_WASM_ENTRY:
case CodeKind::WASM_STACK_ENTRY:
UNREACHABLE();
#endif // V8_ENABLE_WEBASSEMBLY
case CodeKind::REGEXP:
case CodeKind::INTERPRETED_FUNCTION:
case CodeKind::BASELINE:
case CodeKind::MAGLEV:
case CodeKind::TURBOFAN_JS:
UNIMPLEMENTED();
}
UNREACHABLE();
}
void OptimizedCompilationInfo::SetCode(IndirectHandle<Code> code) {
DCHECK_EQ(code->kind(), code_kind());
code_ = code;
}
bool OptimizedCompilationInfo::has_context() const {
return !closure().is_null();
}
Tagged<Context> OptimizedCompilationInfo::context() const {
DCHECK(has_context());
return closure()->context();
}
bool OptimizedCompilationInfo::has_native_context() const {
return !closure().is_null() && !closure()->native_context().is_null();
}
Tagged<NativeContext> OptimizedCompilationInfo::native_context() const {
DCHECK(has_native_context());
return closure()->native_context();
}
bool OptimizedCompilationInfo::has_global_object() const {
return has_native_context();
}
Tagged<JSGlobalObject> OptimizedCompilationInfo::global_object() const {
DCHECK(has_global_object());
return native_context()->global_object();
}
int OptimizedCompilationInfo::AddInlinedFunction(
IndirectHandle<SharedFunctionInfo> inlined_function,
IndirectHandle<BytecodeArray> inlined_bytecode, SourcePosition pos) {
int id = static_cast<int>(inlined_functions_.size());
inlined_functions_.push_back(
InlinedFunctionHolder(inlined_function, inlined_bytecode, pos));
return id;
}
void OptimizedCompilationInfo::SetTracingFlags(bool passes_filter) {
if (!passes_filter) return;
if (v8_flags.trace_turbo) set_trace_turbo_json();
if (v8_flags.trace_turbo_graph) set_trace_turbo_graph();
if (v8_flags.trace_turbo_scheduled) set_trace_turbo_scheduled();
if (v8_flags.trace_turbo_alloc) set_trace_turbo_allocation();
if (v8_flags.trace_heap_broker) set_trace_heap_broker();
if (v8_flags.turboshaft_trace_reduction) set_turboshaft_trace_reduction();
}
void OptimizedCompilationInfo::mark_cancelled() {
was_cancelled_.store(true, std::memory_order_relaxed);
}
OptimizedCompilationInfo::InlinedFunctionHolder::InlinedFunctionHolder(
IndirectHandle<SharedFunctionInfo> inlined_shared_info,
IndirectHandle<BytecodeArray> inlined_bytecode, SourcePosition pos)
: shared_info(inlined_shared_info), bytecode_array(inlined_bytecode) {
position.position = pos;
// initialized when generating the deoptimization literals
position.inlined_function_id = DeoptimizationData::kNotInlinedIndex;
}
} // namespace internal
} // namespace v8