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 #ifndef V8_IC_H_ | 5 #ifndef V8_IC_H_ |
6 #define V8_IC_H_ | 6 #define V8_IC_H_ |
7 | 7 |
8 #include "src/ic/ic-state.h" | 8 #include "src/ic/ic-state.h" |
9 #include "src/macro-assembler.h" | 9 #include "src/macro-assembler.h" |
10 | 10 |
(...skipping 95 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
106 InlineCacheState state = code->ic_state(); | 106 InlineCacheState state = code->ic_state(); |
107 return !FLAG_use_ic || state == UNINITIALIZED || state == PREMONOMORPHIC; | 107 return !FLAG_use_ic || state == UNINITIALIZED || state == PREMONOMORPHIC; |
108 } | 108 } |
109 | 109 |
110 static bool IsCleared(FeedbackNexus* nexus) { | 110 static bool IsCleared(FeedbackNexus* nexus) { |
111 InlineCacheState state = nexus->StateFromFeedback(); | 111 InlineCacheState state = nexus->StateFromFeedback(); |
112 return !FLAG_use_ic || state == UNINITIALIZED || state == PREMONOMORPHIC; | 112 return !FLAG_use_ic || state == UNINITIALIZED || state == PREMONOMORPHIC; |
113 } | 113 } |
114 | 114 |
115 static bool ICUseVector(Code::Kind kind) { | 115 static bool ICUseVector(Code::Kind kind) { |
116 return (FLAG_vector_ics && | 116 return kind == Code::LOAD_IC || kind == Code::KEYED_LOAD_IC || |
117 (kind == Code::LOAD_IC || kind == Code::KEYED_LOAD_IC)) || | |
118 kind == Code::CALL_IC; | 117 kind == Code::CALL_IC; |
119 } | 118 } |
120 | 119 |
121 protected: | 120 protected: |
122 // Get the call-site target; used for determining the state. | 121 // Get the call-site target; used for determining the state. |
123 Handle<Code> target() const { return target_; } | 122 Handle<Code> target() const { return target_; } |
124 | 123 |
125 Address fp() const { return fp_; } | 124 Address fp() const { return fp_; } |
126 Address pc() const { return *pc_address_; } | 125 Address pc() const { return *pc_address_; } |
127 Isolate* isolate() const { return isolate_; } | 126 Isolate* isolate() const { return isolate_; } |
(...skipping 226 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
354 static ExtraICState ComputeExtraICState(ContextualMode contextual_mode) { | 353 static ExtraICState ComputeExtraICState(ContextualMode contextual_mode) { |
355 return LoadICState(contextual_mode).GetExtraICState(); | 354 return LoadICState(contextual_mode).GetExtraICState(); |
356 } | 355 } |
357 | 356 |
358 ContextualMode contextual_mode() const { | 357 ContextualMode contextual_mode() const { |
359 return LoadICState::GetContextualMode(extra_ic_state()); | 358 return LoadICState::GetContextualMode(extra_ic_state()); |
360 } | 359 } |
361 | 360 |
362 LoadIC(FrameDepth depth, Isolate* isolate, FeedbackNexus* nexus = NULL) | 361 LoadIC(FrameDepth depth, Isolate* isolate, FeedbackNexus* nexus = NULL) |
363 : IC(depth, isolate, nexus) { | 362 : IC(depth, isolate, nexus) { |
364 DCHECK(!FLAG_vector_ics || nexus != NULL); | 363 DCHECK(nexus != NULL); |
365 DCHECK(IsLoadStub()); | 364 DCHECK(IsLoadStub()); |
366 } | 365 } |
367 | 366 |
368 // TODO(mvstanton): The for_queries_only is because we have a case where we | 367 // TODO(mvstanton): The for_queries_only is because we have a case where we |
369 // construct an IC only to gather the contextual mode, and we don't have | 368 // construct an IC only to gather the contextual mode, and we don't have |
370 // vector/slot information. for_queries_only is a temporary hack to enable the | 369 // vector/slot information. for_queries_only is a temporary hack to enable the |
371 // strong DCHECK protection around vector/slot. | 370 // strong DCHECK protection around vector/slot. |
372 LoadIC(FrameDepth depth, Isolate* isolate, bool for_queries_only) | 371 LoadIC(FrameDepth depth, Isolate* isolate, bool for_queries_only) |
373 : IC(depth, isolate, NULL, for_queries_only) { | 372 : IC(depth, isolate, NULL, for_queries_only) { |
374 DCHECK(IsLoadStub()); | 373 DCHECK(IsLoadStub()); |
375 } | 374 } |
376 | 375 |
377 // Returns if this IC is for contextual (no explicit receiver) | 376 // Returns if this IC is for contextual (no explicit receiver) |
378 // access to properties. | 377 // access to properties. |
379 bool IsUndeclaredGlobal(Handle<Object> receiver) { | 378 bool IsUndeclaredGlobal(Handle<Object> receiver) { |
380 if (receiver->IsGlobalObject()) { | 379 if (receiver->IsGlobalObject()) { |
381 return contextual_mode() == CONTEXTUAL; | 380 return contextual_mode() == CONTEXTUAL; |
382 } else { | 381 } else { |
383 DCHECK(contextual_mode() != CONTEXTUAL); | 382 DCHECK(contextual_mode() != CONTEXTUAL); |
384 return false; | 383 return false; |
385 } | 384 } |
386 } | 385 } |
387 | 386 |
388 // Code generator routines. | 387 // Code generator routines. |
389 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); } | 388 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); } |
390 static void GeneratePreMonomorphic(MacroAssembler* masm) { | |
391 GenerateMiss(masm); | |
392 } | |
393 static void GenerateMiss(MacroAssembler* masm); | 389 static void GenerateMiss(MacroAssembler* masm); |
394 static void GenerateNormal(MacroAssembler* masm); | 390 static void GenerateNormal(MacroAssembler* masm); |
395 static void GenerateRuntimeGetProperty(MacroAssembler* masm); | 391 static void GenerateRuntimeGetProperty(MacroAssembler* masm); |
396 | 392 |
397 static Handle<Code> initialize_stub(Isolate* isolate, | 393 static Handle<Code> initialize_stub(Isolate* isolate, |
398 ExtraICState extra_state); | 394 ExtraICState extra_state); |
399 static Handle<Code> initialize_stub_in_optimized_code( | 395 static Handle<Code> initialize_stub_in_optimized_code( |
400 Isolate* isolate, ExtraICState extra_state, State initialization_state); | 396 Isolate* isolate, ExtraICState extra_state, State initialization_state); |
401 static Handle<Code> load_global(Isolate* isolate, Handle<GlobalObject> global, | |
402 Handle<String> name); | |
403 | 397 |
404 MUST_USE_RESULT MaybeHandle<Object> Load(Handle<Object> object, | 398 MUST_USE_RESULT MaybeHandle<Object> Load(Handle<Object> object, |
405 Handle<Name> name); | 399 Handle<Name> name); |
406 | 400 |
407 static void Clear(Isolate* isolate, Code* host, LoadICNexus* nexus); | 401 static void Clear(Isolate* isolate, Code* host, LoadICNexus* nexus); |
408 | 402 |
409 protected: | 403 protected: |
410 inline void set_target(Code* code); | 404 inline void set_target(Code* code); |
411 | 405 |
412 Handle<Code> slow_stub() const { | 406 Handle<Code> slow_stub() const { |
413 if (kind() == Code::LOAD_IC) { | 407 if (kind() == Code::LOAD_IC) { |
414 return isolate()->builtins()->LoadIC_Slow(); | 408 return isolate()->builtins()->LoadIC_Slow(); |
415 } else { | 409 } else { |
416 DCHECK_EQ(Code::KEYED_LOAD_IC, kind()); | 410 DCHECK_EQ(Code::KEYED_LOAD_IC, kind()); |
417 return isolate()->builtins()->KeyedLoadIC_Slow(); | 411 return isolate()->builtins()->KeyedLoadIC_Slow(); |
418 } | 412 } |
419 } | 413 } |
420 | 414 |
421 Handle<Code> megamorphic_stub() override; | 415 Handle<Code> megamorphic_stub() override; |
422 | 416 |
423 // Update the inline cache and the global stub cache based on the | 417 // Update the inline cache and the global stub cache based on the |
424 // lookup result. | 418 // lookup result. |
425 void UpdateCaches(LookupIterator* lookup); | 419 void UpdateCaches(LookupIterator* lookup); |
426 | 420 |
427 virtual Handle<Code> CompileHandler(LookupIterator* lookup, | 421 virtual Handle<Code> CompileHandler(LookupIterator* lookup, |
428 Handle<Object> unused, | 422 Handle<Object> unused, |
429 CacheHolderFlag cache_holder) override; | 423 CacheHolderFlag cache_holder) override; |
430 | 424 |
431 private: | 425 private: |
432 virtual Handle<Code> pre_monomorphic_stub() const; | |
433 static Handle<Code> pre_monomorphic_stub(Isolate* isolate, | |
434 ExtraICState extra_state); | |
435 | |
436 Handle<Code> SimpleFieldLoad(FieldIndex index); | 426 Handle<Code> SimpleFieldLoad(FieldIndex index); |
437 | 427 |
438 static void Clear(Isolate* isolate, Address address, Code* target, | 428 static void Clear(Isolate* isolate, Address address, Code* target, |
439 ConstantPoolArray* constant_pool); | 429 ConstantPoolArray* constant_pool); |
440 | 430 |
441 friend class IC; | 431 friend class IC; |
442 }; | 432 }; |
443 | 433 |
444 | 434 |
445 class KeyedLoadIC : public LoadIC { | 435 class KeyedLoadIC : public LoadIC { |
446 public: | 436 public: |
447 // ExtraICState bits (building on IC) | 437 // ExtraICState bits (building on IC) |
448 class IcCheckTypeField : public BitField<IcCheckType, 1, 1> {}; | 438 class IcCheckTypeField : public BitField<IcCheckType, 1, 1> {}; |
449 | 439 |
450 static ExtraICState ComputeExtraICState(ContextualMode contextual_mode, | 440 static ExtraICState ComputeExtraICState(ContextualMode contextual_mode, |
451 IcCheckType key_type) { | 441 IcCheckType key_type) { |
452 return LoadICState(contextual_mode).GetExtraICState() | | 442 return LoadICState(contextual_mode).GetExtraICState() | |
453 IcCheckTypeField::encode(key_type); | 443 IcCheckTypeField::encode(key_type); |
454 } | 444 } |
455 | 445 |
456 static IcCheckType GetKeyType(ExtraICState extra_state) { | 446 static IcCheckType GetKeyType(ExtraICState extra_state) { |
457 return IcCheckTypeField::decode(extra_state); | 447 return IcCheckTypeField::decode(extra_state); |
458 } | 448 } |
459 | 449 |
460 KeyedLoadIC(FrameDepth depth, Isolate* isolate, | 450 KeyedLoadIC(FrameDepth depth, Isolate* isolate, |
461 KeyedLoadICNexus* nexus = NULL) | 451 KeyedLoadICNexus* nexus = NULL) |
462 : LoadIC(depth, isolate, nexus) { | 452 : LoadIC(depth, isolate, nexus) { |
463 DCHECK(!FLAG_vector_ics || nexus != NULL); | 453 DCHECK(nexus != NULL); |
464 DCHECK(target()->is_keyed_load_stub()); | 454 DCHECK(target()->is_keyed_load_stub()); |
465 } | 455 } |
466 | 456 |
467 MUST_USE_RESULT MaybeHandle<Object> Load(Handle<Object> object, | 457 MUST_USE_RESULT MaybeHandle<Object> Load(Handle<Object> object, |
468 Handle<Object> key); | 458 Handle<Object> key); |
469 | 459 |
470 // Code generator routines. | 460 // Code generator routines. |
471 static void GenerateMiss(MacroAssembler* masm); | 461 static void GenerateMiss(MacroAssembler* masm); |
472 static void GenerateRuntimeGetProperty(MacroAssembler* masm); | 462 static void GenerateRuntimeGetProperty(MacroAssembler* masm); |
473 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); } | 463 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); } |
474 static void GeneratePreMonomorphic(MacroAssembler* masm) { | |
475 GenerateMiss(masm); | |
476 } | |
477 static void GenerateMegamorphic(MacroAssembler* masm); | 464 static void GenerateMegamorphic(MacroAssembler* masm); |
478 | 465 |
479 // Bit mask to be tested against bit field for the cases when | 466 // Bit mask to be tested against bit field for the cases when |
480 // generic stub should go into slow case. | 467 // generic stub should go into slow case. |
481 // Access check is necessary explicitly since generic stub does not perform | 468 // Access check is necessary explicitly since generic stub does not perform |
482 // map checks. | 469 // map checks. |
483 static const int kSlowCaseBitFieldMask = | 470 static const int kSlowCaseBitFieldMask = |
484 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor); | 471 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor); |
485 | 472 |
486 static Handle<Code> initialize_stub(Isolate* isolate); | 473 static Handle<Code> initialize_stub(Isolate* isolate); |
487 static Handle<Code> initialize_stub_in_optimized_code( | 474 static Handle<Code> initialize_stub_in_optimized_code( |
488 Isolate* isolate, State initialization_state); | 475 Isolate* isolate, State initialization_state); |
489 static Handle<Code> ChooseMegamorphicStub(Isolate* isolate); | 476 static Handle<Code> ChooseMegamorphicStub(Isolate* isolate); |
490 static Handle<Code> pre_monomorphic_stub(Isolate* isolate); | |
491 | 477 |
492 static void Clear(Isolate* isolate, Code* host, KeyedLoadICNexus* nexus); | 478 static void Clear(Isolate* isolate, Code* host, KeyedLoadICNexus* nexus); |
493 | 479 |
494 protected: | 480 protected: |
495 // receiver is HeapObject because it could be a String or a JSObject | 481 // receiver is HeapObject because it could be a String or a JSObject |
496 Handle<Code> LoadElementStub(Handle<HeapObject> receiver); | 482 Handle<Code> LoadElementStub(Handle<HeapObject> receiver); |
497 virtual Handle<Code> pre_monomorphic_stub() const { | |
498 return pre_monomorphic_stub(isolate()); | |
499 } | |
500 | 483 |
501 private: | 484 private: |
502 static void Clear(Isolate* isolate, Address address, Code* target, | 485 static void Clear(Isolate* isolate, Address address, Code* target, |
503 ConstantPoolArray* constant_pool); | 486 ConstantPoolArray* constant_pool); |
504 | 487 |
505 friend class IC; | 488 friend class IC; |
506 }; | 489 }; |
507 | 490 |
508 | 491 |
509 class StoreIC : public IC { | 492 class StoreIC : public IC { |
(...skipping 256 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
766 | 749 |
767 // Support functions for interceptor handlers. | 750 // Support functions for interceptor handlers. |
768 DECLARE_RUNTIME_FUNCTION(LoadPropertyWithInterceptorOnly); | 751 DECLARE_RUNTIME_FUNCTION(LoadPropertyWithInterceptorOnly); |
769 DECLARE_RUNTIME_FUNCTION(LoadPropertyWithInterceptor); | 752 DECLARE_RUNTIME_FUNCTION(LoadPropertyWithInterceptor); |
770 DECLARE_RUNTIME_FUNCTION(LoadElementWithInterceptor); | 753 DECLARE_RUNTIME_FUNCTION(LoadElementWithInterceptor); |
771 DECLARE_RUNTIME_FUNCTION(StorePropertyWithInterceptor); | 754 DECLARE_RUNTIME_FUNCTION(StorePropertyWithInterceptor); |
772 } | 755 } |
773 } // namespace v8::internal | 756 } // namespace v8::internal |
774 | 757 |
775 #endif // V8_IC_H_ | 758 #endif // V8_IC_H_ |
OLD | NEW |