| OLD | NEW | 
|---|
| 1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 the V8 project authors. All rights reserved. | 
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without | 
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are | 
| 4 // met: | 4 // met: | 
| 5 // | 5 // | 
| 6 //     * Redistributions of source code must retain the above copyright | 6 //     * Redistributions of source code must retain the above copyright | 
| 7 //       notice, this list of conditions and the following disclaimer. | 7 //       notice, this list of conditions and the following disclaimer. | 
| 8 //     * Redistributions in binary form must reproduce the above | 8 //     * Redistributions in binary form must reproduce the above | 
| 9 //       copyright notice, this list of conditions and the following | 9 //       copyright notice, this list of conditions and the following | 
| 10 //       disclaimer in the documentation and/or other materials provided | 10 //       disclaimer in the documentation and/or other materials provided | 
| (...skipping 25 matching lines...) Expand all  Loading... | 
| 36 // independent of the V8 heap and internalizing them later. During parsing, | 36 // independent of the V8 heap and internalizing them later. During parsing, | 
| 37 // AstStrings and AstValues are created and stored outside the heap, in | 37 // AstStrings and AstValues are created and stored outside the heap, in | 
| 38 // AstValueFactory. After parsing, the strings and values are internalized | 38 // AstValueFactory. After parsing, the strings and values are internalized | 
| 39 // (moved into the V8 heap). | 39 // (moved into the V8 heap). | 
| 40 namespace v8 { | 40 namespace v8 { | 
| 41 namespace internal { | 41 namespace internal { | 
| 42 | 42 | 
| 43 class AstString : public ZoneObject { | 43 class AstString : public ZoneObject { | 
| 44  public: | 44  public: | 
| 45   explicit AstString(bool is_raw) | 45   explicit AstString(bool is_raw) | 
| 46       : bit_field_(IsRawStringBits::encode(is_raw)) {} | 46       : next_(nullptr), bit_field_(IsRawStringBits::encode(is_raw)) {} | 
| 47 |  | 
| 48   ~AstString() {} |  | 
| 49 | 47 | 
| 50   int length() const; | 48   int length() const; | 
| 51   bool IsEmpty() const { return length() == 0; } | 49   bool IsEmpty() const { return length() == 0; } | 
| 52 | 50 | 
| 53   // Puts the string into the V8 heap. | 51   // Puts the string into the V8 heap. | 
| 54   void Internalize(Isolate* isolate); | 52   void Internalize(Isolate* isolate); | 
| 55 | 53 | 
| 56   // This function can be called after internalizing. | 54   // This function can be called after internalizing. | 
| 57   V8_INLINE Handle<String> string() const { | 55   V8_INLINE Handle<String> string() const { | 
| 58     DCHECK(!string_.is_null()); | 56     DCHECK(!string_.is_null()); | 
| 59     return string_; | 57     return string_; | 
| 60   } | 58   } | 
| 61 | 59 | 
|  | 60   AstString** next_location() { return &next_; } | 
|  | 61   AstString* next() const { return next_; } | 
|  | 62 | 
| 62  protected: | 63  protected: | 
| 63   // This is null until the string is internalized. | 64   // Handle<String>::null() until internalized. | 
| 64   Handle<String> string_; | 65   Handle<String> string_; | 
|  | 66   AstString* next_; | 
| 65   // Poor-man's virtual dispatch to AstRawString / AstConsString. Takes less | 67   // Poor-man's virtual dispatch to AstRawString / AstConsString. Takes less | 
| 66   // memory. | 68   // memory. | 
| 67   class IsRawStringBits : public BitField<bool, 0, 1> {}; | 69   class IsRawStringBits : public BitField<bool, 0, 1> {}; | 
| 68   int bit_field_; | 70   int bit_field_; | 
| 69 }; | 71 }; | 
| 70 | 72 | 
| 71 | 73 | 
| 72 class AstRawString final : public AstString { | 74 class AstRawString final : public AstString { | 
| 73  public: | 75  public: | 
| 74   int length() const { | 76   int length() const { | 
| 75     if (is_one_byte()) return literal_bytes_.length(); | 77     if (is_one_byte()) return literal_bytes_.length(); | 
| 76     return literal_bytes_.length() / 2; | 78     return literal_bytes_.length() / 2; | 
| 77   } | 79   } | 
| 78 | 80 | 
| 79   int byte_length() const { return literal_bytes_.length(); } | 81   int byte_length() const { return literal_bytes_.length(); } | 
| 80 | 82 | 
| 81   void Internalize(Isolate* isolate); | 83   void Internalize(Isolate* isolate); | 
| 82 | 84 | 
| 83   bool AsArrayIndex(uint32_t* index, HandleDereferenceMode deref_mode = | 85   bool AsArrayIndex(uint32_t* index) const; | 
| 84                                          HandleDereferenceMode::kAllowed) const; |  | 
| 85 | 86 | 
| 86   // The string is not null-terminated, use length() to find out the length. | 87   // The string is not null-terminated, use length() to find out the length. | 
| 87   const unsigned char* raw_data() const { | 88   const unsigned char* raw_data() const { | 
| 88     return literal_bytes_.start(); | 89     return literal_bytes_.start(); | 
| 89   } | 90   } | 
| 90 | 91 | 
| 91   bool is_one_byte() const { return IsOneByteBits::decode(bit_field_); } | 92   bool is_one_byte() const { return IsOneByteBits::decode(bit_field_); } | 
| 92 | 93 | 
| 93   bool IsOneByteEqualTo(const char* data) const; | 94   bool IsOneByteEqualTo(const char* data) const; | 
| 94   uint16_t FirstCharacter() const { | 95   uint16_t FirstCharacter() const { | 
| (...skipping 79 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 174 | 175 | 
| 175   Smi* AsSmi() const { | 176   Smi* AsSmi() const { | 
| 176     CHECK_EQ(SMI, type_); | 177     CHECK_EQ(SMI, type_); | 
| 177     return Smi::FromInt(smi_); | 178     return Smi::FromInt(smi_); | 
| 178   } | 179   } | 
| 179 | 180 | 
| 180   bool EqualsString(const AstRawString* string) const { | 181   bool EqualsString(const AstRawString* string) const { | 
| 181     return type_ == STRING && string_ == string; | 182     return type_ == STRING && string_ == string; | 
| 182   } | 183   } | 
| 183 | 184 | 
| 184   bool IsPropertyName( | 185   bool IsPropertyName() const; | 
| 185       HandleDereferenceMode deref_mode = HandleDereferenceMode::kAllowed) const; |  | 
| 186 | 186 | 
| 187   bool BooleanValue() const; | 187   bool BooleanValue() const; | 
| 188 | 188 | 
| 189   bool IsSmi() const { return type_ == SMI; } | 189   bool IsSmi() const { return type_ == SMI; } | 
| 190   bool IsFalse() const { return type_ == BOOLEAN && !bool_; } | 190   bool IsFalse() const { return type_ == BOOLEAN && !bool_; } | 
| 191   bool IsTrue() const { return type_ == BOOLEAN && bool_; } | 191   bool IsTrue() const { return type_ == BOOLEAN && bool_; } | 
| 192   bool IsUndefined() const { return type_ == UNDEFINED; } | 192   bool IsUndefined() const { return type_ == UNDEFINED; } | 
| 193   bool IsTheHole() const { return type_ == THE_HOLE; } | 193   bool IsTheHole() const { return type_ == THE_HOLE; } | 
| 194   bool IsNull() const { return type_ == NULL_TYPE; } | 194   bool IsNull() const { return type_ == NULL_TYPE; } | 
| 195 | 195 | 
| 196   void Internalize(Isolate* isolate); | 196   void Internalize(Isolate* isolate); | 
| 197 | 197 | 
| 198   // Can be called after Internalize has been called. | 198   // Can be called after Internalize has been called. | 
| 199   V8_INLINE Handle<Object> value() const { | 199   V8_INLINE Handle<Object> value() const { | 
| 200     if (type_ == STRING) { | 200     if (type_ == STRING) { | 
| 201       return string_->string(); | 201       return string_->string(); | 
| 202     } | 202     } | 
| 203     DCHECK(!value_.is_null()); | 203     DCHECK(!value_.is_null()); | 
| 204     return value_; | 204     return value_; | 
| 205   } | 205   } | 
|  | 206   AstValue* next() const { return next_; } | 
|  | 207   void set_next(AstValue* next) { next_ = next; } | 
| 206 | 208 | 
| 207  private: | 209  private: | 
| 208   friend class AstValueFactory; | 210   friend class AstValueFactory; | 
| 209 | 211 | 
| 210   enum Type { | 212   enum Type { | 
| 211     STRING, | 213     STRING, | 
| 212     SYMBOL, | 214     SYMBOL, | 
| 213     NUMBER, | 215     NUMBER, | 
| 214     NUMBER_WITH_DOT, | 216     NUMBER_WITH_DOT, | 
| 215     SMI, | 217     SMI, | 
| 216     BOOLEAN, | 218     BOOLEAN, | 
| 217     NULL_TYPE, | 219     NULL_TYPE, | 
| 218     UNDEFINED, | 220     UNDEFINED, | 
| 219     THE_HOLE | 221     THE_HOLE | 
| 220   }; | 222   }; | 
| 221 | 223 | 
| 222   explicit AstValue(const AstRawString* s) : type_(STRING) { string_ = s; } | 224   explicit AstValue(const AstRawString* s) : type_(STRING), next_(nullptr) { | 
|  | 225     string_ = s; | 
|  | 226   } | 
| 223 | 227 | 
| 224   explicit AstValue(const char* name) : type_(SYMBOL) { symbol_name_ = name; } | 228   explicit AstValue(const char* name) : type_(SYMBOL), next_(nullptr) { | 
|  | 229     symbol_name_ = name; | 
|  | 230   } | 
| 225 | 231 | 
| 226   explicit AstValue(double n, bool with_dot) { | 232   explicit AstValue(double n, bool with_dot) : next_(nullptr) { | 
| 227     if (with_dot) { | 233     if (with_dot) { | 
| 228       type_ = NUMBER_WITH_DOT; | 234       type_ = NUMBER_WITH_DOT; | 
| 229       number_ = n; | 235       number_ = n; | 
| 230     } else { | 236     } else { | 
| 231       int int_value; | 237       int int_value; | 
| 232       if (DoubleToSmiInteger(n, &int_value)) { | 238       if (DoubleToSmiInteger(n, &int_value)) { | 
| 233         type_ = SMI; | 239         type_ = SMI; | 
| 234         smi_ = int_value; | 240         smi_ = int_value; | 
| 235       } else { | 241       } else { | 
| 236         type_ = NUMBER; | 242         type_ = NUMBER; | 
| 237         number_ = n; | 243         number_ = n; | 
| 238       } | 244       } | 
| 239     } | 245     } | 
| 240   } | 246   } | 
| 241 | 247 | 
| 242   AstValue(Type t, int i) : type_(t) { | 248   AstValue(Type t, int i) : type_(t), next_(nullptr) { | 
| 243     DCHECK(type_ == SMI); | 249     DCHECK(type_ == SMI); | 
| 244     smi_ = i; | 250     smi_ = i; | 
| 245   } | 251   } | 
| 246 | 252 | 
| 247   explicit AstValue(bool b) : type_(BOOLEAN) { bool_ = b; } | 253   explicit AstValue(bool b) : type_(BOOLEAN), next_(nullptr) { bool_ = b; } | 
| 248 | 254 | 
| 249   explicit AstValue(Type t) : type_(t) { | 255   explicit AstValue(Type t) : type_(t), next_(nullptr) { | 
| 250     DCHECK(t == NULL_TYPE || t == UNDEFINED || t == THE_HOLE); | 256     DCHECK(t == NULL_TYPE || t == UNDEFINED || t == THE_HOLE); | 
| 251   } | 257   } | 
| 252 | 258 | 
| 253   Type type_; | 259   Type type_; | 
| 254 | 260 | 
| 255   // Uninternalized value. | 261   // Uninternalized value. | 
| 256   union { | 262   union { | 
| 257     const AstRawString* string_; | 263     const AstRawString* string_; | 
| 258     double number_; | 264     double number_; | 
| 259     int smi_; | 265     int smi_; | 
| 260     bool bool_; | 266     bool bool_; | 
| 261     ZoneList<const AstRawString*>* strings_; | 267     const AstRawString* strings_; | 
| 262     const char* symbol_name_; | 268     const char* symbol_name_; | 
| 263   }; | 269   }; | 
| 264 | 270 | 
| 265   // Internalized value (empty before internalized). | 271   // Handle<String>::null() until internalized. | 
| 266   Handle<Object> value_; | 272   Handle<Object> value_; | 
|  | 273   AstValue* next_; | 
| 267 }; | 274 }; | 
| 268 | 275 | 
| 269 | 276 | 
| 270 // For generating constants. | 277 // For generating constants. | 
| 271 #define STRING_CONSTANTS(F)                     \ | 278 #define STRING_CONSTANTS(F)                     \ | 
| 272   F(anonymous_function, "(anonymous function)") \ | 279   F(anonymous_function, "(anonymous function)") \ | 
| 273   F(arguments, "arguments")                     \ | 280   F(arguments, "arguments")                     \ | 
| 274   F(async, "async")                             \ | 281   F(async, "async")                             \ | 
| 275   F(await, "await")                             \ | 282   F(await, "await")                             \ | 
| 276   F(constructor, "constructor")                 \ | 283   F(constructor, "constructor")                 \ | 
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 309   F(true_value)            \ | 316   F(true_value)            \ | 
| 310   F(false_value)           \ | 317   F(false_value)           \ | 
| 311   F(null_value)            \ | 318   F(null_value)            \ | 
| 312   F(undefined_value)       \ | 319   F(undefined_value)       \ | 
| 313   F(the_hole_value) | 320   F(the_hole_value) | 
| 314 | 321 | 
| 315 class AstValueFactory { | 322 class AstValueFactory { | 
| 316  public: | 323  public: | 
| 317   AstValueFactory(Zone* zone, uint32_t hash_seed) | 324   AstValueFactory(Zone* zone, uint32_t hash_seed) | 
| 318       : string_table_(AstRawStringCompare), | 325       : string_table_(AstRawStringCompare), | 
|  | 326         values_(nullptr), | 
|  | 327         strings_end_(&strings_), | 
| 319         zone_(zone), | 328         zone_(zone), | 
| 320         isolate_(NULL), | 329         isolate_(NULL), | 
| 321         hash_seed_(hash_seed) { | 330         hash_seed_(hash_seed) { | 
|  | 331     ResetStrings(); | 
| 322 #define F(name, str) name##_string_ = NULL; | 332 #define F(name, str) name##_string_ = NULL; | 
| 323     STRING_CONSTANTS(F) | 333     STRING_CONSTANTS(F) | 
| 324 #undef F | 334 #undef F | 
| 325 #define F(name) name##_ = NULL; | 335 #define F(name) name##_ = NULL; | 
| 326     OTHER_CONSTANTS(F) | 336     OTHER_CONSTANTS(F) | 
| 327 #undef F | 337 #undef F | 
| 328   } | 338   } | 
| 329 | 339 | 
| 330   Zone* zone() const { return zone_; } | 340   Zone* zone() const { return zone_; } | 
| 331 | 341 | 
| (...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 366   const AstValue* NewSymbol(const char* name); | 376   const AstValue* NewSymbol(const char* name); | 
| 367   const AstValue* NewNumber(double number, bool with_dot = false); | 377   const AstValue* NewNumber(double number, bool with_dot = false); | 
| 368   const AstValue* NewSmi(int number); | 378   const AstValue* NewSmi(int number); | 
| 369   const AstValue* NewBoolean(bool b); | 379   const AstValue* NewBoolean(bool b); | 
| 370   const AstValue* NewStringList(ZoneList<const AstRawString*>* strings); | 380   const AstValue* NewStringList(ZoneList<const AstRawString*>* strings); | 
| 371   const AstValue* NewNull(); | 381   const AstValue* NewNull(); | 
| 372   const AstValue* NewUndefined(); | 382   const AstValue* NewUndefined(); | 
| 373   const AstValue* NewTheHole(); | 383   const AstValue* NewTheHole(); | 
| 374 | 384 | 
| 375  private: | 385  private: | 
|  | 386   AstValue* AddValue(AstValue* value) { | 
|  | 387     if (isolate_) { | 
|  | 388       value->Internalize(isolate_); | 
|  | 389     } else { | 
|  | 390       value->set_next(values_); | 
|  | 391       values_ = value; | 
|  | 392     } | 
|  | 393     return value; | 
|  | 394   } | 
|  | 395   AstString* AddString(AstString* string) { | 
|  | 396     if (isolate_) { | 
|  | 397       string->Internalize(isolate_); | 
|  | 398     } else { | 
|  | 399       *strings_end_ = string; | 
|  | 400       strings_end_ = string->next_location(); | 
|  | 401     } | 
|  | 402     return string; | 
|  | 403   } | 
|  | 404   void ResetStrings() { | 
|  | 405     strings_ = nullptr; | 
|  | 406     strings_end_ = &strings_; | 
|  | 407   } | 
| 376   AstRawString* GetOneByteStringInternal(Vector<const uint8_t> literal); | 408   AstRawString* GetOneByteStringInternal(Vector<const uint8_t> literal); | 
| 377   AstRawString* GetTwoByteStringInternal(Vector<const uint16_t> literal); | 409   AstRawString* GetTwoByteStringInternal(Vector<const uint16_t> literal); | 
| 378   AstRawString* GetString(uint32_t hash, bool is_one_byte, | 410   AstRawString* GetString(uint32_t hash, bool is_one_byte, | 
| 379                           Vector<const byte> literal_bytes); | 411                           Vector<const byte> literal_bytes); | 
| 380 | 412 | 
| 381   static bool AstRawStringCompare(void* a, void* b); | 413   static bool AstRawStringCompare(void* a, void* b); | 
| 382 | 414 | 
| 383   // All strings are copied here, one after another (no NULLs inbetween). | 415   // All strings are copied here, one after another (no NULLs inbetween). | 
| 384   base::HashMap string_table_; | 416   base::HashMap string_table_; | 
| 385   // For keeping track of all AstValues and AstRawStrings we've created (so that | 417   // For keeping track of all AstValues and AstRawStrings we've created (so that | 
| 386   // they can be internalized later). | 418   // they can be internalized later). | 
| 387   List<AstValue*> values_; | 419   AstValue* values_; | 
| 388   List<AstString*> strings_; | 420   // We need to keep track of strings_ in order, since cons strings require | 
|  | 421   // their members to be internalized first. | 
|  | 422   AstString* strings_; | 
|  | 423   AstString** strings_end_; | 
| 389   Zone* zone_; | 424   Zone* zone_; | 
| 390   Isolate* isolate_; | 425   Isolate* isolate_; | 
| 391 | 426 | 
| 392   uint32_t hash_seed_; | 427   uint32_t hash_seed_; | 
| 393 | 428 | 
| 394 #define F(name, str) const AstRawString* name##_string_; | 429 #define F(name, str) const AstRawString* name##_string_; | 
| 395   STRING_CONSTANTS(F) | 430   STRING_CONSTANTS(F) | 
| 396 #undef F | 431 #undef F | 
| 397 | 432 | 
| 398 #define F(name) AstValue* name##_; | 433 #define F(name) AstValue* name##_; | 
| 399   OTHER_CONSTANTS(F) | 434   OTHER_CONSTANTS(F) | 
| 400 #undef F | 435 #undef F | 
| 401 }; | 436 }; | 
| 402 }  // namespace internal | 437 }  // namespace internal | 
| 403 }  // namespace v8 | 438 }  // namespace v8 | 
| 404 | 439 | 
| 405 #undef STRING_CONSTANTS | 440 #undef STRING_CONSTANTS | 
| 406 #undef OTHER_CONSTANTS | 441 #undef OTHER_CONSTANTS | 
| 407 | 442 | 
| 408 #endif  // V8_AST_AST_VALUE_FACTORY_H_ | 443 #endif  // V8_AST_AST_VALUE_FACTORY_H_ | 
| OLD | NEW | 
|---|