OLD | NEW |
(Empty) | |
| 1 // Copyright 2016 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 #include "src/ast/context-slot-cache.h" |
| 6 |
| 7 #include <stdlib.h> |
| 8 |
| 9 #include "src/ast/scopes.h" |
| 10 #include "src/bootstrapper.h" |
| 11 |
| 12 namespace v8 { |
| 13 namespace internal { |
| 14 |
| 15 int ContextSlotCache::Hash(Object* data, String* name) { |
| 16 // Uses only lower 32 bits if pointers are larger. |
| 17 uintptr_t addr_hash = |
| 18 static_cast<uint32_t>(reinterpret_cast<uintptr_t>(data)) >> 2; |
| 19 return static_cast<int>((addr_hash ^ name->Hash()) % kLength); |
| 20 } |
| 21 |
| 22 int ContextSlotCache::Lookup(Object* data, String* name, VariableMode* mode, |
| 23 InitializationFlag* init_flag, |
| 24 MaybeAssignedFlag* maybe_assigned_flag) { |
| 25 int index = Hash(data, name); |
| 26 Key& key = keys_[index]; |
| 27 if ((key.data == data) && key.name->Equals(name)) { |
| 28 Value result(values_[index]); |
| 29 if (mode != nullptr) *mode = result.mode(); |
| 30 if (init_flag != nullptr) *init_flag = result.initialization_flag(); |
| 31 if (maybe_assigned_flag != nullptr) |
| 32 *maybe_assigned_flag = result.maybe_assigned_flag(); |
| 33 return result.index() + kNotFound; |
| 34 } |
| 35 return kNotFound; |
| 36 } |
| 37 |
| 38 void ContextSlotCache::Update(Handle<Object> data, Handle<String> name, |
| 39 VariableMode mode, InitializationFlag init_flag, |
| 40 MaybeAssignedFlag maybe_assigned_flag, |
| 41 int slot_index) { |
| 42 DisallowHeapAllocation no_gc; |
| 43 Handle<String> internalized_name; |
| 44 DCHECK(slot_index > kNotFound); |
| 45 if (StringTable::InternalizeStringIfExists(name->GetIsolate(), name) |
| 46 .ToHandle(&internalized_name)) { |
| 47 int index = Hash(*data, *internalized_name); |
| 48 Key& key = keys_[index]; |
| 49 key.data = *data; |
| 50 key.name = *internalized_name; |
| 51 // Please note value only takes a uint as index. |
| 52 values_[index] = |
| 53 Value(mode, init_flag, maybe_assigned_flag, slot_index - kNotFound) |
| 54 .raw(); |
| 55 #ifdef DEBUG |
| 56 ValidateEntry(data, name, mode, init_flag, maybe_assigned_flag, slot_index); |
| 57 #endif |
| 58 } |
| 59 } |
| 60 |
| 61 void ContextSlotCache::Clear() { |
| 62 for (int index = 0; index < kLength; index++) keys_[index].data = nullptr; |
| 63 } |
| 64 |
| 65 #ifdef DEBUG |
| 66 |
| 67 void ContextSlotCache::ValidateEntry(Handle<Object> data, Handle<String> name, |
| 68 VariableMode mode, |
| 69 InitializationFlag init_flag, |
| 70 MaybeAssignedFlag maybe_assigned_flag, |
| 71 int slot_index) { |
| 72 DisallowHeapAllocation no_gc; |
| 73 Handle<String> internalized_name; |
| 74 if (StringTable::InternalizeStringIfExists(name->GetIsolate(), name) |
| 75 .ToHandle(&internalized_name)) { |
| 76 int index = Hash(*data, *name); |
| 77 Key& key = keys_[index]; |
| 78 DCHECK(key.data == *data); |
| 79 DCHECK(key.name->Equals(*name)); |
| 80 Value result(values_[index]); |
| 81 DCHECK(result.mode() == mode); |
| 82 DCHECK(result.initialization_flag() == init_flag); |
| 83 DCHECK(result.maybe_assigned_flag() == maybe_assigned_flag); |
| 84 DCHECK(result.index() + kNotFound == slot_index); |
| 85 } |
| 86 } |
| 87 |
| 88 #endif // DEBUG |
| 89 |
| 90 } // namespace internal |
| 91 } // namespace v8 |
OLD | NEW |