Revert "[runtime] Add length check in ConvertElementsWithCapacity"
This reverts commit b271648e94
.
Reason for revert: New test fails: https://ci.chromium.org/ui/p/v8/builders/ci/V8%20Linux%20-%20shared/42282/overview
Original change's description:
> [runtime] Add length check in ConvertElementsWithCapacity
>
> This also propagates the exception through all the users of
> ConvertElementsWithCapacity.
>
> Bug: chromium:1201626
> Change-Id: Ie44ba4327a4c3a20f1376477f45d3cd95d0da3b3
> Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2857961
> Commit-Queue: Victor Gomes <victorgomes@chromium.org>
> Reviewed-by: Toon Verwaest <verwaest@chromium.org>
> Cr-Commit-Position: refs/heads/master@{#74412}
Bug: chromium:1201626
Change-Id: I764256e9d0dcc69ea3a2f3c77afaca73a910bb66
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2876861
Auto-Submit: Clemens Backes <clemensb@chromium.org>
Commit-Queue: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Bot-Commit: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Cr-Commit-Position: refs/heads/master@{#74414}
This commit is contained in:
parent
519c82ce36
commit
5bdfd84e17
@ -203,12 +203,7 @@ void Accessors::ArrayLengthSetter(
|
||||
return;
|
||||
}
|
||||
|
||||
if (JSArray::SetLength(array, length).IsNothing()) {
|
||||
// TODO(victorgomes): AccessorNameBooleanSetterCallback does not handle
|
||||
// exceptions.
|
||||
FATAL("Fatal JavaScript invalid array length %u", length);
|
||||
UNREACHABLE();
|
||||
}
|
||||
JSArray::SetLength(array, length);
|
||||
|
||||
uint32_t actual_new_len = 0;
|
||||
CHECK(array->length().ToArrayLength(&actual_new_len));
|
||||
|
@ -173,8 +173,7 @@ V8_WARN_UNUSED_RESULT MaybeHandle<Object> SetLengthProperty(
|
||||
Handle<JSArray> array = Handle<JSArray>::cast(receiver);
|
||||
if (!JSArray::HasReadOnlyLength(array)) {
|
||||
DCHECK_LE(length, kMaxUInt32);
|
||||
MAYBE_RETURN_NULL(
|
||||
JSArray::SetLength(array, static_cast<uint32_t>(length)));
|
||||
JSArray::SetLength(array, static_cast<uint32_t>(length));
|
||||
return receiver;
|
||||
}
|
||||
}
|
||||
@ -386,9 +385,7 @@ BUILTIN(ArrayPush) {
|
||||
}
|
||||
|
||||
ElementsAccessor* accessor = array->GetElementsAccessor();
|
||||
uint32_t new_length;
|
||||
MAYBE_ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
|
||||
isolate, new_length, accessor->Push(array, &args, to_add));
|
||||
uint32_t new_length = accessor->Push(array, &args, to_add);
|
||||
return *isolate->factory()->NewNumberFromUint((new_length));
|
||||
}
|
||||
|
||||
@ -471,8 +468,7 @@ BUILTIN(ArrayPop) {
|
||||
Handle<Object> result;
|
||||
if (IsJSArrayFastElementMovingAllowed(isolate, JSArray::cast(*receiver))) {
|
||||
// Fast Elements Path
|
||||
ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
|
||||
isolate, result, array->GetElementsAccessor()->Pop(array));
|
||||
result = array->GetElementsAccessor()->Pop(array);
|
||||
} else {
|
||||
// Use Slow Lookup otherwise
|
||||
uint32_t new_length = len - 1;
|
||||
@ -487,9 +483,7 @@ BUILTIN(ArrayPop) {
|
||||
isolate->factory()->length_string(),
|
||||
Object::TypeOf(isolate, array), array));
|
||||
}
|
||||
bool set_len_ok;
|
||||
MAYBE_ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
|
||||
isolate, set_len_ok, JSArray::SetLength(array, new_length));
|
||||
JSArray::SetLength(array, new_length);
|
||||
}
|
||||
|
||||
return *result;
|
||||
@ -601,8 +595,7 @@ BUILTIN(ArrayShift) {
|
||||
|
||||
if (CanUseFastArrayShift(isolate, receiver)) {
|
||||
Handle<JSArray> array = Handle<JSArray>::cast(receiver);
|
||||
RETURN_RESULT_OR_FAILURE(isolate,
|
||||
array->GetElementsAccessor()->Shift(array));
|
||||
return *array->GetElementsAccessor()->Shift(array);
|
||||
}
|
||||
|
||||
return GenericArrayShift(isolate, receiver, length);
|
||||
@ -630,9 +623,7 @@ BUILTIN(ArrayUnshift) {
|
||||
DCHECK(!JSArray::HasReadOnlyLength(array));
|
||||
|
||||
ElementsAccessor* accessor = array->GetElementsAccessor();
|
||||
uint32_t new_length;
|
||||
MAYBE_ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
|
||||
isolate, new_length, accessor->Unshift(array, &args, to_add));
|
||||
int new_length = accessor->Unshift(array, &args, to_add);
|
||||
return Smi::FromInt(new_length);
|
||||
}
|
||||
|
||||
|
@ -154,8 +154,7 @@ BUILTIN(TypedArrayPrototypeFill) {
|
||||
DCHECK_LE(end, len);
|
||||
DCHECK_LE(count, len);
|
||||
|
||||
RETURN_RESULT_OR_FAILURE(isolate, ElementsAccessor::ForKind(kind)->Fill(
|
||||
array, obj_value, start, end));
|
||||
return ElementsAccessor::ForKind(kind)->Fill(array, obj_value, start, end);
|
||||
}
|
||||
|
||||
BUILTIN(TypedArrayPrototypeIncludes) {
|
||||
|
@ -639,64 +639,57 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
Maybe<bool> Add(Handle<JSObject> object, uint32_t index, Handle<Object> value,
|
||||
PropertyAttributes attributes, uint32_t new_capacity) final {
|
||||
return Subclass::AddImpl(object, index, value, attributes, new_capacity);
|
||||
void Add(Handle<JSObject> object, uint32_t index, Handle<Object> value,
|
||||
PropertyAttributes attributes, uint32_t new_capacity) final {
|
||||
Subclass::AddImpl(object, index, value, attributes, new_capacity);
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
Maybe<uint32_t> Push(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) final {
|
||||
uint32_t Push(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) final {
|
||||
return Subclass::PushImpl(receiver, args, push_size);
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> PushImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args, uint32_t push_sized) {
|
||||
static uint32_t PushImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_sized) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
Maybe<uint32_t> Unshift(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t unshift_size) final {
|
||||
uint32_t Unshift(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t unshift_size) final {
|
||||
return Subclass::UnshiftImpl(receiver, args, unshift_size);
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> UnshiftImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args,
|
||||
uint32_t unshift_size) {
|
||||
static uint32_t UnshiftImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t unshift_size) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
MaybeHandle<Object> Pop(Handle<JSArray> receiver) final {
|
||||
Handle<Object> Pop(Handle<JSArray> receiver) final {
|
||||
return Subclass::PopImpl(receiver);
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> PopImpl(Handle<JSArray> receiver) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
static Handle<Object> PopImpl(Handle<JSArray> receiver) { UNREACHABLE(); }
|
||||
|
||||
MaybeHandle<Object> Shift(Handle<JSArray> receiver) final {
|
||||
Handle<Object> Shift(Handle<JSArray> receiver) final {
|
||||
return Subclass::ShiftImpl(receiver);
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> ShiftImpl(Handle<JSArray> receiver) {
|
||||
UNREACHABLE();
|
||||
static Handle<Object> ShiftImpl(Handle<JSArray> receiver) { UNREACHABLE(); }
|
||||
|
||||
void SetLength(Handle<JSArray> array, uint32_t length) final {
|
||||
Subclass::SetLengthImpl(array->GetIsolate(), array, length,
|
||||
handle(array->elements(), array->GetIsolate()));
|
||||
}
|
||||
|
||||
Maybe<bool> SetLength(Handle<JSArray> array, uint32_t length) final {
|
||||
return Subclass::SetLengthImpl(
|
||||
array->GetIsolate(), array, length,
|
||||
handle(array->elements(), array->GetIsolate()));
|
||||
}
|
||||
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
DCHECK(!array->SetLengthWouldNormalize(length));
|
||||
DCHECK(IsFastElementsKind(array->GetElementsKind()));
|
||||
uint32_t old_length = 0;
|
||||
@ -742,13 +735,11 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
} else {
|
||||
// Check whether the backing store should be expanded.
|
||||
capacity = std::max(length, JSObject::NewElementsCapacity(capacity));
|
||||
MAYBE_RETURN(Subclass::GrowCapacityAndConvertImpl(array, capacity),
|
||||
Nothing<bool>());
|
||||
Subclass::GrowCapacityAndConvertImpl(array, capacity);
|
||||
}
|
||||
|
||||
array->set_length(Smi::FromInt(length));
|
||||
JSObject::ValidateElements(*array);
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
size_t NumberOfElements(JSObject receiver) final {
|
||||
@ -774,40 +765,22 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
return Subclass::GetMaxIndex(receiver, elements);
|
||||
}
|
||||
|
||||
static MaybeHandle<FixedArrayBase> ConvertElementsWithCapacity(
|
||||
static Handle<FixedArrayBase> ConvertElementsWithCapacity(
|
||||
Handle<JSObject> object, Handle<FixedArrayBase> old_elements,
|
||||
ElementsKind from_kind, uint32_t capacity) {
|
||||
return ConvertElementsWithCapacity(object, old_elements, from_kind,
|
||||
capacity, 0, 0);
|
||||
}
|
||||
|
||||
static MaybeHandle<FixedArrayBase> ConvertElementsWithCapacity(
|
||||
static Handle<FixedArrayBase> ConvertElementsWithCapacity(
|
||||
Handle<JSObject> object, Handle<FixedArrayBase> old_elements,
|
||||
ElementsKind from_kind, uint32_t capacity, uint32_t src_index,
|
||||
uint32_t dst_index) {
|
||||
Isolate* isolate = object->GetIsolate();
|
||||
Handle<FixedArrayBase> new_elements;
|
||||
// TODO(victorgomes): Retrieve native context in optimized code
|
||||
// and remove the fatal errors.
|
||||
if (IsDoubleElementsKind(kind())) {
|
||||
if (capacity < 0 || capacity > FixedDoubleArray::kMaxLength) {
|
||||
if (isolate->context().is_null()) {
|
||||
FATAL("Fatal JavaScript invalid array length");
|
||||
UNREACHABLE();
|
||||
}
|
||||
return isolate->Throw<FixedArrayBase>(isolate->factory()->NewRangeError(
|
||||
MessageTemplate::kInvalidArrayLength));
|
||||
}
|
||||
new_elements = isolate->factory()->NewFixedDoubleArray(capacity);
|
||||
} else {
|
||||
if (capacity < 0 || capacity > FixedArray::kMaxLength) {
|
||||
if (isolate->context().is_null()) {
|
||||
FATAL("Fatal JavaScript invalid array length");
|
||||
UNREACHABLE();
|
||||
}
|
||||
return isolate->Throw<FixedArrayBase>(isolate->factory()->NewRangeError(
|
||||
MessageTemplate::kInvalidArrayLength));
|
||||
}
|
||||
new_elements = isolate->factory()->NewFixedArray(capacity);
|
||||
}
|
||||
|
||||
@ -820,11 +793,11 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
from_kind, dst_index, packed_size,
|
||||
kCopyToEndAndInitializeToHole);
|
||||
|
||||
return MaybeHandle<FixedArrayBase>(new_elements);
|
||||
return new_elements;
|
||||
}
|
||||
|
||||
static Maybe<bool> TransitionElementsKindImpl(Handle<JSObject> object,
|
||||
Handle<Map> to_map) {
|
||||
static void TransitionElementsKindImpl(Handle<JSObject> object,
|
||||
Handle<Map> to_map) {
|
||||
Isolate* isolate = object->GetIsolate();
|
||||
Handle<Map> from_map = handle(object->map(), isolate);
|
||||
ElementsKind from_kind = from_map->elements_kind();
|
||||
@ -849,12 +822,8 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
(IsSmiElementsKind(from_kind) && IsDoubleElementsKind(to_kind)) ||
|
||||
(IsDoubleElementsKind(from_kind) && IsObjectElementsKind(to_kind)));
|
||||
uint32_t capacity = static_cast<uint32_t>(object->elements().length());
|
||||
Handle<FixedArrayBase> elements;
|
||||
ASSIGN_RETURN_ON_EXCEPTION_VALUE(
|
||||
object->GetIsolate(), elements,
|
||||
ConvertElementsWithCapacity(object, from_elements, from_kind,
|
||||
capacity),
|
||||
Nothing<bool>());
|
||||
Handle<FixedArrayBase> elements = ConvertElementsWithCapacity(
|
||||
object, from_elements, from_kind, capacity);
|
||||
JSObject::SetMapAndElements(object, to_map, elements);
|
||||
}
|
||||
if (FLAG_trace_elements_transitions) {
|
||||
@ -863,11 +832,10 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
handle(object->elements(), isolate));
|
||||
}
|
||||
}
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static Maybe<bool> GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
static void GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
ElementsKind from_kind = object->GetElementsKind();
|
||||
if (IsSmiOrObjectElementsKind(from_kind)) {
|
||||
// Array optimizations rely on the prototype lookups of Array objects
|
||||
@ -882,18 +850,15 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
DCHECK(IsDoubleElementsKind(from_kind) != IsDoubleElementsKind(kind()) ||
|
||||
IsDictionaryElementsKind(from_kind) ||
|
||||
static_cast<uint32_t>(old_elements->length()) < capacity);
|
||||
return Subclass::BasicGrowCapacityAndConvertImpl(
|
||||
object, old_elements, from_kind, kind(), capacity);
|
||||
Subclass::BasicGrowCapacityAndConvertImpl(object, old_elements, from_kind,
|
||||
kind(), capacity);
|
||||
}
|
||||
|
||||
static Maybe<bool> BasicGrowCapacityAndConvertImpl(
|
||||
static void BasicGrowCapacityAndConvertImpl(
|
||||
Handle<JSObject> object, Handle<FixedArrayBase> old_elements,
|
||||
ElementsKind from_kind, ElementsKind to_kind, uint32_t capacity) {
|
||||
Handle<FixedArrayBase> elements;
|
||||
ASSIGN_RETURN_ON_EXCEPTION_VALUE(
|
||||
object->GetIsolate(), elements,
|
||||
ConvertElementsWithCapacity(object, old_elements, from_kind, capacity),
|
||||
Nothing<bool>());
|
||||
Handle<FixedArrayBase> elements =
|
||||
ConvertElementsWithCapacity(object, old_elements, from_kind, capacity);
|
||||
|
||||
if (IsHoleyElementsKind(from_kind)) {
|
||||
to_kind = GetHoleyElementsKind(to_kind);
|
||||
@ -908,45 +873,40 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
JSObject::PrintElementsTransition(stdout, object, from_kind, old_elements,
|
||||
to_kind, elements);
|
||||
}
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
Maybe<bool> TransitionElementsKind(Handle<JSObject> object,
|
||||
Handle<Map> map) final {
|
||||
return Subclass::TransitionElementsKindImpl(object, map);
|
||||
void TransitionElementsKind(Handle<JSObject> object, Handle<Map> map) final {
|
||||
Subclass::TransitionElementsKindImpl(object, map);
|
||||
}
|
||||
|
||||
Maybe<bool> GrowCapacityAndConvert(Handle<JSObject> object,
|
||||
uint32_t capacity) final {
|
||||
return Subclass::GrowCapacityAndConvertImpl(object, capacity);
|
||||
void GrowCapacityAndConvert(Handle<JSObject> object,
|
||||
uint32_t capacity) final {
|
||||
Subclass::GrowCapacityAndConvertImpl(object, capacity);
|
||||
}
|
||||
|
||||
Maybe<bool> GrowCapacity(Handle<JSObject> object, uint32_t index) final {
|
||||
bool GrowCapacity(Handle<JSObject> object, uint32_t index) final {
|
||||
// This function is intended to be called from optimized code. We don't
|
||||
// want to trigger lazy deopts there, so refuse to handle cases that would.
|
||||
if (object->map().is_prototype_map() ||
|
||||
object->WouldConvertToSlowElements(index)) {
|
||||
return Just(false);
|
||||
return false;
|
||||
}
|
||||
Handle<FixedArrayBase> old_elements(object->elements(),
|
||||
object->GetIsolate());
|
||||
uint32_t new_capacity = JSObject::NewElementsCapacity(index + 1);
|
||||
DCHECK(static_cast<uint32_t>(old_elements->length()) < new_capacity);
|
||||
Handle<FixedArrayBase> elements;
|
||||
ASSIGN_RETURN_ON_EXCEPTION_VALUE(
|
||||
object->GetIsolate(), elements,
|
||||
ConvertElementsWithCapacity(object, old_elements, kind(), new_capacity),
|
||||
Nothing<bool>());
|
||||
Handle<FixedArrayBase> elements =
|
||||
ConvertElementsWithCapacity(object, old_elements, kind(), new_capacity);
|
||||
|
||||
DCHECK_EQ(object->GetElementsKind(), kind());
|
||||
// Transition through the allocation site as well if present.
|
||||
if (JSObject::UpdateAllocationSite<AllocationSiteUpdateMode::kCheckOnly>(
|
||||
object, kind())) {
|
||||
return Just(false);
|
||||
return false;
|
||||
}
|
||||
|
||||
object->set_elements(*elements);
|
||||
return Just(true);
|
||||
return true;
|
||||
}
|
||||
|
||||
void Delete(Handle<JSObject> obj, InternalIndex entry) final {
|
||||
@ -1260,14 +1220,13 @@ class ElementsAccessorBase : public InternalElementsAccessor {
|
||||
return Subclass::GetCapacityImpl(holder, backing_store);
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> FillImpl(Handle<JSObject> receiver,
|
||||
Handle<Object> obj_value, size_t start,
|
||||
size_t end) {
|
||||
static Object FillImpl(Handle<JSObject> receiver, Handle<Object> obj_value,
|
||||
size_t start, size_t end) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
MaybeHandle<Object> Fill(Handle<JSObject> receiver, Handle<Object> obj_value,
|
||||
size_t start, size_t end) override {
|
||||
Object Fill(Handle<JSObject> receiver, Handle<Object> obj_value, size_t start,
|
||||
size_t end) override {
|
||||
return Subclass::FillImpl(receiver, obj_value, start, end);
|
||||
}
|
||||
|
||||
@ -1389,9 +1348,9 @@ class DictionaryElementsAccessor
|
||||
return dict.NumberOfElements();
|
||||
}
|
||||
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
Handle<NumberDictionary> dict =
|
||||
Handle<NumberDictionary>::cast(backing_store);
|
||||
uint32_t old_length = 0;
|
||||
@ -1442,7 +1401,6 @@ class DictionaryElementsAccessor
|
||||
|
||||
Handle<Object> length_obj = isolate->factory()->NewNumberFromUint(length);
|
||||
array->set_length(*length_obj);
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static void CopyElementsImpl(Isolate* isolate, FixedArrayBase from,
|
||||
@ -1508,10 +1466,9 @@ class DictionaryElementsAccessor
|
||||
dictionary.DetailsAtPut(entry, details);
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
PropertyDetails details(kData, attributes, PropertyCellType::kNoCell);
|
||||
Handle<NumberDictionary> dictionary =
|
||||
object->HasFastElements() || object->HasFastStringWrapperElements()
|
||||
@ -1522,9 +1479,8 @@ class DictionaryElementsAccessor
|
||||
object->GetIsolate(), dictionary, index, value, details);
|
||||
new_dictionary->UpdateMaxNumberKey(index, object);
|
||||
if (attributes != NONE) object->RequireSlowElements(*new_dictionary);
|
||||
if (dictionary.is_identical_to(new_dictionary)) return Just(true);
|
||||
if (dictionary.is_identical_to(new_dictionary)) return;
|
||||
object->set_elements(*new_dictionary);
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static bool HasEntryImpl(Isolate* isolate, FixedArrayBase store,
|
||||
@ -2013,10 +1969,9 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
value, attributes);
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
DCHECK_EQ(NONE, attributes);
|
||||
ElementsKind from_kind = object->GetElementsKind();
|
||||
ElementsKind to_kind = Subclass::kind();
|
||||
@ -2024,8 +1979,7 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
IsDoubleElementsKind(from_kind) != IsDoubleElementsKind(to_kind) ||
|
||||
Subclass::GetCapacityImpl(*object, object->elements()) !=
|
||||
new_capacity) {
|
||||
MAYBE_RETURN(Subclass::GrowCapacityAndConvertImpl(object, new_capacity),
|
||||
Nothing<bool>());
|
||||
Subclass::GrowCapacityAndConvertImpl(object, new_capacity);
|
||||
} else {
|
||||
if (IsFastElementsKind(from_kind) && from_kind != to_kind) {
|
||||
JSObject::TransitionElementsKind(object, to_kind);
|
||||
@ -2036,7 +1990,6 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
}
|
||||
}
|
||||
Subclass::SetImpl(object, InternalIndex(index), *value);
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static void DeleteImpl(Handle<JSObject> obj, InternalIndex entry) {
|
||||
@ -2131,25 +2084,24 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
#endif
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> PopImpl(Handle<JSArray> receiver) {
|
||||
static Handle<Object> PopImpl(Handle<JSArray> receiver) {
|
||||
return Subclass::RemoveElement(receiver, AT_END);
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> ShiftImpl(Handle<JSArray> receiver) {
|
||||
static Handle<Object> ShiftImpl(Handle<JSArray> receiver) {
|
||||
return Subclass::RemoveElement(receiver, AT_START);
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> PushImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args, uint32_t push_size) {
|
||||
static uint32_t PushImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) {
|
||||
Handle<FixedArrayBase> backing_store(receiver->elements(),
|
||||
receiver->GetIsolate());
|
||||
return Subclass::AddArguments(receiver, backing_store, args, push_size,
|
||||
AT_END);
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> UnshiftImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args,
|
||||
uint32_t unshift_size) {
|
||||
static uint32_t UnshiftImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t unshift_size) {
|
||||
Handle<FixedArrayBase> backing_store(receiver->elements(),
|
||||
receiver->GetIsolate());
|
||||
return Subclass::AddArguments(receiver, backing_store, args, unshift_size,
|
||||
@ -2183,9 +2135,8 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
}
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> FillImpl(Handle<JSObject> receiver,
|
||||
Handle<Object> obj_value, size_t start,
|
||||
size_t end) {
|
||||
static Object FillImpl(Handle<JSObject> receiver, Handle<Object> obj_value,
|
||||
size_t start, size_t end) {
|
||||
// Ensure indexes are within array bounds
|
||||
DCHECK_LE(0, start);
|
||||
DCHECK_LE(start, end);
|
||||
@ -2198,8 +2149,8 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
// Make sure we have enough space.
|
||||
DCHECK_LE(end, std::numeric_limits<uint32_t>::max());
|
||||
if (end > Subclass::GetCapacityImpl(*receiver, receiver->elements())) {
|
||||
MAYBE_RETURN_NULL(Subclass::GrowCapacityAndConvertImpl(
|
||||
receiver, static_cast<uint32_t>(end)));
|
||||
Subclass::GrowCapacityAndConvertImpl(receiver,
|
||||
static_cast<uint32_t>(end));
|
||||
CHECK_EQ(Subclass::kind(), receiver->GetElementsKind());
|
||||
}
|
||||
DCHECK_LE(end, Subclass::GetCapacityImpl(*receiver, receiver->elements()));
|
||||
@ -2207,7 +2158,7 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
for (size_t index = start; index < end; ++index) {
|
||||
Subclass::SetImpl(receiver, InternalIndex(index), *obj_value);
|
||||
}
|
||||
return MaybeHandle<Object>(receiver);
|
||||
return *receiver;
|
||||
}
|
||||
|
||||
static Maybe<bool> IncludesValueImpl(Isolate* isolate,
|
||||
@ -2369,8 +2320,8 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
return result;
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> RemoveElement(Handle<JSArray> receiver,
|
||||
Where remove_position) {
|
||||
static Handle<Object> RemoveElement(Handle<JSArray> receiver,
|
||||
Where remove_position) {
|
||||
Isolate* isolate = receiver->GetIsolate();
|
||||
ElementsKind kind = KindTraits::Kind;
|
||||
if (IsSmiOrObjectElementsKind(kind)) {
|
||||
@ -2388,26 +2339,24 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
Subclass::MoveElements(isolate, receiver, backing_store, 0, 1, new_length,
|
||||
0, 0);
|
||||
}
|
||||
MAYBE_RETURN_NULL(
|
||||
Subclass::SetLengthImpl(isolate, receiver, new_length, backing_store));
|
||||
Subclass::SetLengthImpl(isolate, receiver, new_length, backing_store);
|
||||
|
||||
if (IsHoleyElementsKind(kind) && result->IsTheHole(isolate)) {
|
||||
return isolate->factory()->undefined_value();
|
||||
}
|
||||
return MaybeHandle<Object>(result);
|
||||
return result;
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> AddArguments(Handle<JSArray> receiver,
|
||||
Handle<FixedArrayBase> backing_store,
|
||||
BuiltinArguments* args, uint32_t add_size,
|
||||
Where add_position) {
|
||||
static uint32_t AddArguments(Handle<JSArray> receiver,
|
||||
Handle<FixedArrayBase> backing_store,
|
||||
BuiltinArguments* args, uint32_t add_size,
|
||||
Where add_position) {
|
||||
uint32_t length = Smi::ToInt(receiver->length());
|
||||
DCHECK_LT(0, add_size);
|
||||
uint32_t elms_len = backing_store->length();
|
||||
// Check we do not overflow the new_length.
|
||||
DCHECK(add_size <= static_cast<uint32_t>(Smi::kMaxValue - length));
|
||||
uint32_t new_length = length + add_size;
|
||||
Isolate* isolate = receiver->GetIsolate();
|
||||
|
||||
if (new_length > elms_len) {
|
||||
// New backing storage is needed.
|
||||
@ -2415,16 +2364,14 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
// If we add arguments to the start we have to shift the existing objects.
|
||||
int copy_dst_index = add_position == AT_START ? add_size : 0;
|
||||
// Copy over all objects to a new backing_store.
|
||||
ASSIGN_RETURN_ON_EXCEPTION_VALUE(
|
||||
isolate, backing_store,
|
||||
Subclass::ConvertElementsWithCapacity(receiver, backing_store,
|
||||
KindTraits::Kind, capacity, 0,
|
||||
copy_dst_index),
|
||||
Nothing<uint32_t>());
|
||||
backing_store = Subclass::ConvertElementsWithCapacity(
|
||||
receiver, backing_store, KindTraits::Kind, capacity, 0,
|
||||
copy_dst_index);
|
||||
receiver->set_elements(*backing_store);
|
||||
} else if (add_position == AT_START) {
|
||||
// If the backing store has enough capacity and we add elements to the
|
||||
// start we have to shift the existing objects.
|
||||
Isolate* isolate = receiver->GetIsolate();
|
||||
Subclass::MoveElements(isolate, receiver, backing_store, add_size, 0,
|
||||
length, 0, 0);
|
||||
}
|
||||
@ -2434,7 +2381,7 @@ class FastElementsAccessor : public ElementsAccessorBase<Subclass, KindTraits> {
|
||||
Subclass::CopyArguments(args, backing_store, add_size, 1, insertion_index);
|
||||
// Set the length.
|
||||
receiver->set_length(Smi::FromInt(new_length));
|
||||
return Just(new_length);
|
||||
return new_length;
|
||||
}
|
||||
|
||||
static void CopyArguments(BuiltinArguments* args,
|
||||
@ -2620,29 +2567,28 @@ class FastNonextensibleObjectElementsAccessor
|
||||
public:
|
||||
using BackingStore = typename KindTraits::BackingStore;
|
||||
|
||||
static Maybe<uint32_t> PushImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args, uint32_t push_size) {
|
||||
static uint32_t PushImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
// TODO(duongn): refactor this due to code duplication of sealed version.
|
||||
// Consider using JSObject::NormalizeElements(). Also consider follow the fast
|
||||
// element logic instead of changing to dictionary mode.
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
uint32_t old_length = 0;
|
||||
CHECK(array->length().ToArrayIndex(&old_length));
|
||||
if (length == old_length) {
|
||||
// Do nothing.
|
||||
return Just(true);
|
||||
return;
|
||||
}
|
||||
|
||||
// Transition to DICTIONARY_ELEMENTS.
|
||||
@ -2674,8 +2620,8 @@ class FastNonextensibleObjectElementsAccessor
|
||||
|
||||
// Set length.
|
||||
Handle<FixedArrayBase> new_backing_store(array->elements(), isolate);
|
||||
return DictionaryElementsAccessor::SetLengthImpl(isolate, array, length,
|
||||
new_backing_store);
|
||||
DictionaryElementsAccessor::SetLengthImpl(isolate, array, length,
|
||||
new_backing_store);
|
||||
}
|
||||
};
|
||||
|
||||
@ -2714,33 +2660,30 @@ class FastSealedObjectElementsAccessor
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> PopImpl(Handle<JSArray> receiver) {
|
||||
static Handle<Object> PopImpl(Handle<JSArray> receiver) { UNREACHABLE(); }
|
||||
|
||||
static uint32_t PushImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> PushImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args, uint32_t push_size) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
// TODO(duongn): refactor this due to code duplication of nonextensible
|
||||
// version. Consider using JSObject::NormalizeElements(). Also consider follow
|
||||
// the fast element logic instead of changing to dictionary mode.
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
uint32_t old_length = 0;
|
||||
CHECK(array->length().ToArrayIndex(&old_length));
|
||||
if (length == old_length) {
|
||||
// Do nothing.
|
||||
return Just(true);
|
||||
return;
|
||||
}
|
||||
|
||||
// Transition to DICTIONARY_ELEMENTS.
|
||||
@ -2772,8 +2715,8 @@ class FastSealedObjectElementsAccessor
|
||||
|
||||
// Set length
|
||||
Handle<FixedArrayBase> new_backing_store(array->elements(), isolate);
|
||||
return DictionaryElementsAccessor::SetLengthImpl(isolate, array, length,
|
||||
new_backing_store);
|
||||
DictionaryElementsAccessor::SetLengthImpl(isolate, array, length,
|
||||
new_backing_store);
|
||||
}
|
||||
};
|
||||
|
||||
@ -2827,25 +2770,22 @@ class FastFrozenObjectElementsAccessor
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> PopImpl(Handle<JSArray> receiver) {
|
||||
static Handle<Object> PopImpl(Handle<JSArray> receiver) { UNREACHABLE(); }
|
||||
|
||||
static uint32_t PushImpl(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<uint32_t> PushImpl(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args, uint32_t push_size) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
@ -3183,9 +3123,9 @@ class TypedElementsAccessor
|
||||
return false;
|
||||
}
|
||||
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> backing_store) {
|
||||
// External arrays do not support changing their length.
|
||||
UNREACHABLE();
|
||||
}
|
||||
@ -3252,9 +3192,8 @@ class TypedElementsAccessor
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static MaybeHandle<Object> FillImpl(Handle<JSObject> receiver,
|
||||
Handle<Object> value, size_t start,
|
||||
size_t end) {
|
||||
static Object FillImpl(Handle<JSObject> receiver, Handle<Object> value,
|
||||
size_t start, size_t end) {
|
||||
Handle<JSTypedArray> typed_array = Handle<JSTypedArray>::cast(receiver);
|
||||
DCHECK(!typed_array->WasDetached());
|
||||
DCHECK_LE(start, end);
|
||||
@ -3269,7 +3208,7 @@ class TypedElementsAccessor
|
||||
} else {
|
||||
std::fill(data + start, data + end, scalar);
|
||||
}
|
||||
return MaybeHandle<Object>(typed_array);
|
||||
return *typed_array;
|
||||
}
|
||||
|
||||
static Maybe<bool> IncludesValueImpl(Isolate* isolate,
|
||||
@ -4006,13 +3945,13 @@ class SloppyArgumentsElementsAccessor
|
||||
}
|
||||
}
|
||||
|
||||
static Maybe<bool> TransitionElementsKindImpl(Handle<JSObject> object,
|
||||
Handle<Map> map) {
|
||||
static void TransitionElementsKindImpl(Handle<JSObject> object,
|
||||
Handle<Map> map) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
static Maybe<bool> GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
static void GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
UNREACHABLE();
|
||||
}
|
||||
|
||||
@ -4051,9 +3990,9 @@ class SloppyArgumentsElementsAccessor
|
||||
}
|
||||
}
|
||||
|
||||
static Maybe<bool> SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> parameter_map) {
|
||||
static void SetLengthImpl(Isolate* isolate, Handle<JSArray> array,
|
||||
uint32_t length,
|
||||
Handle<FixedArrayBase> parameter_map) {
|
||||
// Sloppy arguments objects are not arrays.
|
||||
UNREACHABLE();
|
||||
}
|
||||
@ -4349,10 +4288,9 @@ class SlowSloppyArgumentsElementsAccessor
|
||||
NumberDictionary::DeleteEntry(isolate, dict, entry.adjust_down(length));
|
||||
elements->set_arguments(*dict);
|
||||
}
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
Isolate* isolate = object->GetIsolate();
|
||||
Handle<SloppyArgumentsElements> elements(
|
||||
SloppyArgumentsElements::cast(object->elements()), isolate);
|
||||
@ -4369,7 +4307,6 @@ class SlowSloppyArgumentsElementsAccessor
|
||||
if (*dictionary != *new_dictionary) {
|
||||
elements->set_arguments(*new_dictionary);
|
||||
}
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static void ReconfigureImpl(Handle<JSObject> object,
|
||||
@ -4465,10 +4402,9 @@ class FastSloppyArgumentsElementsAccessor
|
||||
SlowSloppyArgumentsElementsAccessor::SloppyDeleteImpl(obj, elements, entry);
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
DCHECK_EQ(NONE, attributes);
|
||||
Isolate* isolate = object->GetIsolate();
|
||||
Handle<SloppyArgumentsElements> elements(
|
||||
@ -4476,8 +4412,7 @@ class FastSloppyArgumentsElementsAccessor
|
||||
Handle<FixedArray> old_arguments(elements->arguments(), isolate);
|
||||
if (old_arguments->IsNumberDictionary() ||
|
||||
static_cast<uint32_t>(old_arguments->length()) < new_capacity) {
|
||||
MAYBE_RETURN(GrowCapacityAndConvertImpl(object, new_capacity),
|
||||
Nothing<bool>());
|
||||
GrowCapacityAndConvertImpl(object, new_capacity);
|
||||
}
|
||||
FixedArray arguments = elements->arguments();
|
||||
// For fast holey objects, the entry equals the index. The code above made
|
||||
@ -4487,7 +4422,6 @@ class FastSloppyArgumentsElementsAccessor
|
||||
// kMaxUInt32.
|
||||
FastHoleyObjectElementsAccessor::SetImpl(arguments, InternalIndex(index),
|
||||
*value);
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static void ReconfigureImpl(Handle<JSObject> object,
|
||||
@ -4517,8 +4451,8 @@ class FastSloppyArgumentsElementsAccessor
|
||||
}
|
||||
}
|
||||
|
||||
static Maybe<bool> GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
static void GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
Isolate* isolate = object->GetIsolate();
|
||||
Handle<SloppyArgumentsElements> elements(
|
||||
SloppyArgumentsElements::cast(object->elements()), isolate);
|
||||
@ -4529,17 +4463,13 @@ class FastSloppyArgumentsElementsAccessor
|
||||
// elements.
|
||||
DCHECK(from_kind == SLOW_SLOPPY_ARGUMENTS_ELEMENTS ||
|
||||
static_cast<uint32_t>(old_arguments->length()) < capacity);
|
||||
Handle<FixedArrayBase> arguments;
|
||||
ASSIGN_RETURN_ON_EXCEPTION_VALUE(
|
||||
isolate, arguments,
|
||||
ConvertElementsWithCapacity(object, old_arguments, from_kind, capacity),
|
||||
Nothing<bool>());
|
||||
Handle<FixedArrayBase> arguments =
|
||||
ConvertElementsWithCapacity(object, old_arguments, from_kind, capacity);
|
||||
Handle<Map> new_map = JSObject::GetElementsTransitionMap(
|
||||
object, FAST_SLOPPY_ARGUMENTS_ELEMENTS);
|
||||
JSObject::MigrateToMap(isolate, object, new_map);
|
||||
elements->set_arguments(FixedArray::cast(*arguments));
|
||||
JSObject::ValidateElements(*object);
|
||||
return Just(true);
|
||||
}
|
||||
};
|
||||
|
||||
@ -4611,10 +4541,9 @@ class StringWrapperElementsAccessor
|
||||
value);
|
||||
}
|
||||
|
||||
static Maybe<bool> AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
static void AddImpl(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) {
|
||||
DCHECK(index >= static_cast<uint32_t>(GetString(*object).length()));
|
||||
// Explicitly grow fast backing stores if needed. Dictionaries know how to
|
||||
// extend their capacity themselves.
|
||||
@ -4622,12 +4551,10 @@ class StringWrapperElementsAccessor
|
||||
(object->GetElementsKind() == SLOW_STRING_WRAPPER_ELEMENTS ||
|
||||
BackingStoreAccessor::GetCapacityImpl(*object, object->elements()) !=
|
||||
new_capacity)) {
|
||||
MAYBE_RETURN(GrowCapacityAndConvertImpl(object, new_capacity),
|
||||
Nothing<bool>());
|
||||
GrowCapacityAndConvertImpl(object, new_capacity);
|
||||
}
|
||||
BackingStoreAccessor::AddImpl(object, index, value, attributes,
|
||||
new_capacity);
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
static void ReconfigureImpl(Handle<JSObject> object,
|
||||
@ -4672,8 +4599,8 @@ class StringWrapperElementsAccessor
|
||||
backing_store, keys);
|
||||
}
|
||||
|
||||
static Maybe<bool> GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
static void GrowCapacityAndConvertImpl(Handle<JSObject> object,
|
||||
uint32_t capacity) {
|
||||
Handle<FixedArrayBase> old_elements(object->elements(),
|
||||
object->GetIsolate());
|
||||
ElementsKind from_kind = object->GetElementsKind();
|
||||
@ -4688,9 +4615,9 @@ class StringWrapperElementsAccessor
|
||||
// elements.
|
||||
DCHECK(from_kind == SLOW_STRING_WRAPPER_ELEMENTS ||
|
||||
static_cast<uint32_t>(old_elements->length()) < capacity);
|
||||
return Subclass::BasicGrowCapacityAndConvertImpl(
|
||||
object, old_elements, from_kind, FAST_STRING_WRAPPER_ELEMENTS,
|
||||
capacity);
|
||||
Subclass::BasicGrowCapacityAndConvertImpl(object, old_elements, from_kind,
|
||||
FAST_STRING_WRAPPER_ELEMENTS,
|
||||
capacity);
|
||||
}
|
||||
|
||||
static void CopyElementsImpl(Isolate* isolate, FixedArrayBase from,
|
||||
@ -4775,7 +4702,7 @@ MaybeHandle<Object> ArrayConstructInitializeElements(
|
||||
} else {
|
||||
// Take the argument as the length.
|
||||
JSArray::Initialize(array, 0);
|
||||
MAYBE_RETURN_NULL(JSArray::SetLength(array, length));
|
||||
JSArray::SetLength(array, length);
|
||||
}
|
||||
return array;
|
||||
}
|
||||
|
@ -66,8 +66,7 @@ class ElementsAccessor {
|
||||
// changing array sizes as defined in EcmaScript 5.1 15.4.5.2, i.e. array that
|
||||
// have non-deletable elements can only be shrunk to the size of highest
|
||||
// element that is non-deletable.
|
||||
virtual Maybe<bool> SetLength(Handle<JSArray> holder,
|
||||
uint32_t new_length) = 0;
|
||||
virtual void SetLength(Handle<JSArray> holder, uint32_t new_length) = 0;
|
||||
|
||||
// Copy all indices that have elements from |object| into the given
|
||||
// KeyAccumulator. For Dictionary-based element-kinds we filter out elements
|
||||
@ -97,13 +96,13 @@ class ElementsAccessor {
|
||||
Handle<JSObject> receiver, KeyAccumulator* accumulator,
|
||||
AddKeyConversion convert) = 0;
|
||||
|
||||
virtual Maybe<bool> TransitionElementsKind(Handle<JSObject> object,
|
||||
Handle<Map> map) = 0;
|
||||
virtual Maybe<bool> GrowCapacityAndConvert(Handle<JSObject> object,
|
||||
uint32_t capacity) = 0;
|
||||
virtual void TransitionElementsKind(Handle<JSObject> object,
|
||||
Handle<Map> map) = 0;
|
||||
virtual void GrowCapacityAndConvert(Handle<JSObject> object,
|
||||
uint32_t capacity) = 0;
|
||||
// Unlike GrowCapacityAndConvert do not attempt to convert the backing store
|
||||
// and simply return false in this case.
|
||||
virtual Maybe<bool> GrowCapacity(Handle<JSObject> object, uint32_t index) = 0;
|
||||
virtual bool GrowCapacity(Handle<JSObject> object, uint32_t index) = 0;
|
||||
|
||||
static void InitializeOncePerProcess();
|
||||
static void TearDown();
|
||||
@ -111,31 +110,29 @@ class ElementsAccessor {
|
||||
virtual void Set(Handle<JSObject> holder, InternalIndex entry,
|
||||
Object value) = 0;
|
||||
|
||||
virtual Maybe<bool> Add(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) = 0;
|
||||
virtual void Add(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value, PropertyAttributes attributes,
|
||||
uint32_t new_capacity) = 0;
|
||||
|
||||
static Handle<JSArray> Concat(Isolate* isolate, BuiltinArguments* args,
|
||||
uint32_t concat_size, uint32_t result_length);
|
||||
|
||||
virtual Maybe<uint32_t> Push(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) = 0;
|
||||
virtual uint32_t Push(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t push_size) = 0;
|
||||
|
||||
virtual Maybe<uint32_t> Unshift(Handle<JSArray> receiver,
|
||||
BuiltinArguments* args,
|
||||
uint32_t unshift_size) = 0;
|
||||
virtual uint32_t Unshift(Handle<JSArray> receiver, BuiltinArguments* args,
|
||||
uint32_t unshift_size) = 0;
|
||||
|
||||
virtual MaybeHandle<Object> Pop(Handle<JSArray> receiver) = 0;
|
||||
virtual Handle<Object> Pop(Handle<JSArray> receiver) = 0;
|
||||
|
||||
virtual MaybeHandle<Object> Shift(Handle<JSArray> receiver) = 0;
|
||||
virtual Handle<Object> Shift(Handle<JSArray> receiver) = 0;
|
||||
|
||||
virtual Handle<NumberDictionary> Normalize(Handle<JSObject> object) = 0;
|
||||
|
||||
virtual size_t GetCapacity(JSObject holder, FixedArrayBase backing_store) = 0;
|
||||
|
||||
virtual MaybeHandle<Object> Fill(Handle<JSObject> receiver,
|
||||
Handle<Object> obj_value, size_t start,
|
||||
size_t end) = 0;
|
||||
virtual Object Fill(Handle<JSObject> receiver, Handle<Object> obj_value,
|
||||
size_t start, size_t end) = 0;
|
||||
|
||||
// Check an Object's own elements for an element (using SameValueZero
|
||||
// semantics)
|
||||
|
@ -479,12 +479,7 @@ Handle<JSObject> InnerAddElement(Isolate* isolate, Handle<JSArray> array,
|
||||
field_type_string, NONE);
|
||||
|
||||
JSObject::AddProperty(isolate, element, factory->value_string(), value, NONE);
|
||||
// TODO(victorgomes): Temporarily forcing a fatal error here in case of
|
||||
// overflow, until Intl::AddElement can handle exceptions.
|
||||
if (JSObject::AddDataElement(array, index, element, NONE).IsNothing()) {
|
||||
FATAL("Fatal JavaScript invalid array size when adding element");
|
||||
UNREACHABLE();
|
||||
}
|
||||
JSObject::AddDataElement(array, index, element, NONE);
|
||||
return element;
|
||||
}
|
||||
|
||||
@ -1574,9 +1569,9 @@ std::vector<std::string> BestFitSupportedLocales(
|
||||
}
|
||||
|
||||
// ecma262 #sec-createarrayfromlist
|
||||
MaybeHandle<JSArray> CreateArrayFromList(Isolate* isolate,
|
||||
std::vector<std::string> elements,
|
||||
PropertyAttributes attr) {
|
||||
Handle<JSArray> CreateArrayFromList(Isolate* isolate,
|
||||
std::vector<std::string> elements,
|
||||
PropertyAttributes attr) {
|
||||
Factory* factory = isolate->factory();
|
||||
// Let array be ! ArrayCreate(0).
|
||||
Handle<JSArray> array = factory->NewJSArray(0);
|
||||
@ -1589,11 +1584,10 @@ MaybeHandle<JSArray> CreateArrayFromList(Isolate* isolate,
|
||||
const std::string& part = elements[i];
|
||||
Handle<String> value =
|
||||
factory->NewStringFromUtf8(CStrVector(part.c_str())).ToHandleChecked();
|
||||
MAYBE_RETURN(JSObject::AddDataElement(array, i, value, attr),
|
||||
MaybeHandle<JSArray>());
|
||||
JSObject::AddDataElement(array, i, value, attr);
|
||||
}
|
||||
// 5. Return array.
|
||||
return MaybeHandle<JSArray>(array);
|
||||
return array;
|
||||
}
|
||||
|
||||
// ECMA 402 9.2.9 SupportedLocales(availableLocales, requestedLocales, options)
|
||||
|
@ -58,8 +58,8 @@ class JSArray : public JSObject {
|
||||
// Initializes the array to a certain length.
|
||||
inline bool AllowsSetLength();
|
||||
|
||||
V8_EXPORT_PRIVATE static Maybe<bool> SetLength(Handle<JSArray> array,
|
||||
uint32_t length);
|
||||
V8_EXPORT_PRIVATE static void SetLength(Handle<JSArray> array,
|
||||
uint32_t length);
|
||||
|
||||
// Set the content of the array to the content of storage.
|
||||
static inline void SetContent(Handle<JSArray> array,
|
||||
|
@ -4849,9 +4849,9 @@ static ElementsKind BestFittingFastElementsKind(JSObject object) {
|
||||
}
|
||||
|
||||
// static
|
||||
Maybe<bool> JSObject::AddDataElement(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes) {
|
||||
void JSObject::AddDataElement(Handle<JSObject> object, uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes) {
|
||||
Isolate* isolate = object->GetIsolate();
|
||||
|
||||
DCHECK(object->map(isolate).is_extensible());
|
||||
@ -4894,15 +4894,13 @@ Maybe<bool> JSObject::AddDataElement(Handle<JSObject> object, uint32_t index,
|
||||
}
|
||||
to = GetMoreGeneralElementsKind(kind, to);
|
||||
ElementsAccessor* accessor = ElementsAccessor::ForKind(to);
|
||||
MAYBE_RETURN(accessor->Add(object, index, value, attributes, new_capacity),
|
||||
Nothing<bool>());
|
||||
accessor->Add(object, index, value, attributes, new_capacity);
|
||||
|
||||
if (object->IsJSArray(isolate) && index >= old_length) {
|
||||
Handle<Object> new_length =
|
||||
isolate->factory()->NewNumberFromUint(index + 1);
|
||||
JSArray::cast(*object).set_length(*new_length);
|
||||
}
|
||||
return Just(true);
|
||||
}
|
||||
|
||||
template <AllocationSiteUpdateMode update_or_check>
|
||||
@ -4969,15 +4967,7 @@ void JSObject::TransitionElementsKind(Handle<JSObject> object,
|
||||
DCHECK((IsSmiElementsKind(from_kind) && IsDoubleElementsKind(to_kind)) ||
|
||||
(IsDoubleElementsKind(from_kind) && IsObjectElementsKind(to_kind)));
|
||||
uint32_t c = static_cast<uint32_t>(object->elements().length());
|
||||
if (ElementsAccessor::ForKind(to_kind)
|
||||
->GrowCapacityAndConvert(object, c)
|
||||
.IsNothing()) {
|
||||
// TODO(victorgomes): Temporarily forcing a fatal error here in case of
|
||||
// overflow, until all users of TransitionElementsKind can handle
|
||||
// exceptions.
|
||||
FATAL("Fatal JavaScript invalid array size transitioning elements kind.");
|
||||
UNREACHABLE();
|
||||
}
|
||||
ElementsAccessor::ForKind(to_kind)->GrowCapacityAndConvert(object, c);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -440,9 +440,10 @@ class JSObject : public TorqueGeneratedJSObject<JSObject, JSReceiver> {
|
||||
const char* name, Handle<Object> value,
|
||||
PropertyAttributes attributes);
|
||||
|
||||
V8_EXPORT_PRIVATE static Maybe<bool> AddDataElement(
|
||||
Handle<JSObject> receiver, uint32_t index, Handle<Object> value,
|
||||
PropertyAttributes attributes);
|
||||
V8_EXPORT_PRIVATE static void AddDataElement(Handle<JSObject> receiver,
|
||||
uint32_t index,
|
||||
Handle<Object> value,
|
||||
PropertyAttributes attributes);
|
||||
|
||||
// Extend the receiver with a single fast property appeared first in the
|
||||
// passed map. This also extends the property backing store if necessary.
|
||||
|
@ -2899,9 +2899,8 @@ Maybe<bool> Object::AddDataProperty(LookupIterator* it, Handle<Object> value,
|
||||
}
|
||||
|
||||
Handle<JSObject> receiver_obj = Handle<JSObject>::cast(receiver);
|
||||
MAYBE_RETURN(JSObject::AddDataElement(receiver_obj, it->array_index(),
|
||||
value, attributes),
|
||||
Nothing<bool>());
|
||||
JSObject::AddDataElement(receiver_obj, it->array_index(), value,
|
||||
attributes);
|
||||
JSObject::ValidateElements(*receiver_obj);
|
||||
return Just(true);
|
||||
} else {
|
||||
@ -3419,7 +3418,7 @@ Maybe<bool> JSArray::ArraySetLength(Isolate* isolate, Handle<JSArray> a,
|
||||
// (Not needed.)
|
||||
}
|
||||
// Most of steps 16 through 19 is implemented by JSArray::SetLength.
|
||||
MAYBE_RETURN(JSArray::SetLength(a, new_len), Nothing<bool>());
|
||||
JSArray::SetLength(a, new_len);
|
||||
// Steps 19d-ii, 20.
|
||||
if (!new_writable) {
|
||||
PropertyDescriptor readonly;
|
||||
@ -5103,13 +5102,13 @@ void JSArray::Initialize(Handle<JSArray> array, int capacity, int length) {
|
||||
array, length, capacity, INITIALIZE_ARRAY_ELEMENTS_WITH_HOLE);
|
||||
}
|
||||
|
||||
Maybe<bool> JSArray::SetLength(Handle<JSArray> array, uint32_t new_length) {
|
||||
void JSArray::SetLength(Handle<JSArray> array, uint32_t new_length) {
|
||||
// We should never end in here with a pixel or external array.
|
||||
DCHECK(array->AllowsSetLength());
|
||||
if (array->SetLengthWouldNormalize(new_length)) {
|
||||
JSObject::NormalizeElements(array);
|
||||
}
|
||||
return array->GetElementsAccessor()->SetLength(array, new_length);
|
||||
array->GetElementsAccessor()->SetLength(array, new_length);
|
||||
}
|
||||
|
||||
// ES6: 9.5.2 [[SetPrototypeOf]] (V)
|
||||
|
@ -1548,7 +1548,7 @@ MaybeHandle<JSArray> ValueDeserializer::ReadSparseJSArray() {
|
||||
HandleScope scope(isolate_);
|
||||
Handle<JSArray> array =
|
||||
isolate_->factory()->NewJSArray(0, TERMINAL_FAST_ELEMENTS_KIND);
|
||||
MAYBE_RETURN(JSArray::SetLength(array, length), MaybeHandle<JSArray>());
|
||||
JSArray::SetLength(array, length);
|
||||
AddObjectWithID(id, array);
|
||||
|
||||
uint32_t num_properties;
|
||||
@ -2321,7 +2321,7 @@ ValueDeserializer::ReadObjectUsingEntireBufferForLegacyFormat() {
|
||||
|
||||
Handle<JSArray> js_array =
|
||||
isolate_->factory()->NewJSArray(0, TERMINAL_FAST_ELEMENTS_KIND);
|
||||
MAYBE_RETURN_NULL(JSArray::SetLength(js_array, length));
|
||||
JSArray::SetLength(js_array, length);
|
||||
size_t begin_properties =
|
||||
stack.size() - 2 * static_cast<size_t>(num_properties);
|
||||
if (num_properties &&
|
||||
|
@ -28,14 +28,7 @@ RUNTIME_FUNCTION(Runtime_TransitionElementsKind) {
|
||||
CONVERT_ARG_HANDLE_CHECKED(JSObject, object, 0);
|
||||
CONVERT_ARG_HANDLE_CHECKED(Map, to_map, 1);
|
||||
ElementsKind to_kind = to_map->elements_kind();
|
||||
if (ElementsAccessor::ForKind(to_kind)
|
||||
->TransitionElementsKind(object, to_map)
|
||||
.IsNothing()) {
|
||||
// TODO(victorgomes): EffectControlLinearizer::LowerTransitionElementsKind
|
||||
// does not handle exceptions.
|
||||
FATAL("Fatal JavaScript invalid array size");
|
||||
UNREACHABLE();
|
||||
}
|
||||
ElementsAccessor::ForKind(to_kind)->TransitionElementsKind(object, to_map);
|
||||
return *object;
|
||||
}
|
||||
|
||||
@ -187,11 +180,7 @@ RUNTIME_FUNCTION(Runtime_GrowArrayElements) {
|
||||
uint32_t capacity = static_cast<uint32_t>(object->elements().length());
|
||||
|
||||
if (index >= capacity) {
|
||||
bool has_grown;
|
||||
MAYBE_ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
|
||||
isolate, has_grown,
|
||||
object->GetElementsAccessor()->GrowCapacity(object, index));
|
||||
if (!has_grown) {
|
||||
if (!object->GetElementsAccessor()->GrowCapacity(object, index)) {
|
||||
return Smi::zero();
|
||||
}
|
||||
}
|
||||
|
@ -196,7 +196,6 @@
|
||||
'regress/regress-crbug-808192': [SKIP],
|
||||
'regress/regress-crbug-941743': [SKIP],
|
||||
'regress/regress-create-exception': [SKIP],
|
||||
'regress/regress-crbug-1201626': [SKIP],
|
||||
|
||||
# These tests run out of stack space in debug mode.
|
||||
'big-array-literal': [SKIP],
|
||||
@ -535,9 +534,6 @@
|
||||
# The failed allocation causes an asan/msan/tsan error
|
||||
'es6/typedarray-construct-offset-not-smi': [SKIP],
|
||||
|
||||
# Skip slow tests.
|
||||
'regress/regress-crbug-1201626': [SKIP],
|
||||
|
||||
# Exception thrown during bootstrapping on ASAN builds, see issue 4236.
|
||||
'regress/regress-1132': [SKIP],
|
||||
|
||||
|
@ -1,16 +0,0 @@
|
||||
// Copyright 2021 the V8 project authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style license that can be
|
||||
// found in the LICENSE file.
|
||||
|
||||
function main() {
|
||||
let a = []
|
||||
a.length = 4294967295.0;
|
||||
function f() {
|
||||
a.length = 0;
|
||||
return -1
|
||||
}
|
||||
const len = {valueOf:f};
|
||||
a.fill(1.1,0,len);
|
||||
}
|
||||
|
||||
assertThrows(() => main(), RangeError);
|
Loading…
Reference in New Issue
Block a user