summaryrefslogtreecommitdiff
path: root/deps/v8/src/builtins/array-copywithin.tq
blob: 94d871e8f74c13be93bbe82053c5da139421bcd7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
// Copyright 2018 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.

namespace array_copywithin {
  macro ConvertToRelativeIndex(index: Number, length: Number): Number {
    return index < 0 ? Max(index + length, 0) : Min(index, length);
  }

  // https://tc39.github.io/ecma262/#sec-array.prototype.copyWithin
  transitioning javascript builtin ArrayPrototypeCopyWithin(
      js-implicit context: Context, receiver: Object)(...arguments): Object {
    // 1. Let O be ? ToObject(this value).
    const object: JSReceiver = ToObject_Inline(context, receiver);

    // 2. Let len be ? ToLength(? Get(O, "length")).
    const length: Number = GetLengthProperty(object);

    // 3. Let relativeTarget be ? ToInteger(target).
    const relativeTarget: Number = ToInteger_Inline(context, arguments[0]);

    // 4. If relativeTarget < 0, let to be max((len + relativeTarget), 0);
    //    else let to be min(relativeTarget, len).
    let to: Number = ConvertToRelativeIndex(relativeTarget, length);

    // 5. Let relativeStart be ? ToInteger(start).
    const relativeStart: Number = ToInteger_Inline(context, arguments[1]);

    // 6. If relativeStart < 0, let from be max((len + relativeStart), 0);
    //    else let from be min(relativeStart, len).
    let from: Number = ConvertToRelativeIndex(relativeStart, length);

    // 7. If end is undefined, let relativeEnd be len;
    //    else let relativeEnd be ? ToInteger(end).
    let relativeEnd: Number = length;
    if (arguments[2] != Undefined) {
      relativeEnd = ToInteger_Inline(context, arguments[2]);
    }

    // 8. If relativeEnd < 0, let final be max((len + relativeEnd), 0);
    //    else let final be min(relativeEnd, len).
    const final: Number = ConvertToRelativeIndex(relativeEnd, length);

    // 9. Let count be min(final-from, len-to).
    let count: Number = Min(final - from, length - to);

    // 10. If from<to and to<from+count, then.
    let direction: Number = 1;

    if (from < to && to < (from + count)) {
      // a. Let direction be -1.
      direction = -1;

      // b. Let from be from + count - 1.
      from = from + count - 1;

      // c. Let to be to + count - 1.
      to = to + count - 1;
    }

    // 12. Repeat, while count > 0.
    while (count > 0) {
      // a. Let fromKey be ! ToString(from).
      // b. Let toKey be ! ToString(to).
      // c. Let fromPresent be ? HasProperty(O, fromKey).
      const fromPresent: Boolean = HasProperty(object, from);

      // d. If fromPresent is true, then.
      if (fromPresent == True) {
        // i. Let fromVal be ? Get(O, fromKey).
        const fromVal: Object = GetProperty(object, from);

        // ii. Perform ? Set(O, toKey, fromVal, true).
        SetProperty(object, to, fromVal);
      } else {
        // i. Perform ? DeletePropertyOrThrow(O, toKey).
        DeleteProperty(object, to, kStrict);
      }

      // f. Let from be from + direction.
      from = from + direction;

      // g. Let to be to + direction.
      to = to + direction;

      // h. Let count be count - 1.
      --count;
    }

    // 13. Return O.
    return object;
  }
}