Commit 2aa6751e authored by Mythri A's avatar Mythri A Committed by Commit Bot

[turboprop] Use feedback only for calls to builtins

To reduce the number of deoptimizations in TurboProp use call feedback
only when we know the call target is a builtin. Given that we don't
inline in TurboProp, call feedback isn't really useful and using Generic
lowering doesn't impact performance much. TurboProp still inlines
builtins, so it is important to use this feedback for generating better
optimized code.

BUG: v8:10431
Change-Id: I24d51e43728f9aea3099767deb7800119fea40e2
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2116033
Commit-Queue: Mythri Alle <mythria@chromium.org>
Reviewed-by: 's avatarGeorg Neis <neis@chromium.org>
Reviewed-by: 's avatarJakob Gruber <jgruber@chromium.org>
Cr-Commit-Position: refs/heads/master@{#67468}
parent 218fc557
...@@ -3845,6 +3845,19 @@ bool ShouldUseCallICFeedback(Node* node) { ...@@ -3845,6 +3845,19 @@ bool ShouldUseCallICFeedback(Node* node) {
} // namespace } // namespace
bool JSCallReducer::IsBuiltinOrApiFunction(JSFunctionRef function) const {
if (should_disallow_heap_access() && !function.serialized()) {
TRACE_BROKER_MISSING(broker(), "data for function " << function);
return false;
}
// TODO(neis): Add a way to check if function template info isn't serialized
// and add a warning in such cases. Currently we can't tell if function
// template info doesn't exist or wasn't serialized.
return function.shared().HasBuiltinId() ||
function.shared().function_template_info().has_value();
}
Reduction JSCallReducer::ReduceJSCall(Node* node) { Reduction JSCallReducer::ReduceJSCall(Node* node) {
DCHECK_EQ(IrOpcode::kJSCall, node->opcode()); DCHECK_EQ(IrOpcode::kJSCall, node->opcode());
CallParameters const& p = CallParametersOf(node->op()); CallParameters const& p = CallParametersOf(node->op());
...@@ -3969,7 +3982,7 @@ Reduction JSCallReducer::ReduceJSCall(Node* node) { ...@@ -3969,7 +3982,7 @@ Reduction JSCallReducer::ReduceJSCall(Node* node) {
ProcessedFeedback const& feedback = ProcessedFeedback const& feedback =
broker()->GetFeedbackForCall(p.feedback()); broker()->GetFeedbackForCall(p.feedback());
if (feedback.IsInsufficient()) { if (feedback.IsInsufficient()) {
return ReduceSoftDeoptimize( return ReduceForInsufficientFeedback(
node, DeoptimizeReason::kInsufficientTypeFeedbackForCall); node, DeoptimizeReason::kInsufficientTypeFeedbackForCall);
} }
...@@ -3977,6 +3990,13 @@ Reduction JSCallReducer::ReduceJSCall(Node* node) { ...@@ -3977,6 +3990,13 @@ Reduction JSCallReducer::ReduceJSCall(Node* node) {
if (feedback_target.has_value() && feedback_target->map().is_callable()) { if (feedback_target.has_value() && feedback_target->map().is_callable()) {
Node* target_function = jsgraph()->Constant(*feedback_target); Node* target_function = jsgraph()->Constant(*feedback_target);
if (FLAG_turboprop) {
if (!feedback_target->IsJSFunction()) return NoChange();
if (!IsBuiltinOrApiFunction(feedback_target->AsJSFunction())) {
return NoChange();
}
}
// Check that the {target} is still the {target_function}. // Check that the {target} is still the {target_function}.
Node* check = graph()->NewNode(simplified()->ReferenceEqual(), target, Node* check = graph()->NewNode(simplified()->ReferenceEqual(), target,
target_function); target_function);
...@@ -4003,6 +4023,12 @@ Reduction JSCallReducer::ReduceJSCall(Node* node) { ...@@ -4003,6 +4023,12 @@ Reduction JSCallReducer::ReduceJSCall(Node* node) {
broker(), "feedback vector, not serialized: " << feedback_vector); broker(), "feedback vector, not serialized: " << feedback_vector);
return NoChange(); return NoChange();
} }
if (FLAG_turboprop &&
!feedback_vector.shared_function_info().HasBuiltinId()) {
return NoChange();
}
Node* target_closure = effect = Node* target_closure = effect =
graph()->NewNode(simplified()->CheckClosure(feedback_cell.object()), graph()->NewNode(simplified()->CheckClosure(feedback_cell.object()),
target, effect, control); target, effect, control);
...@@ -4412,7 +4438,7 @@ Reduction JSCallReducer::ReduceJSConstruct(Node* node) { ...@@ -4412,7 +4438,7 @@ Reduction JSCallReducer::ReduceJSConstruct(Node* node) {
ProcessedFeedback const& feedback = ProcessedFeedback const& feedback =
broker()->GetFeedbackForCall(p.feedback()); broker()->GetFeedbackForCall(p.feedback());
if (feedback.IsInsufficient()) { if (feedback.IsInsufficient()) {
return ReduceSoftDeoptimize( return ReduceForInsufficientFeedback(
node, DeoptimizeReason::kInsufficientTypeFeedbackForConstruct); node, DeoptimizeReason::kInsufficientTypeFeedbackForConstruct);
} }
...@@ -4823,9 +4849,15 @@ Reduction JSCallReducer::ReduceReturnReceiver(Node* node) { ...@@ -4823,9 +4849,15 @@ Reduction JSCallReducer::ReduceReturnReceiver(Node* node) {
return Replace(receiver); return Replace(receiver);
} }
Reduction JSCallReducer::ReduceSoftDeoptimize(Node* node, Reduction JSCallReducer::ReduceForInsufficientFeedback(
DeoptimizeReason reason) { Node* node, DeoptimizeReason reason) {
DCHECK(node->opcode() == IrOpcode::kJSCall ||
node->opcode() == IrOpcode::kJSConstruct);
if (!(flags() & kBailoutOnUninitialized)) return NoChange(); if (!(flags() & kBailoutOnUninitialized)) return NoChange();
// TODO(mythria): May be add additional flags to specify if we need to deopt
// on calls / construct rather than checking for TurboProp here. We may need
// it for NativeContextIndependent code too.
if (FLAG_turboprop) return NoChange();
Node* effect = NodeProperties::GetEffectInput(node); Node* effect = NodeProperties::GetEffectInput(node);
Node* control = NodeProperties::GetControlInput(node); Node* control = NodeProperties::GetControlInput(node);
......
...@@ -159,7 +159,7 @@ class V8_EXPORT_PRIVATE JSCallReducer final : public AdvancedReducer { ...@@ -159,7 +159,7 @@ class V8_EXPORT_PRIVATE JSCallReducer final : public AdvancedReducer {
const SharedFunctionInfoRef& shared); const SharedFunctionInfoRef& shared);
Reduction ReduceTypedArrayPrototypeToStringTag(Node* node); Reduction ReduceTypedArrayPrototypeToStringTag(Node* node);
Reduction ReduceSoftDeoptimize(Node* node, DeoptimizeReason reason); Reduction ReduceForInsufficientFeedback(Node* node, DeoptimizeReason reason);
Reduction ReduceMathUnary(Node* node, const Operator* op); Reduction ReduceMathUnary(Node* node, const Operator* op);
Reduction ReduceMathBinary(Node* node, const Operator* op); Reduction ReduceMathBinary(Node* node, const Operator* op);
...@@ -218,6 +218,8 @@ class V8_EXPORT_PRIVATE JSCallReducer final : public AdvancedReducer { ...@@ -218,6 +218,8 @@ class V8_EXPORT_PRIVATE JSCallReducer final : public AdvancedReducer {
Node* control, Node** if_true, Node** if_false); Node* control, Node** if_true, Node** if_false);
Node* LoadReceiverElementsKind(Node* receiver, Node** effect, Node** control); Node* LoadReceiverElementsKind(Node* receiver, Node** effect, Node** control);
bool IsBuiltinOrApiFunction(JSFunctionRef target_ref) const;
Graph* graph() const; Graph* graph() const;
JSGraph* jsgraph() const { return jsgraph_; } JSGraph* jsgraph() const { return jsgraph_; }
JSHeapBroker* broker() const { return broker_; } JSHeapBroker* broker() const { return broker_; }
......
...@@ -568,6 +568,13 @@ Node* JSTypeHintLowering::TryBuildSoftDeopt(FeedbackSlot slot, Node* effect, ...@@ -568,6 +568,13 @@ Node* JSTypeHintLowering::TryBuildSoftDeopt(FeedbackSlot slot, Node* effect,
if (!(flags() & kBailoutOnUninitialized)) return nullptr; if (!(flags() & kBailoutOnUninitialized)) return nullptr;
FeedbackSource source(feedback_vector(), slot); FeedbackSource source(feedback_vector(), slot);
// TODO(mythria): Think of adding flags to specify if we need a soft deopt for
// calls instead of using FLAG_turboprop here.
if (FLAG_turboprop &&
broker()->GetFeedbackSlotKind(source) == FeedbackSlotKind::kCall) {
return nullptr;
}
if (!broker()->FeedbackIsInsufficient(source)) return nullptr; if (!broker()->FeedbackIsInsufficient(source)) return nullptr;
Node* deoptimize = jsgraph()->graph()->NewNode( Node* deoptimize = jsgraph()->graph()->NewNode(
......
...@@ -1090,6 +1090,9 @@ bool SerializerForBackgroundCompilation::BailoutOnUninitialized( ...@@ -1090,6 +1090,9 @@ bool SerializerForBackgroundCompilation::BailoutOnUninitialized(
// OSR entry point. TODO(neis): Support OSR? // OSR entry point. TODO(neis): Support OSR?
return false; return false;
} }
if (FLAG_turboprop && feedback.slot_kind() == FeedbackSlotKind::kCall) {
return false;
}
if (feedback.IsInsufficient()) { if (feedback.IsInsufficient()) {
environment()->Kill(); environment()->Kill();
return true; return true;
......
...@@ -619,6 +619,11 @@ ...@@ -619,6 +619,11 @@
'test-cpu-profiler/DeoptAtFirstLevelInlinedSource': [SKIP], 'test-cpu-profiler/DeoptAtFirstLevelInlinedSource': [SKIP],
'test-cpu-profiler/DeoptAtSecondLevelInlinedSource': [SKIP], 'test-cpu-profiler/DeoptAtSecondLevelInlinedSource': [SKIP],
'test-cpu-profiler/DeoptUntrackedFunction': [SKIP], 'test-cpu-profiler/DeoptUntrackedFunction': [SKIP],
# Turboprop doesn't use call feedback and hence doesn't inline even if
# the inlining flag is explicitly set.
'test-cpu-profiler/DetailedSourcePositionAPI_Inlining': [SKIP],
'serializer-tester/BoundFunctionArguments': [SKIP],
'serializer-tester/BoundFunctionTarget': [SKIP],
}], # variant == turboprop }], # variant == turboprop
############################################################################## ##############################################################################
......
...@@ -1147,6 +1147,7 @@ ...@@ -1147,6 +1147,7 @@
'compiler/opt-higher-order-functions': [SKIP], 'compiler/opt-higher-order-functions': [SKIP],
'regress/regress-1049982-1': [SKIP], 'regress/regress-1049982-1': [SKIP],
'regress/regress-1049982-2': [SKIP], 'regress/regress-1049982-2': [SKIP],
'es6/iterator-eager-deopt': [SKIP],
# interrupt_budget overrides don't work with TurboProp. # interrupt_budget overrides don't work with TurboProp.
'interrupt-budget-override': [SKIP], 'interrupt-budget-override': [SKIP],
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment