Commit 042e3e02 authored by Marja Hölttä's avatar Marja Hölttä Committed by V8 LUCI CQ

[builtins] Fix ArrayPrototypeSlice

The bug was introduced in
https://chromium-review.googlesource.com/c/v8/v8/+/3147910 : We only
want the fast path when "start" is either missing or the number 0, not
when it's something which converts to 0.

Bug: chromium:1248704
Change-Id: I72bb8fa8a9b90a13aae216c6a8e16e7be54285fe
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3157948
Commit-Queue: Marja Hölttä <marja@chromium.org>
Reviewed-by: 's avatarJakob Gruber <jgruber@chromium.org>
Cr-Commit-Position: refs/heads/main@{#76789}
parent 4a02c06b
...@@ -150,12 +150,21 @@ ArrayPrototypeSlice( ...@@ -150,12 +150,21 @@ ArrayPrototypeSlice(
const end: JSAny = arguments[1]; const end: JSAny = arguments[1];
const relativeEnd: Number = end == Undefined ? len : ToInteger_Inline(end); const relativeEnd: Number = end == Undefined ? len : ToInteger_Inline(end);
// Handle array cloning case if the receiver is a fast array. This logic // Handle array cloning case if the receiver is a fast array. In the case
// should be in sync with ArrayPrototypeSlice (to a reasonable degree). This // where relativeStart is 0 but start is not the SMI zero (e.g., start is an
// is because CloneFastJSArray produces arrays which are potentially COW. If // object whose valueOf returns 0) we must not call CloneFastJSArray. This is
// there's a discrepancy, TF generates code which produces a COW array and // because CloneFastArray reloads the array length, and the ToInteger above
// then expects it to be non-COW (or the other way around) -> immediate deopt. // might have called user code which changed it. Thus, calling
if (relativeStart == 0 && end == Undefined) { // CloneFastJSArray here is safe only if we know ToInteger didn't call user
// code.
// This logic should be in sync with ArrayPrototypeSlice (to a reasonable
// degree). This is because CloneFastJSArray produces arrays which are
// potentially COW. If there's a discrepancy, TF generates code which produces
// a COW array and then expects it to be non-COW (or the other way around) ->
// immediate deopt.
if ((start == Undefined || TaggedEqual(start, SmiConstant(0))) &&
end == Undefined) {
typeswitch (receiver) { typeswitch (receiver) {
case (a: FastJSArrayForCopy): { case (a: FastJSArrayForCopy): {
return CloneFastJSArray(context, a); return CloneFastJSArray(context, a);
......
// 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.
let evil = {
valueOf: function () {
array.length = 1;
}
};
let array = [1, 2, 3];
let newArray = array.slice(evil);
assertEquals(3, newArray.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