[maglev] Omit calling default ctors

I.e., implement the Maglev handler for the FindNonDefaultConstructor
bytecode.

Bug: v8:13091
Change-Id: I6d9905227875fe4efd460434b650fc48d008e7bf
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3925196
Reviewed-by: Leszek Swirski <leszeks@chromium.org>
Commit-Queue: Marja Hölttä <marja@chromium.org>
Cr-Commit-Position: refs/heads/main@{#83470}
This commit is contained in:
Marja Hölttä 2022-09-28 11:38:32 +02:00 committed by V8 LUCI CQ
parent d8990fdc76
commit ece2746346
10 changed files with 790 additions and 15 deletions

View File

@ -2054,8 +2054,14 @@ void MaglevGraphBuilder::VisitGetSuperConstructor() {
}
void MaglevGraphBuilder::VisitFindNonDefaultConstructor() {
// TODO(v8:13091): Implement.
CHECK(false);
ValueNode* this_function = LoadRegisterTagged(0);
ValueNode* new_target = LoadRegisterTagged(1);
CallBuiltin* call_builtin =
BuildCallBuiltin<Builtin::kFindNonDefaultConstructor>(
{this_function, new_target});
auto result = iterator_.GetRegisterPairOperand(2);
StoreRegisterPair(result, call_builtin);
}
void MaglevGraphBuilder::InlineCallFromRegisters(

View File

@ -728,6 +728,8 @@ class MaglevGraphBuilder {
// would be emitted between these two nodes.
if (result->opcode() == Opcode::kCallRuntime) {
DCHECK_EQ(result->Cast<CallRuntime>()->ReturnCount(), 2);
} else if (result->opcode() == Opcode::kCallBuiltin) {
DCHECK_EQ(result->Cast<CallBuiltin>()->ReturnCount(), 2);
} else {
DCHECK_EQ(result->opcode(), Opcode::kForInPrepare);
}

View File

@ -3084,7 +3084,6 @@ void CallBuiltin::AllocateVreg(MaglevVregAllocationState* vreg_state) {
if (has_context) {
UseFixed(input(i), kContextRegister);
}
DCHECK_EQ(descriptor.GetReturnCount(), 1);
DefineAsFixed(vreg_state, this, kReturnRegister0);
}

View File

@ -3460,6 +3460,10 @@ class CallBuiltin : public ValueNodeT<CallBuiltin> {
void set_arg(int i, ValueNode* node) { set_input(i, node); }
int ReturnCount() const {
return Builtins::CallInterfaceDescriptorFor(builtin_).GetReturnCount();
}
DECL_NODE_INTERFACE()
private:

View File

@ -2,11 +2,9 @@
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --omit-default-ctors --allow-natives-syntax --no-maglev --turbofan
// Flags: --omit-default-ctors --allow-natives-syntax --turbofan
// Flags: --no-always-turbofan
// TODO(v8:13091): Enable Maglev.
// This behavior is not spec compliant, see crbug.com/v8/13249.
(function ArrayIteratorMonkeyPatched() {
let iterationCount = 0;

View File

@ -2,11 +2,9 @@
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --omit-default-ctors --allow-natives-syntax --no-maglev --turbofan
// Flags: --omit-default-ctors --allow-natives-syntax --turbofan
// Flags: --no-always-turbofan
// TODO(v8:13091): Enable Maglev.
(function OmitDefaultBaseCtor() {
class A {}; // default base ctor -> will be omitted
class B extends A {};

View File

@ -0,0 +1,32 @@
// Copyright 2022 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.
// Flags: --omit-default-ctors --allow-natives-syntax --maglev
// This behavior is not spec compliant, see crbug.com/v8/13249.
(function ArrayIteratorMonkeyPatched() {
let iterationCount = 0;
const oldIterator = Array.prototype[Symbol.iterator];
Array.prototype[Symbol.iterator] =
function () { ++iterationCount; return oldIterator.call(this); };
class A {}
class B extends A {}
class C extends B {}
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
// C default ctor doing "...args" and B default ctor doing "...args".
assertEquals(2, iterationCount);
new C();
// C default ctor doing "...args" and B default ctor doing "...args".
assertEquals(4, iterationCount);
assertTrue(isMaglevved(C)); // No deopt.
Array.prototype[Symbol.iterator] = oldIterator;
})();

View File

@ -0,0 +1,740 @@
// Copyright 2022 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.
// Flags: --omit-default-ctors --allow-natives-syntax --maglev
(function OmitDefaultBaseCtor() {
class A {}; // default base ctor -> will be omitted
class B extends A {};
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const o = new B();
assertSame(B.prototype, o.__proto__);
assertTrue(isMaglevved(B)); // No deopt.
})();
(function OmitDefaultDerivedCtor() {
class A { constructor() {} };
class B extends A {}; // default derived ctor -> will be omitted
class C extends B {};
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
const o = new C();
assertSame(C.prototype, o.__proto__);
assertTrue(isMaglevved(C)); // No deopt.
})();
(function OmitDefaultBaseAndDerivedCtor() {
class A {}; // default base ctor -> will be omitted
class B extends A {}; // default derived ctor -> will be omitted
class C extends B {};
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
const o = new C();
assertSame(C.prototype, o.__proto__);
assertTrue(isMaglevved(C)); // No deopt.
})();
(function OmitDefaultBaseCtorWithExplicitSuper() {
class A {}; // default base ctor -> will be omitted
class B extends A { constructor() { super(); } };
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const o = new B();
assertSame(B.prototype, o.__proto__);
assertTrue(isMaglevved(B)); // No deopt.
})();
(function OmitDefaultDerivedCtorWithExplicitSuper() {
class A { constructor() {} };
class B extends A {}; // default derived ctor -> will be omitted
class C extends B { constructor() { super(); } };
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
const o = new C();
assertSame(C.prototype, o.__proto__);
assertTrue(isMaglevved(C)); // No deopt.
})();
(function OmitDefaultBaseAndDerivedCtorWithExplicitSuper() {
class A {}; // default base ctor -> will be omitted
class B extends A {}; // default derived ctor -> will be omitted
class C extends B { constructor() { super(); } };
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
const o = new C();
assertSame(C.prototype, o.__proto__);
assertTrue(isMaglevved(C)); // No deopt.
})();
(function OmitDefaultBaseCtorWithExplicitSuperAndNonFinalSpread() {
class A {}; // default base ctor -> will be omitted
class B extends A { constructor(...args) { super(1, ...args, 2); } };
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const o = new B(3, 4);
assertSame(B.prototype, o.__proto__);
// See https://bugs.chromium.org/p/v8/issues/detail?id=13337
// assertTrue(isMaglevved(B)); // No deopt.
// This assert will fail when the above bug is fixed:
assertFalse(isMaglevved(B));
})();
(function OmitDefaultDerivedCtorWithExplicitSuperAndNonFinalSpread() {
class A { constructor() {} };
class B extends A {}; // default derived ctor -> will be omitted
class C extends B { constructor(...args) { super(1, ...args, 2); } };
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
const o = new C(3, 4);
assertSame(C.prototype, o.__proto__);
// See https://bugs.chromium.org/p/v8/issues/detail?id=13337
// assertTrue(isMaglevved(C)); // No deopt.
// This assert will fail when the above bug is fixed:
assertFalse(isMaglevved(C));
})();
(function OmitDefaultBaseAndDerivedCtorWithExplicitSuperAndNonFinalSpread() {
class A {}; // default base ctor -> will be omitted
class B extends A {}; // default derived ctor -> will be omitted
class C extends B { constructor(...args) { super(1, ...args, 2); } };
%PrepareFunctionForOptimization(C);
new C();
%OptimizeMaglevOnNextCall(C);
const o = new C(3, 4);
assertSame(C.prototype, o.__proto__);
// See https://bugs.chromium.org/p/v8/issues/detail?id=13337
// assertTrue(isMaglevved(C)); // No deopt.
// This assert will fail when the above bug is fixed:
assertFalse(isMaglevved(C));
})();
(function NonDefaultBaseConstructorCalled() {
let ctorCallCount = 0;
let lastArgs;
class Base {
constructor(...args) {
++ctorCallCount;
this.baseTagged = true;
lastArgs = args;
}
};
// Nothing will be omitted.
class A extends Base {};
%PrepareFunctionForOptimization(A);
new A();
%OptimizeMaglevOnNextCall(A);
const a = new A(1, 2, 3);
assertEquals(2, ctorCallCount);
assertEquals([1, 2, 3], lastArgs);
assertTrue(a.baseTagged);
assertTrue(isMaglevved(A)); // No deopt.
// 'A' default ctor will be omitted.
class B1 extends A {};
%PrepareFunctionForOptimization(B1);
new B1();
%OptimizeMaglevOnNextCall(B1);
const b1 = new B1(4, 5, 6);
assertEquals(4, ctorCallCount);
assertEquals([4, 5, 6], lastArgs);
assertTrue(b1.baseTagged);
assertTrue(isMaglevved(B1)); // No deopt.
// The same test with non-final spread; 'A' default ctor will be omitted.
class B2 extends A {
constructor(...args) { super(1, ...args, 2); }
};
%PrepareFunctionForOptimization(B2);
new B2();
%OptimizeMaglevOnNextCall(B2);
const b2 = new B2(4, 5, 6);
assertEquals(6, ctorCallCount);
assertEquals([1, 4, 5, 6, 2], lastArgs);
assertTrue(b2.baseTagged);
// See https://bugs.chromium.org/p/v8/issues/detail?id=13337
// assertTrue(isMaglevved(B2)); // No deopt.
// This assert will fail when the above bug is fixed:
assertFalse(isMaglevved(B2)); // No deopt.
})();
(function NonDefaultDerivedConstructorCalled() {
let ctorCallCount = 0;
let lastArgs;
class Base {};
class Derived extends Base {
constructor(...args) {
super();
++ctorCallCount;
this.derivedTagged = true;
lastArgs = args;
}
};
// Nothing will be omitted.
class A extends Derived {};
%PrepareFunctionForOptimization(A);
new A();
%OptimizeMaglevOnNextCall(A);
const a = new A(1, 2, 3);
assertEquals(2, ctorCallCount);
assertEquals([1, 2, 3], lastArgs);
assertTrue(a.derivedTagged);
assertTrue(isMaglevved(A)); // No deopt.
// 'A' default ctor will be omitted.
class B1 extends A {};
%PrepareFunctionForOptimization(B1);
new B1();
%OptimizeMaglevOnNextCall(B1);
const b1 = new B1(4, 5, 6);
assertEquals(4, ctorCallCount);
assertEquals([4, 5, 6], lastArgs);
assertTrue(b1.derivedTagged);
assertTrue(isMaglevved(B1)); // No deopt.
// The same test with non-final spread. 'A' default ctor will be omitted.
class B2 extends A {
constructor(...args) { super(1, ...args, 2); }
};
%PrepareFunctionForOptimization(B2);
new B2();
%OptimizeMaglevOnNextCall(B2);
const b2 = new B2(4, 5, 6);
assertEquals(6, ctorCallCount);
assertEquals([1, 4, 5, 6, 2], lastArgs);
assertTrue(b2.derivedTagged);
// See https://bugs.chromium.org/p/v8/issues/detail?id=13337
// assertTrue(isMaglevved(B2)); // No deopt.
// This assert will fail when the above bug is fixed:
assertFalse(isMaglevved(B2)); // No deopt.
})();
(function BaseFunctionCalled() {
let baseFunctionCallCount = 0;
function BaseFunction() {
++baseFunctionCallCount;
this.baseTagged = true;
}
class A1 extends BaseFunction {};
%PrepareFunctionForOptimization(A1);
new A1();
%OptimizeMaglevOnNextCall(A1);
const a1 = new A1();
assertEquals(2, baseFunctionCallCount);
assertTrue(a1.baseTagged);
assertTrue(isMaglevved(A1)); // No deopt.
class A2 extends BaseFunction {
constructor(...args) { super(1, ...args, 2); }
};
%PrepareFunctionForOptimization(A2);
new A2();
%OptimizeMaglevOnNextCall(A2);
const a2 = new A2();
assertEquals(4, baseFunctionCallCount);
assertTrue(a2.baseTagged);
assertTrue(isMaglevved(A2)); // No deopt.
})();
(function NonSuperclassCtor() {
class A {};
class B extends A {};
class C extends B {};
class D1 extends C {};
class D2 extends C { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C);
%PrepareFunctionForOptimization(D1);
%PrepareFunctionForOptimization(D2);
new C();
new D1();
new D2();
%OptimizeMaglevOnNextCall(C);
%OptimizeMaglevOnNextCall(D1);
%OptimizeMaglevOnNextCall(D2);
// Install an object which is not a constructor into the class hierarchy.
C.__proto__ = {};
assertThrows(() => { new C(); }, TypeError);
assertThrows(() => { new D1(); }, TypeError);
assertThrows(() => { new D2(); }, TypeError);
})();
(function ArgumentsEvaluatedBeforeNonSuperclassCtorDetected() {
class A {};
class B extends A {};
class C extends B {};
class D1 extends C {};
class D2 extends C { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C);
%PrepareFunctionForOptimization(D1);
%PrepareFunctionForOptimization(D2);
new C();
new D1();
new D2();
%OptimizeMaglevOnNextCall(C);
%OptimizeMaglevOnNextCall(D1);
%OptimizeMaglevOnNextCall(D2);
// Install an object which is not a constructor into the class hierarchy.
C.__proto__ = {};
let callCount = 0;
function foo() {
++callCount;
}
assertThrows(() => { new C(foo()); }, TypeError);
assertEquals(1, callCount);
assertThrows(() => { new D1(foo()); }, TypeError);
assertEquals(2, callCount);
assertThrows(() => { new D2(foo()); }, TypeError);
assertEquals(3, callCount);
})();
(function ArgumentsEvaluatedBeforeNonSuperclassCtorDetected2() {
class A {};
class B extends A {};
class C extends B {};
class D1 extends C {
constructor() {
super(foo());
}
};
class D2 extends C {
constructor(...args) {
super(...args, foo());
}
};
let callCount = 0;
function foo() {
++callCount;
}
%PrepareFunctionForOptimization(D1);
%PrepareFunctionForOptimization(D2);
new D1();
new D2();
%OptimizeMaglevOnNextCall(D1);
%OptimizeMaglevOnNextCall(D2);
assertEquals(2, callCount);
// Install an object which is not a constructor into the class hierarchy.
C.__proto__ = {};
assertThrows(() => { new D1(); }, TypeError);
assertEquals(3, callCount);
assertThrows(() => { new D2(); }, TypeError);
assertEquals(4, callCount);
})();
(function EvaluatingArgumentsChangesClassHierarchy() {
let ctorCallCount = 0;
class A {};
class B extends A { constructor() {
super();
++ctorCallCount;
}};
class C extends B {};
class D extends C {
constructor() {
super(foo());
}
};
let fooCallCount = 0;
let changeHierarchy = false;
function foo() {
if (changeHierarchy) {
C.__proto__ = A;
C.prototype.__proto__ = A.prototype;
}
++fooCallCount;
}
%PrepareFunctionForOptimization(D);
new D();
assertEquals(1, fooCallCount);
assertEquals(1, ctorCallCount);
%OptimizeMaglevOnNextCall(D);
changeHierarchy = true;
new D();
assertEquals(2, fooCallCount);
assertEquals(1, ctorCallCount);
// No deopt (Maglev doesn't depend on the prototype chain not being mutated).
assertTrue(isMaglevved(D));
})();
// The same test as the previous one, but with a ctor with a non-final spread.
(function EvaluatingArgumentsChangesClassHierarchyThisTimeWithNonFinalSpread() {
let ctorCallCount = 0;
class A {};
class B extends A { constructor() {
super();
++ctorCallCount;
}};
class C extends B {};
class D extends C {
constructor(...args) {
super(...args, foo());
}
};
let fooCallCount = 0;
let changeHierarchy = false;
function foo() {
if (changeHierarchy) {
C.__proto__ = A;
C.prototype.__proto__ = A.prototype;
}
++fooCallCount;
}
%PrepareFunctionForOptimization(D);
new D();
assertEquals(1, fooCallCount);
assertEquals(1, ctorCallCount);
%OptimizeMaglevOnNextCall(D);
changeHierarchy = true;
new D();
assertEquals(2, fooCallCount);
assertEquals(1, ctorCallCount);
// No deopt (Maglev doesn't depend on the prototype chain not being mutated).
assertTrue(isMaglevved(D));
})();
(function BasePrivateField() {
class A {
#aBrand = true;
isA() {
return #aBrand in this;
}
};
class B extends A {};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const b = new B();
assertTrue(b.isA());
assertTrue(isMaglevved(B)); // No deopt.
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertTrue(c1.isA());
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertTrue(c2.isA());
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function DerivedPrivateField() {
class A {};
class B extends A {
#bBrand = true;
isB() {
return #bBrand in this;
}
};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertTrue(c1.isB());
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertTrue(c2.isB());
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function BasePrivateMethod() {
class A {
#m() { return 'private'; }
callPrivate() {
return this.#m();
}
};
class B extends A {};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const b = new B();
assertEquals('private', b.callPrivate());
assertTrue(isMaglevved(B)); // No deopt.
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertEquals('private', c1.callPrivate());
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertEquals('private', c2.callPrivate());
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function DerivedPrivateMethod() {
class A {};
class B extends A {
#m() { return 'private'; }
callPrivate() {
return this.#m();
}
};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertEquals('private', c1.callPrivate());
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertEquals('private', c2.callPrivate());
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function BasePrivateGetter() {
class A {
get #p() { return 'private'; }
getPrivate() {
return this.#p;
}
};
class B extends A {};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const b = new B();
assertEquals('private', b.getPrivate());
assertTrue(isMaglevved(B)); // No deopt.
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertEquals('private', c1.getPrivate());
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertEquals('private', c2.getPrivate());
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function DerivedPrivateGetter() {
class A {};
class B extends A {
get #p() { return 'private'; }
getPrivate() {
return this.#p;
}
};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertEquals('private', c1.getPrivate());
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertEquals('private', c2.getPrivate());
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function BasePrivateSetter() {
class A {
set #p(value) { this.secret = value; }
setPrivate() {
this.#p = 'private';
}
};
class B extends A {};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const b = new B();
b.setPrivate();
assertEquals('private', b.secret);
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
c1.setPrivate();
assertEquals('private', c1.secret);
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
c2.setPrivate();
assertEquals('private', c2.secret);
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function DerivedPrivateSetter() {
class A {};
class B extends A {
set #p(value) { this.secret = value; }
setPrivate() {
this.#p = 'private';
}
};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
c1.setPrivate();
assertEquals('private', c1.secret);
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
c2.setPrivate();
assertEquals('private', c2.secret);
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function BaseClassFields() {
class A {
aField = true;
};
class B extends A {};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(B);
new B();
%OptimizeMaglevOnNextCall(B);
const b = new B();
assertTrue(b.aField);
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertTrue(c1.aField);
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertTrue(c2.aField);
assertTrue(isMaglevved(C2)); // No deopt.
})();
(function DerivedClassFields() {
class A {};
class B extends A {
bField = true;
};
class C1 extends B {};
class C2 extends B { constructor(...args) { super(1, ...args, 2); }};
%PrepareFunctionForOptimization(C1);
new C1();
%OptimizeMaglevOnNextCall(C1);
const c1 = new C1();
assertTrue(c1.bField);
assertTrue(isMaglevved(C1)); // No deopt.
%PrepareFunctionForOptimization(C2);
new C2();
%OptimizeMaglevOnNextCall(C2);
const c2 = new C2();
assertTrue(c2.bField);
assertTrue(isMaglevved(C2)); // No deopt.
})();

View File

@ -2,9 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --omit-default-ctors --no-maglev
// TODO(v8:13091): Enable Maglev.
// Flags: --omit-default-ctors
// This behavior is not spec compliant, see crbug.com/v8/13249.
(function ArrayIteratorMonkeyPatched() {

View File

@ -2,9 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --omit-default-ctors --no-maglev
// TODO(v8:13091): Enable Maglev.
// Flags: --omit-default-ctors
(function OmitDefaultBaseCtor() {
class A {} // default base ctor -> will be omitted