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/ic/ic.h" | 5 #include "src/ic/ic.h" |
6 | 6 |
7 #include "src/accessors.h" | 7 #include "src/accessors.h" |
8 #include "src/api-arguments-inl.h" | 8 #include "src/api-arguments-inl.h" |
9 #include "src/api.h" | 9 #include "src/api.h" |
10 #include "src/arguments.h" | 10 #include "src/arguments.h" |
(...skipping 534 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
545 Handle<Object> handler) { | 545 Handle<Object> handler) { |
546 DCHECK(UseVector()); | 546 DCHECK(UseVector()); |
547 if (kind() == Code::LOAD_IC) { | 547 if (kind() == Code::LOAD_IC) { |
548 LoadICNexus* nexus = casted_nexus<LoadICNexus>(); | 548 LoadICNexus* nexus = casted_nexus<LoadICNexus>(); |
549 nexus->ConfigureMonomorphic(map, handler); | 549 nexus->ConfigureMonomorphic(map, handler); |
550 } else if (kind() == Code::LOAD_GLOBAL_IC) { | 550 } else if (kind() == Code::LOAD_GLOBAL_IC) { |
551 LoadGlobalICNexus* nexus = casted_nexus<LoadGlobalICNexus>(); | 551 LoadGlobalICNexus* nexus = casted_nexus<LoadGlobalICNexus>(); |
552 nexus->ConfigureHandlerMode(Handle<Code>::cast(handler)); | 552 nexus->ConfigureHandlerMode(Handle<Code>::cast(handler)); |
553 } else if (kind() == Code::KEYED_LOAD_IC) { | 553 } else if (kind() == Code::KEYED_LOAD_IC) { |
554 KeyedLoadICNexus* nexus = casted_nexus<KeyedLoadICNexus>(); | 554 KeyedLoadICNexus* nexus = casted_nexus<KeyedLoadICNexus>(); |
555 nexus->ConfigureMonomorphic(name, map, Handle<Code>::cast(handler)); | 555 nexus->ConfigureMonomorphic(name, map, handler); |
556 } else if (kind() == Code::STORE_IC) { | 556 } else if (kind() == Code::STORE_IC) { |
557 StoreICNexus* nexus = casted_nexus<StoreICNexus>(); | 557 StoreICNexus* nexus = casted_nexus<StoreICNexus>(); |
558 nexus->ConfigureMonomorphic(map, Handle<Code>::cast(handler)); | 558 nexus->ConfigureMonomorphic(map, Handle<Code>::cast(handler)); |
559 } else { | 559 } else { |
560 DCHECK(kind() == Code::KEYED_STORE_IC); | 560 DCHECK(kind() == Code::KEYED_STORE_IC); |
561 KeyedStoreICNexus* nexus = casted_nexus<KeyedStoreICNexus>(); | 561 KeyedStoreICNexus* nexus = casted_nexus<KeyedStoreICNexus>(); |
562 nexus->ConfigureMonomorphic(name, map, Handle<Code>::cast(handler)); | 562 nexus->ConfigureMonomorphic(name, map, Handle<Code>::cast(handler)); |
563 } | 563 } |
564 | 564 |
565 vector_set_ = true; | 565 vector_set_ = true; |
(...skipping 208 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
774 Map* transitioned_map = nullptr; | 774 Map* transitioned_map = nullptr; |
775 if (more_general_transition) { | 775 if (more_general_transition) { |
776 MapHandleList map_list; | 776 MapHandleList map_list; |
777 map_list.Add(handle(target_map)); | 777 map_list.Add(handle(target_map)); |
778 transitioned_map = source_map->FindElementsKindTransitionedMap(&map_list); | 778 transitioned_map = source_map->FindElementsKindTransitionedMap(&map_list); |
779 } | 779 } |
780 return transitioned_map == target_map; | 780 return transitioned_map == target_map; |
781 } | 781 } |
782 | 782 |
783 void IC::PatchCache(Handle<Name> name, Handle<Object> code) { | 783 void IC::PatchCache(Handle<Name> name, Handle<Object> code) { |
784 DCHECK(code->IsCode() || (kind() == Code::LOAD_IC && code->IsSmi())); | 784 DCHECK(code->IsCode() || (code->IsSmi() && (kind() == Code::LOAD_IC || |
| 785 kind() == Code::KEYED_LOAD_IC))); |
785 switch (state()) { | 786 switch (state()) { |
786 case UNINITIALIZED: | 787 case UNINITIALIZED: |
787 case PREMONOMORPHIC: | 788 case PREMONOMORPHIC: |
788 UpdateMonomorphicIC(code, name); | 789 UpdateMonomorphicIC(code, name); |
789 break; | 790 break; |
790 case RECOMPUTE_HANDLER: | 791 case RECOMPUTE_HANDLER: |
791 case MONOMORPHIC: | 792 case MONOMORPHIC: |
792 if (kind() == Code::LOAD_GLOBAL_IC) { | 793 if (kind() == Code::LOAD_GLOBAL_IC) { |
793 UpdateMonomorphicIC(code, name); | 794 UpdateMonomorphicIC(code, name); |
794 break; | 795 break; |
(...skipping 26 matching lines...) Expand all Loading... |
821 return LoadICTFStub(isolate).GetCode(); | 822 return LoadICTFStub(isolate).GetCode(); |
822 } | 823 } |
823 return LoadICStub(isolate).GetCode(); | 824 return LoadICStub(isolate).GetCode(); |
824 } | 825 } |
825 | 826 |
826 Handle<Code> LoadGlobalIC::initialize_stub_in_optimized_code( | 827 Handle<Code> LoadGlobalIC::initialize_stub_in_optimized_code( |
827 Isolate* isolate, ExtraICState extra_state) { | 828 Isolate* isolate, ExtraICState extra_state) { |
828 return LoadGlobalICStub(isolate, LoadGlobalICState(extra_state)).GetCode(); | 829 return LoadGlobalICStub(isolate, LoadGlobalICState(extra_state)).GetCode(); |
829 } | 830 } |
830 | 831 |
831 Handle<Code> KeyedLoadIC::initialize_stub_in_optimized_code( | 832 Handle<Code> KeyedLoadIC::initialize_stub_in_optimized_code(Isolate* isolate) { |
832 Isolate* isolate, ExtraICState extra_state) { | 833 if (FLAG_tf_load_ic_stub) { |
833 // TODO(ishell): remove extra_ic_state | 834 return KeyedLoadICTFStub(isolate).GetCode(); |
| 835 } |
834 return KeyedLoadICStub(isolate).GetCode(); | 836 return KeyedLoadICStub(isolate).GetCode(); |
835 } | 837 } |
836 | 838 |
837 Handle<Code> KeyedStoreIC::initialize_stub_in_optimized_code( | 839 Handle<Code> KeyedStoreIC::initialize_stub_in_optimized_code( |
838 Isolate* isolate, LanguageMode language_mode) { | 840 Isolate* isolate, LanguageMode language_mode) { |
839 StoreICState state = StoreICState(language_mode); | 841 StoreICState state = StoreICState(language_mode); |
840 return KeyedStoreICStub(isolate, state).GetCode(); | 842 return KeyedStoreICStub(isolate, state).GetCode(); |
841 } | 843 } |
842 | 844 |
843 | 845 |
844 Handle<Code> KeyedStoreIC::ChooseMegamorphicStub(Isolate* isolate, | 846 Handle<Code> KeyedStoreIC::ChooseMegamorphicStub(Isolate* isolate, |
845 ExtraICState extra_state) { | 847 ExtraICState extra_state) { |
846 LanguageMode mode = StoreICState::GetLanguageMode(extra_state); | 848 LanguageMode mode = StoreICState::GetLanguageMode(extra_state); |
847 return is_strict(mode) | 849 return is_strict(mode) |
848 ? isolate->builtins()->KeyedStoreIC_Megamorphic_Strict() | 850 ? isolate->builtins()->KeyedStoreIC_Megamorphic_Strict() |
849 : isolate->builtins()->KeyedStoreIC_Megamorphic(); | 851 : isolate->builtins()->KeyedStoreIC_Megamorphic(); |
850 } | 852 } |
851 | 853 |
852 Handle<Object> LoadIC::SimpleFieldLoad(FieldIndex index) { | 854 Handle<Object> LoadIC::SimpleFieldLoad(FieldIndex index) { |
853 if (kind() == Code::LOAD_IC && FLAG_tf_load_ic_stub) { | 855 if (FLAG_tf_load_ic_stub) { |
854 return handle(Smi::FromInt(index.GetLoadByFieldOffset()), isolate()); | 856 return handle(Smi::FromInt(index.GetLoadByFieldOffset()), isolate()); |
855 } | 857 } |
856 DCHECK(kind() == Code::KEYED_LOAD_IC || !FLAG_tf_load_ic_stub); | |
857 TRACE_HANDLER_STATS(isolate(), LoadIC_LoadFieldStub); | 858 TRACE_HANDLER_STATS(isolate(), LoadIC_LoadFieldStub); |
858 LoadFieldStub stub(isolate(), index); | 859 LoadFieldStub stub(isolate(), index); |
859 return stub.GetCode(); | 860 return stub.GetCode(); |
860 } | 861 } |
861 | 862 |
862 | 863 |
863 bool IsCompatibleReceiver(LookupIterator* lookup, Handle<Map> receiver_map) { | 864 bool IsCompatibleReceiver(LookupIterator* lookup, Handle<Map> receiver_map) { |
864 DCHECK(lookup->state() == LookupIterator::ACCESSOR); | 865 DCHECK(lookup->state() == LookupIterator::ACCESSOR); |
865 Isolate* isolate = lookup->isolate(); | 866 Isolate* isolate = lookup->isolate(); |
866 Handle<Object> accessors = lookup->GetAccessors(); | 867 Handle<Object> accessors = lookup->GetAccessors(); |
(...skipping 480 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1347 | 1348 |
1348 void KeyedLoadIC::UpdateLoadElement(Handle<HeapObject> receiver) { | 1349 void KeyedLoadIC::UpdateLoadElement(Handle<HeapObject> receiver) { |
1349 Handle<Map> receiver_map(receiver->map(), isolate()); | 1350 Handle<Map> receiver_map(receiver->map(), isolate()); |
1350 DCHECK(receiver_map->instance_type() != JS_VALUE_TYPE && | 1351 DCHECK(receiver_map->instance_type() != JS_VALUE_TYPE && |
1351 receiver_map->instance_type() != JS_PROXY_TYPE); // Checked by caller. | 1352 receiver_map->instance_type() != JS_PROXY_TYPE); // Checked by caller. |
1352 MapHandleList target_receiver_maps; | 1353 MapHandleList target_receiver_maps; |
1353 TargetMaps(&target_receiver_maps); | 1354 TargetMaps(&target_receiver_maps); |
1354 | 1355 |
1355 if (target_receiver_maps.length() == 0) { | 1356 if (target_receiver_maps.length() == 0) { |
1356 Handle<Code> handler = | 1357 Handle<Code> handler = |
1357 PropertyICCompiler::ComputeKeyedLoadMonomorphicHandler( | 1358 ElementHandlerCompiler::GetKeyedLoadHandler(receiver_map, isolate()); |
1358 receiver_map, extra_ic_state()); | |
1359 return ConfigureVectorState(Handle<Name>(), receiver_map, handler); | 1359 return ConfigureVectorState(Handle<Name>(), receiver_map, handler); |
1360 } | 1360 } |
1361 | 1361 |
1362 for (int i = 0; i < target_receiver_maps.length(); i++) { | 1362 for (int i = 0; i < target_receiver_maps.length(); i++) { |
1363 Handle<Map> map = target_receiver_maps.at(i); | 1363 Handle<Map> map = target_receiver_maps.at(i); |
1364 if (map.is_null()) continue; | 1364 if (map.is_null()) continue; |
1365 if (map->instance_type() == JS_VALUE_TYPE) { | 1365 if (map->instance_type() == JS_VALUE_TYPE) { |
1366 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "JSValue"); | 1366 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "JSValue"); |
1367 return; | 1367 return; |
1368 } | 1368 } |
1369 if (map->instance_type() == JS_PROXY_TYPE) { | 1369 if (map->instance_type() == JS_PROXY_TYPE) { |
1370 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "JSProxy"); | 1370 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "JSProxy"); |
1371 return; | 1371 return; |
1372 } | 1372 } |
1373 } | 1373 } |
1374 | 1374 |
1375 // The first time a receiver is seen that is a transitioned version of the | 1375 // The first time a receiver is seen that is a transitioned version of the |
1376 // previous monomorphic receiver type, assume the new ElementsKind is the | 1376 // previous monomorphic receiver type, assume the new ElementsKind is the |
1377 // monomorphic type. This benefits global arrays that only transition | 1377 // monomorphic type. This benefits global arrays that only transition |
1378 // once, and all call sites accessing them are faster if they remain | 1378 // once, and all call sites accessing them are faster if they remain |
1379 // monomorphic. If this optimistic assumption is not true, the IC will | 1379 // monomorphic. If this optimistic assumption is not true, the IC will |
1380 // miss again and it will become polymorphic and support both the | 1380 // miss again and it will become polymorphic and support both the |
1381 // untransitioned and transitioned maps. | 1381 // untransitioned and transitioned maps. |
1382 if (state() == MONOMORPHIC && !receiver->IsString() && | 1382 if (state() == MONOMORPHIC && !receiver->IsString() && |
1383 IsMoreGeneralElementsKindTransition( | 1383 IsMoreGeneralElementsKindTransition( |
1384 target_receiver_maps.at(0)->elements_kind(), | 1384 target_receiver_maps.at(0)->elements_kind(), |
1385 Handle<JSObject>::cast(receiver)->GetElementsKind())) { | 1385 Handle<JSObject>::cast(receiver)->GetElementsKind())) { |
1386 Handle<Code> handler = | 1386 Handle<Code> handler = |
1387 PropertyICCompiler::ComputeKeyedLoadMonomorphicHandler( | 1387 ElementHandlerCompiler::GetKeyedLoadHandler(receiver_map, isolate()); |
1388 receiver_map, extra_ic_state()); | |
1389 return ConfigureVectorState(Handle<Name>(), receiver_map, handler); | 1388 return ConfigureVectorState(Handle<Name>(), receiver_map, handler); |
1390 } | 1389 } |
1391 | 1390 |
1392 DCHECK(state() != GENERIC); | 1391 DCHECK(state() != GENERIC); |
1393 | 1392 |
1394 // Determine the list of receiver maps that this call site has seen, | 1393 // Determine the list of receiver maps that this call site has seen, |
1395 // adding the map that was just encountered. | 1394 // adding the map that was just encountered. |
1396 if (!AddOneReceiverMapIfMissing(&target_receiver_maps, receiver_map)) { | 1395 if (!AddOneReceiverMapIfMissing(&target_receiver_maps, receiver_map)) { |
1397 // If the miss wasn't due to an unseen map, a polymorphic stub | 1396 // If the miss wasn't due to an unseen map, a polymorphic stub |
1398 // won't help, use the generic stub. | 1397 // won't help, use the generic stub. |
1399 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "same map added twice"); | 1398 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "same map added twice"); |
1400 return; | 1399 return; |
1401 } | 1400 } |
1402 | 1401 |
1403 // If the maximum number of receiver maps has been exceeded, use the generic | 1402 // If the maximum number of receiver maps has been exceeded, use the generic |
1404 // version of the IC. | 1403 // version of the IC. |
1405 if (target_receiver_maps.length() > kMaxKeyedPolymorphism) { | 1404 if (target_receiver_maps.length() > kMaxKeyedPolymorphism) { |
1406 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "max polymorph exceeded"); | 1405 TRACE_GENERIC_IC(isolate(), "KeyedLoadIC", "max polymorph exceeded"); |
1407 return; | 1406 return; |
1408 } | 1407 } |
1409 | 1408 |
1410 List<Handle<Object>> handlers(target_receiver_maps.length()); | 1409 List<Handle<Object>> handlers(target_receiver_maps.length()); |
1411 TRACE_HANDLER_STATS(isolate(), KeyedLoadIC_PolymorphicElement); | |
1412 ElementHandlerCompiler compiler(isolate()); | 1410 ElementHandlerCompiler compiler(isolate()); |
1413 compiler.CompileElementHandlers(&target_receiver_maps, &handlers); | 1411 compiler.CompileElementHandlers(&target_receiver_maps, &handlers); |
1414 ConfigureVectorState(Handle<Name>(), &target_receiver_maps, &handlers); | 1412 ConfigureVectorState(Handle<Name>(), &target_receiver_maps, &handlers); |
1415 } | 1413 } |
1416 | 1414 |
1417 | 1415 |
1418 MaybeHandle<Object> KeyedLoadIC::Load(Handle<Object> object, | 1416 MaybeHandle<Object> KeyedLoadIC::Load(Handle<Object> object, |
1419 Handle<Object> key) { | 1417 Handle<Object> key) { |
1420 if (MigrateDeprecated(object)) { | 1418 if (MigrateDeprecated(object)) { |
1421 Handle<Object> result; | 1419 Handle<Object> result; |
(...skipping 1619 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3041 DCHECK_EQ(FeedbackVectorSlotKind::KEYED_LOAD_IC, | 3039 DCHECK_EQ(FeedbackVectorSlotKind::KEYED_LOAD_IC, |
3042 vector->GetKind(vector_slot)); | 3040 vector->GetKind(vector_slot)); |
3043 KeyedLoadICNexus nexus(vector, vector_slot); | 3041 KeyedLoadICNexus nexus(vector, vector_slot); |
3044 KeyedLoadIC ic(IC::EXTRA_CALL_FRAME, isolate, &nexus); | 3042 KeyedLoadIC ic(IC::EXTRA_CALL_FRAME, isolate, &nexus); |
3045 ic.UpdateState(receiver, key); | 3043 ic.UpdateState(receiver, key); |
3046 RETURN_RESULT_OR_FAILURE(isolate, ic.Load(receiver, key)); | 3044 RETURN_RESULT_OR_FAILURE(isolate, ic.Load(receiver, key)); |
3047 } | 3045 } |
3048 } | 3046 } |
3049 } // namespace internal | 3047 } // namespace internal |
3050 } // namespace v8 | 3048 } // namespace v8 |
OLD | NEW |