OLD | NEW |
---|---|
1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 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/type-feedback-vector.h" | 5 #include "src/type-feedback-vector.h" |
6 | 6 |
7 #include "src/code-stubs.h" | 7 #include "src/code-stubs.h" |
8 #include "src/ic/ic.h" | 8 #include "src/ic/ic.h" |
9 #include "src/ic/ic-state.h" | 9 #include "src/ic/ic-state.h" |
10 #include "src/objects.h" | 10 #include "src/objects.h" |
(...skipping 19 matching lines...) Expand all Loading... | |
30 } | 30 } |
31 | 31 |
32 | 32 |
33 FeedbackVectorSlotKind TypeFeedbackMetadata::GetKind( | 33 FeedbackVectorSlotKind TypeFeedbackMetadata::GetKind( |
34 FeedbackVectorSlot slot) const { | 34 FeedbackVectorSlot slot) const { |
35 int index = VectorICComputer::index(kReservedIndexCount, slot.ToInt()); | 35 int index = VectorICComputer::index(kReservedIndexCount, slot.ToInt()); |
36 int data = Smi::cast(get(index))->value(); | 36 int data = Smi::cast(get(index))->value(); |
37 return VectorICComputer::decode(data, slot.ToInt()); | 37 return VectorICComputer::decode(data, slot.ToInt()); |
38 } | 38 } |
39 | 39 |
40 String* TypeFeedbackMetadata::GetName(FeedbackVectorSlot slot) const { | |
41 DCHECK(SlotRequiresName(GetKind(slot))); | |
42 FixedArray* names = FixedArray::cast(get(kNamesTableIndex)); | |
43 const int kEntrySize = 2; | |
44 for (int i = 0; i < names->length(); i += kEntrySize) { | |
45 int current_slot = Smi::cast(names->get(i))->value(); | |
46 if (current_slot == slot.ToInt()) { | |
47 Object* name = names->get(i + 1); | |
48 if (!name->IsSmi()) { | |
49 return String::cast(name); | |
50 } else { | |
51 break; | |
52 } | |
53 } | |
54 } | |
55 return GetHeap()->empty_string(); | |
56 } | |
40 | 57 |
41 void TypeFeedbackMetadata::SetKind(FeedbackVectorSlot slot, | 58 void TypeFeedbackMetadata::SetKind(FeedbackVectorSlot slot, |
42 FeedbackVectorSlotKind kind) { | 59 FeedbackVectorSlotKind kind) { |
43 int index = VectorICComputer::index(kReservedIndexCount, slot.ToInt()); | 60 int index = VectorICComputer::index(kReservedIndexCount, slot.ToInt()); |
44 int data = Smi::cast(get(index))->value(); | 61 int data = Smi::cast(get(index))->value(); |
45 int new_data = VectorICComputer::encode(data, slot.ToInt(), kind); | 62 int new_data = VectorICComputer::encode(data, slot.ToInt(), kind); |
46 set(index, Smi::FromInt(new_data)); | 63 set(index, Smi::FromInt(new_data)); |
47 } | 64 } |
48 | 65 |
49 | 66 |
50 template Handle<TypeFeedbackMetadata> TypeFeedbackMetadata::New( | 67 template Handle<TypeFeedbackMetadata> TypeFeedbackMetadata::New( |
51 Isolate* isolate, const StaticFeedbackVectorSpec* spec); | 68 Isolate* isolate, const StaticFeedbackVectorSpec* spec); |
52 template Handle<TypeFeedbackMetadata> TypeFeedbackMetadata::New( | 69 template Handle<TypeFeedbackMetadata> TypeFeedbackMetadata::New( |
53 Isolate* isolate, const FeedbackVectorSpec* spec); | 70 Isolate* isolate, const FeedbackVectorSpec* spec); |
54 | 71 |
55 | 72 |
56 // static | 73 // static |
57 template <typename Spec> | 74 template <typename Spec> |
58 Handle<TypeFeedbackMetadata> TypeFeedbackMetadata::New(Isolate* isolate, | 75 Handle<TypeFeedbackMetadata> TypeFeedbackMetadata::New(Isolate* isolate, |
59 const Spec* spec) { | 76 const Spec* spec) { |
77 Factory* factory = isolate->factory(); | |
78 | |
60 const int slot_count = spec->slots(); | 79 const int slot_count = spec->slots(); |
61 const int slot_kinds_length = VectorICComputer::word_count(slot_count); | 80 const int slot_kinds_length = VectorICComputer::word_count(slot_count); |
62 const int length = slot_kinds_length + kReservedIndexCount; | 81 const int length = slot_kinds_length + kReservedIndexCount; |
63 if (length == kReservedIndexCount) { | 82 if (length == kReservedIndexCount) { |
64 return Handle<TypeFeedbackMetadata>::cast( | 83 return Handle<TypeFeedbackMetadata>::cast(factory->empty_fixed_array()); |
65 isolate->factory()->empty_fixed_array()); | |
66 } | 84 } |
67 #ifdef DEBUG | 85 #ifdef DEBUG |
68 for (int i = 0; i < slot_count;) { | 86 for (int i = 0; i < slot_count;) { |
69 FeedbackVectorSlotKind kind = spec->GetKind(i); | 87 FeedbackVectorSlotKind kind = spec->GetKind(i); |
70 int entry_size = TypeFeedbackMetadata::GetSlotSize(kind); | 88 int entry_size = TypeFeedbackMetadata::GetSlotSize(kind); |
71 for (int j = 1; j < entry_size; j++) { | 89 for (int j = 1; j < entry_size; j++) { |
72 FeedbackVectorSlotKind kind = spec->GetKind(i + j); | 90 FeedbackVectorSlotKind kind = spec->GetKind(i + j); |
73 DCHECK_EQ(FeedbackVectorSlotKind::INVALID, kind); | 91 DCHECK_EQ(FeedbackVectorSlotKind::INVALID, kind); |
74 } | 92 } |
75 i += entry_size; | 93 i += entry_size; |
76 } | 94 } |
77 #endif | 95 #endif |
78 | 96 |
79 Handle<FixedArray> array = isolate->factory()->NewFixedArray(length, TENURED); | 97 Handle<FixedArray> array = factory->NewFixedArray(length, TENURED); |
80 array->set(kSlotsCountIndex, Smi::FromInt(slot_count)); | 98 array->set(kSlotsCountIndex, Smi::FromInt(slot_count)); |
81 // Fill the bit-vector part with zeros. | 99 // Fill the bit-vector part with zeros. |
82 for (int i = 0; i < slot_kinds_length; i++) { | 100 for (int i = 0; i < slot_kinds_length; i++) { |
83 array->set(kReservedIndexCount + i, Smi::FromInt(0)); | 101 array->set(kReservedIndexCount + i, Smi::FromInt(0)); |
84 } | 102 } |
85 | 103 |
86 Handle<TypeFeedbackMetadata> metadata = | 104 Handle<TypeFeedbackMetadata> metadata = |
87 Handle<TypeFeedbackMetadata>::cast(array); | 105 Handle<TypeFeedbackMetadata>::cast(array); |
106 | |
107 // Add names to NamesTable. | |
108 const int kEntrySize = 2; | |
109 Handle<FixedArray> names = factory->empty_fixed_array(); | |
110 | |
111 int name_index = 0; | |
88 for (int i = 0; i < slot_count; i++) { | 112 for (int i = 0; i < slot_count; i++) { |
89 metadata->SetKind(FeedbackVectorSlot(i), spec->GetKind(i)); | 113 FeedbackVectorSlotKind kind = spec->GetKind(i); |
114 metadata->SetKind(FeedbackVectorSlot(i), kind); | |
115 if (SlotRequiresName(kind)) { | |
116 Handle<String> name = spec->GetName(name_index); | |
117 | |
118 // Add entry to names. | |
119 int length = names->length(); | |
120 if (length < (name_index + 1) * kEntrySize) { | |
121 Handle<FixedArray> tmp = | |
mvstanton
2016/06/24 10:04:11
maybe call this newNames.
Igor Sheludko
2016/06/24 13:30:50
Done.
| |
122 factory->NewFixedArray(length == 0 ? 16 : length * 2); | |
mvstanton
2016/06/24 10:04:12
Did you do anything like figure out average # of g
Igor Sheludko
2016/06/24 13:30:50
Good point! Although we don't waste space in the e
| |
123 names->CopyTo(0, *tmp, 0, length); | |
124 names = tmp; | |
125 } | |
126 names->set(name_index * 2, Smi::FromInt(i)); | |
127 names->set(name_index * 2 + 1, | |
128 !name.is_null() ? Object::cast(*name) : Smi::FromInt(0)); | |
129 name_index++; | |
130 } | |
90 } | 131 } |
132 if (name_index > 0) { | |
133 names->Shrink(name_index * 2); | |
134 } | |
135 metadata->set(kNamesTableIndex, *names); | |
91 | 136 |
92 // It's important that the TypeFeedbackMetadata have a COW map, since it's | 137 // It's important that the TypeFeedbackMetadata have a COW map, since it's |
93 // pointed to by both a SharedFunctionInfo and indirectly by closures through | 138 // pointed to by both a SharedFunctionInfo and indirectly by closures through |
94 // the TypeFeedbackVector. The serializer uses the COW map type to decide | 139 // the TypeFeedbackVector. The serializer uses the COW map type to decide |
95 // this object belongs in the startup snapshot and not the partial | 140 // this object belongs in the startup snapshot and not the partial |
96 // snapshot(s). | 141 // snapshot(s). |
97 metadata->set_map(isolate->heap()->fixed_cow_array_map()); | 142 metadata->set_map(isolate->heap()->fixed_cow_array_map()); |
98 | 143 |
99 return metadata; | 144 return metadata; |
100 } | 145 } |
101 | 146 |
102 | 147 |
103 bool TypeFeedbackMetadata::SpecDiffersFrom( | 148 bool TypeFeedbackMetadata::SpecDiffersFrom( |
104 const FeedbackVectorSpec* other_spec) const { | 149 const FeedbackVectorSpec* other_spec) const { |
105 if (other_spec->slots() != slot_count()) { | 150 if (other_spec->slots() != slot_count()) { |
106 return true; | 151 return true; |
107 } | 152 } |
108 | 153 |
109 int slots = slot_count(); | 154 int slots = slot_count(); |
110 for (int i = 0; i < slots; i++) { | 155 int name_index = 0; |
111 if (GetKind(FeedbackVectorSlot(i)) != other_spec->GetKind(i)) { | 156 for (int i = 0; i < slots;) { |
157 FeedbackVectorSlot slot(i); | |
158 FeedbackVectorSlotKind kind = GetKind(slot); | |
159 int entry_size = TypeFeedbackMetadata::GetSlotSize(kind); | |
160 | |
161 if (kind != other_spec->GetKind(i)) { | |
112 return true; | 162 return true; |
113 } | 163 } |
164 if (SlotRequiresName(kind)) { | |
165 String* name = GetName(slot); | |
166 DCHECK(name != GetHeap()->empty_string()); | |
167 String* other_name = *other_spec->GetName(name_index++); | |
168 if (name != other_name) { | |
169 return true; | |
170 } | |
171 } | |
172 i += entry_size; | |
114 } | 173 } |
115 return false; | 174 return false; |
116 } | 175 } |
117 | 176 |
118 bool TypeFeedbackMetadata::DiffersFrom( | 177 bool TypeFeedbackMetadata::DiffersFrom( |
119 const TypeFeedbackMetadata* other_metadata) const { | 178 const TypeFeedbackMetadata* other_metadata) const { |
120 if (other_metadata->slot_count() != slot_count()) { | 179 if (other_metadata->slot_count() != slot_count()) { |
121 return true; | 180 return true; |
122 } | 181 } |
123 | 182 |
124 int slots = slot_count(); | 183 int slots = slot_count(); |
125 for (int i = 0; i < slots; i++) { | 184 for (int i = 0; i < slots;) { |
126 FeedbackVectorSlot slot(i); | 185 FeedbackVectorSlot slot(i); |
186 FeedbackVectorSlotKind kind = GetKind(slot); | |
187 int entry_size = TypeFeedbackMetadata::GetSlotSize(kind); | |
127 if (GetKind(slot) != other_metadata->GetKind(slot)) { | 188 if (GetKind(slot) != other_metadata->GetKind(slot)) { |
128 return true; | 189 return true; |
129 } | 190 } |
191 if (SlotRequiresName(kind)) { | |
192 if (GetName(slot) != other_metadata->GetName(slot)) { | |
193 return true; | |
194 } | |
195 } | |
196 i += entry_size; | |
130 } | 197 } |
131 return false; | 198 return false; |
132 } | 199 } |
133 | 200 |
134 const char* TypeFeedbackMetadata::Kind2String(FeedbackVectorSlotKind kind) { | 201 const char* TypeFeedbackMetadata::Kind2String(FeedbackVectorSlotKind kind) { |
135 switch (kind) { | 202 switch (kind) { |
136 case FeedbackVectorSlotKind::INVALID: | 203 case FeedbackVectorSlotKind::INVALID: |
137 return "INVALID"; | 204 return "INVALID"; |
138 case FeedbackVectorSlotKind::CALL_IC: | 205 case FeedbackVectorSlotKind::CALL_IC: |
139 return "CALL_IC"; | 206 return "CALL_IC"; |
(...skipping 15 matching lines...) Expand all Loading... | |
155 UNREACHABLE(); | 222 UNREACHABLE(); |
156 return "?"; | 223 return "?"; |
157 } | 224 } |
158 | 225 |
159 FeedbackVectorSlotKind TypeFeedbackVector::GetKind( | 226 FeedbackVectorSlotKind TypeFeedbackVector::GetKind( |
160 FeedbackVectorSlot slot) const { | 227 FeedbackVectorSlot slot) const { |
161 DCHECK(!is_empty()); | 228 DCHECK(!is_empty()); |
162 return metadata()->GetKind(slot); | 229 return metadata()->GetKind(slot); |
163 } | 230 } |
164 | 231 |
232 String* TypeFeedbackVector::GetName(FeedbackVectorSlot slot) const { | |
233 DCHECK(!is_empty()); | |
234 return metadata()->GetName(slot); | |
235 } | |
236 | |
165 // static | 237 // static |
166 Handle<TypeFeedbackVector> TypeFeedbackVector::New( | 238 Handle<TypeFeedbackVector> TypeFeedbackVector::New( |
167 Isolate* isolate, Handle<TypeFeedbackMetadata> metadata) { | 239 Isolate* isolate, Handle<TypeFeedbackMetadata> metadata) { |
168 Factory* factory = isolate->factory(); | 240 Factory* factory = isolate->factory(); |
169 | 241 |
170 const int slot_count = metadata->slot_count(); | 242 const int slot_count = metadata->slot_count(); |
171 const int length = slot_count + kReservedIndexCount; | 243 const int length = slot_count + kReservedIndexCount; |
172 if (length == kReservedIndexCount) { | 244 if (length == kReservedIndexCount) { |
173 return Handle<TypeFeedbackVector>::cast(factory->empty_fixed_array()); | 245 return Handle<TypeFeedbackVector>::cast(factory->empty_fixed_array()); |
174 } | 246 } |
175 | 247 |
176 Handle<FixedArray> array = factory->NewFixedArray(length, TENURED); | 248 Handle<FixedArray> array = factory->NewFixedArray(length, TENURED); |
177 array->set(kMetadataIndex, *metadata); | 249 array->set(kMetadataIndex, *metadata); |
178 | 250 |
251 DisallowHeapAllocation no_gc; | |
252 | |
179 // Ensure we can skip the write barrier | 253 // Ensure we can skip the write barrier |
180 Handle<Object> uninitialized_sentinel = UninitializedSentinel(isolate); | 254 Handle<Object> uninitialized_sentinel = UninitializedSentinel(isolate); |
181 DCHECK_EQ(*factory->uninitialized_symbol(), *uninitialized_sentinel); | 255 DCHECK_EQ(*factory->uninitialized_symbol(), *uninitialized_sentinel); |
182 for (int i = 0; i < slot_count;) { | 256 for (int i = 0; i < slot_count;) { |
183 FeedbackVectorSlot slot(i); | 257 FeedbackVectorSlot slot(i); |
184 FeedbackVectorSlotKind kind = metadata->GetKind(slot); | 258 FeedbackVectorSlotKind kind = metadata->GetKind(slot); |
185 int index = TypeFeedbackVector::GetIndex(slot); | 259 int index = TypeFeedbackVector::GetIndex(slot); |
186 int entry_size = TypeFeedbackMetadata::GetSlotSize(kind); | 260 int entry_size = TypeFeedbackMetadata::GetSlotSize(kind); |
187 | 261 |
188 Object* value; | 262 Object* value; |
(...skipping 763 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
952 | 1026 |
953 IcCheckType KeyedStoreICNexus::GetKeyType() const { | 1027 IcCheckType KeyedStoreICNexus::GetKeyType() const { |
954 Object* feedback = GetFeedback(); | 1028 Object* feedback = GetFeedback(); |
955 if (feedback == *TypeFeedbackVector::MegamorphicSentinel(GetIsolate())) { | 1029 if (feedback == *TypeFeedbackVector::MegamorphicSentinel(GetIsolate())) { |
956 return static_cast<IcCheckType>(Smi::cast(GetFeedbackExtra())->value()); | 1030 return static_cast<IcCheckType>(Smi::cast(GetFeedbackExtra())->value()); |
957 } | 1031 } |
958 return IsPropertyNameFeedback(feedback) ? PROPERTY : ELEMENT; | 1032 return IsPropertyNameFeedback(feedback) ? PROPERTY : ELEMENT; |
959 } | 1033 } |
960 } // namespace internal | 1034 } // namespace internal |
961 } // namespace v8 | 1035 } // namespace v8 |
OLD | NEW |