Commit b1076996 authored by Mythri A's avatar Mythri A Committed by Commit Bot

Also hold BytecodeArrays for optimized functions in PrepareFunctionForOptimization

In PrepareFunctionForOptimization, for functions that are already optimized
we didn't hold on to the bytecode array strongly. If these functions get
deoptimized before we call OptimizeFunctionOnNextCall, then they need to
be re-optimized again. So we should hold the bytecode arrays for optimized
functions as well. OptimizeFunctionOnNextCall removes it from the table if
the function is still optimized.


Bug: v8:8801
Change-Id: I7f3d94d9842223d85843c9ddb109c8bc9f414891
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1599388
Commit-Queue: Mythri Alle <mythria@chromium.org>
Reviewed-by: 's avatarRoss McIlroy <rmcilroy@chromium.org>
Cr-Commit-Position: refs/heads/master@{#61303}
parent b804bdd4
...@@ -217,6 +217,28 @@ RUNTIME_FUNCTION(Runtime_IsConcurrentRecompilationSupported) { ...@@ -217,6 +217,28 @@ RUNTIME_FUNCTION(Runtime_IsConcurrentRecompilationSupported) {
isolate->concurrent_recompilation_enabled()); isolate->concurrent_recompilation_enabled());
} }
namespace {
void RemoveBytecodeFromPendingOptimizeTable(v8::internal::Isolate* isolate,
Handle<JSFunction> function) {
// TODO(mythria): Remove the check for undefined, once we fix all tests to
// add PrepareForOptimization when using OptimizeFunctionOnNextCall.
if (isolate->heap()->pending_optimize_for_test_bytecode()->IsUndefined()) {
return;
}
Handle<ObjectHashTable> table =
handle(ObjectHashTable::cast(
isolate->heap()->pending_optimize_for_test_bytecode()),
isolate);
bool was_present;
table = table->Remove(isolate, table, handle(function->shared(), isolate),
&was_present);
isolate->heap()->SetPendingOptimizeForTestBytecode(*table);
}
} // namespace
RUNTIME_FUNCTION(Runtime_OptimizeFunctionOnNextCall) { RUNTIME_FUNCTION(Runtime_OptimizeFunctionOnNextCall) {
HandleScope scope(isolate); HandleScope scope(isolate);
...@@ -233,15 +255,6 @@ RUNTIME_FUNCTION(Runtime_OptimizeFunctionOnNextCall) { ...@@ -233,15 +255,6 @@ RUNTIME_FUNCTION(Runtime_OptimizeFunctionOnNextCall) {
} }
Handle<JSFunction> function = Handle<JSFunction>::cast(function_object); Handle<JSFunction> function = Handle<JSFunction>::cast(function_object);
// Check we called PrepareFunctionForOptimization and hold the bytecode
// array to prevent it from getting flushed.
// TODO(mythria): Enable this check once we add PrepareForOptimization in all
// tests before calling OptimizeFunctionOnNextCall.
// CHECK(!ObjectHashTable::cast(
// isolate->heap()->pending_optimize_for_test_bytecode())
// ->Lookup(handle(function->shared(), isolate))
// ->IsTheHole());
// The following conditions were lifted (in part) from the DCHECK inside // The following conditions were lifted (in part) from the DCHECK inside
// JSFunction::MarkForOptimization(). // JSFunction::MarkForOptimization().
...@@ -263,14 +276,24 @@ RUNTIME_FUNCTION(Runtime_OptimizeFunctionOnNextCall) { ...@@ -263,14 +276,24 @@ RUNTIME_FUNCTION(Runtime_OptimizeFunctionOnNextCall) {
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
// If the function is already optimized, just return. if (function->shared()->HasAsmWasmData()) {
if (function->IsOptimized() || function->shared()->HasAsmWasmData()) {
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
// If the function has optimized code, ensure that we check for it and return. // Check we called PrepareFunctionForOptimization and hold the bytecode
// array to prevent it from getting flushed.
// TODO(mythria): Enable this check once we add PrepareForOptimization in all
// tests before calling OptimizeFunctionOnNextCall.
// CHECK(!ObjectHashTable::cast(
// isolate->heap()->pending_optimize_for_test_bytecode())
// ->Lookup(handle(function->shared(), isolate))
// ->IsTheHole());
if (function->HasOptimizedCode()) { if (function->HasOptimizedCode()) {
DCHECK(function->ChecksOptimizationMarker()); DCHECK(function->IsOptimized() || function->ChecksOptimizationMarker());
// If function is already optimized, remove the bytecode array from the
// pending optimize for test table and return.
RemoveBytecodeFromPendingOptimizeTable(isolate, function);
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
...@@ -358,16 +381,8 @@ RUNTIME_FUNCTION(Runtime_PrepareFunctionForOptimization) { ...@@ -358,16 +381,8 @@ RUNTIME_FUNCTION(Runtime_PrepareFunctionForOptimization) {
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
// If the function is already optimized, return without making it pending // We don't optimize Asm/Wasm functions.
// optimize for test. if (function->shared()->HasAsmWasmData()) {
if (function->IsOptimized() || function->shared()->HasAsmWasmData()) {
return ReadOnlyRoots(isolate).undefined_value();
}
// If the function has optimized code, ensure that we check for it and then
// return without making it pending optimize for test.
if (function->HasOptimizedCode()) {
DCHECK(function->ChecksOptimizationMarker());
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
...@@ -402,25 +417,37 @@ RUNTIME_FUNCTION(Runtime_OptimizeOsr) { ...@@ -402,25 +417,37 @@ RUNTIME_FUNCTION(Runtime_OptimizeOsr) {
if (!it.done()) function = handle(it.frame()->function(), isolate); if (!it.done()) function = handle(it.frame()->function(), isolate);
if (function.is_null()) return ReadOnlyRoots(isolate).undefined_value(); if (function.is_null()) return ReadOnlyRoots(isolate).undefined_value();
// If the function is already optimized, just return.
if (function->IsOptimized()) return ReadOnlyRoots(isolate).undefined_value();
if (function->shared()->optimization_disabled() && if (function->shared()->optimization_disabled() &&
function->shared()->disable_optimization_reason() == function->shared()->disable_optimization_reason() ==
BailoutReason::kNeverOptimize) { BailoutReason::kNeverOptimize) {
return ReadOnlyRoots(isolate).undefined_value(); return ReadOnlyRoots(isolate).undefined_value();
} }
// Check we called PrepareFunctionForOptimization and hold the bytecode
// array to prevent it from getting flushed.
// TODO(mythria): Enable this check once we add PrepareForOptimization in all
// tests before calling OptimizeOsr.
// CHECK(!ObjectHashTable::cast(
// isolate->heap()->pending_optimize_for_test_bytecode())
// ->Lookup(handle(function->shared(), isolate))
// ->IsTheHole());
if (function->HasOptimizedCode()) {
DCHECK(function->IsOptimized() || function->ChecksOptimizationMarker());
// If function is already optimized, remove the bytecode array from the
// pending optimize for test table and return.
RemoveBytecodeFromPendingOptimizeTable(isolate, function);
return ReadOnlyRoots(isolate).undefined_value();
}
// Ensure that the function is marked for non-concurrent optimization, so that // Ensure that the function is marked for non-concurrent optimization, so that
// subsequent runs don't also optimize. // subsequent runs don't also optimize.
if (!function->HasOptimizedCode()) {
if (FLAG_trace_osr) { if (FLAG_trace_osr) {
PrintF("[OSR - OptimizeOsr marking "); PrintF("[OSR - OptimizeOsr marking ");
function->ShortPrint(); function->ShortPrint();
PrintF(" for non-concurrent optimization]\n"); PrintF(" for non-concurrent optimization]\n");
} }
function->MarkForOptimization(ConcurrencyMode::kNotConcurrent); function->MarkForOptimization(ConcurrencyMode::kNotConcurrent);
}
// Make the profiler arm all back edges in unoptimized code. // Make the profiler arm all back edges in unoptimized code.
if (it.frame()->type() == StackFrame::INTERPRETED) { if (it.frame()->type() == StackFrame::INTERPRETED) {
......
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