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/v8.h" | 5 #include "src/v8.h" |
6 | 6 |
7 #include "src/code-stubs.h" | 7 #include "src/code-stubs.h" |
8 #include "src/field-index.h" | 8 #include "src/field-index.h" |
9 #include "src/hydrogen.h" | 9 #include "src/hydrogen.h" |
10 #include "src/lithium.h" | 10 #include "src/lithium.h" |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
56 return arguments_length_; | 56 return arguments_length_; |
57 } | 57 } |
58 CompilationInfo* info() { return &info_; } | 58 CompilationInfo* info() { return &info_; } |
59 HydrogenCodeStub* stub() { return info_.code_stub(); } | 59 HydrogenCodeStub* stub() { return info_.code_stub(); } |
60 HContext* context() { return context_; } | 60 HContext* context() { return context_; } |
61 Isolate* isolate() { return info_.isolate(); } | 61 Isolate* isolate() { return info_.isolate(); } |
62 | 62 |
63 HLoadNamedField* BuildLoadNamedField(HValue* object, | 63 HLoadNamedField* BuildLoadNamedField(HValue* object, |
64 FieldIndex index); | 64 FieldIndex index); |
65 void BuildStoreNamedField(HValue* object, HValue* value, FieldIndex index, | 65 void BuildStoreNamedField(HValue* object, HValue* value, FieldIndex index, |
66 Representation representation); | 66 Representation representation, |
| 67 bool transition_to_field); |
67 | 68 |
68 enum ArgumentClass { | 69 enum ArgumentClass { |
69 NONE, | 70 NONE, |
70 SINGLE, | 71 SINGLE, |
71 MULTIPLE | 72 MULTIPLE |
72 }; | 73 }; |
73 | 74 |
74 HValue* BuildArrayConstructor(ElementsKind kind, | 75 HValue* BuildArrayConstructor(ElementsKind kind, |
75 AllocationSiteOverrideMode override_mode, | 76 AllocationSiteOverrideMode override_mode, |
76 ArgumentClass argument_class); | 77 ArgumentClass argument_class); |
(...skipping 518 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
595 return Add<HLoadNamedField>(descriptors, static_cast<HValue*>(NULL), | 596 return Add<HLoadNamedField>(descriptors, static_cast<HValue*>(NULL), |
596 value_access); | 597 value_access); |
597 } | 598 } |
598 | 599 |
599 | 600 |
600 Handle<Code> LoadConstantStub::GenerateCode() { return DoGenerateCode(this); } | 601 Handle<Code> LoadConstantStub::GenerateCode() { return DoGenerateCode(this); } |
601 | 602 |
602 | 603 |
603 void CodeStubGraphBuilderBase::BuildStoreNamedField( | 604 void CodeStubGraphBuilderBase::BuildStoreNamedField( |
604 HValue* object, HValue* value, FieldIndex index, | 605 HValue* object, HValue* value, FieldIndex index, |
605 Representation representation) { | 606 Representation representation, bool transition_to_field) { |
606 DCHECK(!index.is_double() || representation.IsDouble()); | 607 DCHECK(!index.is_double() || representation.IsDouble()); |
607 int offset = index.offset(); | 608 int offset = index.offset(); |
608 HObjectAccess access = | 609 HObjectAccess access = |
609 index.is_inobject() | 610 index.is_inobject() |
610 ? HObjectAccess::ForObservableJSObjectOffset(offset, representation) | 611 ? HObjectAccess::ForObservableJSObjectOffset(offset, representation) |
611 : HObjectAccess::ForBackingStoreOffset(offset, representation); | 612 : HObjectAccess::ForBackingStoreOffset(offset, representation); |
612 | 613 |
613 if (representation.IsDouble()) { | 614 if (representation.IsDouble()) { |
614 // Load the heap number. | 615 HObjectAccess heap_number_access = |
615 object = Add<HLoadNamedField>( | 616 access.WithRepresentation(Representation::Tagged()); |
616 object, static_cast<HValue*>(NULL), | 617 if (transition_to_field) { |
617 access.WithRepresentation(Representation::Tagged())); | 618 // The store requires a mutable HeapNumber to be allocated. |
618 // Store the double value into it. | 619 NoObservableSideEffectsScope no_side_effects(this); |
619 access = HObjectAccess::ForHeapNumberValue(); | 620 HInstruction* heap_number_size = Add<HConstant>(HeapNumber::kSize); |
| 621 |
| 622 // TODO(hpayer): Allocation site pretenuring support. |
| 623 HInstruction* heap_number = |
| 624 Add<HAllocate>(heap_number_size, HType::HeapObject(), NOT_TENURED, |
| 625 MUTABLE_HEAP_NUMBER_TYPE); |
| 626 AddStoreMapConstant(heap_number, |
| 627 isolate()->factory()->mutable_heap_number_map()); |
| 628 Add<HStoreNamedField>(heap_number, HObjectAccess::ForHeapNumberValue(), |
| 629 value); |
| 630 // Store the new mutable heap number into the object. |
| 631 access = heap_number_access; |
| 632 value = heap_number; |
| 633 } else { |
| 634 // Load the heap number. |
| 635 object = Add<HLoadNamedField>(object, static_cast<HValue*>(NULL), |
| 636 heap_number_access); |
| 637 // Store the double value into it. |
| 638 access = HObjectAccess::ForHeapNumberValue(); |
| 639 } |
620 } else if (representation.IsHeapObject()) { | 640 } else if (representation.IsHeapObject()) { |
621 BuildCheckHeapObject(value); | 641 BuildCheckHeapObject(value); |
622 } | 642 } |
623 | 643 |
624 Add<HStoreNamedField>(object, access, value, INITIALIZING_STORE); | 644 Add<HStoreNamedField>(object, access, value, INITIALIZING_STORE); |
625 } | 645 } |
626 | 646 |
627 | 647 |
628 template <> | 648 template <> |
629 HValue* CodeStubGraphBuilder<StoreFieldStub>::BuildCodeStub() { | 649 HValue* CodeStubGraphBuilder<StoreFieldStub>::BuildCodeStub() { |
630 BuildStoreNamedField(GetParameter(0), GetParameter(2), casted_stub()->index(), | 650 BuildStoreNamedField(GetParameter(0), GetParameter(2), casted_stub()->index(), |
631 casted_stub()->representation()); | 651 casted_stub()->representation(), false); |
632 return GetParameter(2); | 652 return GetParameter(2); |
633 } | 653 } |
634 | 654 |
635 | 655 |
636 Handle<Code> StoreFieldStub::GenerateCode() { return DoGenerateCode(this); } | 656 Handle<Code> StoreFieldStub::GenerateCode() { return DoGenerateCode(this); } |
637 | 657 |
638 | 658 |
639 template <> | 659 template <> |
| 660 HValue* CodeStubGraphBuilder<ExtendStorageStub>::BuildCodeStub() { |
| 661 HValue* object = GetParameter(ExtendStorageDescriptor::kReceiverIndex); |
| 662 HValue* properties = |
| 663 Add<HLoadNamedField>(object, static_cast<HValue*>(NULL), |
| 664 HObjectAccess::ForPropertiesPointer()); |
| 665 HValue* length = AddLoadFixedArrayLength(properties); |
| 666 HValue* delta = Add<HConstant>(static_cast<int32_t>(JSObject::kFieldsAdded)); |
| 667 HValue* new_capacity = AddUncasted<HAdd>(length, delta); |
| 668 |
| 669 // Grow properties array. |
| 670 ElementsKind kind = FAST_ELEMENTS; |
| 671 Add<HBoundsCheck>(new_capacity, |
| 672 Add<HConstant>((Page::kMaxRegularHeapObjectSize - |
| 673 FixedArray::kHeaderSize) >> |
| 674 ElementsKindToShiftSize(kind))); |
| 675 |
| 676 // Reuse this code for properties backing store allocation. |
| 677 HValue* new_properties = BuildAllocateAndInitializeArray(kind, new_capacity); |
| 678 |
| 679 BuildCopyProperties(properties, new_properties, length, new_capacity); |
| 680 |
| 681 // Store the new value into the "extended" object. |
| 682 Add<HStoreNamedField>(object, HObjectAccess::ForPropertiesPointer(), |
| 683 new_properties); |
| 684 |
| 685 BuildStoreNamedField( |
| 686 object, GetParameter(ExtendStorageDescriptor::kValueIndex), |
| 687 casted_stub()->index(), casted_stub()->representation(), true); |
| 688 |
| 689 // And finally update the map after the new field is added. |
| 690 Add<HStoreNamedField>(object, HObjectAccess::ForMap(), |
| 691 GetParameter(ExtendStorageDescriptor::kMapIndex)); |
| 692 |
| 693 return GetParameter(ExtendStorageDescriptor::kValueIndex); |
| 694 } |
| 695 |
| 696 |
| 697 Handle<Code> ExtendStorageStub::GenerateCode() { return DoGenerateCode(this); } |
| 698 |
| 699 |
| 700 template <> |
640 HValue* CodeStubGraphBuilder<StringLengthStub>::BuildCodeStub() { | 701 HValue* CodeStubGraphBuilder<StringLengthStub>::BuildCodeStub() { |
641 HValue* string = BuildLoadNamedField(GetParameter(0), | 702 HValue* string = BuildLoadNamedField(GetParameter(0), |
642 FieldIndex::ForInObjectOffset(JSValue::kValueOffset)); | 703 FieldIndex::ForInObjectOffset(JSValue::kValueOffset)); |
643 return BuildLoadNamedField(string, | 704 return BuildLoadNamedField(string, |
644 FieldIndex::ForInObjectOffset(String::kLengthOffset)); | 705 FieldIndex::ForInObjectOffset(String::kLengthOffset)); |
645 } | 706 } |
646 | 707 |
647 | 708 |
648 Handle<Code> StringLengthStub::GenerateCode() { | 709 Handle<Code> StringLengthStub::GenerateCode() { |
649 return DoGenerateCode(this); | 710 return DoGenerateCode(this); |
(...skipping 1173 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1823 | 1884 |
1824 // Probe the stub cache. | 1885 // Probe the stub cache. |
1825 Code::Flags flags = Code::RemoveTypeAndHolderFromFlags( | 1886 Code::Flags flags = Code::RemoveTypeAndHolderFromFlags( |
1826 Code::ComputeHandlerFlags(Code::LOAD_IC)); | 1887 Code::ComputeHandlerFlags(Code::LOAD_IC)); |
1827 Add<HTailCallThroughMegamorphicCache>(receiver, name, flags); | 1888 Add<HTailCallThroughMegamorphicCache>(receiver, name, flags); |
1828 | 1889 |
1829 // We never continue. | 1890 // We never continue. |
1830 return graph()->GetConstant0(); | 1891 return graph()->GetConstant0(); |
1831 } | 1892 } |
1832 } } // namespace v8::internal | 1893 } } // namespace v8::internal |
OLD | NEW |