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 #if V8_TARGET_ARCH_IA32 | 7 #if V8_TARGET_ARCH_IA32 |
8 | 8 |
9 #include "src/base/bits.h" | 9 #include "src/base/bits.h" |
10 #include "src/bootstrapper.h" | 10 #include "src/bootstrapper.h" |
(...skipping 635 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
646 __ bind(&done); | 646 __ bind(&done); |
647 __ IncrementCounter(counters->math_pow(), 1); | 647 __ IncrementCounter(counters->math_pow(), 1); |
648 __ ret(0); | 648 __ ret(0); |
649 } | 649 } |
650 } | 650 } |
651 | 651 |
652 | 652 |
653 void FunctionPrototypeStub::Generate(MacroAssembler* masm) { | 653 void FunctionPrototypeStub::Generate(MacroAssembler* masm) { |
654 Label miss; | 654 Label miss; |
655 Register receiver = LoadDescriptor::ReceiverRegister(); | 655 Register receiver = LoadDescriptor::ReceiverRegister(); |
656 if (FLAG_vector_ics) { | 656 // With careful management, we won't have to save slot and vector on |
657 // With careful management, we won't have to save slot and vector on | 657 // the stack. Simply handle the possibly missing case first. |
658 // the stack. Simply handle the possibly missing case first. | 658 // TODO(mvstanton): this code can be more efficient. |
659 // TODO(mvstanton): this code can be more efficient. | 659 __ cmp(FieldOperand(receiver, JSFunction::kPrototypeOrInitialMapOffset), |
660 __ cmp(FieldOperand(receiver, JSFunction::kPrototypeOrInitialMapOffset), | 660 Immediate(isolate()->factory()->the_hole_value())); |
661 Immediate(isolate()->factory()->the_hole_value())); | 661 __ j(equal, &miss); |
662 __ j(equal, &miss); | 662 __ TryGetFunctionPrototype(receiver, eax, ebx, &miss); |
663 __ TryGetFunctionPrototype(receiver, eax, ebx, &miss); | 663 __ ret(0); |
664 __ ret(0); | |
665 } else { | |
666 NamedLoadHandlerCompiler::GenerateLoadFunctionPrototype(masm, receiver, eax, | |
667 ebx, &miss); | |
668 } | |
669 | 664 |
670 __ bind(&miss); | 665 __ bind(&miss); |
671 PropertyAccessCompiler::TailCallBuiltin( | 666 PropertyAccessCompiler::TailCallBuiltin( |
672 masm, PropertyAccessCompiler::MissBuiltin(Code::LOAD_IC)); | 667 masm, PropertyAccessCompiler::MissBuiltin(Code::LOAD_IC)); |
673 } | 668 } |
674 | 669 |
675 | 670 |
676 void LoadIndexedInterceptorStub::Generate(MacroAssembler* masm) { | 671 void LoadIndexedInterceptorStub::Generate(MacroAssembler* masm) { |
677 // Return address is on the stack. | 672 // Return address is on the stack. |
678 Label slow; | 673 Label slow; |
(...skipping 27 matching lines...) Expand all Loading... |
706 void LoadIndexedStringStub::Generate(MacroAssembler* masm) { | 701 void LoadIndexedStringStub::Generate(MacroAssembler* masm) { |
707 // Return address is on the stack. | 702 // Return address is on the stack. |
708 Label miss; | 703 Label miss; |
709 | 704 |
710 Register receiver = LoadDescriptor::ReceiverRegister(); | 705 Register receiver = LoadDescriptor::ReceiverRegister(); |
711 Register index = LoadDescriptor::NameRegister(); | 706 Register index = LoadDescriptor::NameRegister(); |
712 Register scratch = edi; | 707 Register scratch = edi; |
713 DCHECK(!scratch.is(receiver) && !scratch.is(index)); | 708 DCHECK(!scratch.is(receiver) && !scratch.is(index)); |
714 Register result = eax; | 709 Register result = eax; |
715 DCHECK(!result.is(scratch)); | 710 DCHECK(!result.is(scratch)); |
716 DCHECK(!FLAG_vector_ics || | 711 DCHECK(!scratch.is(VectorLoadICDescriptor::VectorRegister()) && |
717 (!scratch.is(VectorLoadICDescriptor::VectorRegister()) && | 712 result.is(VectorLoadICDescriptor::SlotRegister())); |
718 result.is(VectorLoadICDescriptor::SlotRegister()))); | |
719 | 713 |
720 // StringCharAtGenerator doesn't use the result register until it's passed | 714 // StringCharAtGenerator doesn't use the result register until it's passed |
721 // the different miss possibilities. If it did, we would have a conflict | 715 // the different miss possibilities. If it did, we would have a conflict |
722 // when FLAG_vector_ics is true. | 716 // when FLAG_vector_ics is true. |
723 StringCharAtGenerator char_at_generator(receiver, index, scratch, result, | 717 StringCharAtGenerator char_at_generator(receiver, index, scratch, result, |
724 &miss, // When not a string. | 718 &miss, // When not a string. |
725 &miss, // When not a number. | 719 &miss, // When not a number. |
726 &miss, // When index out of range. | 720 &miss, // When index out of range. |
727 STRING_INDEX_IS_ARRAY_INDEX, | 721 STRING_INDEX_IS_ARRAY_INDEX, |
728 RECEIVER_IS_STRING); | 722 RECEIVER_IS_STRING); |
(...skipping 2226 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2955 __ Abort(kUnexpectedFallthroughToCharCodeAtSlowCase); | 2949 __ Abort(kUnexpectedFallthroughToCharCodeAtSlowCase); |
2956 | 2950 |
2957 // Index is not a smi. | 2951 // Index is not a smi. |
2958 __ bind(&index_not_smi_); | 2952 __ bind(&index_not_smi_); |
2959 // If index is a heap number, try converting it to an integer. | 2953 // If index is a heap number, try converting it to an integer. |
2960 __ CheckMap(index_, | 2954 __ CheckMap(index_, |
2961 masm->isolate()->factory()->heap_number_map(), | 2955 masm->isolate()->factory()->heap_number_map(), |
2962 index_not_number_, | 2956 index_not_number_, |
2963 DONT_DO_SMI_CHECK); | 2957 DONT_DO_SMI_CHECK); |
2964 call_helper.BeforeCall(masm); | 2958 call_helper.BeforeCall(masm); |
2965 if (FLAG_vector_ics && embed_mode == PART_OF_IC_HANDLER) { | 2959 if (embed_mode == PART_OF_IC_HANDLER) { |
2966 __ push(VectorLoadICDescriptor::VectorRegister()); | 2960 __ push(VectorLoadICDescriptor::VectorRegister()); |
2967 __ push(VectorLoadICDescriptor::SlotRegister()); | 2961 __ push(VectorLoadICDescriptor::SlotRegister()); |
2968 } | 2962 } |
2969 __ push(object_); | 2963 __ push(object_); |
2970 __ push(index_); // Consumed by runtime conversion function. | 2964 __ push(index_); // Consumed by runtime conversion function. |
2971 if (index_flags_ == STRING_INDEX_IS_NUMBER) { | 2965 if (index_flags_ == STRING_INDEX_IS_NUMBER) { |
2972 __ CallRuntime(Runtime::kNumberToIntegerMapMinusZero, 1); | 2966 __ CallRuntime(Runtime::kNumberToIntegerMapMinusZero, 1); |
2973 } else { | 2967 } else { |
2974 DCHECK(index_flags_ == STRING_INDEX_IS_ARRAY_INDEX); | 2968 DCHECK(index_flags_ == STRING_INDEX_IS_ARRAY_INDEX); |
2975 // NumberToSmi discards numbers that are not exact integers. | 2969 // NumberToSmi discards numbers that are not exact integers. |
2976 __ CallRuntime(Runtime::kNumberToSmi, 1); | 2970 __ CallRuntime(Runtime::kNumberToSmi, 1); |
2977 } | 2971 } |
2978 if (!index_.is(eax)) { | 2972 if (!index_.is(eax)) { |
2979 // Save the conversion result before the pop instructions below | 2973 // Save the conversion result before the pop instructions below |
2980 // have a chance to overwrite it. | 2974 // have a chance to overwrite it. |
2981 __ mov(index_, eax); | 2975 __ mov(index_, eax); |
2982 } | 2976 } |
2983 __ pop(object_); | 2977 __ pop(object_); |
2984 if (FLAG_vector_ics && embed_mode == PART_OF_IC_HANDLER) { | 2978 if (embed_mode == PART_OF_IC_HANDLER) { |
2985 __ pop(VectorLoadICDescriptor::SlotRegister()); | 2979 __ pop(VectorLoadICDescriptor::SlotRegister()); |
2986 __ pop(VectorLoadICDescriptor::VectorRegister()); | 2980 __ pop(VectorLoadICDescriptor::VectorRegister()); |
2987 } | 2981 } |
2988 // Reload the instance type. | 2982 // Reload the instance type. |
2989 __ mov(result_, FieldOperand(object_, HeapObject::kMapOffset)); | 2983 __ mov(result_, FieldOperand(object_, HeapObject::kMapOffset)); |
2990 __ movzx_b(result_, FieldOperand(result_, Map::kInstanceTypeOffset)); | 2984 __ movzx_b(result_, FieldOperand(result_, Map::kInstanceTypeOffset)); |
2991 call_helper.AfterCall(masm); | 2985 call_helper.AfterCall(masm); |
2992 // If index is still not a smi, it must be out of range. | 2986 // If index is still not a smi, it must be out of range. |
2993 STATIC_ASSERT(kSmiTag == 0); | 2987 STATIC_ASSERT(kSmiTag == 0); |
2994 __ JumpIfNotSmi(index_, index_out_of_range_); | 2988 __ JumpIfNotSmi(index_, index_out_of_range_); |
(...skipping 2424 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
5419 ApiParameterOperand(2), kStackSpace, nullptr, | 5413 ApiParameterOperand(2), kStackSpace, nullptr, |
5420 Operand(ebp, 7 * kPointerSize), NULL); | 5414 Operand(ebp, 7 * kPointerSize), NULL); |
5421 } | 5415 } |
5422 | 5416 |
5423 | 5417 |
5424 #undef __ | 5418 #undef __ |
5425 | 5419 |
5426 } } // namespace v8::internal | 5420 } } // namespace v8::internal |
5427 | 5421 |
5428 #endif // V8_TARGET_ARCH_IA32 | 5422 #endif // V8_TARGET_ARCH_IA32 |
OLD | NEW |