| 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/ic/ic-compiler.h" | 5 #include "src/ic/ic-compiler.h" |
| 6 | 6 |
| 7 #include "src/cpu-profiler.h" | 7 #include "src/cpu-profiler.h" |
| 8 #include "src/ic/handler-compiler.h" | 8 #include "src/ic/handler-compiler.h" |
| 9 #include "src/ic/ic-inl.h" | 9 #include "src/ic/ic-inl.h" |
| 10 | 10 |
| (...skipping 101 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 112 stub = LoadFastElementStub(isolate, is_js_array, elements_kind, | 112 stub = LoadFastElementStub(isolate, is_js_array, elements_kind, |
| 113 convert_hole_to_undefined).GetCode(); | 113 convert_hole_to_undefined).GetCode(); |
| 114 } else { | 114 } else { |
| 115 stub = LoadDictionaryElementStub(isolate, LoadICState(extra_ic_state)) | 115 stub = LoadDictionaryElementStub(isolate, LoadICState(extra_ic_state)) |
| 116 .GetCode(); | 116 .GetCode(); |
| 117 } | 117 } |
| 118 return stub; | 118 return stub; |
| 119 } | 119 } |
| 120 | 120 |
| 121 | 121 |
| 122 Handle<Code> PropertyICCompiler::ComputeKeyedStoreMonomorphicHandler( |
| 123 Handle<Map> receiver_map, LanguageMode language_mode, |
| 124 KeyedAccessStoreMode store_mode) { |
| 125 Isolate* isolate = receiver_map->GetIsolate(); |
| 126 ExtraICState extra_state = |
| 127 KeyedStoreIC::ComputeExtraICState(language_mode, store_mode); |
| 128 |
| 129 DCHECK(store_mode == STANDARD_STORE || |
| 130 store_mode == STORE_AND_GROW_NO_TRANSITION || |
| 131 store_mode == STORE_NO_TRANSITION_IGNORE_OUT_OF_BOUNDS || |
| 132 store_mode == STORE_NO_TRANSITION_HANDLE_COW); |
| 133 |
| 134 PropertyICCompiler compiler(isolate, Code::KEYED_STORE_IC, extra_state); |
| 135 Handle<Code> code = |
| 136 compiler.CompileKeyedStoreMonomorphicHandler(receiver_map, store_mode); |
| 137 return code; |
| 138 } |
| 139 |
| 140 |
| 122 Handle<Code> PropertyICCompiler::ComputeKeyedStoreMonomorphic( | 141 Handle<Code> PropertyICCompiler::ComputeKeyedStoreMonomorphic( |
| 123 Handle<Map> receiver_map, LanguageMode language_mode, | 142 Handle<Map> receiver_map, LanguageMode language_mode, |
| 124 KeyedAccessStoreMode store_mode) { | 143 KeyedAccessStoreMode store_mode) { |
| 125 Isolate* isolate = receiver_map->GetIsolate(); | 144 Isolate* isolate = receiver_map->GetIsolate(); |
| 126 ExtraICState extra_state = | 145 ExtraICState extra_state = |
| 127 KeyedStoreIC::ComputeExtraICState(language_mode, store_mode); | 146 KeyedStoreIC::ComputeExtraICState(language_mode, store_mode); |
| 128 Code::Flags flags = | 147 Code::Flags flags = |
| 129 Code::ComputeMonomorphicFlags(Code::KEYED_STORE_IC, extra_state); | 148 Code::ComputeMonomorphicFlags(Code::KEYED_STORE_IC, extra_state); |
| 130 | 149 |
| 131 DCHECK(store_mode == STANDARD_STORE || | 150 DCHECK(store_mode == STANDARD_STORE || |
| (...skipping 81 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 213 Handle<Code> ic = stub->GetCodeCopy(pattern); | 232 Handle<Code> ic = stub->GetCodeCopy(pattern); |
| 214 | 233 |
| 215 if (!receiver_map->is_dictionary_map()) { | 234 if (!receiver_map->is_dictionary_map()) { |
| 216 Map::UpdateCodeCache(receiver_map, name, ic); | 235 Map::UpdateCodeCache(receiver_map, name, ic); |
| 217 } | 236 } |
| 218 | 237 |
| 219 return ic; | 238 return ic; |
| 220 } | 239 } |
| 221 | 240 |
| 222 | 241 |
| 223 Handle<Code> PropertyICCompiler::ComputeKeyedLoadPolymorphic( | |
| 224 MapHandleList* receiver_maps, LanguageMode language_mode) { | |
| 225 Isolate* isolate = receiver_maps->at(0)->GetIsolate(); | |
| 226 DCHECK(KeyedLoadIC::GetKeyType(kNoExtraICState) == ELEMENT); | |
| 227 Code::Flags flags = Code::ComputeFlags(Code::KEYED_LOAD_IC, POLYMORPHIC); | |
| 228 Handle<PolymorphicCodeCache> cache = | |
| 229 isolate->factory()->polymorphic_code_cache(); | |
| 230 Handle<Object> probe = cache->Lookup(receiver_maps, flags); | |
| 231 if (probe->IsCode()) return Handle<Code>::cast(probe); | |
| 232 | |
| 233 CodeHandleList handlers(receiver_maps->length()); | |
| 234 ElementHandlerCompiler compiler(isolate); | |
| 235 compiler.CompileElementHandlers(receiver_maps, &handlers, language_mode); | |
| 236 PropertyICCompiler ic_compiler(isolate, Code::KEYED_LOAD_IC); | |
| 237 Handle<Code> code = ic_compiler.CompilePolymorphic( | |
| 238 receiver_maps, &handlers, isolate->factory()->empty_string(), | |
| 239 Code::NORMAL, ELEMENT); | |
| 240 | |
| 241 isolate->counters()->keyed_load_polymorphic_stubs()->Increment(); | |
| 242 | |
| 243 PolymorphicCodeCache::Update(cache, receiver_maps, flags, code); | |
| 244 return code; | |
| 245 } | |
| 246 | |
| 247 | |
| 248 Handle<Code> PropertyICCompiler::ComputePolymorphic( | 242 Handle<Code> PropertyICCompiler::ComputePolymorphic( |
| 249 Code::Kind kind, MapHandleList* maps, CodeHandleList* handlers, | 243 Code::Kind kind, MapHandleList* maps, CodeHandleList* handlers, |
| 250 int valid_maps, Handle<Name> name, ExtraICState extra_ic_state) { | 244 int valid_maps, Handle<Name> name, ExtraICState extra_ic_state) { |
| 251 Handle<Code> handler = handlers->at(0); | 245 Handle<Code> handler = handlers->at(0); |
| 252 Code::StubType type = valid_maps == 1 ? handler->type() : Code::NORMAL; | 246 Code::StubType type = valid_maps == 1 ? handler->type() : Code::NORMAL; |
| 253 DCHECK(kind == Code::LOAD_IC || kind == Code::STORE_IC); | 247 DCHECK(kind == Code::LOAD_IC || kind == Code::STORE_IC); |
| 254 PropertyICCompiler ic_compiler(name->GetIsolate(), kind, extra_ic_state); | 248 PropertyICCompiler ic_compiler(name->GetIsolate(), kind, extra_ic_state); |
| 255 return ic_compiler.CompilePolymorphic(maps, handlers, name, type, PROPERTY); | 249 return ic_compiler.CompilePolymorphic(maps, handlers, name, type, PROPERTY); |
| 256 } | 250 } |
| 257 | 251 |
| 258 | 252 |
| 253 void PropertyICCompiler::ComputeKeyedStorePolymorphicHandlers( |
| 254 MapHandleList* receiver_maps, MapHandleList* transitioned_maps, |
| 255 CodeHandleList* handlers, KeyedAccessStoreMode store_mode, |
| 256 LanguageMode language_mode) { |
| 257 Isolate* isolate = receiver_maps->at(0)->GetIsolate(); |
| 258 DCHECK(store_mode == STANDARD_STORE || |
| 259 store_mode == STORE_AND_GROW_NO_TRANSITION || |
| 260 store_mode == STORE_NO_TRANSITION_IGNORE_OUT_OF_BOUNDS || |
| 261 store_mode == STORE_NO_TRANSITION_HANDLE_COW); |
| 262 ExtraICState extra_state = |
| 263 KeyedStoreIC::ComputeExtraICState(language_mode, store_mode); |
| 264 PropertyICCompiler compiler(isolate, Code::KEYED_STORE_IC, extra_state); |
| 265 compiler.CompileKeyedStorePolymorphicHandlers( |
| 266 receiver_maps, transitioned_maps, handlers, store_mode); |
| 267 } |
| 268 |
| 269 |
| 259 Handle<Code> PropertyICCompiler::ComputeKeyedStorePolymorphic( | 270 Handle<Code> PropertyICCompiler::ComputeKeyedStorePolymorphic( |
| 260 MapHandleList* receiver_maps, KeyedAccessStoreMode store_mode, | 271 MapHandleList* receiver_maps, KeyedAccessStoreMode store_mode, |
| 261 LanguageMode language_mode) { | 272 LanguageMode language_mode) { |
| 262 Isolate* isolate = receiver_maps->at(0)->GetIsolate(); | 273 Isolate* isolate = receiver_maps->at(0)->GetIsolate(); |
| 263 DCHECK(store_mode == STANDARD_STORE || | 274 DCHECK(store_mode == STANDARD_STORE || |
| 264 store_mode == STORE_AND_GROW_NO_TRANSITION || | 275 store_mode == STORE_AND_GROW_NO_TRANSITION || |
| 265 store_mode == STORE_NO_TRANSITION_IGNORE_OUT_OF_BOUNDS || | 276 store_mode == STORE_NO_TRANSITION_IGNORE_OUT_OF_BOUNDS || |
| 266 store_mode == STORE_NO_TRANSITION_HANDLE_COW); | 277 store_mode == STORE_NO_TRANSITION_HANDLE_COW); |
| 267 Handle<PolymorphicCodeCache> cache = | 278 Handle<PolymorphicCodeCache> cache = |
| 268 isolate->factory()->polymorphic_code_cache(); | 279 isolate->factory()->polymorphic_code_cache(); |
| (...skipping 62 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 331 Code::ComputeFlags(kind, state, extra_ic_state_, type, cache_holder()); | 342 Code::ComputeFlags(kind, state, extra_ic_state_, type, cache_holder()); |
| 332 Handle<Code> code = GetCodeWithFlags(flags, name); | 343 Handle<Code> code = GetCodeWithFlags(flags, name); |
| 333 PROFILE(isolate(), CodeCreateEvent(log_kind(code), *code, *name)); | 344 PROFILE(isolate(), CodeCreateEvent(log_kind(code), *code, *name)); |
| 334 #ifdef DEBUG | 345 #ifdef DEBUG |
| 335 code->VerifyEmbeddedObjects(); | 346 code->VerifyEmbeddedObjects(); |
| 336 #endif | 347 #endif |
| 337 return code; | 348 return code; |
| 338 } | 349 } |
| 339 | 350 |
| 340 | 351 |
| 341 Handle<Code> PropertyICCompiler::CompileKeyedStorePolymorphic( | 352 void PropertyICCompiler::CompileKeyedStorePolymorphicHandlers( |
| 342 MapHandleList* receiver_maps, KeyedAccessStoreMode store_mode) { | 353 MapHandleList* receiver_maps, MapHandleList* transitioned_maps, |
| 343 // Collect MONOMORPHIC stubs for all |receiver_maps|. | 354 CodeHandleList* handlers, KeyedAccessStoreMode store_mode) { |
| 344 CodeHandleList handlers(receiver_maps->length()); | |
| 345 MapHandleList transitioned_maps(receiver_maps->length()); | |
| 346 for (int i = 0; i < receiver_maps->length(); ++i) { | 355 for (int i = 0; i < receiver_maps->length(); ++i) { |
| 347 Handle<Map> receiver_map(receiver_maps->at(i)); | 356 Handle<Map> receiver_map(receiver_maps->at(i)); |
| 348 Handle<Code> cached_stub; | 357 Handle<Code> cached_stub; |
| 349 Handle<Map> transitioned_map = | 358 Handle<Map> transitioned_map = |
| 350 Map::FindTransitionedMap(receiver_map, receiver_maps); | 359 Map::FindTransitionedMap(receiver_map, receiver_maps); |
| 351 | 360 |
| 352 // TODO(mvstanton): The code below is doing pessimistic elements | 361 // TODO(mvstanton): The code below is doing pessimistic elements |
| 353 // transitions. I would like to stop doing that and rely on Allocation Site | 362 // transitions. I would like to stop doing that and rely on Allocation Site |
| 354 // Tracking to do a better job of ensuring the data types are what they need | 363 // Tracking to do a better job of ensuring the data types are what they need |
| 355 // to be. Not all the elements are in place yet, pessimistic elements | 364 // to be. Not all the elements are in place yet, pessimistic elements |
| (...skipping 16 matching lines...) Expand all Loading... |
| 372 } else if (receiver_map->has_fast_elements() || | 381 } else if (receiver_map->has_fast_elements() || |
| 373 receiver_map->has_fixed_typed_array_elements()) { | 382 receiver_map->has_fixed_typed_array_elements()) { |
| 374 cached_stub = StoreFastElementStub(isolate(), is_js_array, | 383 cached_stub = StoreFastElementStub(isolate(), is_js_array, |
| 375 elements_kind, store_mode).GetCode(); | 384 elements_kind, store_mode).GetCode(); |
| 376 } else { | 385 } else { |
| 377 cached_stub = | 386 cached_stub = |
| 378 StoreElementStub(isolate(), elements_kind, store_mode).GetCode(); | 387 StoreElementStub(isolate(), elements_kind, store_mode).GetCode(); |
| 379 } | 388 } |
| 380 } | 389 } |
| 381 DCHECK(!cached_stub.is_null()); | 390 DCHECK(!cached_stub.is_null()); |
| 382 handlers.Add(cached_stub); | 391 handlers->Add(cached_stub); |
| 383 transitioned_maps.Add(transitioned_map); | 392 transitioned_maps->Add(transitioned_map); |
| 384 } | 393 } |
| 394 } |
| 395 |
| 396 |
| 397 Handle<Code> PropertyICCompiler::CompileKeyedStorePolymorphic( |
| 398 MapHandleList* receiver_maps, KeyedAccessStoreMode store_mode) { |
| 399 // Collect MONOMORPHIC stubs for all |receiver_maps|. |
| 400 CodeHandleList handlers(receiver_maps->length()); |
| 401 MapHandleList transitioned_maps(receiver_maps->length()); |
| 402 CompileKeyedStorePolymorphicHandlers(receiver_maps, &transitioned_maps, |
| 403 &handlers, store_mode); |
| 385 | 404 |
| 386 Handle<Code> code = CompileKeyedStorePolymorphic(receiver_maps, &handlers, | 405 Handle<Code> code = CompileKeyedStorePolymorphic(receiver_maps, &handlers, |
| 387 &transitioned_maps); | 406 &transitioned_maps); |
| 388 isolate()->counters()->keyed_store_polymorphic_stubs()->Increment(); | 407 isolate()->counters()->keyed_store_polymorphic_stubs()->Increment(); |
| 389 PROFILE(isolate(), CodeCreateEvent(log_kind(code), *code, 0)); | 408 PROFILE(isolate(), CodeCreateEvent(log_kind(code), *code, 0)); |
| 390 return code; | 409 return code; |
| 391 } | 410 } |
| 392 | 411 |
| 393 | 412 |
| 394 #define __ ACCESS_MASM(masm()) | 413 #define __ ACCESS_MASM(masm()) |
| 395 | 414 |
| 396 | 415 |
| 397 Handle<Code> PropertyICCompiler::CompileKeyedStoreMonomorphic( | 416 Handle<Code> PropertyICCompiler::CompileKeyedStoreMonomorphicHandler( |
| 398 Handle<Map> receiver_map, KeyedAccessStoreMode store_mode) { | 417 Handle<Map> receiver_map, KeyedAccessStoreMode store_mode) { |
| 399 ElementsKind elements_kind = receiver_map->elements_kind(); | 418 ElementsKind elements_kind = receiver_map->elements_kind(); |
| 400 bool is_jsarray = receiver_map->instance_type() == JS_ARRAY_TYPE; | 419 bool is_jsarray = receiver_map->instance_type() == JS_ARRAY_TYPE; |
| 401 Handle<Code> stub; | 420 Handle<Code> stub; |
| 402 if (receiver_map->has_sloppy_arguments_elements()) { | 421 if (receiver_map->has_sloppy_arguments_elements()) { |
| 403 stub = KeyedStoreSloppyArgumentsStub(isolate(), store_mode).GetCode(); | 422 stub = KeyedStoreSloppyArgumentsStub(isolate(), store_mode).GetCode(); |
| 404 } else if (receiver_map->has_fast_elements() || | 423 } else if (receiver_map->has_fast_elements() || |
| 405 receiver_map->has_fixed_typed_array_elements()) { | 424 receiver_map->has_fixed_typed_array_elements()) { |
| 406 stub = StoreFastElementStub(isolate(), is_jsarray, elements_kind, | 425 stub = StoreFastElementStub(isolate(), is_jsarray, elements_kind, |
| 407 store_mode).GetCode(); | 426 store_mode).GetCode(); |
| 408 } else { | 427 } else { |
| 409 stub = StoreElementStub(isolate(), elements_kind, store_mode).GetCode(); | 428 stub = StoreElementStub(isolate(), elements_kind, store_mode).GetCode(); |
| 410 } | 429 } |
| 430 return stub; |
| 431 } |
| 432 |
| 433 |
| 434 Handle<Code> PropertyICCompiler::CompileKeyedStoreMonomorphic( |
| 435 Handle<Map> receiver_map, KeyedAccessStoreMode store_mode) { |
| 436 Handle<Code> stub = |
| 437 CompileKeyedStoreMonomorphicHandler(receiver_map, store_mode); |
| 411 | 438 |
| 412 Handle<WeakCell> cell = Map::WeakCellForMap(receiver_map); | 439 Handle<WeakCell> cell = Map::WeakCellForMap(receiver_map); |
| 413 | 440 |
| 414 __ DispatchWeakMap(receiver(), scratch1(), scratch2(), cell, stub, | 441 __ DispatchWeakMap(receiver(), scratch1(), scratch2(), cell, stub, |
| 415 DO_SMI_CHECK); | 442 DO_SMI_CHECK); |
| 416 | 443 |
| 417 TailCallBuiltin(masm(), Builtins::kKeyedStoreIC_Miss); | 444 TailCallBuiltin(masm(), Builtins::kKeyedStoreIC_Miss); |
| 418 | 445 |
| 419 return GetCode(kind(), Code::NORMAL, factory()->empty_string()); | 446 return GetCode(kind(), Code::NORMAL, factory()->empty_string()); |
| 420 } | 447 } |
| 421 | 448 |
| 422 | 449 |
| 423 #undef __ | 450 #undef __ |
| 424 } // namespace internal | 451 } // namespace internal |
| 425 } // namespace v8 | 452 } // namespace v8 |
| OLD | NEW |