Commit a6e45ce7 authored by rossberg@chromium.org's avatar rossberg@chromium.org

Introduce some predicates over variable modes.

These should be handy when we add more declaration forms for Harmony.

R=svenpanne@chromium.org
BUG=

Review URL: https://chromiumcodereview.appspot.com/10897010

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12404 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 5419ee7b
...@@ -832,10 +832,9 @@ void FullCodeGenerator::VisitVariableDeclaration( ...@@ -832,10 +832,9 @@ void FullCodeGenerator::VisitVariableDeclaration(
Comment cmnt(masm_, "[ VariableDeclaration"); Comment cmnt(masm_, "[ VariableDeclaration");
__ mov(r2, Operand(variable->name())); __ mov(r2, Operand(variable->name()));
// Declaration nodes are always introduced in one of four modes. // Declaration nodes are always introduced in one of four modes.
ASSERT(mode == VAR || mode == LET || ASSERT(IsDeclaredVariableMode(mode));
mode == CONST || mode == CONST_HARMONY); PropertyAttributes attr =
PropertyAttributes attr = (mode == CONST || mode == CONST_HARMONY) IsImmutableVariableMode(mode) ? READ_ONLY : NONE;
? READ_ONLY : NONE;
__ mov(r1, Operand(Smi::FromInt(attr))); __ mov(r1, Operand(Smi::FromInt(attr)));
// Push initial value, if any. // Push initial value, if any.
// Note: For variables we must not push an initial value (such as // Note: For variables we must not push an initial value (such as
......
...@@ -454,10 +454,7 @@ class Declaration: public AstNode { ...@@ -454,10 +454,7 @@ class Declaration: public AstNode {
: proxy_(proxy), : proxy_(proxy),
mode_(mode), mode_(mode),
scope_(scope) { scope_(scope) {
ASSERT(mode == VAR || ASSERT(IsDeclaredVariableMode(mode));
mode == CONST ||
mode == CONST_HARMONY ||
mode == LET);
} }
private: private:
......
...@@ -4815,8 +4815,9 @@ void HGraphBuilder::VisitVariableProxy(VariableProxy* expr) { ...@@ -4815,8 +4815,9 @@ void HGraphBuilder::VisitVariableProxy(VariableProxy* expr) {
Variable* variable = expr->var(); Variable* variable = expr->var();
switch (variable->location()) { switch (variable->location()) {
case Variable::UNALLOCATED: { case Variable::UNALLOCATED: {
if (variable->mode() == LET || variable->mode() == CONST_HARMONY) { if (IsLexicalVariableMode(variable->mode())) {
return Bailout("reference to global harmony declared variable"); // TODO(rossberg): should this be an ASSERT?
return Bailout("reference to global lexical variable");
} }
// Handle known global constants like 'undefined' specially to avoid a // Handle known global constants like 'undefined' specially to avoid a
// load from a global cell for them. // load from a global cell for them.
...@@ -4861,9 +4862,8 @@ void HGraphBuilder::VisitVariableProxy(VariableProxy* expr) { ...@@ -4861,9 +4862,8 @@ void HGraphBuilder::VisitVariableProxy(VariableProxy* expr) {
case Variable::LOCAL: { case Variable::LOCAL: {
HValue* value = environment()->Lookup(variable); HValue* value = environment()->Lookup(variable);
if (value == graph()->GetConstantHole()) { if (value == graph()->GetConstantHole()) {
ASSERT(variable->mode() == CONST || ASSERT(IsDeclaredVariableMode(variable->mode()) &&
variable->mode() == CONST_HARMONY || variable->mode() != VAR);
variable->mode() == LET);
return Bailout("reference to uninitialized variable"); return Bailout("reference to uninitialized variable");
} }
return ast_context()->ReturnValue(value); return ast_context()->ReturnValue(value);
...@@ -8115,8 +8115,7 @@ void HGraphBuilder::VisitCountOperation(CountOperation* expr) { ...@@ -8115,8 +8115,7 @@ void HGraphBuilder::VisitCountOperation(CountOperation* expr) {
} }
HValue* context = BuildContextChainWalk(var); HValue* context = BuildContextChainWalk(var);
HStoreContextSlot::Mode mode = HStoreContextSlot::Mode mode = IsLexicalVariableMode(var->mode())
(var->mode() == LET || var->mode() == CONST_HARMONY)
? HStoreContextSlot::kCheckDeoptimize : HStoreContextSlot::kNoCheck; ? HStoreContextSlot::kCheckDeoptimize : HStoreContextSlot::kNoCheck;
HStoreContextSlot* instr = HStoreContextSlot* instr =
new(zone()) HStoreContextSlot(context, var->index(), mode, after); new(zone()) HStoreContextSlot(context, var->index(), mode, after);
......
...@@ -810,10 +810,9 @@ void FullCodeGenerator::VisitVariableDeclaration( ...@@ -810,10 +810,9 @@ void FullCodeGenerator::VisitVariableDeclaration(
__ push(esi); __ push(esi);
__ push(Immediate(variable->name())); __ push(Immediate(variable->name()));
// VariableDeclaration nodes are always introduced in one of four modes. // VariableDeclaration nodes are always introduced in one of four modes.
ASSERT(mode == VAR || mode == LET || ASSERT(IsDeclaredVariableMode(mode));
mode == CONST || mode == CONST_HARMONY); PropertyAttributes attr =
PropertyAttributes attr = (mode == CONST || mode == CONST_HARMONY) IsImmutableVariableMode(mode) ? READ_ONLY : NONE;
? READ_ONLY : NONE;
__ push(Immediate(Smi::FromInt(attr))); __ push(Immediate(Smi::FromInt(attr)));
// Push initial value, if any. // Push initial value, if any.
// Note: For variables we must not push an initial value (such as // Note: For variables we must not push an initial value (such as
......
...@@ -849,10 +849,9 @@ void FullCodeGenerator::VisitVariableDeclaration( ...@@ -849,10 +849,9 @@ void FullCodeGenerator::VisitVariableDeclaration(
Comment cmnt(masm_, "[ VariableDeclaration"); Comment cmnt(masm_, "[ VariableDeclaration");
__ li(a2, Operand(variable->name())); __ li(a2, Operand(variable->name()));
// Declaration nodes are always introduced in one of four modes. // Declaration nodes are always introduced in one of four modes.
ASSERT(mode == VAR || mode == LET || ASSERT(IsDeclaredVariableMode(mode));
mode == CONST || mode == CONST_HARMONY); PropertyAttributes attr =
PropertyAttributes attr = (mode == CONST || mode == CONST_HARMONY) IsImmutableVariableMode(mode) ? READ_ONLY : NONE;
? READ_ONLY : NONE;
__ li(a1, Operand(Smi::FromInt(attr))); __ li(a1, Operand(Smi::FromInt(attr)));
// Push initial value, if any. // Push initial value, if any.
// Note: For variables we must not push an initial value (such as // Note: For variables we must not push an initial value (such as
......
...@@ -1802,8 +1802,8 @@ void Parser::Declare(Declaration* declaration, bool resolve, bool* ok) { ...@@ -1802,8 +1802,8 @@ void Parser::Declare(Declaration* declaration, bool resolve, bool* ok) {
name, mode, declaration->initialization(), proxy->interface()); name, mode, declaration->initialization(), proxy->interface());
} else if ((mode != VAR || var->mode() != VAR) && } else if ((mode != VAR || var->mode() != VAR) &&
(!declaration_scope->is_global_scope() || (!declaration_scope->is_global_scope() ||
(mode != VAR && mode != CONST) || IsLexicalVariableMode(mode) ||
(var->mode() != VAR && var->mode() != CONST))) { IsLexicalVariableMode(var->mode()))) {
// The name was declared in this scope before; check for conflicting // The name was declared in this scope before; check for conflicting
// re-declarations. We have a conflict if either of the declarations is // re-declarations. We have a conflict if either of the declarations is
// not a var (in the global scope, we also have to ignore legacy const for // not a var (in the global scope, we also have to ignore legacy const for
...@@ -1817,11 +1817,7 @@ void Parser::Declare(Declaration* declaration, bool resolve, bool* ok) { ...@@ -1817,11 +1817,7 @@ void Parser::Declare(Declaration* declaration, bool resolve, bool* ok) {
// //
// because the var declaration is hoisted to the function scope where 'x' // because the var declaration is hoisted to the function scope where 'x'
// is already bound. // is already bound.
// We only have vars, consts and lets in declarations. ASSERT(IsDeclaredVariableMode(var->mode()));
ASSERT(var->mode() == VAR ||
var->mode() == CONST ||
var->mode() == CONST_HARMONY ||
var->mode() == LET);
if (is_extended_mode()) { if (is_extended_mode()) {
// In harmony mode we treat re-declarations as early errors. See // In harmony mode we treat re-declarations as early errors. See
// ES5 16 for a definition of early errors. // ES5 16 for a definition of early errors.
...@@ -2341,7 +2337,7 @@ Block* Parser::ParseVariableDeclarations( ...@@ -2341,7 +2337,7 @@ Block* Parser::ParseVariableDeclarations(
// browsers where the global object (window) has lots of // browsers where the global object (window) has lots of
// properties defined in prototype objects. // properties defined in prototype objects.
if (initialization_scope->is_global_scope() && if (initialization_scope->is_global_scope() &&
mode != LET && mode != CONST_HARMONY) { !IsLexicalVariableMode(mode)) {
// Compute the arguments for the runtime call. // Compute the arguments for the runtime call.
ZoneList<Expression*>* arguments = ZoneList<Expression*>* arguments =
new(zone()) ZoneList<Expression*>(3, zone()); new(zone()) ZoneList<Expression*>(3, zone());
......
...@@ -579,7 +579,7 @@ class Parser { ...@@ -579,7 +579,7 @@ class Parser {
return top_scope_->is_extended_mode(); return top_scope_->is_extended_mode();
} }
Scope* DeclarationScope(VariableMode mode) { Scope* DeclarationScope(VariableMode mode) {
return (mode == LET || mode == CONST_HARMONY) return IsLexicalVariableMode(mode)
? top_scope_ : top_scope_->DeclarationScope(); ? top_scope_ : top_scope_->DeclarationScope();
} }
......
...@@ -485,10 +485,7 @@ Variable* Scope::DeclareLocal(Handle<String> name, ...@@ -485,10 +485,7 @@ Variable* Scope::DeclareLocal(Handle<String> name,
// This function handles VAR and CONST modes. DYNAMIC variables are // This function handles VAR and CONST modes. DYNAMIC variables are
// introduces during variable allocation, INTERNAL variables are allocated // introduces during variable allocation, INTERNAL variables are allocated
// explicitly, and TEMPORARY variables are allocated via NewTemporary(). // explicitly, and TEMPORARY variables are allocated via NewTemporary().
ASSERT(mode == VAR || ASSERT(IsDeclaredVariableMode(mode));
mode == CONST ||
mode == CONST_HARMONY ||
mode == LET);
++num_var_or_const_; ++num_var_or_const_;
return variables_.Declare( return variables_.Declare(
this, name, mode, true, Variable::NORMAL, init_flag, interface); this, name, mode, true, Variable::NORMAL, init_flag, interface);
...@@ -1179,8 +1176,7 @@ bool Scope::MustAllocateInContext(Variable* var) { ...@@ -1179,8 +1176,7 @@ bool Scope::MustAllocateInContext(Variable* var) {
// catch-bound variables are always allocated in a context. // catch-bound variables are always allocated in a context.
if (var->mode() == TEMPORARY) return false; if (var->mode() == TEMPORARY) return false;
if (is_catch_scope() || is_block_scope() || is_module_scope()) return true; if (is_catch_scope() || is_block_scope() || is_module_scope()) return true;
if (is_global_scope() && (var->mode() == LET || var->mode() == CONST_HARMONY)) if (is_global_scope() && IsLexicalVariableMode(var->mode())) return true;
return true;
return var->has_forced_context_allocation() || return var->has_forced_context_allocation() ||
scope_calls_eval_ || scope_calls_eval_ ||
inner_scope_calls_eval_ || inner_scope_calls_eval_ ||
......
...@@ -479,16 +479,17 @@ const uint64_t kLastNonNaNInt64 = ...@@ -479,16 +479,17 @@ const uint64_t kLastNonNaNInt64 =
(static_cast<uint64_t>(kNaNOrInfinityLowerBoundUpper32) << 32); (static_cast<uint64_t>(kNaNOrInfinityLowerBoundUpper32) << 32);
// The order of this enum has to be kept in sync with the predicates below.
enum VariableMode { enum VariableMode {
// User declared variables: // User declared variables:
VAR, // declared via 'var', and 'function' declarations VAR, // declared via 'var', and 'function' declarations
CONST, // declared via 'const' declarations CONST, // declared via 'const' declarations
CONST_HARMONY, // declared via 'const' declarations in harmony mode
LET, // declared via 'let' declarations LET, // declared via 'let' declarations
CONST_HARMONY, // declared via 'const' declarations in harmony mode
// Variables introduced by the compiler: // Variables introduced by the compiler:
DYNAMIC, // always require dynamic lookup (we don't know DYNAMIC, // always require dynamic lookup (we don't know
// the declaration) // the declaration)
...@@ -510,6 +511,26 @@ enum VariableMode { ...@@ -510,6 +511,26 @@ enum VariableMode {
}; };
inline bool IsDynamicVariableMode(VariableMode mode) {
return mode >= DYNAMIC && mode <= DYNAMIC_LOCAL;
}
inline bool IsDeclaredVariableMode(VariableMode mode) {
return mode >= VAR && mode <= CONST_HARMONY;
}
inline bool IsLexicalVariableMode(VariableMode mode) {
return mode >= LET && mode <= CONST_HARMONY;
}
inline bool IsImmutableVariableMode(VariableMode mode) {
return mode == CONST || mode == CONST_HARMONY;
}
// ES6 Draft Rev3 10.2 specifies declarative environment records with mutable // ES6 Draft Rev3 10.2 specifies declarative environment records with mutable
// and immutable bindings that can be in two states: initialized and // and immutable bindings that can be in two states: initialized and
// uninitialized. In ES5 only immutable bindings have these two states. When // uninitialized. In ES5 only immutable bindings have these two states. When
......
...@@ -41,7 +41,7 @@ const char* Variable::Mode2String(VariableMode mode) { ...@@ -41,7 +41,7 @@ const char* Variable::Mode2String(VariableMode mode) {
switch (mode) { switch (mode) {
case VAR: return "VAR"; case VAR: return "VAR";
case CONST: return "CONST"; case CONST: return "CONST";
case CONST_HARMONY: return "CONST"; case CONST_HARMONY: return "CONST_HARMONY";
case LET: return "LET"; case LET: return "LET";
case DYNAMIC: return "DYNAMIC"; case DYNAMIC: return "DYNAMIC";
case DYNAMIC_GLOBAL: return "DYNAMIC_GLOBAL"; case DYNAMIC_GLOBAL: return "DYNAMIC_GLOBAL";
...@@ -84,7 +84,7 @@ Variable::Variable(Scope* scope, ...@@ -84,7 +84,7 @@ Variable::Variable(Scope* scope,
bool Variable::IsGlobalObjectProperty() const { bool Variable::IsGlobalObjectProperty() const {
// Temporaries are never global, they must always be allocated in the // Temporaries are never global, they must always be allocated in the
// activation frame. // activation frame.
return mode_ != TEMPORARY && mode_ != LET && mode_ != CONST_HARMONY return mode_ != TEMPORARY && !IsLexicalVariableMode(mode_)
&& scope_ != NULL && scope_->is_global_scope(); && scope_ != NULL && scope_->is_global_scope();
} }
......
...@@ -120,15 +120,8 @@ class Variable: public ZoneObject { ...@@ -120,15 +120,8 @@ class Variable: public ZoneObject {
bool IsLookupSlot() const { return location_ == LOOKUP; } bool IsLookupSlot() const { return location_ == LOOKUP; }
bool IsGlobalObjectProperty() const; bool IsGlobalObjectProperty() const;
bool is_dynamic() const { bool is_dynamic() const { return IsDynamicVariableMode(mode_); }
return (mode_ == DYNAMIC || bool is_const_mode() const { return IsImmutableVariableMode(mode_); }
mode_ == DYNAMIC_GLOBAL ||
mode_ == DYNAMIC_LOCAL);
}
bool is_const_mode() const {
return (mode_ == CONST ||
mode_ == CONST_HARMONY);
}
bool binding_needs_init() const { bool binding_needs_init() const {
return initialization_flag_ == kNeedsInitialization; return initialization_flag_ == kNeedsInitialization;
} }
......
...@@ -816,10 +816,9 @@ void FullCodeGenerator::VisitVariableDeclaration( ...@@ -816,10 +816,9 @@ void FullCodeGenerator::VisitVariableDeclaration(
__ push(rsi); __ push(rsi);
__ Push(variable->name()); __ Push(variable->name());
// Declaration nodes are always introduced in one of four modes. // Declaration nodes are always introduced in one of four modes.
ASSERT(mode == VAR || mode == LET || ASSERT(IsDeclaredVariableMode(mode));
mode == CONST || mode == CONST_HARMONY);
PropertyAttributes attr = PropertyAttributes attr =
(mode == CONST || mode == CONST_HARMONY) ? READ_ONLY : NONE; IsImmutableVariableMode(mode) ? READ_ONLY : NONE;
__ Push(Smi::FromInt(attr)); __ Push(Smi::FromInt(attr));
// Push initial value, if any. // Push initial value, if any.
// Note: For variables we must not push an initial value (such as // Note: For variables we must not push an initial value (such as
......
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