Keep track of the addition order of variables explicitly.
This avoids needing to allocate a zonelist on the fly later, sorting variables_, for which we also need to keep track of order in the hashmap. In a later phase we can make sure that Variable is always uniquely in either of params_, temps_ and ordered_variables_. In that case we can use a linked list through Variable. BUG=v8:5209 Review-Url: https://codereview.chromium.org/2264053003 Cr-Commit-Position: refs/heads/master@{#38810}
This commit is contained in:
parent
5e08f43531
commit
6ea8b4f216
@ -30,14 +30,16 @@ Variable* VariableMap::Declare(Zone* zone, Scope* scope,
|
||||
const AstRawString* name, VariableMode mode,
|
||||
Variable::Kind kind,
|
||||
InitializationFlag initialization_flag,
|
||||
MaybeAssignedFlag maybe_assigned_flag) {
|
||||
MaybeAssignedFlag maybe_assigned_flag,
|
||||
bool* added) {
|
||||
// AstRawStrings are unambiguous, i.e., the same string is always represented
|
||||
// by the same AstRawString*.
|
||||
// FIXME(marja): fix the type of Lookup.
|
||||
Entry* p =
|
||||
ZoneHashMap::LookupOrInsert(const_cast<AstRawString*>(name), name->hash(),
|
||||
ZoneAllocationPolicy(zone));
|
||||
if (p->value == NULL) {
|
||||
if (added) *added = p->value == nullptr;
|
||||
if (p->value == nullptr) {
|
||||
// The variable has not been declared yet -> insert it.
|
||||
DCHECK(p->key == name);
|
||||
p->value = new (zone) Variable(scope, name, mode, kind, initialization_flag,
|
||||
@ -79,6 +81,7 @@ Scope::Scope(Zone* zone, Scope* outer_scope, ScopeType scope_type)
|
||||
: zone_(zone),
|
||||
outer_scope_(outer_scope),
|
||||
variables_(zone),
|
||||
ordered_variables_(4, zone),
|
||||
decls_(4, zone),
|
||||
scope_type_(scope_type) {
|
||||
SetDefaults();
|
||||
@ -125,6 +128,7 @@ Scope::Scope(Zone* zone, Scope* inner_scope, ScopeType scope_type,
|
||||
: zone_(zone),
|
||||
outer_scope_(nullptr),
|
||||
variables_(zone),
|
||||
ordered_variables_(0, zone),
|
||||
decls_(0, zone),
|
||||
scope_info_(scope_info),
|
||||
scope_type_(scope_type) {
|
||||
@ -160,6 +164,7 @@ Scope::Scope(Zone* zone, Scope* inner_scope,
|
||||
: zone_(zone),
|
||||
outer_scope_(nullptr),
|
||||
variables_(zone),
|
||||
ordered_variables_(0, zone),
|
||||
decls_(0, zone),
|
||||
scope_type_(CATCH_SCOPE) {
|
||||
SetDefaults();
|
||||
@ -426,7 +431,7 @@ void DeclarationScope::DeclareThis(AstValueFactory* ast_value_factory) {
|
||||
DCHECK(has_this_declaration());
|
||||
|
||||
bool subclass_constructor = IsSubclassConstructor(function_kind_);
|
||||
Variable* var = variables_.Declare(
|
||||
Variable* var = Declare(
|
||||
zone(), this, ast_value_factory->this_string(),
|
||||
subclass_constructor ? CONST : VAR, Variable::THIS,
|
||||
subclass_constructor ? kNeedsInitialization : kCreatedInitialized);
|
||||
@ -440,18 +445,16 @@ void DeclarationScope::DeclareDefaultFunctionVariables(
|
||||
// Declare 'arguments' variable which exists in all non arrow functions.
|
||||
// Note that it might never be accessed, in which case it won't be
|
||||
// allocated during variable allocation.
|
||||
arguments_ =
|
||||
variables_.Declare(zone(), this, ast_value_factory->arguments_string(),
|
||||
VAR, Variable::ARGUMENTS, kCreatedInitialized);
|
||||
arguments_ = Declare(zone(), this, ast_value_factory->arguments_string(), VAR,
|
||||
Variable::ARGUMENTS, kCreatedInitialized);
|
||||
|
||||
new_target_ =
|
||||
variables_.Declare(zone(), this, ast_value_factory->new_target_string(),
|
||||
new_target_ = Declare(zone(), this, ast_value_factory->new_target_string(),
|
||||
CONST, Variable::NORMAL, kCreatedInitialized);
|
||||
|
||||
if (IsConciseMethod(function_kind_) || IsClassConstructor(function_kind_) ||
|
||||
IsAccessorFunction(function_kind_)) {
|
||||
this_function_ = variables_.Declare(
|
||||
zone(), this, ast_value_factory->this_function_string(), CONST,
|
||||
this_function_ =
|
||||
Declare(zone(), this, ast_value_factory->this_function_string(), CONST,
|
||||
Variable::NORMAL, kCreatedInitialized);
|
||||
}
|
||||
}
|
||||
@ -670,7 +673,7 @@ Variable* DeclarationScope::DeclareParameter(
|
||||
if (mode == TEMPORARY) {
|
||||
var = NewTemporary(name);
|
||||
} else {
|
||||
var = variables_.Declare(zone(), this, name, mode, Variable::NORMAL,
|
||||
var = Declare(zone(), this, name, mode, Variable::NORMAL,
|
||||
kCreatedInitialized);
|
||||
// TODO(wingo): Avoid O(n^2) check.
|
||||
*is_duplicate = IsDeclaredParameter(name);
|
||||
@ -698,15 +701,14 @@ Variable* Scope::DeclareLocal(const AstRawString* name, VariableMode mode,
|
||||
// introduced during variable allocation, and TEMPORARY variables are
|
||||
// allocated via NewTemporary().
|
||||
DCHECK(IsDeclaredVariableMode(mode));
|
||||
return variables_.Declare(zone(), this, name, mode, kind, init_flag,
|
||||
return Declare(zone(), this, name, mode, kind, init_flag,
|
||||
maybe_assigned_flag);
|
||||
}
|
||||
|
||||
Variable* DeclarationScope::DeclareDynamicGlobal(const AstRawString* name,
|
||||
Variable::Kind kind) {
|
||||
DCHECK(is_script_scope());
|
||||
return variables_.Declare(zone(), this, name, DYNAMIC_GLOBAL, kind,
|
||||
kCreatedInitialized);
|
||||
return Declare(zone(), this, name, DYNAMIC_GLOBAL, kind, kCreatedInitialized);
|
||||
}
|
||||
|
||||
|
||||
@ -796,20 +798,6 @@ Declaration* Scope::CheckLexDeclarationsConflictingWith(
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
class VarAndOrder {
|
||||
public:
|
||||
VarAndOrder(Variable* var, int order) : var_(var), order_(order) { }
|
||||
Variable* var() const { return var_; }
|
||||
int order() const { return order_; }
|
||||
static int Compare(const VarAndOrder* a, const VarAndOrder* b) {
|
||||
return a->order_ - b->order_;
|
||||
}
|
||||
|
||||
private:
|
||||
Variable* var_;
|
||||
int order_;
|
||||
};
|
||||
|
||||
void Scope::CollectStackAndContextLocals(ZoneList<Variable*>* stack_locals,
|
||||
ZoneList<Variable*>* context_locals,
|
||||
ZoneList<Variable*>* context_globals) {
|
||||
@ -836,20 +824,8 @@ void Scope::CollectStackAndContextLocals(ZoneList<Variable*>* stack_locals,
|
||||
}
|
||||
}
|
||||
|
||||
// Collect declared local variables.
|
||||
ZoneList<VarAndOrder> vars(variables_.occupancy(), zone());
|
||||
for (VariableMap::Entry* p = variables_.Start();
|
||||
p != NULL;
|
||||
p = variables_.Next(p)) {
|
||||
Variable* var = reinterpret_cast<Variable*>(p->value);
|
||||
if (var->is_used()) {
|
||||
vars.Add(VarAndOrder(var, p->order), zone());
|
||||
}
|
||||
}
|
||||
vars.Sort(VarAndOrder::Compare);
|
||||
int var_count = vars.length();
|
||||
for (int i = 0; i < var_count; i++) {
|
||||
Variable* var = vars[i].var();
|
||||
for (int i = 0; i < ordered_variables_.length(); i++) {
|
||||
Variable* var = ordered_variables_[i];
|
||||
if (var->IsStackLocal()) {
|
||||
stack_locals->Add(var, zone());
|
||||
} else if (var->IsContextSlot()) {
|
||||
@ -1634,22 +1610,13 @@ void Scope::AllocateNonParameterLocalsAndDeclaredGlobals() {
|
||||
}
|
||||
}
|
||||
|
||||
ZoneList<VarAndOrder> vars(variables_.occupancy(), zone());
|
||||
for (VariableMap::Entry* p = variables_.Start();
|
||||
p != NULL;
|
||||
p = variables_.Next(p)) {
|
||||
Variable* var = reinterpret_cast<Variable*>(p->value);
|
||||
vars.Add(VarAndOrder(var, p->order), zone());
|
||||
}
|
||||
vars.Sort(VarAndOrder::Compare);
|
||||
int var_count = vars.length();
|
||||
for (int i = 0; i < var_count; i++) {
|
||||
AllocateNonParameterLocal(vars[i].var());
|
||||
for (int i = 0; i < ordered_variables_.length(); i++) {
|
||||
AllocateNonParameterLocal(ordered_variables_[i]);
|
||||
}
|
||||
|
||||
if (FLAG_global_var_shortcuts) {
|
||||
for (int i = 0; i < var_count; i++) {
|
||||
AllocateDeclaredGlobal(vars[i].var());
|
||||
for (int i = 0; i < ordered_variables_.length(); i++) {
|
||||
AllocateDeclaredGlobal(ordered_variables_[i]);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -23,7 +23,8 @@ class VariableMap: public ZoneHashMap {
|
||||
Variable* Declare(Zone* zone, Scope* scope, const AstRawString* name,
|
||||
VariableMode mode, Variable::Kind kind,
|
||||
InitializationFlag initialization_flag,
|
||||
MaybeAssignedFlag maybe_assigned_flag = kNotAssigned);
|
||||
MaybeAssignedFlag maybe_assigned_flag = kNotAssigned,
|
||||
bool* added = nullptr);
|
||||
|
||||
Variable* Lookup(const AstRawString* name);
|
||||
};
|
||||
@ -447,6 +448,17 @@ class Scope: public ZoneObject {
|
||||
}
|
||||
|
||||
private:
|
||||
Variable* Declare(Zone* zone, Scope* scope, const AstRawString* name,
|
||||
VariableMode mode, Variable::Kind kind,
|
||||
InitializationFlag initialization_flag,
|
||||
MaybeAssignedFlag maybe_assigned_flag = kNotAssigned) {
|
||||
bool added;
|
||||
Variable* var =
|
||||
variables_.Declare(zone, scope, name, mode, kind, initialization_flag,
|
||||
maybe_assigned_flag, &added);
|
||||
if (added) ordered_variables_.Add(var, zone);
|
||||
return var;
|
||||
}
|
||||
Zone* zone_;
|
||||
|
||||
// Scope tree.
|
||||
@ -460,6 +472,10 @@ class Scope: public ZoneObject {
|
||||
// variables may be implicitly 'declared' by being used (possibly in
|
||||
// an inner scope) with no intervening with statements or eval calls.
|
||||
VariableMap variables_;
|
||||
// In case of non-scopeinfo-backed scopes, this contains the variables of the
|
||||
// map above in order of addition.
|
||||
// TODO(verwaest): Thread through Variable.
|
||||
ZoneList<Variable*> ordered_variables_;
|
||||
// Variables that must be looked up dynamically.
|
||||
DynamicScopePart* dynamics_;
|
||||
// Unresolved variables referred to from this scope. The proxies themselves
|
||||
|
@ -48,7 +48,6 @@ class TemplateHashMapImpl {
|
||||
void* key;
|
||||
void* value;
|
||||
uint32_t hash; // The full hash value for key
|
||||
int order; // If you never remove entries this is the insertion order.
|
||||
};
|
||||
|
||||
// If an entry with matching key is found, returns that entry.
|
||||
@ -152,7 +151,6 @@ TemplateHashMapImpl<AllocationPolicy>::InsertNew(void* key, uint32_t hash,
|
||||
p->key = key;
|
||||
p->value = NULL;
|
||||
p->hash = hash;
|
||||
p->order = occupancy_;
|
||||
occupancy_++;
|
||||
|
||||
// Grow the map if we reached >= 80% occupancy.
|
||||
@ -300,7 +298,6 @@ void TemplateHashMapImpl<AllocationPolicy>::Resize(AllocationPolicy allocator) {
|
||||
if (p->key != NULL) {
|
||||
Entry* entry = LookupOrInsert(p->key, p->hash, allocator);
|
||||
entry->value = p->value;
|
||||
entry->order = p->order;
|
||||
n--;
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user