// 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. // Flags: --allow-natives-syntax --expose-gc --turbofan --no-concurrent-recompilation // Flags: --no-stress-opt --no-always-turbofan --no-assert-types // This weak ref is for checking whether the closure-allocated object o got // collected as it should. var weak; // Return a function which refers to a variable from its enclosing function. function makeFn() { var o = {}; weak = new WeakRef(o); o.num = 0; return () => { return ++o.num; }; } // Simple wrapper function which will cause inlining. function g(f) { f(); } // Optimize g so that it inlines a specific function instance created by makeFn, // and then drop all user-visible references to that function instance. (function () { var fn = makeFn(); %PrepareFunctionForOptimization(g); %PrepareFunctionForOptimization(fn); g(fn); %OptimizeFunctionOnNextCall(g); g(fn); })(); // WeakRefs created during the current microtask are strong, so defer the rest. setTimeout(() => { // Since the function inlined by g no longer exists, it should deopt and // release the inner function. gc(); // Check that the memory leak described in v8:4578 no longer happens. assertEquals(undefined, weak.deref()); // Next, let's check the opposite case: if an optimized function's Code is // currently running at the time of gc, then it must keep its deoptimization // data alive. // Another simple wrapper function, but this one causes a GC internally. function h(f) { gcAndCheckState(); // Non-inlined call f(); // Inlined call } var doCheck = false; function gcAndCheckState() { if (!doCheck) return; // It's possible that h has already deoptimized by this point if // --stress-incremental-marking caused additional gc cycles. var optimized = isOptimized(h); gc(); // If the optimized code for h is still on the stack, then it must not clear // out its own deoptimization data. Eventually h will deopt due to the wrong // function being passed, but only after this function has returned. if (optimized) { assertNotEquals(undefined, weak.deref()); } else { assertEquals(undefined, weak.deref()); } } // Don't inline gcAndCheckState, to avoid the possibility that its content // could cause h to deoptimize. %NeverOptimizeFunction(gcAndCheckState); // Optimize h to inline a specific function instance, and then drop all // user-visible references to that inlined function. (function () { var fn = makeFn(); %PrepareFunctionForOptimization(h); %PrepareFunctionForOptimization(fn); h(fn); %OptimizeFunctionOnNextCall(h); h(fn); })(); // Defer again so the WeakRef can expire. setTimeout(() => { doCheck = true; h(() => {}); }, 0); }, 0);