Commit 8bc54052 authored by Leszek Swirski's avatar Leszek Swirski Committed by Commit Bot

[csa] Merge in- and out-of-object store paths

Rather than duplicating code paths for in- and out-of-object stores,
have one code path which checks whether it needs to load the property
store (and change the storage location to the HeapNumber value for
unboxed doubles).

As a drive-by, change the representation dispatch into a switch, and
inline the representation checks into that switch, to make explicit
what checks for what and which paths transform the value. Also, TNodify
some of the surrounding functions.

Change-Id: Ia1bf698b4cec3ffce9aaa5732cda2e3be9efd8e8
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1795345Reviewed-by: 's avatarToon Verwaest <verwaest@chromium.org>
Commit-Queue: Leszek Swirski <leszeks@chromium.org>
Cr-Commit-Position: refs/heads/master@{#63652}
parent d7d25d2a
...@@ -5609,45 +5609,22 @@ void CodeStubAssembler::InitializeAllocationMemento(Node* base, ...@@ -5609,45 +5609,22 @@ void CodeStubAssembler::InitializeAllocationMemento(Node* base,
Comment("]"); Comment("]");
} }
Node* CodeStubAssembler::TryTaggedToFloat64(Node* value, TNode<Float64T> CodeStubAssembler::TryTaggedToFloat64(
Label* if_valueisnotnumber) { TNode<Object> value, Label* if_valueisnotnumber) {
Label out(this); return Select<Float64T>(
VARIABLE(var_result, MachineRepresentation::kFloat64); TaggedIsSmi(value), [&]() { return SmiToFloat64(CAST(value)); },
[&]() {
// Check if the {value} is a Smi or a HeapObject. GotoIfNot(IsHeapNumber(CAST(value)), if_valueisnotnumber);
Label if_valueissmi(this), if_valueisnotsmi(this); return LoadHeapNumberValue(CAST(value));
Branch(TaggedIsSmi(value), &if_valueissmi, &if_valueisnotsmi); });
BIND(&if_valueissmi);
{
// Convert the Smi {value}.
var_result.Bind(SmiToFloat64(value));
Goto(&out);
}
BIND(&if_valueisnotsmi);
{
// Check if {value} is a HeapNumber.
Label if_valueisheapnumber(this);
Branch(IsHeapNumber(value), &if_valueisheapnumber, if_valueisnotnumber);
BIND(&if_valueisheapnumber);
{
// Load the floating point value.
var_result.Bind(LoadHeapNumberValue(value));
Goto(&out);
}
}
BIND(&out);
return var_result.value();
} }
Node* CodeStubAssembler::TruncateTaggedToFloat64(Node* context, Node* value) { TNode<Float64T> CodeStubAssembler::TruncateTaggedToFloat64(
SloppyTNode<Context> context, SloppyTNode<Object> value) {
// We might need to loop once due to ToNumber conversion. // We might need to loop once due to ToNumber conversion.
VARIABLE(var_value, MachineRepresentation::kTagged); TVARIABLE(Object, var_value, value);
VARIABLE(var_result, MachineRepresentation::kFloat64); TVARIABLE(Float64T, var_result);
Label loop(this, &var_value), done_loop(this, &var_result); Label loop(this, &var_value), done_loop(this, &var_result);
var_value.Bind(value);
Goto(&loop); Goto(&loop);
BIND(&loop); BIND(&loop);
{ {
...@@ -5658,14 +5635,13 @@ Node* CodeStubAssembler::TruncateTaggedToFloat64(Node* context, Node* value) { ...@@ -5658,14 +5635,13 @@ Node* CodeStubAssembler::TruncateTaggedToFloat64(Node* context, Node* value) {
// Convert {value} to Float64 if it is a number and convert it to a number // Convert {value} to Float64 if it is a number and convert it to a number
// otherwise. // otherwise.
Node* const result = TryTaggedToFloat64(value, &if_valueisnotnumber); var_result = TryTaggedToFloat64(value, &if_valueisnotnumber);
var_result.Bind(result);
Goto(&done_loop); Goto(&done_loop);
BIND(&if_valueisnotnumber); BIND(&if_valueisnotnumber);
{ {
// Convert the {value} to a Number first. // Convert the {value} to a Number first.
var_value.Bind(CallBuiltin(Builtins::kNonNumberToNumber, context, value)); var_value = CallBuiltin(Builtins::kNonNumberToNumber, context, value);
Goto(&loop); Goto(&loop);
} }
} }
...@@ -5673,13 +5649,14 @@ Node* CodeStubAssembler::TruncateTaggedToFloat64(Node* context, Node* value) { ...@@ -5673,13 +5649,14 @@ Node* CodeStubAssembler::TruncateTaggedToFloat64(Node* context, Node* value) {
return var_result.value(); return var_result.value();
} }
Node* CodeStubAssembler::TruncateTaggedToWord32(Node* context, Node* value) { TNode<Word32T> CodeStubAssembler::TruncateTaggedToWord32(
SloppyTNode<Context> context, SloppyTNode<Object> value) {
VARIABLE(var_result, MachineRepresentation::kWord32); VARIABLE(var_result, MachineRepresentation::kWord32);
Label done(this); Label done(this);
TaggedToWord32OrBigIntImpl<Object::Conversion::kToNumber>(context, value, TaggedToWord32OrBigIntImpl<Object::Conversion::kToNumber>(context, value,
&done, &var_result); &done, &var_result);
BIND(&done); BIND(&done);
return var_result.value(); return UncheckedCast<Word32T>(var_result.value());
} }
// Truncate {value} to word32 and jump to {if_number} if it is a Number, // Truncate {value} to word32 and jump to {if_number} if it is a Number,
...@@ -10945,7 +10922,7 @@ void CodeStubAssembler::EmitElementStore(Node* object, Node* key, Node* value, ...@@ -10945,7 +10922,7 @@ void CodeStubAssembler::EmitElementStore(Node* object, Node* key, Node* value,
if (IsSmiElementsKind(elements_kind)) { if (IsSmiElementsKind(elements_kind)) {
GotoIfNot(TaggedIsSmi(value), bailout); GotoIfNot(TaggedIsSmi(value), bailout);
} else if (IsDoubleElementsKind(elements_kind)) { } else if (IsDoubleElementsKind(elements_kind)) {
value = TryTaggedToFloat64(value, bailout); value = TryTaggedToFloat64(CAST(value), bailout);
} }
if (IsGrowStoreMode(store_mode) && if (IsGrowStoreMode(store_mode) &&
......
...@@ -2242,9 +2242,12 @@ class V8_EXPORT_PRIVATE CodeStubAssembler ...@@ -2242,9 +2242,12 @@ class V8_EXPORT_PRIVATE CodeStubAssembler
Node* base_allocation_size, Node* base_allocation_size,
Node* allocation_site); Node* allocation_site);
Node* TryTaggedToFloat64(Node* value, Label* if_valueisnotnumber); TNode<Float64T> TryTaggedToFloat64(TNode<Object> value,
Node* TruncateTaggedToFloat64(Node* context, Node* value); Label* if_valueisnotnumber);
Node* TruncateTaggedToWord32(Node* context, Node* value); TNode<Float64T> TruncateTaggedToFloat64(SloppyTNode<Context> context,
SloppyTNode<Object> value);
TNode<Word32T> TruncateTaggedToWord32(SloppyTNode<Context> context,
SloppyTNode<Object> value);
void TaggedToWord32OrBigInt(Node* context, Node* value, Label* if_number, void TaggedToWord32OrBigInt(Node* context, Node* value, Label* if_number,
Variable* var_word32, Label* if_bigint, Variable* var_word32, Label* if_bigint,
Variable* var_bigint); Variable* var_bigint);
......
This diff is collapsed.
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
#ifndef V8_IC_ACCESSOR_ASSEMBLER_H_ #ifndef V8_IC_ACCESSOR_ASSEMBLER_H_
#define V8_IC_ACCESSOR_ASSEMBLER_H_ #define V8_IC_ACCESSOR_ASSEMBLER_H_
#include "src/base/optional.h"
#include "src/codegen/code-stub-assembler.h" #include "src/codegen/code-stub-assembler.h"
namespace v8 { namespace v8 {
...@@ -314,22 +315,24 @@ class V8_EXPORT_PRIVATE AccessorAssembler : public CodeStubAssembler { ...@@ -314,22 +315,24 @@ class V8_EXPORT_PRIVATE AccessorAssembler : public CodeStubAssembler {
TNode<StoreHandler> handler, Label* miss, TNode<StoreHandler> handler, Label* miss,
ICMode ic_mode, ICMode ic_mode,
ElementSupport support_elements); ElementSupport support_elements);
void HandleStoreICSmiHandlerCase(Node* handler_word, Node* holder, void HandleStoreICSmiHandlerCase(SloppyTNode<Word32T> handler_word,
Node* value, Label* miss); SloppyTNode<JSObject> holder,
void HandleStoreFieldAndReturn(Node* handler_word, Node* holder, SloppyTNode<Object> value, Label* miss);
Representation representation, Node* value, void HandleStoreFieldAndReturn(TNode<Word32T> handler_word,
Label* miss); TNode<JSObject> holder, TNode<Object> value,
base::Optional<TNode<Float64T>> double_value,
Representation representation, Label* miss);
void CheckPrototypeValidityCell(TNode<Object> maybe_validity_cell, void CheckPrototypeValidityCell(TNode<Object> maybe_validity_cell,
Label* miss); Label* miss);
void HandleStoreICNativeDataProperty(const StoreICParameters* p, Node* holder, void HandleStoreICNativeDataProperty(const StoreICParameters* p, Node* holder,
Node* handler_word); TNode<Word32T> handler_word);
void HandleStoreToProxy(const StoreICParameters* p, Node* proxy, Label* miss, void HandleStoreToProxy(const StoreICParameters* p, Node* proxy, Label* miss,
ElementSupport support_elements); ElementSupport support_elements);
void HandleStoreAccessor(const StoreICParameters* p, Node* holder, void HandleStoreAccessor(const StoreICParameters* p, Node* holder,
Node* handler_word); TNode<Word32T> handler_word);
// KeyedLoadIC_Generic implementation. // KeyedLoadIC_Generic implementation.
...@@ -355,18 +358,20 @@ class V8_EXPORT_PRIVATE AccessorAssembler : public CodeStubAssembler { ...@@ -355,18 +358,20 @@ class V8_EXPORT_PRIVATE AccessorAssembler : public CodeStubAssembler {
const OnFoundOnReceiver& on_found_on_receiver, const OnFoundOnReceiver& on_found_on_receiver,
Label* miss, ICMode ic_mode); Label* miss, ICMode ic_mode);
Node* PrepareValueForStore(Node* handler_word, Node* holder, void CheckHeapObjectTypeMatchesDescriptor(TNode<Word32T> handler_word,
Representation representation, Node* value, TNode<JSObject> holder,
Label* bailout); TNode<Object> value,
Label* bailout);
// Double fields store double values in a mutable box, where stores are
// writes into this box rather than HeapNumber assignment.
void CheckDescriptorConsidersNumbersMutable(TNode<Word32T> handler_word,
TNode<JSObject> holder,
Label* bailout);
// Extends properties backing store by JSObject::kFieldsAdded elements, // Extends properties backing store by JSObject::kFieldsAdded elements,
// returns updated properties backing store. // returns updated properties backing store.
Node* ExtendPropertiesBackingStore(Node* object, Node* index); Node* ExtendPropertiesBackingStore(Node* object, Node* index);
void StoreNamedField(Node* handler_word, Node* object, bool is_inobject,
Representation representation, Node* value,
Label* bailout);
void EmitFastElementsBoundsCheck(Node* object, Node* elements, void EmitFastElementsBoundsCheck(Node* object, Node* elements,
Node* intptr_index, Node* intptr_index,
Node* is_jsarray_condition, Label* miss); Node* is_jsarray_condition, Label* miss);
......
...@@ -145,29 +145,12 @@ Handle<Smi> StoreHandler::StoreProxy(Isolate* isolate) { ...@@ -145,29 +145,12 @@ Handle<Smi> StoreHandler::StoreProxy(Isolate* isolate) {
Handle<Smi> StoreHandler::StoreField(Isolate* isolate, Kind kind, Handle<Smi> StoreHandler::StoreField(Isolate* isolate, Kind kind,
int descriptor, FieldIndex field_index, int descriptor, FieldIndex field_index,
Representation representation) { Representation representation) {
FieldRepresentation field_rep; DCHECK(!representation.IsNone());
switch (representation.kind()) {
case Representation::kSmi:
field_rep = kSmi;
break;
case Representation::kDouble:
field_rep = kDouble;
break;
case Representation::kHeapObject:
field_rep = kHeapObject;
break;
case Representation::kTagged:
field_rep = kTagged;
break;
default:
UNREACHABLE();
}
DCHECK(kind == kField || kind == kConstField); DCHECK(kind == kField || kind == kConstField);
int config = KindBits::encode(kind) | int config = KindBits::encode(kind) |
IsInobjectBits::encode(field_index.is_inobject()) | IsInobjectBits::encode(field_index.is_inobject()) |
FieldRepresentationBits::encode(field_rep) | RepresentationBits::encode(representation.kind()) |
DescriptorBits::encode(descriptor) | DescriptorBits::encode(descriptor) |
FieldIndexBits::encode(field_index.index()); FieldIndexBits::encode(field_index.index());
return handle(Smi::FromInt(config), isolate); return handle(Smi::FromInt(config), isolate);
......
...@@ -204,8 +204,6 @@ class StoreHandler final : public DataHandler { ...@@ -204,8 +204,6 @@ class StoreHandler final : public DataHandler {
}; };
using KindBits = BitField<Kind, 0, 4>; using KindBits = BitField<Kind, 0, 4>;
enum FieldRepresentation { kSmi, kDouble, kHeapObject, kTagged };
// Applicable to kGlobalProxy, kProxy kinds. // Applicable to kGlobalProxy, kProxy kinds.
// Defines whether access rights check should be done on receiver object. // Defines whether access rights check should be done on receiver object.
...@@ -233,10 +231,10 @@ class StoreHandler final : public DataHandler { ...@@ -233,10 +231,10 @@ class StoreHandler final : public DataHandler {
// Encoding when KindBits contains kField or kTransitionToField. // Encoding when KindBits contains kField or kTransitionToField.
// //
using IsInobjectBits = DescriptorBits::Next<bool, 1>; using IsInobjectBits = DescriptorBits::Next<bool, 1>;
using FieldRepresentationBits = IsInobjectBits::Next<FieldRepresentation, 2>; using RepresentationBits = IsInobjectBits::Next<Representation::Kind, 3>;
// +1 here is to cover all possible JSObject header sizes. // +1 here is to cover all possible JSObject header sizes.
using FieldIndexBits = using FieldIndexBits =
FieldRepresentationBits::Next<unsigned, kDescriptorIndexBitCount + 1>; RepresentationBits::Next<unsigned, kDescriptorIndexBitCount + 1>;
// Make sure we don't overflow the smi. // Make sure we don't overflow the smi.
STATIC_ASSERT(FieldIndexBits::kLastUsedBit < kSmiValueSize); STATIC_ASSERT(FieldIndexBits::kLastUsedBit < kSmiValueSize);
......
...@@ -94,7 +94,7 @@ class KeyedStoreGenericAssembler : public AccessorAssembler { ...@@ -94,7 +94,7 @@ class KeyedStoreGenericAssembler : public AccessorAssembler {
void StoreElementWithCapacity(Node* receiver, TNode<Map> receiver_map, void StoreElementWithCapacity(Node* receiver, TNode<Map> receiver_map,
SloppyTNode<FixedArrayBase> elements, SloppyTNode<FixedArrayBase> elements,
TNode<Word32T> elements_kind, TNode<Word32T> elements_kind,
TNode<IntPtrT> index, Node* value, TNode<IntPtrT> index, SloppyTNode<Object> value,
Node* context, Label* slow, Node* context, Label* slow,
UpdateLength update_length); UpdateLength update_length);
...@@ -306,7 +306,7 @@ void KeyedStoreGenericAssembler::MaybeUpdateLengthAndReturn( ...@@ -306,7 +306,7 @@ void KeyedStoreGenericAssembler::MaybeUpdateLengthAndReturn(
void KeyedStoreGenericAssembler::StoreElementWithCapacity( void KeyedStoreGenericAssembler::StoreElementWithCapacity(
Node* receiver, TNode<Map> receiver_map, Node* receiver, TNode<Map> receiver_map,
SloppyTNode<FixedArrayBase> elements, TNode<Word32T> elements_kind, SloppyTNode<FixedArrayBase> elements, TNode<Word32T> elements_kind,
TNode<IntPtrT> index, Node* value, Node* context, Label* slow, TNode<IntPtrT> index, SloppyTNode<Object> value, Node* context, Label* slow,
UpdateLength update_length) { UpdateLength update_length) {
if (update_length != kDontChangeLength) { if (update_length != kDontChangeLength) {
CSA_ASSERT(this, InstanceTypeEqual(LoadMapInstanceType(receiver_map), CSA_ASSERT(this, InstanceTypeEqual(LoadMapInstanceType(receiver_map),
...@@ -388,7 +388,7 @@ void KeyedStoreGenericAssembler::StoreElementWithCapacity( ...@@ -388,7 +388,7 @@ void KeyedStoreGenericAssembler::StoreElementWithCapacity(
{ {
Label transition_to_double(this), transition_to_object(this); Label transition_to_double(this), transition_to_object(this);
TNode<Context> native_context = LoadNativeContext(context); TNode<Context> native_context = LoadNativeContext(context);
Branch(TaggedEqual(LoadMap(value), HeapNumberMapConstant()), Branch(TaggedEqual(LoadMap(CAST(value)), HeapNumberMapConstant()),
&transition_to_double, &transition_to_object); &transition_to_double, &transition_to_object);
BIND(&transition_to_double); BIND(&transition_to_double);
{ {
...@@ -405,7 +405,7 @@ void KeyedStoreGenericAssembler::StoreElementWithCapacity( ...@@ -405,7 +405,7 @@ void KeyedStoreGenericAssembler::StoreElementWithCapacity(
index, PACKED_DOUBLE_ELEMENTS, INTPTR_PARAMETERS, kHeaderSize); index, PACKED_DOUBLE_ELEMENTS, INTPTR_PARAMETERS, kHeaderSize);
// Make sure we do not store signalling NaNs into double arrays. // Make sure we do not store signalling NaNs into double arrays.
TNode<Float64T> double_value = TNode<Float64T> double_value =
Float64SilenceNaN(LoadHeapNumberValue(value)); Float64SilenceNaN(LoadHeapNumberValue(CAST(value)));
StoreNoWriteBarrier(MachineRepresentation::kFloat64, double_elements, StoreNoWriteBarrier(MachineRepresentation::kFloat64, double_elements,
double_offset, double_value); double_offset, double_value);
MaybeUpdateLengthAndReturn(receiver, index, value, update_length); MaybeUpdateLengthAndReturn(receiver, index, value, update_length);
......
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