| OLD | NEW | 
|---|
| 1 // Copyright 2015 the V8 project authors. All rights reserved. | 1 // Copyright 2015 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/compiler/js-type-feedback.h" | 5 #include "src/compiler/js-type-feedback.h" | 
| 6 | 6 | 
| 7 #include "src/property-details.h" | 7 #include "src/property-details.h" | 
| 8 | 8 | 
| 9 #include "src/accessors.h" | 9 #include "src/accessors.h" | 
| 10 #include "src/ast.h" | 10 #include "src/ast.h" | 
| 11 #include "src/compiler.h" | 11 #include "src/compiler.h" | 
| 12 #include "src/type-info.h" | 12 #include "src/type-info.h" | 
| 13 | 13 | 
| 14 #include "src/compiler/access-builder.h" | 14 #include "src/compiler/access-builder.h" | 
| 15 #include "src/compiler/common-operator.h" | 15 #include "src/compiler/common-operator.h" | 
| 16 #include "src/compiler/frame-states.h" | 16 #include "src/compiler/frame-states.h" | 
| 17 #include "src/compiler/node-aux-data.h" | 17 #include "src/compiler/node-aux-data.h" | 
| 18 #include "src/compiler/node-matchers.h" | 18 #include "src/compiler/node-matchers.h" | 
| 19 #include "src/compiler/operator-properties.h" | 19 #include "src/compiler/operator-properties.h" | 
| 20 #include "src/compiler/simplified-operator.h" | 20 #include "src/compiler/simplified-operator.h" | 
| 21 | 21 | 
| 22 namespace v8 { | 22 namespace v8 { | 
| 23 namespace internal { | 23 namespace internal { | 
| 24 namespace compiler { | 24 namespace compiler { | 
| 25 | 25 | 
| 26 enum LoadOrStore { LOAD, STORE }; | 26 enum LoadOrStore { LOAD, STORE }; | 
| 27 | 27 | 
|  | 28 // TODO(turbofan): fix deoptimization problems | 
|  | 29 #define ENABLE_FAST_PROPERTY_LOADS false | 
|  | 30 #define ENABLE_FAST_PROPERTY_STORES false | 
|  | 31 | 
| 28 JSTypeFeedbackTable::JSTypeFeedbackTable(Zone* zone) | 32 JSTypeFeedbackTable::JSTypeFeedbackTable(Zone* zone) | 
| 29     : map_(TypeFeedbackIdMap::key_compare(), | 33     : type_feedback_id_map_(TypeFeedbackIdMap::key_compare(), | 
| 30            TypeFeedbackIdMap::allocator_type(zone)) {} | 34                             TypeFeedbackIdMap::allocator_type(zone)), | 
|  | 35       feedback_vector_ic_slot_map_(TypeFeedbackIdMap::key_compare(), | 
|  | 36                                    TypeFeedbackIdMap::allocator_type(zone)) {} | 
| 31 | 37 | 
| 32 | 38 | 
| 33 void JSTypeFeedbackTable::Record(Node* node, TypeFeedbackId id) { | 39 void JSTypeFeedbackTable::Record(Node* node, TypeFeedbackId id) { | 
| 34   map_.insert(std::make_pair(node->id(), id)); | 40   type_feedback_id_map_.insert(std::make_pair(node->id(), id)); | 
| 35 } | 41 } | 
| 36 | 42 | 
| 37 | 43 | 
|  | 44 void JSTypeFeedbackTable::Record(Node* node, FeedbackVectorICSlot slot) { | 
|  | 45   feedback_vector_ic_slot_map_.insert(std::make_pair(node->id(), slot)); | 
|  | 46 } | 
|  | 47 | 
|  | 48 | 
| 38 Reduction JSTypeFeedbackSpecializer::Reduce(Node* node) { | 49 Reduction JSTypeFeedbackSpecializer::Reduce(Node* node) { | 
| 39   switch (node->opcode()) { | 50   switch (node->opcode()) { | 
| 40     case IrOpcode::kJSLoadProperty: { | 51     case IrOpcode::kJSLoadProperty: { | 
| 41       HeapObjectMatcher<Name> match(node->InputAt(1)); | 52       HeapObjectMatcher<Name> match(node->InputAt(1)); | 
| 42       if (match.HasValue() && match.Value().handle()->IsName()) { | 53       if (match.HasValue() && match.Value().handle()->IsName()) { | 
| 43         // LoadProperty(o, "constant") => LoadNamed["constant"](o). | 54         // LoadProperty(o, "constant") => LoadNamed["constant"](o). | 
| 44         Unique<Name> name = match.Value(); | 55         Unique<Name> name = match.Value(); | 
| 45         const VectorSlotPair& feedback = | 56         const VectorSlotPair& feedback = | 
| 46             LoadPropertyParametersOf(node->op()).feedback(); | 57             LoadPropertyParametersOf(node->op()).feedback(); | 
| 47         node->set_op(jsgraph()->javascript()->LoadNamed(name, feedback, | 58         node->set_op(jsgraph()->javascript()->LoadNamed(name, feedback, | 
| (...skipping 81 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 129     if (is_smi || is_double) { | 140     if (is_smi || is_double) { | 
| 130       // TODO(turbofan): check type and deopt for SMI/double stores. | 141       // TODO(turbofan): check type and deopt for SMI/double stores. | 
| 131       return false; | 142       return false; | 
| 132     } | 143     } | 
| 133   } | 144   } | 
| 134 | 145 | 
| 135   int index = map->instance_descriptors()->GetFieldIndex(number); | 146   int index = map->instance_descriptors()->GetFieldIndex(number); | 
| 136   FieldIndex field_index = FieldIndex::ForPropertyIndex(*map, index, is_double); | 147   FieldIndex field_index = FieldIndex::ForPropertyIndex(*map, index, is_double); | 
| 137 | 148 | 
| 138   if (field_index.is_inobject()) { | 149   if (field_index.is_inobject()) { | 
|  | 150     if (is_double && !map->IsUnboxedDoubleField(field_index)) { | 
|  | 151       // TODO(turbofan): support for out-of-line (MutableHeapNumber) loads. | 
|  | 152       return false; | 
|  | 153     } | 
| 139     access->offset = field_index.offset(); | 154     access->offset = field_index.offset(); | 
| 140     return true; | 155     return true; | 
| 141   } | 156   } | 
| 142 | 157 | 
| 143   // TODO(turbofan): handle out of object properties. | 158   // TODO(turbofan): handle out of object properties. | 
| 144   return false; | 159   return false; | 
| 145 } | 160 } | 
| 146 | 161 | 
| 147 | 162 | 
| 148 static bool IsGlobalObject(Node* node) { | 163 static bool IsGlobalObject(Node* node) { | 
| 149   return NodeProperties::IsTyped(node) && | 164   return NodeProperties::IsTyped(node) && | 
| 150          NodeProperties::GetBounds(node).upper->Is(Type::GlobalObject()); | 165          NodeProperties::GetBounds(node).upper->Is(Type::GlobalObject()); | 
| 151 } | 166 } | 
| 152 | 167 | 
| 153 | 168 | 
| 154 Reduction JSTypeFeedbackSpecializer::ReduceJSLoadNamed(Node* node) { | 169 Reduction JSTypeFeedbackSpecializer::ReduceJSLoadNamed(Node* node) { | 
| 155   DCHECK(node->opcode() == IrOpcode::kJSLoadNamed); | 170   DCHECK(node->opcode() == IrOpcode::kJSLoadNamed); | 
| 156   Node* receiver = node->InputAt(0); | 171   Node* receiver = node->InputAt(0); | 
| 157   if (IsGlobalObject(receiver)) { | 172   if (IsGlobalObject(receiver)) { | 
| 158     return ReduceJSLoadNamedForGlobalVariable(node); | 173     return ReduceJSLoadNamedForGlobalVariable(node); | 
| 159   } | 174   } | 
| 160 | 175 | 
| 161   if (!FLAG_turbo_deoptimization) return NoChange(); | 176   if (!FLAG_turbo_deoptimization) return NoChange(); | 
| 162   Node* frame_state_before = GetFrameStateBefore(node); | 177   Node* frame_state_before = GetFrameStateBefore(node); | 
| 163   if (frame_state_before == nullptr) return NoChange(); | 178   if (frame_state_before == nullptr) return NoChange(); | 
| 164 | 179 | 
| 165   // TODO(turbofan): handle vector-based type feedback. | 180   const LoadNamedParameters& p = LoadNamedParametersOf(node->op()); | 
| 166   TypeFeedbackId id = js_type_feedback_->find(node); | 181   Handle<Name> name = p.name().handle(); | 
| 167   if (id.IsNone() || oracle()->LoadInlineCacheState(id) == UNINITIALIZED) { | 182   SmallMapList maps; | 
|  | 183 | 
|  | 184   FeedbackVectorICSlot slot = js_type_feedback_->FindFeedbackVectorICSlot(node); | 
|  | 185   if (slot.IsInvalid() || | 
|  | 186       oracle()->LoadInlineCacheState(slot) == UNINITIALIZED) { | 
|  | 187     // No type feedback ids or the load is uninitialized. | 
| 168     return NoChange(); | 188     return NoChange(); | 
| 169   } | 189   } | 
|  | 190   if (p.load_ic() == NAMED) { | 
|  | 191     oracle()->PropertyReceiverTypes(slot, name, &maps); | 
|  | 192   } else { | 
|  | 193     // The load named was originally a load property. | 
|  | 194     bool is_string;        // Unused. | 
|  | 195     IcCheckType key_type;  // Unused. | 
|  | 196     oracle()->KeyedPropertyReceiverTypes(slot, &maps, &is_string, &key_type); | 
|  | 197   } | 
| 170 | 198 | 
| 171   const LoadNamedParameters& p = LoadNamedParametersOf(node->op()); |  | 
| 172   SmallMapList maps; |  | 
| 173   Handle<Name> name = p.name().handle(); |  | 
| 174   Node* effect = NodeProperties::GetEffectInput(node); | 199   Node* effect = NodeProperties::GetEffectInput(node); | 
| 175   GatherReceiverTypes(receiver, effect, id, name, &maps); |  | 
| 176 | 200 | 
| 177   if (maps.length() != 1) return NoChange();  // TODO(turbofan): polymorphism | 201   if (maps.length() != 1) return NoChange();  // TODO(turbofan): polymorphism | 
|  | 202   if (!ENABLE_FAST_PROPERTY_LOADS) return NoChange(); | 
| 178 | 203 | 
| 179   Handle<Map> map = maps.first(); | 204   Handle<Map> map = maps.first(); | 
| 180   FieldAccess field_access; | 205   FieldAccess field_access; | 
| 181   if (!GetInObjectFieldAccess(LOAD, map, name, &field_access)) { | 206   if (!GetInObjectFieldAccess(LOAD, map, name, &field_access)) { | 
| 182     return NoChange(); | 207     return NoChange(); | 
| 183   } | 208   } | 
| 184 | 209 | 
| 185   Node* control = NodeProperties::GetControlInput(node); | 210   Node* control = NodeProperties::GetControlInput(node); | 
| 186   Node* check_success; | 211   Node* check_success; | 
| 187   Node* check_failed; | 212   Node* check_failed; | 
| (...skipping 81 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 269 } | 294 } | 
| 270 | 295 | 
| 271 | 296 | 
| 272 Reduction JSTypeFeedbackSpecializer::ReduceJSLoadProperty(Node* node) { | 297 Reduction JSTypeFeedbackSpecializer::ReduceJSLoadProperty(Node* node) { | 
| 273   return NoChange(); | 298   return NoChange(); | 
| 274 } | 299 } | 
| 275 | 300 | 
| 276 | 301 | 
| 277 Reduction JSTypeFeedbackSpecializer::ReduceJSStoreNamed(Node* node) { | 302 Reduction JSTypeFeedbackSpecializer::ReduceJSStoreNamed(Node* node) { | 
| 278   DCHECK(node->opcode() == IrOpcode::kJSStoreNamed); | 303   DCHECK(node->opcode() == IrOpcode::kJSStoreNamed); | 
| 279   if (true) return NoChange();  // TODO(titzer): storenamed is broken |  | 
| 280   Node* frame_state_before = GetFrameStateBefore(node); | 304   Node* frame_state_before = GetFrameStateBefore(node); | 
| 281   if (frame_state_before == nullptr) return NoChange(); | 305   if (frame_state_before == nullptr) return NoChange(); | 
| 282 | 306 | 
| 283   TypeFeedbackId id = js_type_feedback_->find(node); | 307   const StoreNamedParameters& p = StoreNamedParametersOf(node->op()); | 
| 284   if (id.IsNone() || oracle()->StoreIsUninitialized(id)) return NoChange(); | 308   Handle<Name> name = p.name().handle(); | 
|  | 309   SmallMapList maps; | 
|  | 310   TypeFeedbackId id = js_type_feedback_->FindTypeFeedbackId(node); | 
|  | 311   if (id.IsNone() || oracle()->StoreIsUninitialized(id) == UNINITIALIZED) { | 
|  | 312     // No type feedback ids or the store is uninitialized. | 
|  | 313     // TODO(titzer): no feedback from vector ICs from stores. | 
|  | 314     return NoChange(); | 
|  | 315   } else { | 
|  | 316     if (p.store_ic() == NAMED) { | 
|  | 317       oracle()->PropertyReceiverTypes(id, name, &maps); | 
|  | 318     } else { | 
|  | 319       // The named store was originally a store property. | 
|  | 320       bool is_string;        // Unused. | 
|  | 321       IcCheckType key_type;  // Unused. | 
|  | 322       oracle()->KeyedPropertyReceiverTypes(id, &maps, &is_string, &key_type); | 
|  | 323     } | 
|  | 324   } | 
| 285 | 325 | 
| 286   const StoreNamedParameters& p = StoreNamedParametersOf(node->op()); |  | 
| 287   SmallMapList maps; |  | 
| 288   Handle<Name> name = p.name().handle(); |  | 
| 289   Node* receiver = node->InputAt(0); | 326   Node* receiver = node->InputAt(0); | 
| 290   Node* effect = NodeProperties::GetEffectInput(node); | 327   Node* effect = NodeProperties::GetEffectInput(node); | 
| 291   GatherReceiverTypes(receiver, effect, id, name, &maps); |  | 
| 292 | 328 | 
| 293   if (maps.length() != 1) return NoChange();  // TODO(turbofan): polymorphism | 329   if (maps.length() != 1) return NoChange();  // TODO(turbofan): polymorphism | 
| 294 | 330 | 
|  | 331   if (!ENABLE_FAST_PROPERTY_STORES) return NoChange(); | 
|  | 332 | 
| 295   Handle<Map> map = maps.first(); | 333   Handle<Map> map = maps.first(); | 
| 296   FieldAccess field_access; | 334   FieldAccess field_access; | 
| 297   if (!GetInObjectFieldAccess(STORE, map, name, &field_access)) { | 335   if (!GetInObjectFieldAccess(STORE, map, name, &field_access)) { | 
| 298     return NoChange(); | 336     return NoChange(); | 
| 299   } | 337   } | 
| 300 | 338 | 
| 301   Node* control = NodeProperties::GetControlInput(node); | 339   Node* control = NodeProperties::GetControlInput(node); | 
| 302   Node* check_success; | 340   Node* check_success; | 
| 303   Node* check_failed; | 341   Node* check_failed; | 
| 304   BuildMapCheck(receiver, map, true, effect, control, &check_success, | 342   BuildMapCheck(receiver, map, true, effect, control, &check_success, | 
| (...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 346       graph()->NewNode(common()->Branch(BranchHint::kTrue), cmp, control); | 384       graph()->NewNode(common()->Branch(BranchHint::kTrue), cmp, control); | 
| 347   *success = graph()->NewNode(common()->IfTrue(), branch); | 385   *success = graph()->NewNode(common()->IfTrue(), branch); | 
| 348   *fail = graph()->NewNode(common()->IfFalse(), branch); | 386   *fail = graph()->NewNode(common()->IfFalse(), branch); | 
| 349 | 387 | 
| 350   if (if_smi) { | 388   if (if_smi) { | 
| 351     *fail = graph()->NewNode(common()->Merge(2), *fail, if_smi); | 389     *fail = graph()->NewNode(common()->Merge(2), *fail, if_smi); | 
| 352   } | 390   } | 
| 353 } | 391 } | 
| 354 | 392 | 
| 355 | 393 | 
| 356 void JSTypeFeedbackSpecializer::GatherReceiverTypes(Node* receiver, |  | 
| 357                                                     Node* effect, |  | 
| 358                                                     TypeFeedbackId id, |  | 
| 359                                                     Handle<Name> name, |  | 
| 360                                                     SmallMapList* maps) { |  | 
| 361   // TODO(turbofan): filter maps by initial receiver map if known |  | 
| 362   // TODO(turbofan): filter maps by native context (if specializing) |  | 
| 363   // TODO(turbofan): filter maps by effect chain |  | 
| 364   oracle()->PropertyReceiverTypes(id, name, maps); |  | 
| 365 } |  | 
| 366 |  | 
| 367 |  | 
| 368 // Get the frame state before an operation if it exists and has a valid | 394 // Get the frame state before an operation if it exists and has a valid | 
| 369 // bailout id. | 395 // bailout id. | 
| 370 Node* JSTypeFeedbackSpecializer::GetFrameStateBefore(Node* node) { | 396 Node* JSTypeFeedbackSpecializer::GetFrameStateBefore(Node* node) { | 
| 371   int count = OperatorProperties::GetFrameStateInputCount(node->op()); | 397   int count = OperatorProperties::GetFrameStateInputCount(node->op()); | 
| 372   DCHECK_LE(count, 2); | 398   DCHECK_LE(count, 2); | 
| 373   if (count == 2) { | 399   if (count == 2) { | 
| 374     Node* frame_state = NodeProperties::GetFrameStateInput(node, 1); | 400     Node* frame_state = NodeProperties::GetFrameStateInput(node, 1); | 
| 375     if (frame_state->opcode() == IrOpcode::kFrameState) { | 401     if (frame_state->opcode() == IrOpcode::kFrameState) { | 
| 376       BailoutId id = OpParameter<FrameStateCallInfo>(node).bailout_id(); | 402       BailoutId id = OpParameter<FrameStateCallInfo>(node).bailout_id(); | 
| 377       if (id != BailoutId::None()) return frame_state; | 403       if (id != BailoutId::None()) return frame_state; | 
| 378     } | 404     } | 
| 379   } | 405   } | 
| 380   return nullptr; | 406   return nullptr; | 
| 381 } | 407 } | 
| 382 | 408 | 
| 383 }  // namespace compiler | 409 }  // namespace compiler | 
| 384 }  // namespace internal | 410 }  // namespace internal | 
| 385 }  // namespace v8 | 411 }  // namespace v8 | 
| OLD | NEW | 
|---|