Home | History | Annotate | Download | only in ast
      1 // Copyright 2011 the V8 project authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #ifndef V8_AST_VARIABLES_H_
      6 #define V8_AST_VARIABLES_H_
      7 
      8 #include "src/ast/ast-value-factory.h"
      9 #include "src/globals.h"
     10 #include "src/zone/zone.h"
     11 
     12 namespace v8 {
     13 namespace internal {
     14 
     15 // The AST refers to variables via VariableProxies - placeholders for the actual
     16 // variables. Variables themselves are never directly referred to from the AST,
     17 // they are maintained by scopes, and referred to from VariableProxies and Slots
     18 // after binding and variable allocation.
     19 class Variable final : public ZoneObject {
     20  public:
     21   Variable(Scope* scope, const AstRawString* name, VariableMode mode,
     22            VariableKind kind, InitializationFlag initialization_flag,
     23            MaybeAssignedFlag maybe_assigned_flag = kNotAssigned)
     24       : scope_(scope),
     25         name_(name),
     26         local_if_not_shadowed_(nullptr),
     27         next_(nullptr),
     28         index_(-1),
     29         initializer_position_(kNoSourcePosition),
     30         bit_field_(MaybeAssignedFlagField::encode(maybe_assigned_flag) |
     31                    InitializationFlagField::encode(initialization_flag) |
     32                    VariableModeField::encode(mode) |
     33                    IsUsedField::encode(false) |
     34                    ForceContextAllocationField::encode(false) |
     35                    ForceHoleInitializationField::encode(false) |
     36                    LocationField::encode(VariableLocation::UNALLOCATED) |
     37                    VariableKindField::encode(kind)) {
     38     // Var declared variables never need initialization.
     39     DCHECK(!(mode == VariableMode::kVar &&
     40              initialization_flag == kNeedsInitialization));
     41   }
     42 
     43   explicit Variable(Variable* other);
     44 
     45   // The source code for an eval() call may refer to a variable that is
     46   // in an outer scope about which we don't know anything (it may not
     47   // be the script scope). scope() is nullptr in that case. Currently the
     48   // scope is only used to follow the context chain length.
     49   Scope* scope() const { return scope_; }
     50 
     51   // This is for adjusting the scope of temporaries used when desugaring
     52   // parameter initializers.
     53   void set_scope(Scope* scope) { scope_ = scope; }
     54 
     55   Handle<String> name() const { return name_->string(); }
     56   const AstRawString* raw_name() const { return name_; }
     57   VariableMode mode() const { return VariableModeField::decode(bit_field_); }
     58   bool has_forced_context_allocation() const {
     59     return ForceContextAllocationField::decode(bit_field_);
     60   }
     61   void ForceContextAllocation() {
     62     DCHECK(IsUnallocated() || IsContextSlot() ||
     63            location() == VariableLocation::MODULE);
     64     bit_field_ = ForceContextAllocationField::update(bit_field_, true);
     65   }
     66   bool is_used() { return IsUsedField::decode(bit_field_); }
     67   void set_is_used() { bit_field_ = IsUsedField::update(bit_field_, true); }
     68   MaybeAssignedFlag maybe_assigned() const {
     69     return MaybeAssignedFlagField::decode(bit_field_);
     70   }
     71   void set_maybe_assigned() {
     72     bit_field_ = MaybeAssignedFlagField::update(bit_field_, kMaybeAssigned);
     73   }
     74 
     75   int initializer_position() { return initializer_position_; }
     76   void set_initializer_position(int pos) { initializer_position_ = pos; }
     77 
     78   bool IsUnallocated() const {
     79     return location() == VariableLocation::UNALLOCATED;
     80   }
     81   bool IsParameter() const { return location() == VariableLocation::PARAMETER; }
     82   bool IsStackLocal() const { return location() == VariableLocation::LOCAL; }
     83   bool IsStackAllocated() const { return IsParameter() || IsStackLocal(); }
     84   bool IsContextSlot() const { return location() == VariableLocation::CONTEXT; }
     85   bool IsLookupSlot() const { return location() == VariableLocation::LOOKUP; }
     86   bool IsGlobalObjectProperty() const;
     87 
     88   bool is_dynamic() const { return IsDynamicVariableMode(mode()); }
     89 
     90   // Returns the InitializationFlag this Variable was created with.
     91   // Scope analysis may allow us to relax this initialization
     92   // requirement, which will be reflected in the return value of
     93   // binding_needs_init().
     94   InitializationFlag initialization_flag() const {
     95     return InitializationFlagField::decode(bit_field_);
     96   }
     97 
     98   // Whether this variable needs to be initialized with the hole at
     99   // declaration time. Only returns valid results after scope analysis.
    100   bool binding_needs_init() const {
    101     DCHECK_IMPLIES(initialization_flag() == kNeedsInitialization,
    102                    IsLexicalVariableMode(mode()));
    103     DCHECK_IMPLIES(ForceHoleInitializationField::decode(bit_field_),
    104                    initialization_flag() == kNeedsInitialization);
    105 
    106     // Always initialize if hole initialization was forced during
    107     // scope analysis.
    108     if (ForceHoleInitializationField::decode(bit_field_)) return true;
    109 
    110     // If initialization was not forced, no need for initialization
    111     // for stack allocated variables, since UpdateNeedsHoleCheck()
    112     // in scopes.cc has proven that no VariableProxy refers to
    113     // this variable in such a way that a runtime hole check
    114     // would be generated.
    115     if (IsStackAllocated()) return false;
    116 
    117     // Otherwise, defer to the flag set when this Variable was constructed.
    118     return initialization_flag() == kNeedsInitialization;
    119   }
    120 
    121   // Called during scope analysis when a VariableProxy is found to
    122   // reference this Variable in such a way that a hole check will
    123   // be required at runtime.
    124   void ForceHoleInitialization() {
    125     DCHECK_EQ(kNeedsInitialization, initialization_flag());
    126     DCHECK(IsLexicalVariableMode(mode()));
    127     bit_field_ = ForceHoleInitializationField::update(bit_field_, true);
    128   }
    129 
    130   bool throw_on_const_assignment(LanguageMode language_mode) const {
    131     return kind() != SLOPPY_FUNCTION_NAME_VARIABLE || is_strict(language_mode);
    132   }
    133 
    134   bool is_function() const { return kind() == FUNCTION_VARIABLE; }
    135   bool is_this() const { return kind() == THIS_VARIABLE; }
    136   bool is_sloppy_function_name() const {
    137     return kind() == SLOPPY_FUNCTION_NAME_VARIABLE;
    138   }
    139 
    140   Variable* local_if_not_shadowed() const {
    141     DCHECK(mode() == VariableMode::kDynamicLocal &&
    142            local_if_not_shadowed_ != nullptr);
    143     return local_if_not_shadowed_;
    144   }
    145 
    146   void set_local_if_not_shadowed(Variable* local) {
    147     local_if_not_shadowed_ = local;
    148   }
    149 
    150   VariableLocation location() const {
    151     return LocationField::decode(bit_field_);
    152   }
    153   VariableKind kind() const { return VariableKindField::decode(bit_field_); }
    154 
    155   int index() const { return index_; }
    156 
    157   bool IsReceiver() const {
    158     DCHECK(IsParameter());
    159 
    160     return index_ == -1;
    161   }
    162 
    163   bool IsExport() const {
    164     DCHECK_EQ(location(), VariableLocation::MODULE);
    165     DCHECK_NE(index(), 0);
    166     return index() > 0;
    167   }
    168 
    169   void AllocateTo(VariableLocation location, int index) {
    170     DCHECK(IsUnallocated() ||
    171            (this->location() == location && this->index() == index));
    172     DCHECK_IMPLIES(location == VariableLocation::MODULE, index != 0);
    173     bit_field_ = LocationField::update(bit_field_, location);
    174     DCHECK_EQ(location, this->location());
    175     index_ = index;
    176   }
    177 
    178   static InitializationFlag DefaultInitializationFlag(VariableMode mode) {
    179     DCHECK(IsDeclaredVariableMode(mode));
    180     return mode == VariableMode::kVar ? kCreatedInitialized
    181                                       : kNeedsInitialization;
    182   }
    183 
    184   typedef ThreadedList<Variable> List;
    185 
    186  private:
    187   Scope* scope_;
    188   const AstRawString* name_;
    189 
    190   // If this field is set, this variable references the stored locally bound
    191   // variable, but it might be shadowed by variable bindings introduced by
    192   // sloppy 'eval' calls between the reference scope (inclusive) and the
    193   // binding scope (exclusive).
    194   Variable* local_if_not_shadowed_;
    195   Variable* next_;
    196   int index_;
    197   int initializer_position_;
    198   uint16_t bit_field_;
    199 
    200   class VariableModeField : public BitField16<VariableMode, 0, 3> {};
    201   class VariableKindField
    202       : public BitField16<VariableKind, VariableModeField::kNext, 3> {};
    203   class LocationField
    204       : public BitField16<VariableLocation, VariableKindField::kNext, 3> {};
    205   class ForceContextAllocationField
    206       : public BitField16<bool, LocationField::kNext, 1> {};
    207   class IsUsedField
    208       : public BitField16<bool, ForceContextAllocationField::kNext, 1> {};
    209   class InitializationFlagField
    210       : public BitField16<InitializationFlag, IsUsedField::kNext, 1> {};
    211   class ForceHoleInitializationField
    212       : public BitField16<bool, InitializationFlagField::kNext, 1> {};
    213   class MaybeAssignedFlagField
    214       : public BitField16<MaybeAssignedFlag,
    215                           ForceHoleInitializationField::kNext, 1> {};
    216   Variable** next() { return &next_; }
    217   friend List;
    218 };
    219 }  // namespace internal
    220 }  // namespace v8
    221 
    222 #endif  // V8_AST_VARIABLES_H_
    223