Commit a85d74a3 authored by Leszek Swirski's avatar Leszek Swirski Committed by Commit Bot

[parser] Fix cache scope recursion for `with`

The fix in https://crrev.com/c/1997135 didn't properly recurse the cache
scope after a with scope, passing the current scope rather than the
original cache scope up the recursion. Now the "use external cache" check
is done in LookupWith (and, analogously, LookupSloppyEval) while passing
the given cache scope through the Lookup recursion.

Fixed: chromium:1041210
Fixed: chromium:1041616
Change-Id: I5ac9ddc6c16d63b59aa034721fccec2f7781c4f8
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2000133
Commit-Queue: Leszek Swirski <leszeks@chromium.org>
Reviewed-by: 's avatarToon Verwaest <verwaest@chromium.org>
Auto-Submit: Leszek Swirski <leszeks@chromium.org>
Cr-Commit-Position: refs/heads/master@{#65754}
parent 72f7fe1b
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#include <set> #include <set>
#include "src/ast/ast.h" #include "src/ast/ast.h"
#include "src/base/logging.h"
#include "src/base/optional.h" #include "src/base/optional.h"
#include "src/builtins/accessors.h" #include "src/builtins/accessors.h"
#include "src/common/message-template.h" #include "src/common/message-template.h"
...@@ -1899,8 +1900,9 @@ template <Scope::ScopeLookupMode mode> ...@@ -1899,8 +1900,9 @@ template <Scope::ScopeLookupMode mode>
Variable* Scope::Lookup(VariableProxy* proxy, Scope* scope, Variable* Scope::Lookup(VariableProxy* proxy, Scope* scope,
Scope* outer_scope_end, Scope* cache_scope, Scope* outer_scope_end, Scope* cache_scope,
bool force_context_allocation) { bool force_context_allocation) {
// If we have already passed it in earlier recursions though, so we should // If we have already passed the cache scope in earlier recursions, we should
// first quickly check if it is an outer scope before continuing. // first quickly check if the current scope uses the cache scope before
// continuing.
if (mode == kDeserializedScope && if (mode == kDeserializedScope &&
scope->deserialized_scope_uses_external_cache()) { scope->deserialized_scope_uses_external_cache()) {
Variable* var = cache_scope->variables_.Lookup(proxy->raw_name()); Variable* var = cache_scope->variables_.Lookup(proxy->raw_name());
...@@ -1918,6 +1920,8 @@ Variable* Scope::Lookup(VariableProxy* proxy, Scope* scope, ...@@ -1918,6 +1920,8 @@ Variable* Scope::Lookup(VariableProxy* proxy, Scope* scope,
// the scopes in which it's evaluating. // the scopes in which it's evaluating.
if (mode == kDeserializedScope && if (mode == kDeserializedScope &&
V8_UNLIKELY(scope->is_debug_evaluate_scope_)) { V8_UNLIKELY(scope->is_debug_evaluate_scope_)) {
DCHECK(scope->deserialized_scope_uses_external_cache() ||
scope == cache_scope);
return cache_scope->NonLocal(proxy->raw_name(), VariableMode::kDynamic); return cache_scope->NonLocal(proxy->raw_name(), VariableMode::kDynamic);
} }
...@@ -2034,10 +2038,16 @@ Variable* Scope::LookupWith(VariableProxy* proxy, Scope* scope, ...@@ -2034,10 +2038,16 @@ Variable* Scope::LookupWith(VariableProxy* proxy, Scope* scope,
var->ForceContextAllocation(); var->ForceContextAllocation();
if (proxy->is_assigned()) var->SetMaybeAssigned(); if (proxy->is_assigned()) var->SetMaybeAssigned();
} }
if (cache_scope != nullptr) cache_scope->variables_.Remove(var); Scope* target_scope;
Scope* target = cache_scope == nullptr ? scope : cache_scope; if (scope->deserialized_scope_uses_external_cache()) {
DCHECK_NOT_NULL(cache_scope);
cache_scope->variables_.Remove(var);
target_scope = cache_scope;
} else {
target_scope = scope;
}
Variable* dynamic = Variable* dynamic =
target->NonLocal(proxy->raw_name(), VariableMode::kDynamic); target_scope->NonLocal(proxy->raw_name(), VariableMode::kDynamic);
dynamic->set_local_if_not_shadowed(var); dynamic->set_local_if_not_shadowed(var);
return dynamic; return dynamic;
} }
...@@ -2063,6 +2073,14 @@ Variable* Scope::LookupSloppyEval(VariableProxy* proxy, Scope* scope, ...@@ -2063,6 +2073,14 @@ Variable* Scope::LookupSloppyEval(VariableProxy* proxy, Scope* scope,
outer_scope_end, entry_cache); outer_scope_end, entry_cache);
if (var == nullptr) return var; if (var == nullptr) return var;
// We may not want to use the cache scope, change it back to the given scope
// if necessary.
if (!scope->deserialized_scope_uses_external_cache()) {
// For a deserialized scope, we'll be replacing the cache_scope.
DCHECK_IMPLIES(!scope->scope_info_.is_null(), cache_scope != nullptr);
cache_scope = scope;
}
// A variable binding may have been found in an outer scope, but the current // A variable binding may have been found in an outer scope, but the current
// scope makes a sloppy 'eval' call, so the found variable may not be the // scope makes a sloppy 'eval' call, so the found variable may not be the
// correct one (the 'eval' may introduce a binding with the same name). In // correct one (the 'eval' may introduce a binding with the same name). In
......
// 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: --stress-lazy-source-positions
with ({}) {
(() => eval())();
}
// 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.
global = 0;
(function foo() {
function bar() {
let context_allocated = 0;
with ({}) {
f = function() { ++global; }
}
function baz() { return foo(context_allocated); };
f();
};
bar();
})();
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