cdcf3c3134
Bug: v8:9237 Change-Id: I1e308a5a325b01fcdc9236f13080653f3be6d328 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2527095 Commit-Queue: Marja Hölttä <marja@chromium.org> Reviewed-by: Georg Neis <neis@chromium.org> Reviewed-by: Jakob Gruber <jgruber@chromium.org> Cr-Commit-Position: refs/heads/master@{#71155}
181 lines
4.9 KiB
JavaScript
181 lines
4.9 KiB
JavaScript
// Copyright 2020 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: --allow-natives-syntax --super-ic --opt
|
|
// Flags: --no-always-opt --no-stress-opt --deopt-every-n-times=0
|
|
|
|
// This file contains tests which are disabled for TurboProp. TurboProp deopts
|
|
// differently than TurboFan, so the assertions about when a function is
|
|
// deoptimized won't hold. In addition, TurboProp doesn't inline, so all
|
|
// tests that rely on inlining need to be in this file.
|
|
|
|
(function TestPropertyIsConstant() {
|
|
// Test for a case where the property is a constant found in the lookup start
|
|
// object.
|
|
class A {}
|
|
A.prototype.bar = "wrong value: A.prototype.bar";
|
|
|
|
class B extends A {};
|
|
B.prototype.bar = "correct value";
|
|
|
|
class C extends B {
|
|
foo() { return super.bar; }
|
|
}
|
|
C.prototype.bar = "wrong value: C.prototype.bar";
|
|
%PrepareFunctionForOptimization(C.prototype.foo);
|
|
|
|
let o = new C();
|
|
o.bar = "wrong value: o.bar";
|
|
|
|
// Fill in the feedback.
|
|
r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
%OptimizeFunctionOnNextCall(C.prototype.foo);
|
|
|
|
// Test the optimized function.
|
|
r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
// Assert that the function was not deoptimized.
|
|
assertOptimized(C.prototype.foo);
|
|
|
|
// Change the property value.
|
|
B.prototype.bar = "new value";
|
|
r = o.foo();
|
|
assertEquals("new value", r);
|
|
|
|
// Assert that the function was deoptimized (dependency to the constant
|
|
// value).
|
|
assertFalse(isOptimized(C.prototype.foo));
|
|
})();
|
|
|
|
(function TestSuperpropertyAccessInlined() {
|
|
class A {}
|
|
A.prototype.bar = "wrong value: A.prototype.bar";
|
|
|
|
class B extends A {}
|
|
B.prototype.bar = "correct value";
|
|
|
|
class C extends B {}
|
|
|
|
let inline_this_is_being_interpreted;
|
|
class D extends C {
|
|
inline_this() {
|
|
inline_this_is_being_interpreted = %IsBeingInterpreted();
|
|
return super.bar;
|
|
}
|
|
foo() { return this.inline_this(); }
|
|
}
|
|
%PrepareFunctionForOptimization(D.prototype.inline_this);
|
|
%PrepareFunctionForOptimization(D.prototype.foo);
|
|
D.prototype.bar = "wrong value: D.prototype.bar";
|
|
|
|
const o = new D();
|
|
|
|
// Fill in the feedback.
|
|
let r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
%OptimizeFunctionOnNextCall(D.prototype.foo);
|
|
|
|
// Test the optimized function.
|
|
r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
// Assert that the function was not deoptimized.
|
|
assertOptimized(D.prototype.foo);
|
|
|
|
// Assert that inline_this really got inlined.
|
|
assertFalse(inline_this_is_being_interpreted);
|
|
})();
|
|
|
|
(function TestMegamorphicInlined() {
|
|
class A {}
|
|
A.prototype.bar = "wrong value: A.prototype.bar";
|
|
|
|
class B extends A {}
|
|
B.prototype.bar = "correct value";
|
|
|
|
class C extends B {}
|
|
|
|
let inline_this_is_being_interpreted;
|
|
class D extends C {
|
|
inline_this() {
|
|
inline_this_is_being_interpreted = %IsBeingInterpreted();
|
|
return super.bar;
|
|
}
|
|
foo() { return this.inline_this(); }
|
|
}
|
|
%PrepareFunctionForOptimization(D.prototype.inline_this);
|
|
%PrepareFunctionForOptimization(D.prototype.foo);
|
|
D.prototype.bar = "wrong value: D.prototype.bar";
|
|
|
|
const o = new D();
|
|
|
|
// Create objects which will act as the "home object's prototype" later.
|
|
const prototypes = [{"a": 0}, {"b": 0}, {"c": 0}, {"d": 0}, {"e": 0},
|
|
{"f": 0}, {"g": 0}, {"e": 0}];
|
|
for (p of prototypes) {
|
|
p.__proto__ = B.prototype;
|
|
}
|
|
|
|
// Fill in the feedback (megamorphic).
|
|
for (p of prototypes) {
|
|
D.prototype.__proto__ = p;
|
|
const r = o.foo();
|
|
assertEquals("correct value", r);
|
|
}
|
|
|
|
%OptimizeFunctionOnNextCall(D.prototype.foo);
|
|
|
|
// Test the optimized function - don't change the home object's proto any
|
|
// more.
|
|
let r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
// Assert that the function was not deoptimized.
|
|
assertOptimized(D.prototype.foo);
|
|
|
|
// Assert that inline_this really got inlined.
|
|
assertFalse(inline_this_is_being_interpreted);
|
|
})();
|
|
|
|
(function TestHomeObjectProtoIsGlobalThisGetterPropertyInlined() {
|
|
class A {}
|
|
|
|
let inline_this_is_being_interpreted;
|
|
class B extends A {
|
|
inline_this() {
|
|
inline_this_is_being_interpreted = %IsBeingInterpreted();
|
|
return super.bar;
|
|
}
|
|
foo() { return this.inline_this(); }
|
|
}
|
|
B.prototype.__proto__ = globalThis;
|
|
Object.defineProperty(globalThis, "bar",
|
|
{get: function() { return this.this_value; }});
|
|
%PrepareFunctionForOptimization(B.prototype.inline_this);
|
|
%PrepareFunctionForOptimization(B.prototype.foo);
|
|
|
|
let o = new B();
|
|
o.this_value = "correct value";
|
|
|
|
// Fill in the feedback.
|
|
let r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
%OptimizeFunctionOnNextCall(B.prototype.foo);
|
|
|
|
// Test the optimized function.
|
|
r = o.foo();
|
|
assertEquals("correct value", r);
|
|
|
|
// Assert that the function was not deoptimized.
|
|
assertOptimized(B.prototype.foo);
|
|
|
|
// Assert that inline_this really got inlined.
|
|
assertFalse(inline_this_is_being_interpreted);
|
|
})();
|