[maglev] Don't trigger TF OSR before TF
Wait for TF optimisation to be requested (or to have happened) before cranking up TF OSR urgency; this prevents us from getting into a situation where we repeatedly call Maglev code which then OSRs into TurboFan lots of times before finally tiering up to TurboFan properly. Since we are waiting for TF optimisation to be requested, we also need a mechanism to allow TF optimisation to be requested even when Maglev code has been requested hasn't yet run (for direct Baseline->Turbofan tierup, since Maglev can't OSR). Do so by re-trying the optimisation check if it spits out Maglev but Maglev is already requested. As a drive-by, clean up some naming around OSR code. Bug: v8:7700 Change-Id: I483a412dd92fe13ee21f8aa46d86572bcf3f8a61 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3942385 Reviewed-by: Toon Verwaest <verwaest@chromium.org> Auto-Submit: Leszek Swirski <leszeks@chromium.org> Commit-Queue: Leszek Swirski <leszeks@chromium.org> Cr-Commit-Position: refs/heads/main@{#83630}
This commit is contained in:
parent
8610cc4ee3
commit
325853cf50
@ -1833,6 +1833,15 @@ static constexpr uint32_t kNoneOrInProgressMask = 0b110;
|
|||||||
TIERING_STATE_LIST(V)
|
TIERING_STATE_LIST(V)
|
||||||
#undef V
|
#undef V
|
||||||
|
|
||||||
|
constexpr bool IsRequestMaglev(TieringState state) {
|
||||||
|
return IsRequestMaglev_Concurrent(state) ||
|
||||||
|
IsRequestMaglev_Synchronous(state);
|
||||||
|
}
|
||||||
|
constexpr bool IsRequestTurbofan(TieringState state) {
|
||||||
|
return IsRequestTurbofan_Concurrent(state) ||
|
||||||
|
IsRequestTurbofan_Synchronous(state);
|
||||||
|
}
|
||||||
|
|
||||||
constexpr const char* ToString(TieringState marker) {
|
constexpr const char* ToString(TieringState marker) {
|
||||||
switch (marker) {
|
switch (marker) {
|
||||||
#define V(Name, Value) \
|
#define V(Name, Value) \
|
||||||
|
@ -11,12 +11,14 @@
|
|||||||
#include "src/codegen/compilation-cache.h"
|
#include "src/codegen/compilation-cache.h"
|
||||||
#include "src/codegen/compiler.h"
|
#include "src/codegen/compiler.h"
|
||||||
#include "src/codegen/pending-optimization-table.h"
|
#include "src/codegen/pending-optimization-table.h"
|
||||||
|
#include "src/common/globals.h"
|
||||||
#include "src/diagnostics/code-tracer.h"
|
#include "src/diagnostics/code-tracer.h"
|
||||||
#include "src/execution/execution.h"
|
#include "src/execution/execution.h"
|
||||||
#include "src/execution/frames-inl.h"
|
#include "src/execution/frames-inl.h"
|
||||||
#include "src/handles/global-handles.h"
|
#include "src/handles/global-handles.h"
|
||||||
#include "src/init/bootstrapper.h"
|
#include "src/init/bootstrapper.h"
|
||||||
#include "src/interpreter/interpreter.h"
|
#include "src/interpreter/interpreter.h"
|
||||||
|
#include "src/objects/code-kind.h"
|
||||||
#include "src/objects/code.h"
|
#include "src/objects/code.h"
|
||||||
#include "src/tracing/trace-event.h"
|
#include "src/tracing/trace-event.h"
|
||||||
|
|
||||||
@ -261,7 +263,7 @@ void TieringManager::RequestOsrAtNextOpportunity(JSFunction function) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void TieringManager::MaybeOptimizeFrame(JSFunction function,
|
void TieringManager::MaybeOptimizeFrame(JSFunction function,
|
||||||
CodeKind code_kind) {
|
CodeKind calling_code_kind) {
|
||||||
const TieringState tiering_state = function.feedback_vector().tiering_state();
|
const TieringState tiering_state = function.feedback_vector().tiering_state();
|
||||||
const TieringState osr_tiering_state =
|
const TieringState osr_tiering_state =
|
||||||
function.feedback_vector().osr_tiering_state();
|
function.feedback_vector().osr_tiering_state();
|
||||||
@ -288,24 +290,15 @@ void TieringManager::MaybeOptimizeFrame(JSFunction function,
|
|||||||
// Continue below and do a normal optimized compile as well.
|
// Continue below and do a normal optimized compile as well.
|
||||||
}
|
}
|
||||||
|
|
||||||
const bool is_marked_for_any_optimization =
|
|
||||||
(static_cast<uint32_t>(tiering_state) & kNoneOrInProgressMask) != 0;
|
|
||||||
// Baseline OSR uses a separate mechanism and must not be considered here,
|
// Baseline OSR uses a separate mechanism and must not be considered here,
|
||||||
// therefore we limit to kOptimizedJSFunctionCodeKindsMask.
|
// therefore we limit to kOptimizedJSFunctionCodeKindsMask.
|
||||||
// TODO(v8:7700): Change the condition below for Maglev OSR once it is
|
// TODO(v8:7700): Change the condition below for Maglev OSR once it is
|
||||||
// implemented.
|
// implemented.
|
||||||
if (is_marked_for_any_optimization ||
|
if (IsRequestTurbofan(tiering_state) ||
|
||||||
function.HasAvailableHigherTierCodeThanWithFilter(
|
function.HasAvailableCodeKind(CodeKind::TURBOFAN)) {
|
||||||
code_kind, kOptimizedJSFunctionCodeKindsMask)) {
|
|
||||||
// OSR kicks in only once we've previously decided to tier up, but we are
|
// OSR kicks in only once we've previously decided to tier up, but we are
|
||||||
// still in the lower-tier frame (this implies a long-running loop).
|
// still in a lower-tier frame (this implies a long-running loop).
|
||||||
//
|
if (SmallEnoughForOSR(isolate_, function, calling_code_kind)) {
|
||||||
// TODO(v8:7700): In the presence of Maglev, OSR is triggered much earlier
|
|
||||||
// than with the old pipeline since we tier up to Maglev earlier which
|
|
||||||
// affects both conditions above. This *seems* fine (when stuck in a loop
|
|
||||||
// we want to tier up, regardless of the active tier), but we may want to
|
|
||||||
// think about this again at some point.
|
|
||||||
if (SmallEnoughForOSR(isolate_, function, code_kind)) {
|
|
||||||
TryIncrementOsrUrgency(isolate_, function);
|
TryIncrementOsrUrgency(isolate_, function);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -314,20 +307,33 @@ void TieringManager::MaybeOptimizeFrame(JSFunction function,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
DCHECK(!is_marked_for_any_optimization &&
|
DCHECK(!IsRequestTurbofan(tiering_state));
|
||||||
!function.HasAvailableHigherTierCodeThanWithFilter(
|
DCHECK(!function.HasAvailableCodeKind(CodeKind::TURBOFAN));
|
||||||
code_kind, kOptimizedJSFunctionCodeKindsMask));
|
OptimizationDecision d = ShouldOptimize(function, calling_code_kind);
|
||||||
OptimizationDecision d = ShouldOptimize(function, code_kind);
|
// We might be stuck in a baseline frame that wants to tier up to Maglev, but
|
||||||
|
// is in a loop, and can't OSR, because Maglev doesn't have OSR. Allow it to
|
||||||
|
// skip over Maglev by re-checking ShouldOptimize as if we were in Maglev.
|
||||||
|
// TODO(v8:7700): Remove this when Maglev can OSR.
|
||||||
|
static_assert(!CodeKindCanOSR(CodeKind::MAGLEV));
|
||||||
|
if (d.should_optimize() && d.code_kind == CodeKind::MAGLEV) {
|
||||||
|
bool is_marked_for_maglev_optimization =
|
||||||
|
IsRequestMaglev(tiering_state) ||
|
||||||
|
function.HasAvailableCodeKind(CodeKind::MAGLEV);
|
||||||
|
if (is_marked_for_maglev_optimization) {
|
||||||
|
d = ShouldOptimize(function, CodeKind::MAGLEV);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (d.should_optimize()) Optimize(function, d);
|
if (d.should_optimize()) Optimize(function, d);
|
||||||
}
|
}
|
||||||
|
|
||||||
OptimizationDecision TieringManager::ShouldOptimize(JSFunction function,
|
OptimizationDecision TieringManager::ShouldOptimize(
|
||||||
CodeKind code_kind) {
|
JSFunction function, CodeKind calling_code_kind) {
|
||||||
if (TiersUpToMaglev(code_kind) &&
|
if (TiersUpToMaglev(calling_code_kind) &&
|
||||||
function.shared().PassesFilter(v8_flags.maglev_filter) &&
|
function.shared().PassesFilter(v8_flags.maglev_filter) &&
|
||||||
!function.shared(isolate_).maglev_compilation_failed()) {
|
!function.shared(isolate_).maglev_compilation_failed()) {
|
||||||
return OptimizationDecision::Maglev();
|
return OptimizationDecision::Maglev();
|
||||||
} else if (code_kind == CodeKind::TURBOFAN) {
|
} else if (calling_code_kind == CodeKind::TURBOFAN) {
|
||||||
// Already in the top tier.
|
// Already in the top tier.
|
||||||
return OptimizationDecision::DoNotOptimize();
|
return OptimizationDecision::DoNotOptimize();
|
||||||
}
|
}
|
||||||
|
@ -3558,6 +3558,7 @@ void AttemptOnStackReplacement(MaglevAssembler* masm, Label* return_label,
|
|||||||
// See also: InterpreterAssembler::OnStackReplacement.
|
// See also: InterpreterAssembler::OnStackReplacement.
|
||||||
|
|
||||||
baseline::BaselineAssembler basm(masm);
|
baseline::BaselineAssembler basm(masm);
|
||||||
|
__ AssertFeedbackVector(scratch0);
|
||||||
|
|
||||||
// Case 1).
|
// Case 1).
|
||||||
Label deopt;
|
Label deopt;
|
||||||
@ -3569,7 +3570,6 @@ void AttemptOnStackReplacement(MaglevAssembler* masm, Label* return_label,
|
|||||||
|
|
||||||
// Case 2).
|
// Case 2).
|
||||||
{
|
{
|
||||||
__ AssertFeedbackVector(scratch0);
|
|
||||||
__ movb(scratch0, FieldOperand(scratch0, FeedbackVector::kOsrStateOffset));
|
__ movb(scratch0, FieldOperand(scratch0, FeedbackVector::kOsrStateOffset));
|
||||||
__ DecodeField<FeedbackVector::OsrUrgencyBits>(scratch0);
|
__ DecodeField<FeedbackVector::OsrUrgencyBits>(scratch0);
|
||||||
basm.JumpIfByte(baseline::Condition::kUnsignedLessThanEqual, scratch0,
|
basm.JumpIfByte(baseline::Condition::kUnsignedLessThanEqual, scratch0,
|
||||||
@ -3601,18 +3601,19 @@ void AttemptOnStackReplacement(MaglevAssembler* masm, Label* return_label,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
DCHECK(!snapshot.live_registers.has(maybe_target_code));
|
||||||
SaveRegisterStateForCall save_register_state(masm, snapshot);
|
SaveRegisterStateForCall save_register_state(masm, snapshot);
|
||||||
__ Move(kContextRegister, masm->native_context().object());
|
__ Move(kContextRegister, masm->native_context().object());
|
||||||
__ Push(Smi::FromInt(osr_offset.ToInt()));
|
__ Push(Smi::FromInt(osr_offset.ToInt()));
|
||||||
__ CallRuntime(Runtime::kCompileOptimizedOSRFromMaglev, 1);
|
__ CallRuntime(Runtime::kCompileOptimizedOSRFromMaglev, 1);
|
||||||
save_register_state.DefineSafepoint();
|
save_register_state.DefineSafepoint();
|
||||||
__ Move(scratch0, rax);
|
__ Move(maybe_target_code, kReturnRegister0);
|
||||||
}
|
}
|
||||||
|
|
||||||
// A `0` return value means there is no OSR code available yet. Fall
|
// A `0` return value means there is no OSR code available yet. Fall
|
||||||
// through for now, OSR code will be picked up once it exists and is
|
// through for now, OSR code will be picked up once it exists and is
|
||||||
// cached on the feedback vector.
|
// cached on the feedback vector.
|
||||||
__ testq(scratch0, scratch0);
|
__ Cmp(maybe_target_code, 0);
|
||||||
__ j(equal, return_label, Label::kNear);
|
__ j(equal, return_label, Label::kNear);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,13 +5,25 @@
|
|||||||
// Flags: --allow-natives-syntax --maglev --no-stress-opt
|
// Flags: --allow-natives-syntax --maglev --no-stress-opt
|
||||||
// Flags: --no-baseline-batch-compilation --use-osr --turbofan
|
// Flags: --no-baseline-batch-compilation --use-osr --turbofan
|
||||||
|
|
||||||
let keep_going = 100000; // A counter to avoid test hangs on failure.
|
let keep_going = 10000000; // A counter to avoid test hangs on failure.
|
||||||
|
|
||||||
function f() {
|
function f() {
|
||||||
let reached_tf = false;
|
let reached_tf = false;
|
||||||
|
let prev_status = 0;
|
||||||
while (!reached_tf && --keep_going) {
|
while (!reached_tf && --keep_going) {
|
||||||
// This loop should trigger OSR.
|
// This loop should trigger OSR.
|
||||||
reached_tf = %CurrentFrameIsTurbofan();
|
reached_tf = %CurrentFrameIsTurbofan();
|
||||||
|
let status = %GetOptimizationStatus(f);
|
||||||
|
if (status !== prev_status) {
|
||||||
|
let p = []
|
||||||
|
for (let k in V8OptimizationStatus) {
|
||||||
|
if (V8OptimizationStatus[k] & status) {
|
||||||
|
p.push(k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
print(p.join(","));
|
||||||
|
prev_status = status;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user