| OLD | NEW | 
|---|
| 1 // Copyright 2011 the V8 project authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be | 
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. | 
| 4 | 4 | 
| 5 #include "v8.h" | 5 #include "v8.h" | 
| 6 | 6 | 
| 7 #include "ast.h" | 7 #include "ast.h" | 
| 8 #include "func-name-inferrer.h" | 8 #include "func-name-inferrer.h" | 
| 9 #include "list-inl.h" | 9 #include "list-inl.h" | 
|  | 10 #include "parser-symbol-table.h" | 
| 10 | 11 | 
| 11 namespace v8 { | 12 namespace v8 { | 
| 12 namespace internal { | 13 namespace internal { | 
| 13 | 14 | 
| 14 FuncNameInferrer::FuncNameInferrer(Isolate* isolate, Zone* zone) | 15 FuncNameInferrer::FuncNameInferrer(ParserSymbolTable* symbol_table, Zone* zone) | 
| 15     : isolate_(isolate), | 16     : symbol_table_(symbol_table), | 
| 16       entries_stack_(10, zone), | 17       entries_stack_(10, zone), | 
| 17       names_stack_(5, zone), | 18       names_stack_(5, zone), | 
| 18       funcs_to_infer_(4, zone), | 19       funcs_to_infer_(4, zone), | 
| 19       zone_(zone) { | 20       zone_(zone) { | 
| 20 } | 21 } | 
| 21 | 22 | 
| 22 | 23 | 
| 23 void FuncNameInferrer::PushEnclosingName(Handle<String> name) { | 24 void FuncNameInferrer::PushEnclosingName(ParserSymbolTable::Symbol* name) { | 
| 24   // Enclosing name is a name of a constructor function. To check | 25   // Enclosing name is a name of a constructor function. To check | 
| 25   // that it is really a constructor, we check that it is not empty | 26   // that it is really a constructor, we check that it is not empty | 
| 26   // and starts with a capital letter. | 27   // and starts with a capital letter. | 
| 27   if (name->length() > 0 && Runtime::IsUpperCaseChar( | 28   if (name->literal_bytes.length() > 0 && | 
| 28           isolate()->runtime_state(), name->Get(0))) { | 29       unibrow::Uppercase::Is(name->literal_bytes[0])) { | 
| 29     names_stack_.Add(Name(name, kEnclosingConstructorName), zone()); | 30     names_stack_.Add(Name(name, kEnclosingConstructorName), zone()); | 
| 30   } | 31   } | 
| 31 } | 32 } | 
| 32 | 33 | 
| 33 | 34 | 
| 34 void FuncNameInferrer::PushLiteralName(Handle<String> name) { | 35 void FuncNameInferrer::PushLiteralName(ParserSymbolTable::Symbol* name) { | 
| 35   if (IsOpen() && | 36   if (IsOpen() && name != symbol_table_->prototype_string()) { | 
| 36       !String::Equals(isolate()->factory()->prototype_string(), name)) { |  | 
| 37     names_stack_.Add(Name(name, kLiteralName), zone()); | 37     names_stack_.Add(Name(name, kLiteralName), zone()); | 
| 38   } | 38   } | 
| 39 } | 39 } | 
| 40 | 40 | 
| 41 | 41 | 
| 42 void FuncNameInferrer::PushVariableName(Handle<String> name) { | 42 void FuncNameInferrer::PushVariableName(ParserSymbolTable::Symbol* name) { | 
| 43   if (IsOpen() && | 43   if (IsOpen() && name != symbol_table_->dot_result_string()) { | 
| 44       !String::Equals(isolate()->factory()->dot_result_string(), name)) { |  | 
| 45     names_stack_.Add(Name(name, kVariableName), zone()); | 44     names_stack_.Add(Name(name, kVariableName), zone()); | 
| 46   } | 45   } | 
| 47 } | 46 } | 
| 48 | 47 | 
| 49 | 48 | 
| 50 Handle<String> FuncNameInferrer::MakeNameFromStack() { | 49 ParserSymbolTable::Symbol* FuncNameInferrer::MakeNameFromStack() { | 
| 51   return MakeNameFromStackHelper(0, isolate()->factory()->empty_string()); | 50   // First see how many names we will use. | 
| 52 } | 51   int length = 0; | 
| 53 | 52   bool one_byte = true; | 
| 54 | 53   int pos = 0; | 
| 55 Handle<String> FuncNameInferrer::MakeNameFromStackHelper(int pos, | 54   while (pos < names_stack_.length()) { | 
| 56                                                          Handle<String> prev) { | 55     if (pos < names_stack_.length() - 1 && | 
| 57   if (pos >= names_stack_.length()) return prev; | 56         names_stack_.at(pos).type == kVariableName && | 
| 58   if (pos < names_stack_.length() - 1 && | 57         names_stack_.at(pos + 1).type == kVariableName) { | 
| 59       names_stack_.at(pos).type == kVariableName && | 58       // Skip consecutive variable declarations. | 
| 60       names_stack_.at(pos + 1).type == kVariableName) { | 59       ++pos; | 
| 61     // Skip consecutive variable declarations. | 60       continue; | 
| 62     return MakeNameFromStackHelper(pos + 1, prev); |  | 
| 63   } else { |  | 
| 64     if (prev->length() > 0) { |  | 
| 65       Handle<String> name = names_stack_.at(pos).name; |  | 
| 66       if (prev->length() + name->length() + 1 > String::kMaxLength) return prev; |  | 
| 67       Factory* factory = isolate()->factory(); |  | 
| 68       Handle<String> curr = |  | 
| 69           factory->NewConsString(factory->dot_string(), name).ToHandleChecked(); |  | 
| 70       curr = factory->NewConsString(prev, curr).ToHandleChecked(); |  | 
| 71       return MakeNameFromStackHelper(pos + 1, curr); |  | 
| 72     } else { |  | 
| 73       return MakeNameFromStackHelper(pos + 1, names_stack_.at(pos).name); |  | 
| 74     } | 61     } | 
|  | 62     int cur_length = names_stack_.at(pos).name->literal_bytes.length(); | 
|  | 63     if (length + 1 + cur_length > String::kMaxLength) { | 
|  | 64       break; | 
|  | 65     } | 
|  | 66     if (length == 0) { | 
|  | 67       length = cur_length; | 
|  | 68     } else {  // Add the . between names. | 
|  | 69       length += (1 + cur_length); | 
|  | 70     } | 
|  | 71     one_byte = one_byte && names_stack_.at(pos).name->is_one_byte; | 
|  | 72     ++pos; | 
| 75   } | 73   } | 
|  | 74   ParserSymbolTable::Symbol* to_return = NULL; | 
|  | 75   if (one_byte) { | 
|  | 76     Vector<uint8_t> new_name = Vector<uint8_t>::New(length); | 
|  | 77     int name_pos = 0; | 
|  | 78     const char* dot = "."; | 
|  | 79     for (int i = 0; i < pos; ++i) { | 
|  | 80       if (i < names_stack_.length() - 1 && | 
|  | 81           names_stack_.at(i).type == kVariableName && | 
|  | 82           names_stack_.at(i + 1).type == kVariableName) { | 
|  | 83         continue; | 
|  | 84       } | 
|  | 85       if (name_pos != 0) { | 
|  | 86         CopyChars(new_name.start() + name_pos, dot, 1); | 
|  | 87         ++name_pos; | 
|  | 88       } | 
|  | 89       CopyChars(new_name.start() + name_pos, | 
|  | 90                 names_stack_.at(i).name->literal_bytes.start(), | 
|  | 91                 names_stack_.at(i).name->literal_bytes.length()); | 
|  | 92       name_pos += names_stack_.at(i).name->literal_bytes.length(); | 
|  | 93     } | 
|  | 94     to_return = symbol_table_->GetOneByteSymbol(Vector<const uint8_t>( | 
|  | 95         reinterpret_cast<const uint8_t*>(new_name.start()), | 
|  | 96         new_name.length())); | 
|  | 97     new_name.Dispose(); | 
|  | 98   } | 
|  | 99   // FIXME: do it for two-byte too. | 
|  | 100   return to_return; | 
| 76 } | 101 } | 
| 77 | 102 | 
| 78 | 103 | 
| 79 void FuncNameInferrer::InferFunctionsNames() { | 104 void FuncNameInferrer::InferFunctionsNames() { | 
| 80   Handle<String> func_name = MakeNameFromStack(); | 105   ParserSymbolTable::Symbol* func_name = MakeNameFromStack(); | 
| 81   for (int i = 0; i < funcs_to_infer_.length(); ++i) { | 106   for (int i = 0; i < funcs_to_infer_.length(); ++i) { | 
| 82     funcs_to_infer_[i]->set_inferred_name(func_name); | 107     funcs_to_infer_[i]->set_raw_inferred_name(func_name); | 
| 83   } | 108   } | 
| 84   funcs_to_infer_.Rewind(0); | 109   funcs_to_infer_.Rewind(0); | 
| 85 } | 110 } | 
| 86 | 111 | 
| 87 | 112 | 
| 88 } }  // namespace v8::internal | 113 } }  // namespace v8::internal | 
| OLD | NEW | 
|---|