| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 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 "src/runtime/runtime.h" | 5 #include "src/runtime/runtime.h" |
| 6 | 6 |
| 7 #include "src/assembler.h" | 7 #include "src/assembler.h" |
| 8 #include "src/base/hashmap.h" |
| 8 #include "src/contexts.h" | 9 #include "src/contexts.h" |
| 9 #include "src/handles-inl.h" | 10 #include "src/handles-inl.h" |
| 10 #include "src/heap/heap.h" | 11 #include "src/heap/heap.h" |
| 11 #include "src/isolate.h" | 12 #include "src/isolate.h" |
| 12 #include "src/runtime/runtime-utils.h" | 13 #include "src/runtime/runtime-utils.h" |
| 13 | 14 |
| 14 namespace v8 { | 15 namespace v8 { |
| 15 namespace internal { | 16 namespace internal { |
| 16 | 17 |
| 17 // Header of runtime functions. | 18 // Header of runtime functions. |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 50 , | 51 , |
| 51 | 52 |
| 52 static const Runtime::Function kIntrinsicFunctions[] = { | 53 static const Runtime::Function kIntrinsicFunctions[] = { |
| 53 FOR_EACH_INTRINSIC(F) | 54 FOR_EACH_INTRINSIC(F) |
| 54 FOR_EACH_INTRINSIC(I) | 55 FOR_EACH_INTRINSIC(I) |
| 55 }; | 56 }; |
| 56 | 57 |
| 57 #undef I | 58 #undef I |
| 58 #undef F | 59 #undef F |
| 59 | 60 |
| 61 namespace { |
| 60 | 62 |
| 61 void Runtime::InitializeIntrinsicFunctionNames(Isolate* isolate, | 63 V8_DECLARE_ONCE(initialize_function_name_map_once); |
| 62 Handle<NameDictionary> dict) { | 64 static const base::HashMap* kRuntimeFunctionNameMap; |
| 63 DCHECK(dict->NumberOfElements() == 0); | 65 |
| 64 HandleScope scope(isolate); | 66 struct IntrinsicFunctionIdentifier { |
| 65 for (int i = 0; i < kNumFunctions; ++i) { | 67 IntrinsicFunctionIdentifier(const unsigned char* data, const int length) |
| 66 const char* name = kIntrinsicFunctions[i].name; | 68 : data_(data), length_(length) {} |
| 67 if (name == NULL) continue; | 69 |
| 68 Handle<NameDictionary> new_dict = NameDictionary::Add( | 70 static bool Match(void* key1, void* key2) { |
| 69 dict, isolate->factory()->InternalizeUtf8String(name), | 71 const IntrinsicFunctionIdentifier* lhs = |
| 70 Handle<Smi>(Smi::FromInt(i), isolate), PropertyDetails::Empty()); | 72 static_cast<IntrinsicFunctionIdentifier*>(key1); |
| 71 // The dictionary does not need to grow. | 73 const IntrinsicFunctionIdentifier* rhs = |
| 72 CHECK(new_dict.is_identical_to(dict)); | 74 static_cast<IntrinsicFunctionIdentifier*>(key2); |
| 75 if (lhs->length_ != rhs->length_) return false; |
| 76 return CompareCharsUnsigned(reinterpret_cast<const uint8_t*>(lhs->data_), |
| 77 reinterpret_cast<const uint8_t*>(rhs->data_), |
| 78 rhs->length_) == 0; |
| 73 } | 79 } |
| 80 |
| 81 uint32_t Hash() { |
| 82 return StringHasher::HashSequentialString<uint8_t>( |
| 83 data_, length_, v8::internal::kZeroHashSeed); |
| 84 } |
| 85 |
| 86 const unsigned char* data_; |
| 87 const int length_; |
| 88 }; |
| 89 |
| 90 void InitializeIntrinsicFunctionNames() { |
| 91 base::HashMap* function_name_map = |
| 92 new base::HashMap(IntrinsicFunctionIdentifier::Match); |
| 93 for (size_t i = 0; i < arraysize(kIntrinsicFunctions); ++i) { |
| 94 const Runtime::Function* function = &kIntrinsicFunctions[i]; |
| 95 IntrinsicFunctionIdentifier* identifier = new IntrinsicFunctionIdentifier( |
| 96 reinterpret_cast<const unsigned char*>(function->name), |
| 97 static_cast<int>(strlen(function->name))); |
| 98 base::HashMap::Entry* entry = |
| 99 function_name_map->InsertNew(identifier, identifier->Hash()); |
| 100 entry->value = const_cast<Runtime::Function*>(function); |
| 101 } |
| 102 kRuntimeFunctionNameMap = function_name_map; |
| 74 } | 103 } |
| 75 | 104 |
| 105 } // namespace |
| 76 | 106 |
| 77 const Runtime::Function* Runtime::FunctionForName(Handle<String> name) { | 107 const Runtime::Function* Runtime::FunctionForName(const unsigned char* name, |
| 78 Heap* heap = name->GetHeap(); | 108 int length) { |
| 79 int entry = heap->intrinsic_function_names()->FindEntry(name); | 109 base::CallOnce(&initialize_function_name_map_once, |
| 80 if (entry != kNotFound) { | 110 &InitializeIntrinsicFunctionNames); |
| 81 Object* smi_index = heap->intrinsic_function_names()->ValueAt(entry); | 111 IntrinsicFunctionIdentifier identifier(name, length); |
| 82 int function_index = Smi::cast(smi_index)->value(); | 112 base::HashMap::Entry* entry = |
| 83 return &(kIntrinsicFunctions[function_index]); | 113 kRuntimeFunctionNameMap->Lookup(&identifier, identifier.Hash()); |
| 114 if (entry) { |
| 115 return reinterpret_cast<Function*>(entry->value); |
| 84 } | 116 } |
| 85 return NULL; | 117 return NULL; |
| 86 } | 118 } |
| 87 | 119 |
| 88 | 120 |
| 89 const Runtime::Function* Runtime::FunctionForEntry(Address entry) { | 121 const Runtime::Function* Runtime::FunctionForEntry(Address entry) { |
| 90 for (size_t i = 0; i < arraysize(kIntrinsicFunctions); ++i) { | 122 for (size_t i = 0; i < arraysize(kIntrinsicFunctions); ++i) { |
| 91 if (entry == kIntrinsicFunctions[i].entry) { | 123 if (entry == kIntrinsicFunctions[i].entry) { |
| 92 return &(kIntrinsicFunctions[i]); | 124 return &(kIntrinsicFunctions[i]); |
| 93 } | 125 } |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 126 } | 158 } |
| 127 | 159 |
| 128 | 160 |
| 129 std::ostream& operator<<(std::ostream& os, Runtime::FunctionId id) { | 161 std::ostream& operator<<(std::ostream& os, Runtime::FunctionId id) { |
| 130 return os << Runtime::FunctionForId(id)->name; | 162 return os << Runtime::FunctionForId(id)->name; |
| 131 } | 163 } |
| 132 | 164 |
| 133 | 165 |
| 134 } // namespace internal | 166 } // namespace internal |
| 135 } // namespace v8 | 167 } // namespace v8 |
| OLD | NEW |