Commit 80f7c4a8 authored by Benedikt Meurer's avatar Benedikt Meurer Committed by Commit Bot

[es9] Fix object cloning wrt. MutableHeapNumbers.

Previously the object cloning fast-path had a single loop which would
initialize the object _and_ at the same time clone MutableHeapNumbers.
But since that can trigger GCs, the heap verifier was a bit sad to see
double fields holding undefined values. This was flushed out by the CL
https://chromium-review.googlesource.com/1655291, which changed the GC
timing slightly and thus made the test crash in the verifier.

So instead of the one loop, we now have a second loop that takes care
of cloning any MutableHeapNumbers. This has the advantage that the first
loop can always run without write barriers.

Bug: chromium:964748, chromium:973045, v8:7611, v8:9114, v8:9183, v8:9343
Change-Id: I724a1c1e534243ce9ecde95bf0c07ca26363b515
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1655307
Commit-Queue: Benedikt Meurer <bmeurer@chromium.org>
Auto-Submit: Benedikt Meurer <bmeurer@chromium.org>
Reviewed-by: 's avatarIgor Sheludko <ishell@chromium.org>
Cr-Commit-Position: refs/heads/master@{#62114}
parent 89ad50be
...@@ -3753,8 +3753,6 @@ void AccessorAssembler::GenerateCloneObjectIC() { ...@@ -3753,8 +3753,6 @@ void AccessorAssembler::GenerateCloneObjectIC() {
ReturnIf(IsNullOrUndefined(source), object); ReturnIf(IsNullOrUndefined(source), object);
// Lastly, clone any in-object properties. // Lastly, clone any in-object properties.
// Determine the inobject property capacity of both objects, and copy the
// smaller number into the resulting object.
TNode<IntPtrT> source_start = TNode<IntPtrT> source_start =
LoadMapInobjectPropertiesStartInWords(source_map); LoadMapInobjectPropertiesStartInWords(source_map);
TNode<IntPtrT> source_size = LoadMapInstanceSizeInWords(source_map); TNode<IntPtrT> source_size = LoadMapInstanceSizeInWords(source_map);
...@@ -3763,35 +3761,48 @@ void AccessorAssembler::GenerateCloneObjectIC() { ...@@ -3763,35 +3761,48 @@ void AccessorAssembler::GenerateCloneObjectIC() {
TNode<IntPtrT> field_offset_difference = TNode<IntPtrT> field_offset_difference =
TimesTaggedSize(IntPtrSub(result_start, source_start)); TimesTaggedSize(IntPtrSub(result_start, source_start));
// If MutableHeapNumbers may be present in-object, allocations may occur // Just copy the fields as raw data (pretending that there are no
// within this loop, thus the write barrier is required. // MutableHeapNumbers). This doesn't need write barriers.
//
// TODO(caitp): skip the write barrier until the first MutableHeapNumber
// field is found
const bool may_use_mutable_heap_numbers = !FLAG_unbox_double_fields;
BuildFastLoop( BuildFastLoop(
source_start, source_size, source_start, source_size,
[=](Node* field_index) { [=](Node* field_index) {
TNode<IntPtrT> field_offset = TNode<IntPtrT> field_offset =
TimesTaggedSize(UncheckedCast<IntPtrT>(field_index)); TimesTaggedSize(UncheckedCast<IntPtrT>(field_index));
TNode<IntPtrT> field =
if (may_use_mutable_heap_numbers) { LoadObjectField<IntPtrT>(CAST(source), field_offset);
TNode<Object> field = LoadObjectField(CAST(source), field_offset); TNode<IntPtrT> result_offset =
field = CloneIfMutablePrimitive(field); IntPtrAdd(field_offset, field_offset_difference);
TNode<IntPtrT> result_offset = StoreObjectFieldNoWriteBarrier(object, result_offset, field);
IntPtrAdd(field_offset, field_offset_difference);
StoreObjectField(object, result_offset, field);
} else {
// Copy fields as raw data.
TNode<IntPtrT> field =
LoadObjectField<IntPtrT>(CAST(source), field_offset);
TNode<IntPtrT> result_offset =
IntPtrAdd(field_offset, field_offset_difference);
StoreObjectFieldNoWriteBarrier(object, result_offset, field);
}
}, },
1, INTPTR_PARAMETERS, IndexAdvanceMode::kPost); 1, INTPTR_PARAMETERS, IndexAdvanceMode::kPost);
// If MutableHeapNumbers can occur, we need to go through the {object}
// again here and properly clone them. We use a second loop here to
// ensure that the GC (and heap verifier) always sees properly initialized
// objects, i.e. never hits undefined values in double fields.
if (!FLAG_unbox_double_fields) {
BuildFastLoop(
result_start, source_size,
[=](Node* field_index) {
TNode<IntPtrT> result_offset =
TimesTaggedSize(UncheckedCast<IntPtrT>(field_index));
TNode<Object> field = LoadObjectField(object, result_offset);
Label if_done(this), if_mutableheapnumber(this, Label::kDeferred);
GotoIf(TaggedIsSmi(field), &if_done);
Branch(IsMutableHeapNumber(CAST(field)), &if_mutableheapnumber,
&if_done);
BIND(&if_mutableheapnumber);
{
TNode<Object> value = AllocateMutableHeapNumberWithValue(
LoadHeapNumberValue(UncheckedCast<HeapNumber>(field)));
StoreObjectField(object, result_offset, value);
Goto(&if_done);
}
BIND(&if_done);
},
1, INTPTR_PARAMETERS, IndexAdvanceMode::kPost);
}
Return(object); Return(object);
} }
......
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