Commit 7ba8e662 authored by Maya Lekova's avatar Maya Lekova Committed by Commit Bot

[turbofan] Make code and comment match in FastFunctionPrototypeBind

Additionally, used number of own descriptors in both CSA and the
reduction of Function.prototype.bind.

Change-Id: I7b86e059d20faa1160cdc0126932fff924226eee
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1714655
Commit-Queue: Maya Lekova <mslekova@chromium.org>
Reviewed-by: 's avatarJakob Gruber <jgruber@chromium.org>
Reviewed-by: 's avatarGeorg Neis <neis@chromium.org>
Cr-Commit-Position: refs/heads/master@{#62885}
parent 2172f16c
...@@ -27,7 +27,7 @@ TF_BUILTIN(FastFunctionPrototypeBind, CodeStubAssembler) { ...@@ -27,7 +27,7 @@ TF_BUILTIN(FastFunctionPrototypeBind, CodeStubAssembler) {
Node* receiver = args.GetReceiver(); Node* receiver = args.GetReceiver();
GotoIf(TaggedIsSmi(receiver), &slow); GotoIf(TaggedIsSmi(receiver), &slow);
Node* receiver_map = LoadMap(receiver); TNode<Map> receiver_map = LoadMap(receiver);
{ {
Node* instance_type = LoadMapInstanceType(receiver_map); Node* instance_type = LoadMapInstanceType(receiver_map);
GotoIfNot( GotoIfNot(
...@@ -45,20 +45,20 @@ TF_BUILTIN(FastFunctionPrototypeBind, CodeStubAssembler) { ...@@ -45,20 +45,20 @@ TF_BUILTIN(FastFunctionPrototypeBind, CodeStubAssembler) {
// AccessorInfo objects. In that case, their value can be recomputed even if // AccessorInfo objects. In that case, their value can be recomputed even if
// the actual value on the object changes. // the actual value on the object changes.
Comment("Check descriptor array length"); Comment("Check descriptor array length");
TNode<DescriptorArray> descriptors = LoadMapDescriptors(receiver_map);
// Minimum descriptor array length required for fast path. // Minimum descriptor array length required for fast path.
const int min_nof_descriptors = i::Max(JSFunction::kLengthDescriptorIndex, const int min_nof_descriptors = i::Max(JSFunction::kLengthDescriptorIndex,
JSFunction::kNameDescriptorIndex); JSFunction::kNameDescriptorIndex) +
TNode<Int32T> nof_descriptors = LoadNumberOfDescriptors(descriptors); 1;
GotoIf( TNode<Int32T> nof_descriptors = LoadNumberOfOwnDescriptors(receiver_map);
Int32LessThanOrEqual(nof_descriptors, Int32Constant(min_nof_descriptors)), GotoIf(Int32LessThan(nof_descriptors, Int32Constant(min_nof_descriptors)),
&slow); &slow);
// Check whether the length and name properties are still present as // Check whether the length and name properties are still present as
// AccessorInfo objects. In that case, their value can be recomputed even if // AccessorInfo objects. In that case, their value can be recomputed even if
// the actual value on the object changes. // the actual value on the object changes.
Comment("Check name and length properties"); Comment("Check name and length properties");
{ {
TNode<DescriptorArray> descriptors = LoadMapDescriptors(receiver_map);
const int length_index = JSFunction::kLengthDescriptorIndex; const int length_index = JSFunction::kLengthDescriptorIndex;
TNode<Name> maybe_length = TNode<Name> maybe_length =
LoadKeyByDescriptorEntry(descriptors, length_index); LoadKeyByDescriptorEntry(descriptors, length_index);
......
...@@ -1547,6 +1547,12 @@ TNode<Int32T> CodeStubAssembler::LoadNumberOfDescriptors( ...@@ -1547,6 +1547,12 @@ TNode<Int32T> CodeStubAssembler::LoadNumberOfDescriptors(
MachineType::Int16())); MachineType::Int16()));
} }
TNode<Int32T> CodeStubAssembler::LoadNumberOfOwnDescriptors(TNode<Map> map) {
Node* bit_field3 = LoadMapBitField3(map);
return UncheckedCast<Int32T>(
DecodeWord32<Map::NumberOfOwnDescriptorsBits>(bit_field3));
}
TNode<Int32T> CodeStubAssembler::LoadMapBitField(SloppyTNode<Map> map) { TNode<Int32T> CodeStubAssembler::LoadMapBitField(SloppyTNode<Map> map) {
CSA_SLOW_ASSERT(this, IsMap(map)); CSA_SLOW_ASSERT(this, IsMap(map));
return UncheckedCast<Int32T>( return UncheckedCast<Int32T>(
......
...@@ -940,6 +940,8 @@ class V8_EXPORT_PRIVATE CodeStubAssembler ...@@ -940,6 +940,8 @@ class V8_EXPORT_PRIVATE CodeStubAssembler
SloppyTNode<WeakFixedArray> array); SloppyTNode<WeakFixedArray> array);
// Load the number of descriptors in DescriptorArray. // Load the number of descriptors in DescriptorArray.
TNode<Int32T> LoadNumberOfDescriptors(TNode<DescriptorArray> array); TNode<Int32T> LoadNumberOfDescriptors(TNode<DescriptorArray> array);
// Load the number of own descriptors of a map.
TNode<Int32T> LoadNumberOfOwnDescriptors(TNode<Map> map);
// Load the bit field of a Map. // Load the bit field of a Map.
TNode<Int32T> LoadMapBitField(SloppyTNode<Map> map); TNode<Int32T> LoadMapBitField(SloppyTNode<Map> map);
// Load bit field 2 of a map. // Load bit field 2 of a map.
......
...@@ -550,7 +550,13 @@ Reduction JSCallReducer::ReduceFunctionPrototypeBind(Node* node) { ...@@ -550,7 +550,13 @@ Reduction JSCallReducer::ReduceFunctionPrototypeBind(Node* node) {
// runtime otherwise. // runtime otherwise.
Handle<DescriptorArray> descriptors( Handle<DescriptorArray> descriptors(
receiver_map.object()->instance_descriptors(), isolate()); receiver_map.object()->instance_descriptors(), isolate());
if (descriptors->number_of_descriptors() < 2) return inference.NoChange(); int minimum_nof_descriptors = std::max(JSFunction::kLengthDescriptorIndex,
JSFunction::kNameDescriptorIndex) +
1;
if (receiver_map.object()->NumberOfOwnDescriptors() <
minimum_nof_descriptors) {
return inference.NoChange();
}
if (descriptors->GetKey(JSFunction::kLengthDescriptorIndex) != if (descriptors->GetKey(JSFunction::kLengthDescriptorIndex) !=
ReadOnlyRoots(isolate()).length_string()) { ReadOnlyRoots(isolate()).length_string()) {
return inference.NoChange(); return inference.NoChange();
......
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